.got和.got.plt部分有什么区别?

scd*_*dmb 13 linux elf dynamic-linking

ELF格式的.got和.got.plt部分有什么区别?

Joh*_*ugo 12

我之前的评论转向正确,下面的例子让事情变得清晰.

这些是我的32位i686-linux /lib/libm.so的重定位

Relocation section '.rel.dyn' at offset 0x32b8 contains 8 entries:
 Offset     Info    Type            Sym.Value  Sym. Name
00025030  00000008 R_386_RELATIVE   
00024fd8  00005706 R_386_GLOB_DAT    00025034   _LIB_VERSION
00024fdc  00000406 R_386_GLOB_DAT    00000000   __gmon_start__
00024fe0  00000506 R_386_GLOB_DAT    00000000   _Jv_RegisterClasses
00024fe4  00000806 R_386_GLOB_DAT    00000000   _rtld_global_ro
00024fe8  00000906 R_386_GLOB_DAT    00000000   stderr
00024fec  00013006 R_386_GLOB_DAT    0002507c   signgam
00024ff0  00000e06 R_386_GLOB_DAT    00000000   __cxa_finalize

Relocation section '.rel.plt' at offset 0x32f8 contains 12 entries:
 Offset     Info    Type            Sym.Value  Sym. Name
00025000  00000107 R_386_JUMP_SLOT   00000000   fputs
00025004  00000207 R_386_JUMP_SLOT   00000000   __errno_location
00025008  00000307 R_386_JUMP_SLOT   00000000   sprintf
0002500c  00000407 R_386_JUMP_SLOT   00000000   __gmon_start__
00025010  00000607 R_386_JUMP_SLOT   00000000   strtod
00025014  00000707 R_386_JUMP_SLOT   00000000   __assert_fail
00025018  00000a07 R_386_JUMP_SLOT   00000000   strlen
0002501c  00000b07 R_386_JUMP_SLOT   00000000   strtof
00025020  00000c07 R_386_JUMP_SLOT   00000000   fwrite
00025024  00000d07 R_386_JUMP_SLOT   00000000   strtold
00025028  00005e07 R_386_JUMP_SLOT   00005970   matherr
0002502c  00000e07 R_386_JUMP_SLOT   00000000   __cxa_finalize
Run Code Online (Sandbox Code Playgroud)

看看你注意到有两个重定位部分,即.rel.dyn和.rel.plt.你可以看到,对于.rel.plt所有的重定位类型R_386_JUMP_SLOT的,这意味着他们是在另一只手.rel.dyn几乎所有的重定位分支重定位R_386_GLOB_DAT这意味着全局变量搬迁.

.symtab和.dynsym之间存在另一个细微差别.虽然第一个包含静态链接编辑期间使用的所有符号的引用,但后者仅包含动态链接所需的那些符号.因此,上面提到的重定位仅涉及.dynsym部分.

  • IIRC,对于已获取地址的函数,还需要一个“.got”条目;编译器通常选择要求早期绑定,以便可以从 GOT 加载最终地址,而不是仅仅将 PLT 存根的地址作为函数指针。 (2认同)