婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > 從實(shí)例分析ELF格式的.gnu.hash區(qū)與glibc的符號查詢?nèi)^程

從實(shí)例分析ELF格式的.gnu.hash區(qū)與glibc的符號查詢?nèi)^程

熱門標(biāo)簽:防城港市ai電銷機(jī)器人 臨滄移動外呼系統(tǒng)哪家有 不同的地圖標(biāo)注 寧夏保險智能外呼系統(tǒng)哪家好 隨州銷售外呼系統(tǒng)平臺 溫嶺代理外呼系統(tǒng) 交行外呼系統(tǒng)有哪些 怎么更改地圖標(biāo)注電話 激戰(zhàn)黃昏地圖標(biāo)注說明

前言

ELF格式的.gnu.hash節(jié)在設(shè)計(jì)上比較復(fù)雜,直接從glibc源碼進(jìn)行分析的難度也比較大。今天靜下心來看了這篇精彩的文章,終于將布隆濾波器、算數(shù)運(yùn)算轉(zhuǎn)為位運(yùn)算等一系列細(xì)節(jié)搞懂了(值得一提的是,這篇博客十分值得花一些時間讀懂,它不僅對總體有一個較好的描述,而且還涉及了許多有益的實(shí)現(xiàn)細(xì)節(jié))。但本人愚鈍異常,沒有一個完整的walkthrough就不能覺得自己真的搞懂了一個東西。所以本文從查找一個符號的真實(shí)情況出發(fā),把ELF格式是如何組織一個符號,以及動態(tài)鏈接器如何讀取并處理這些信息以進(jìn)行符號查詢的全過程詳細(xì)地講清楚。
本文假定讀者已經(jīng)讀過上文中提到的博客,并理解布隆濾波器,GNU hash采用的單一哈希策略,把取模轉(zhuǎn)為取與這些名詞。在后續(xù)有時間時我可能會對它們進(jìn)行簡單介紹,但珠玉在前讓人確實(shí)不想獻(xiàn)丑。

本文的實(shí)現(xiàn)以及so文件均以glibc 2.31為準(zhǔn)。

符號哈希,符號表與字符表

一個符號的相關(guān)信息會在ELF文件中dynamic section的三塊出現(xiàn):.gnu.hash對應(yīng)的符號哈希,.dynsym對應(yīng)的動態(tài)符號表,.dynstr對應(yīng)的字符表。在查找符號時,動態(tài)鏈接器首先從.gnu.hash中進(jìn)行查詢,得到該符號在動態(tài)符號表中的偏移。動態(tài)鏈接器根據(jù)這個偏移讀出一個符號,并找到這個符號的名字在字符表中的偏移。從字符表中讀出符號的名稱如果與要查找的符號匹配,則找到了這個符號,再從符號表中讀出符號的相關(guān)信息并返回。

64位ELF格式的符號定義如下:

// in elf.h>
typedef struct
{
  // 32 bits
  Elf64_Word	st_name;		/* Symbol name (string tbl index) */
  // 8 bit
  unsigned char	st_info;		/* Symbol type and binding */
  // 8 bit
  unsigned char st_other;		/* Symbol visibility */
  // 16 bits
  Elf64_Section	st_shndx;		/* Section index */
  // 64 bits
  Elf64_Addr	st_value;		/* Symbol value */
  // 64 bits
  Elf64_Xword	st_size;		/* Symbol size */
} Elf64_Sym;

這個數(shù)據(jù)結(jié)構(gòu)占用內(nèi)存的大小為24B,這也是合理安排成員順序以節(jié)約文件大小的一個例子。

.gnu.hash的結(jié)構(gòu)

glibc使用如下函數(shù)從ELF文件中讀取符號哈希相關(guān)信息:

