- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我尝试用 gcc 编译非 PIC 代码,我注意到 GCC 生成的汇编代码没有使用纯函数地址来调用,而是添加了一个奇怪的偏移量。
我将 GCC 9.3.0 用作 gcc test.c -o test-nopic -mcmodel=large -no-pie -O0
,代码如下。我遗漏了 -fPIC
。
#include <stdio.h>
int var1 = 1;
int var2 = 2;
void putstr(int* ptr) {
printf("val: %d\n", *ptr);
}
int main() {
putstr(&var1);
putstr(&var2);
}
这里列出了 objdump -wdrC -M intel test-nopic
中的 main()
代码。
000000000040117e <main>:
40117e: 55 push rbp
40117f: 48 89 e5 mov rbp,rsp
401182: 53 push rbx
401183: 48 83 ec 08 sub rsp,0x8
401187: 48 8d 1d f9 ff ff ff lea rbx,[rip+0xfffffffffffffff9] # 401187 <main+0x9>
40118e: 49 bb 79 2e 00 00 00 00 00 00 movabs r11,0x2e79
401198: 4c 01 db add rbx,r11
40119b: 48 b8 30 00 00 00 00 00 00 00 movabs rax,0x30
4011a5: 48 8d 3c 03 lea rdi,[rbx+rax*1]
4011a9: 48 b8 26 d1 ff ff ff ff ff ff movabs rax,0xffffffffffffd126
4011b3: 48 8d 04 03 lea rax,[rbx+rax*1]
4011b7: ff d0 call rax
4011b9: 48 b8 34 00 00 00 00 00 00 00 movabs rax,0x34
4011c3: 48 8d 3c 03 lea rdi,[rbx+rax*1]
4011c7: 48 b8 26 d1 ff ff ff ff ff ff movabs rax,0xffffffffffffd126
4011d1: 48 8d 04 03 lea rax,[rbx+rax*1]
4011d5: ff d0 call rax
4011d7: b8 00 00 00 00 mov eax,0x0
4011dc: 48 83 c4 08 add rsp,0x8
4011e0: 5b pop rbx
4011e1: 5d pop rbp
4011e2: c3 ret
pustr(int*) 的地址是0x401126
。 readelf -l test-nopic
显示文件类型为 EXEC 和以下 header :
Type Offset VirtAddr PhysAddr
FileSiz MemSiz Flags Align
PHDR 0x0000000000000040 0x0000000000400040 0x0000000000400040
0x0000000000000268 0x0000000000000268 R 0x8
INTERP 0x00000000000002a8 0x00000000004002a8 0x00000000004002a8
0x000000000000001c 0x000000000000001c R 0x1
[Requesting program interpreter: /lib64/ld-linux-x86-64.so.2]
LOAD 0x0000000000000000 0x0000000000400000 0x0000000000400000
0x00000000000004d8 0x00000000000004d8 R 0x1000
LOAD 0x0000000000001000 0x0000000000401000 0x0000000000401000
0x0000000000000275 0x0000000000000275 R E 0x1000
LOAD 0x0000000000002000 0x0000000000402000 0x0000000000402000
0x0000000000000168 0x0000000000000168 R 0x1000
LOAD 0x0000000000002e00 0x0000000000403e00 0x0000000000403e00
0x0000000000000238 0x0000000000000240 RW 0x1000
DYNAMIC 0x0000000000002e10 0x0000000000403e10 0x0000000000403e10
0x00000000000001d0 0x00000000000001d0 RW 0x8
NOTE 0x00000000000002c4 0x00000000004002c4 0x00000000004002c4
0x0000000000000044 0x0000000000000044 R 0x4
GNU_EH_FRAME 0x0000000000002010 0x0000000000402010 0x0000000000402010
0x0000000000000044 0x0000000000000044 R 0x4
GNU_STACK 0x0000000000000000 0x0000000000000000 0x0000000000000000
0x0000000000000000 0x0000000000000000 RW 0x10
GNU_RELRO 0x0000000000002e00 0x0000000000403e00 0x0000000000403e00
0x0000000000000200 0x0000000000000200 R 0x1
movabs rax, 0x401126
?最佳答案
随着@Jester 的评论,我解开了这个谜团。我还必须使用 -fno-pic
标志进行编译,以禁用大多数现代 GNU/Linux 发行版中默认启用的 PIE 代码生成。 -no-pie
只是一个链接器选项,-fno-pic
或 -fno-pie
是代码生成选项。参见 32-bit absolute addresses no longer allowed in x86-64 Linux?
问题中的代码(使用 -mcmodel=large -no-pie -O0
编译)使用了对从 rax
寄存器获取的绝对地址的调用。该地址是使用以下代码从 rip
寄存器中计算出来的。
401187: 48 8d 1d f9 ff ff ff lea rbx,[rip+0xfffffffffffffff9] # 401187 <main+0x9>
40118e: 49 bb 79 2e 00 00 00 00 00 00 movabs r11,0x2e79
401198: 4c 01 db add rbx,r11
40119b: 48 b8 30 00 00 00 00 00 00 00 movabs rax,0x30
4011a5: 48 8d 3c 03 lea rdi,[rbx+rax*1]
4011a9: 48 b8 26 d1 ff ff ff ff ff ff movabs rax,0xffffffffffffd126
4011b3: 48 8d 04 03 lea rax,[rbx+rax*1]
4011b7: ff d0 call rax
我计算了存储在 rip
中的地址,它看起来指向 0x40118e。它用于计算函数及其参数的地址(var1 的地址存储在 rdi
寄存器中,它指向 RW LOAD 段)。使用 -fno-pic
标志,函数调用看起来就像我想要的那样。
40115c: 48 bf 30 40 40 00 00 00 00 00 movabs rdi,0x404030
401166: 48 b8 26 11 40 00 00 00 00 00 movabs rax,0x401126
401170: ff d0 call rax
large
):没有 -mcmodel=large
标志 (-no-pie -fno-pic -O0
) 它看起来不一样。静态数据和代码可以通过 32 位相对位移访问,甚至可以通过非 PIE 代码中的 32 位绝对位移访问。这样效率更高,尤其是对于代码;尽可能避免 -mcmodel=large
。如果您只需要一些巨大的静态数组,请使用 -mcmodel=medium
。
这是一个call在相关版本中:对于位置相关代码,它可以将静态地址放入寄存器中,具有高效的mov r32, imm32
( How to load address of function or label into register in GNU Assembler )
401150: bf 30 40 40 00 mov edi,0x404030
401155: e8 cc ff ff ff call 401126 <putstr>
这是一个只有 -fpie
的代码(在我的配置中默认启用)。
1165: 48 8d 3d c4 2e 00 00 lea rdi,[rip+0x2ec4] # 4030 <var1>
116c: e8 c8 ff ff ff call 1139 <putstr>
并且在添加 -fpic
标志后还可以为全局函数启用符号插入,例如共享库:链接后没有真正的区别,只是一个额外的不必要的 mov
首先将 arg 放入 rdi
中。 (这是-O0
的神器:编译快,不好)
1165: 48 8d 05 c4 2e 00 00 lea rax,[rip+0x2ec4] # 4030 <var1>
116c: 48 89 c7 mov rdi,rax
116f: e8 c5 ff ff ff call 1139 <putstr>
gcc -O0
如果我们将 var1
声明为 static
(What does “static” mean in C? ).或者更简单地说,至少使用 -Og
启用优化,更常见的是 -O2
或 -O3
。未优化的代码充满了浪费的指令。
关于c - 为什么 GCC -mcmodel=large 在没有 -fPIC 的情况下即使使用 -no-pie 标志也会向函数调用添加偏移量?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/61986265/
使用 Qt 时,CMake 会自动将 -fPIC 标志添加到编译选项。我想使用 -fpic,所以我检查了所有 Cmake 变量并将 -fPIC 替换为 -fpic。 cmake_minimum_req
环境:Ubuntu 16.04 在我的实验中,我运行了以下命令: gcc -c 1.c gcc -c -fPIC 2.c gcc -shared 1.o 2.o -o libmyxxx.so 我需要公
我已经阅读了 gcc 手册页,但我仍然无法理解 -fpic 和 -fPIC 之间的区别。有人可以用非常简单明了的方式解释一下吗? 相关问题: What does -fPIC mean when bui
海湾合作委员会错误: relocation R_X86_64_32S against `ebx' can not be used when making a shared object; recomp
我已经下载了一些我想从源代码编译的 C++。从命令行运行 cmake ../src 和 make 后,在构建目录中,出现以下错误: /usr/bin/ld: /usr/local/lib/libBul
我在 Google 上找不到直接的答案,而且由于我已经很长时间没有在 Linux 上做过任何事情,所以希望能在这里找到帮助。构建共享对象时,我在 Ubuntu 上遇到链接错误。链接器告诉我应该使用 -
我目前正在 cmake 环境中构建共享库 (lib1.so)。lib1.so 依赖于外部静态库 libLASlib.a(如有必要,我可以重新编译)。到目前为止,一切都在 Windows 上运行良好,但
我的应用程序需要在运行时加载一个或多个算法,为此我使用 .so。问题是这些库除了我的应用程序之外没有被任何其他进程使用,所以不需要与其他人共享 .text 部分。 .so 的某些部分来 self 预先
我正在尝试使用 luarocks 安装 numlua: luarocks make numlua-0.3-1.rockspec 但出现以下错误: /usr/bin/ld: numlua.o: relo
即使我可能不打算将生成的可重定位目标文件链接到共享库中,始终使用 -fPIC 进行编译是否有任何危害? 最佳答案 好吧,总是使用 -fPIC 进行编译会产生更臃肿和更慢的代码。 差异在 x64 目标上
我想检查共享库是否使用 -fPIC 标志编译。有哪些可能的方法(在 Linux 上,x86_64)检查这个? -fPIC 是否隐含(从而使检查变得多余?) 最佳答案 是的,GCC 不允许您在没有 -f
我知道 -fPIC 是共享库所必需的,并且知道为什么。 但是,我对这个问题不是很清楚: -fPIC 不应在构建可执行文件或静态库期间使用吗? 最佳答案 Should -fPIC never be us
我有 MPICH 3.0.4安装在我的机器上(Ubuntu 12.04)。我正在尝试安装一个名为 Qthreads 的库我之前使用过并成功安装过它(除了安装了 MPICH2 包)。配置工作正常: ./
我正在构建一个共享库。我只需要一个函数就可以公开。 共享库由几个目标文件和几个静态库构建而成。链接器提示一切都应该用 -fPIC 构建.所有目标文件和大多数静态库都是在没有这个选项的情况下构建的。 这
-fpic 标志有什么作用? 我想安装一个库( OpenSFM ),据说它需要使用 -fPIC 编译标志构建和安装 Ceres Solver。 问题是我已经安装了没有 -fPIC 标志的 Ceres
我想将目标文件和静态库组合成一个共享库,但是静态库不能暴露,它只在进入共享库的目标文件中引用。我认为在这种情况下,我不需要使用 -fPIC 编译静态库,但我不知道如何告诉链接器我不会使用静态库中的符号
今天,当尝试使用 mongodb c++ 客户端构建我的 so lib 项目时,出现错误: /usr/bin/ld: /usr/lib/gcc/x86_64-linux-gnu/4.8/../../.
我正在为一个类编写一些 C 代码,但我遇到了需要在 make 文件中完成的编译器问题,当它运行时它总是给出错误 usr/bin/ld: main.o: 重定位 R_X86_64_32S 对符号“被积函
在centos中./configure asterisk后我输入make命令但是出现了以下错误 /usr/bin/ld:/usr/local/lib/liblua.a(lapi.o): 创建共享对象时
我正在尝试在 linux 中创建一个共享库,但在尝试编译时出现以下错误: relocation R_X86_64_32S against `.rodata' can not be used when
我是一名优秀的程序员,十分优秀!