我正在学习Linux中的链接器是如何工作的。下面是我的代码示例:
// main.c
int printf(const char *format, ...);
void func1(int i)
{
printf("%d\n", i);
}
int main(void)
{
func1(1);
return 0;
}
我执行命令gcc -c main.c
并获得一个名为main.o
的obj文件。
然后我执行命令objdump -r main.o
,输出如下:
main.o: file format elf64-x86-64
RELOCATION RECORDS FOR [.text]:
OFFSET TYPE VALUE
0000000000000011 R_X86_64_32 .rodata
000000000000001b R_X86_64_PC32 printf-0x0000000000000004
000000000000002c R_X86_64_PC32 func1-0x0000000000000004
RELOCATION RECORDS FOR [.eh_frame]:
OFFSET TYPE VALUE
0000000000000020 R_X86_64_PC32 .text
0000000000000040 R_X86_64_PC32 .text+0x0000000000000022
如果我是对的,objdump -r
将向我们显示obj文件中的所有重定位表。在本例中,printf
和func1
都被放入重定位表中。
printf
没有在这个C文件中定义,所以需要重新定位它,但是为什么func1
也可以在重定位表中找到呢?据我所知,func1
应该定义良好,可以在.text
部分找到,它不需要重新定位,对吗?
发布于 2020-02-21 21:25:06
重定位记录是确定区段地址时需要调整的位置。在包括x86在内的许多体系结构中,可以编写位置无关的代码,通过从区段内的一个位置到另一个位置的偏移量来引用对象或函数(因此,当整个区段移动时,偏移量不会改变),并且可以通过具有一些附加的链接器辅助的跨区段的偏移量来引用对象或函数(因此,在链接之后,偏移量不会改变,并且整个组合图像可以在内存中移位)。
然而,这是可能的事实并不意味着您的构建工具正在使用位置无关的代码。无论出于什么原因,您所使用的工具以及您所使用的开关似乎都不能生成与位置无关的代码。因此,当存在对printf
或func1
的引用时,会生成对它的可重定位引用,从而导致必须由链接器或加载器调整的重定位记录。
https://stackoverflow.com/questions/60334861
复制相似问题