// in elf/dl-lookup.c
void
_dl_setup_hash (struct link_map *map)
{
  Elf_Symndx *hash;

  if (__glibc_likely (map->l_info[ELF_MACHINE_GNU_HASH_ADDRIDX] != NULL))
    {
      // 一個指向32位長內(nèi)存的指針,用來讀取哈希相關(guān)變量,故名hash32
      Elf32_Word *hash32
	= (void *) D_PTR (map, l_info[ELF_MACHINE_GNU_HASH_ADDRIDX]);
      map->l_nbuckets = *hash32++;
      Elf32_Word symbias = *hash32++;
      Elf32_Word bitmask_nwords = *hash32++;
      /* Must be a power of two.  */
      assert ((bitmask_nwords  (bitmask_nwords - 1)) == 0);
      map->l_gnu_bitmask_idxbits = bitmask_nwords - 1;
      map->l_gnu_shift = *hash32++;

      map->l_gnu_bitmask = (ElfW(Addr) *) hash32;
      hash32 += __ELF_NATIVE_CLASS / 32 * bitmask_nwords;

      map->l_gnu_buckets = hash32;
      hash32 += map->l_nbuckets;
      map->l_gnu_chain_zero = hash32 - symbias;

      /* Initialize MIPS xhash translation table.  */
      ELF_MACHINE_XHASH_SETUP (hash32, symbias, map);

      return;
    }
  // 以下處理古老的DT_HASH項(xiàng),現(xiàn)已不用
  if (!map->l_info[DT_HASH])
    return;
  hash = (void *) D_PTR (map, l_info[DT_HASH]);//Q: what about some non-GNU ELFs

  map->l_nbuckets = *hash++;
  /* Skip nchain.  */
  hash++;
  map->l_buckets = hash;
  hash += map->l_nbuckets;
  map->l_chain = hash;
}

上述代碼讀取了關(guān)鍵變量賦值:l_nbuckets,symbias,bitmask_nwords,l_gnu_shift,l_gnu_buckets,l_gnu_chain_zero。其中,以“l(fā)”開頭的變量存儲在ELF文件的link_map中,具體定義見link.h>。還有不是從文件中讀出的變量l_gnu_bitmask_idxbits,它們的具體含義為:

  • l_nbuckets:使用哈希桶的數(shù)量
  • symbias:動態(tài)符號表中外部不能訪問的符號數(shù)量,但它們?nèi)匀徽加昧藙討B(tài)符號表項(xiàng)
  • bitmask_nwords:使用bitmask_nwords個字作為布隆濾波器的向量
  • l_gnu_shift:為使用同一哈希函數(shù)實(shí)現(xiàn)k=2的布隆濾波器,需要右移的位數(shù)
  • l_gnu_buckets:哈希桶的開始地址
  • l_gnu_chain_zero:符號哈希值的開始地址
  • l_gnu_bitmask_idxbits:為對bitmask_nwords取模化為取與,由bitmask_nwords-1而來

為了便于理解,將.gnu.hash節(jié)中的內(nèi)容畫成示意圖:

以libc為例。檢查對應(yīng)字段的值:

$ objdump -s /lib/x86_64-linux-gnu/libc.so.6 | grep .gnu.hash -A 5

Contents of section .gnu.hash:
 38a0 (f3030000)        (0c000000)      (00010000)            (0e000000)     ................
      ->l_nbuckets=1011 ->symbias=12    ->bitmask_nwords=256  ->l_gnu_shift=14
 38b0 (00301044 a0200201) (8803e690 c5458c00)  .0.D. .......E..
      ->第一個bloom word 0x010220a044103000                  
 38c0 c4005800 07840070 c280010d 8a0c4104  ..X....p......A.
 38d0 10008840 32082a40 88543c2d 200e3248  ...@2.*@.T- .2H
 38e0 2684c08c 04080002 020ea1ac 1a0666c8  .............f.

可以看到symbias=12,即有12個內(nèi)部符號:

$ readelf -s /lib/x86_64-linux-gnu/libc.so.6 | head -n 20

Symbol table '.dynsym' contains 2367 entries:
   Num:    Value          Size Type    Bind   Vis      Ndx Name
     0: 0000000000000000     0 NOTYPE  LOCAL  DEFAULT  UND 
     1: 0000000000000000     0 NOTYPE  WEAK   DEFAULT  UND __libpthread_freeres
     2: 0000000000000000     0 OBJECT  GLOBAL DEFAULT  UND _rtld_global@GLIBC_PRIVATE (33)
     3: 0000000000000000     0 OBJECT  GLOBAL DEFAULT  UND __libc_enable_secure@GLIBC_PRIVATE (33)
     4: 0000000000000000     0 FUNC    GLOBAL DEFAULT  UND __tls_get_addr@GLIBC_2.3 (34)
     5: 0000000000000000     0 FUNC    GLOBAL DEFAULT  UND _dl_exception_create@GLIBC_PRIVATE (33)
     6: 0000000000000000     0 OBJECT  GLOBAL DEFAULT  UND _rtld_global_ro@GLIBC_PRIVATE (33)
     7: 0000000000000000     0 FUNC    GLOBAL DEFAULT  UND __tunable_get_val@GLIBC_PRIVATE (33)
     8: 0000000000000000     0 FUNC    GLOBAL DEFAULT  UND _dl_find_dso_for_object@GLIBC_PRIVATE (33)
     9: 0000000000000000     0 NOTYPE  WEAK   DEFAULT  UND _dl_starting_up
    10: 0000000000000000     0 NOTYPE  WEAK   DEFAULT  UND __libdl_freeres
    11: 0000000000000000     0 OBJECT  GLOBAL DEFAULT  UND _dl_argv@GLIBC_PRIVATE (33)
    12: 00000000000ab970    33 FUNC    GLOBAL DEFAULT   16 __strspn_c1@GLIBC_2.2.5
    13: 0000000000089260   352 FUNC    GLOBAL DEFAULT   16 putwchar@@GLIBC_2.2.5
    14: 00000000001324f0    20 FUNC    GLOBAL DEFAULT   16 __gethostname_chk@@GLIBC_2.4
    15: 00000000000ab9a0    44 FUNC    GLOBAL DEFAULT   16 __strspn_c2@GLIBC_2.2.5
    16: 000000000014f580   218 FUNC    GLOBAL DEFAULT   16 setrpcent@@GLIBC_2.2.5

可見符號0-11為內(nèi)部符號。

查找符號

下面以查找符號printf為例,介紹符號查找的過程。

首先使用下面的哈希函數(shù)生成符號的32位哈希:

// in elf/dl-lookup.c
static uint_fast32_t
dl_new_hash (const char *s)
{
  uint_fast32_t h = 5381;
  for (unsigned char c = *s; c != '\0'; c = *++s)
    h = h * 33 + c;
  return h  0xffffffff;
}

得到printf的哈希值為0x156b2bb8。
隨后計(jì)算布隆濾波器需要的兩個hashbit:

unsigned int hashbit1 = new_hash  (__ELF_NATIVE_CLASS - 1);
    unsigned int hashbit2 = ((new_hash >> l->l_gnu_shift)  (__ELF_NATIVE_CLASS - 1));

得到hashbit1 = 56,hashbit2 = 44。
找到該hash對應(yīng)的bloom word:

  const Elf64_Addr *bitmask = l->l_gnu_bitmask;
    // l->l_gnu_bitmask_idxbits = bitmask_nwords - 1,將取模變?yōu)槿∨c
    // (new_hash / __ELF_NATIVE_CLASS)  l->l_gnu_bitmask_idxbits = 174
    Elf64_Addr bitmask_word = bitmask[(new_hash / __ELF_NATIVE_CLASS)  l->l_gnu_bitmask_idxbits];

printf對應(yīng)的hash在第174個bloom word處,它的值位于bloom word的開始地址0x38b0+174*8=3e20
檢查3e20處對應(yīng)的值:

$ objdump -s /lib/x86_64-linux-gnu/libc.so.6 | grep " 3e20 "

 3e20 d0884a41 c0703429 10ec4303 92003103  ..JA.p4)..C...1.

其bloom word為0x293470c0414a88d0。
將其右移56位:0b0010 1001
將其右移44位:0b10 1001 0011 0100 0111
二者的最后一位均為1,說明布隆濾波器不能拒絕這個哈希值。

這時在對應(yīng)的哈希桶上進(jìn)行尋找:

Elf32_Word bucket = l->l_gnu_buckets[new_hash % l->l_nbuckets];

由于0x156b2bb8 % 1011 = 295,需要找到第296個哈希桶。
而哈希桶的起始地址為l_gnu_bitmask + 64 / 32 * bitmask_nwords = 0x40b0,對應(yīng)哈希桶的地址為0x40b0+295*4=0x454c。
查看0x454c處對應(yīng)的哈希桶內(nèi)容:

$ objdump -s /lib/x86_64-linux-gnu/libc.so.6 | grep " 4540 "

 4540 77020000 00000000 7a020000 **7c020000**  w.......z...|...

哈希桶的內(nèi)容為0x27c。
而l_gnu_chain_zero的地址為:

  l_gnu_chain_zero = l_gnu_buckets + l_nbuckets - symbias;

可計(jì)算出l_gnu_chain_zero的地址為0x504c,所以第296個哈希桶包含的真正哈希位于0x504c+27c*4=0x5a3c
查看具體的哈希內(nèi)容:

$ objdump -s /lib/x86_64-linux-gnu/libc.so.6 | grep " 5a30 " -A 2

 5a30 ade8dbbb 142dcb13 bb86f85f e6952000  .....-....._.. .
 5a40 **b82b6b15** 0a05f1d5 deb6427f 856177fd  .+k.......B..aw.
 5a50 1ae585e7 ec296fa8 1ae585e7 29ce248f  .....)o.....).$.

于0x5a40處找到我們之前計(jì)算的哈希0x156b2bb8(注意小端序)。
此時,這個符號在.gnu.hash的下標(biāo),就是它在動態(tài)符號表中的(下標(biāo)-symbias)。但由于之前l(fā)_gnu_chain_zero已經(jīng)整體減掉了symbias,所以此處用該符號的地址減掉l_gnu_chain_zero可直接得到符號在符號表中的下標(biāo)。
0x5a40 - 0x504c = 0x9f4 = 2548,由于一個哈希值為4字節(jié),故下標(biāo)為2548 / 4 = 637

找到動態(tài)符號表的起始地址:

$ objdump -s /lib/x86_64-linux-gnu/libc.so.6 | grep .dynsym -A 1

Contents of section .dynsym:
 07548 00000000 00000000 00000000 00000000  ................

上文中提到,64位ELF文件中一個符號的長度位24字節(jié),故符號在符號表上的起始地址應(yīng)當(dāng)為0x7548 + 24*637 = 0xb100
找到動態(tài)符號表對應(yīng)位置的內(nèi)容:

$ objdump -s /lib/x86_64-linux-gnu/libc.so.6 | grep " 0b0f8 " -A 1

 0b0f8 16000000 00000000 **f3040000** 12001000  ................
 0b108 104e0600 00000000 cc000000 00000000  .N..............

讀出符號在字符表上的偏移量為0x4f3。
找到字符表的起始地址:

$ objdump -s /lib/x86_64-linux-gnu/libc.so.6 | grep .dynstr -A 1

Contents of section .dynstr:
 15330 00786472 5f755f6c 6f6e6700 5f5f7763  .xdr_u_long.__wc

起始地址為0x15330,故該符號的地址為0x15330 + 0x4f3 = 0x15823

讀出字符表對應(yīng)位置的值:

$ objdump -s /lib/x86_64-linux-gnu/libc.so.6 | grep " 15820 " -A 1

 15820 494f5f**70** 72696e74 66007265 67697374  IO_printf.regist
 15830 65725f70 72696e74 665f6675 6e637469  er_printf_functi

查找到了符號printf,它是IO_printf的別名,在字符表中為了節(jié)省空間將二者合并了。

這樣,就完成了一次符號查詢的全過程。

以上就是從實(shí)例分析ELF格式的.gnu.hash區(qū)與glibc的符號查找的詳細(xì)內(nèi)容,更多關(guān)于ELF格式的.gnu.hash區(qū)與glibc的符號查找的資料請關(guān)注腳本之家其它相關(guān)文章!

標(biāo)簽:青海 紅河 阜陽 河源 忻州 無錫 哈密 沈陽

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《從實(shí)例分析ELF格式的.gnu.hash區(qū)與glibc的符號查詢?nèi)^程》,本文關(guān)鍵詞  從,實(shí)例分析,ELF,格式,的,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《從實(shí)例分析ELF格式的.gnu.hash區(qū)與glibc的符號查詢?nèi)^程》相關(guān)的同類信息!
  • 本頁收集關(guān)于從實(shí)例分析ELF格式的.gnu.hash區(qū)與glibc的符號查詢?nèi)^程的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    欧美三级日韩三级国产三级| 亚洲欧美综合网| 欧美日韩一区二区电影| 欧美一区二区三区的| 久久先锋影音av| 亚洲欧洲成人自拍| 奇米亚洲午夜久久精品| 粉嫩欧美一区二区三区高清影视| 91麻豆福利精品推荐| 日韩片之四级片| 亚洲欧美国产三级| 另类欧美日韩国产在线| 91麻豆6部合集magnet| 精品国产伦理网| 亚洲在线视频一区| 国产高清不卡二三区| 久久久噜噜噜久久人人看| 亚洲综合av网| 大白屁股一区二区视频| 日韩视频一区二区三区在线播放| 国产精品免费久久久久| 免费的成人av| 在线精品观看国产| 国产精品嫩草影院com| 久久精品免费观看| 欧美日韩精品欧美日韩精品一| 欧美国产在线观看| 免费成人你懂的| 欧美三级电影在线观看| 中文字幕av不卡| 国产乱人伦精品一区二区在线观看| 欧美日韩在线一区二区| 亚洲欧美在线观看| 国产成人av一区二区三区在线 | 日韩一区二区三区四区五区六区| 欧美激情一区二区三区| 精品综合免费视频观看| 91精品国产91热久久久做人人| 亚洲精品国产品国语在线app| 成人一区二区三区| 久久久精品综合| 国产美女一区二区三区| 日韩欧美国产一区在线观看| 日韩成人一级片| 欧美丰满嫩嫩电影| 婷婷一区二区三区| 欧美美女bb生活片| 日本网站在线观看一区二区三区| 色八戒一区二区三区| 成人免费在线播放视频| 成人美女视频在线观看| 国产日韩欧美在线一区| 国产精品99久久不卡二区| 欧美成人aa大片| 精品亚洲国产成人av制服丝袜| 欧美一级欧美三级在线观看| 日本欧美加勒比视频| 欧美一区二区三区思思人 | 91精品国产91久久久久久最新毛片| 亚洲精品日日夜夜| 一本大道久久a久久精二百 | av高清久久久| 国产精品理论在线观看| 国产成人av福利| 国产精品乱人伦中文| 99精品久久久久久| 亚洲综合在线视频| 欧美日本在线一区| 日本不卡视频一二三区| 久久久久久久电影| 波多野结衣在线aⅴ中文字幕不卡| 中文字幕乱码久久午夜不卡| 成人手机电影网| 一区二区三区蜜桃| 欧美一级高清片在线观看| 国产麻豆视频一区二区| 国产精品视频一二三| 久久婷婷国产综合精品青草| 国产aⅴ精品一区二区三区色成熟| 欧美国产精品专区| 国产suv精品一区二区883| 亚洲精品免费在线播放| 欧美日韩国产免费一区二区| 国产精品一区在线观看乱码| 国产精品麻豆视频| 69堂亚洲精品首页| 国产精品一区二区在线播放 | 国产精品一二二区| 亚洲欧美日韩中文播放| 欧美二区在线观看| 成人在线视频首页| 午夜精品久久久久久不卡8050 | 久久er99热精品一区二区| 中文字幕va一区二区三区| 欧美日韩国产a| 国产激情精品久久久第一区二区| 亚洲视频在线观看三级| 678五月天丁香亚洲综合网| 国产永久精品大片wwwapp| 亚洲免费色视频| 日韩精品中文字幕一区 | 中文字幕亚洲电影| 日韩一区二区精品在线观看| 972aa.com艺术欧美| 日本欧洲一区二区| 一区二区三区在线高清| 久久久777精品电影网影网| 欧美日韩在线播| 成人高清视频在线观看| 久久99久久久久久久久久久| 亚洲与欧洲av电影| 亚洲图片欧美激情| 久久亚洲综合av| 制服丝袜中文字幕亚洲| 一本大道久久a久久综合婷婷| 国产成人一区二区精品非洲| 婷婷综合另类小说色区| 亚洲视频在线一区| 久久先锋资源网| 欧美一区二区三级| 欧美调教femdomvk| 91视频免费观看| 国产精品一二三区在线| 国产伦理精品不卡| 久久国产福利国产秒拍| 日韩av一区二区三区| 一区二区三区精品在线| 亚洲丝袜自拍清纯另类| 国产免费久久精品| 国产三级久久久| 久久久久国产精品麻豆| 久久众筹精品私拍模特| 日韩欧美国产精品| 日韩一二三四区| 欧美一级高清大全免费观看| 91麻豆精品国产91久久久更新时间| 日本韩国欧美在线| 欧洲国内综合视频| 色老汉一区二区三区| 色老汉av一区二区三区| 91麻豆精品秘密| 在线一区二区视频| 欧美在线视频你懂得| 欧美色网一区二区| 欧美久久久久免费| 日韩精品中文字幕一区二区三区| 日韩免费电影一区| 久久综合给合久久狠狠狠97色69| 26uuu精品一区二区在线观看| 日本va欧美va精品| 日韩成人午夜电影| 九九**精品视频免费播放| 国产一区二三区好的| 成人av在线资源网站| 色又黄又爽网站www久久| 欧美三级中文字幕| 7777精品伊人久久久大香线蕉经典版下载 | 国产精品系列在线播放| 成人免费毛片嘿嘿连载视频| 暴力调教一区二区三区| 欧美少妇一区二区| 9191国产精品| 国产欧美日韩中文久久| 中文字幕日韩一区| 手机精品视频在线观看| 久久99热狠狠色一区二区| 成人视屏免费看| 在线观看亚洲一区| 日韩精品一区国产麻豆| 国产欧美日产一区| 亚洲永久精品国产| 久久国产视频网| 91浏览器入口在线观看| 欧美日韩成人综合天天影院| 亚洲精品一区二区三区99| 亚洲欧美怡红院| 美女一区二区视频| 成人高清免费在线播放| 欧美日韩mp4| 中文幕一区二区三区久久蜜桃| 亚洲一级在线观看| 国产一区二区三区| 欧美日韩国产系列| 国产亚洲一本大道中文在线| 亚洲第一福利一区| 成人免费福利片| 精品久久久久久久久久久院品网| 欧美日韩一区精品| 国产精品美女一区二区在线观看| 久久久久高清精品| 亚洲大尺度视频在线观看| 国产v日产∨综合v精品视频| 欧美日韩高清在线| 亚洲欧美aⅴ...| 国产乱码一区二区三区| 欧美体内she精高潮| 国产精品欧美综合在线| 激情文学综合丁香| 欧美日韩免费在线视频| 亚洲欧美另类在线|