- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我正在使用 NASM 编译器在 Linux 中编译一个 NASM 64 位共享对象并与 ld 链接。它使用以下字符串编译为目标文件:
sudo nasm -felf64 Test_File.asm
我用ld链接:
sudo ld -shared Test_File.o -o Test_File.so
我收到以下错误:
Relocation R_X86_64_32S against '.data' can not be used when making a shared object; recompile with -fPIC
ld: final link failed: Nonrepresentable section on output
不幸的是,NASM 编译器没有 -fPIC 选项。
在阅读了许多有关在 Linux 中为 64 位共享库编写位置无关代码的资源后,我非常了解这个问题,但我仍然不清楚我需要对哪些指令进行更改才能定位-独立于 64 位 NASM。例如,是否所有涉及命名变量的指令都需要是“rel”——例如,movsd xmm0,[rel abc] 而不是 movsd xmm0,[abc]?我知道 R_X86_64_32S 表示 32 位寻址,但我的代码中没有任何 32 位寻址。
此外,32 位和 64 位在如何编写与位置无关的代码方面存在显着差异,一些资源仅集中在 32 位代码上。甚至 NASM 手册中的第 9.2 节编写 NetBSD/FreeBSD/OpenBSD 和 Linux/ELF 共享库也不清楚如何将 64 位代码更改为与位置无关的代码。该部分重点介绍 32 位代码(使用全局偏移表),它不(基于其他研究)用于 64 位代码。
根据需要,文件以 [BITS 64] 和 [default rel] 开头。
数据段声明为section .data align=16
.data 段中的每个变量都定义为 dq,例如,number: dq 0。
文件顶部包含以下格式的导出:global ABC:function。
我怀疑只有数据移动指令会受到影响——数学指令不会。对于 realloc 的外部调用,我添加了 wrt ..plt 特殊符号,但我仍然遇到相同的错误。
这是我的问题:
是否所有的 mov 指令都需要用“rel”关键字重写,例如,mov rax,[rel abc] 而不是 mov rax,[abc]?
lea 指令是否需要更改(例如,lea rdi,[rel abc])?
还有其他需要特殊处理的指令类型吗?
我不会在这里发布完整的(很长的)nasm 代码 list ,因为我不想逐行分析。我只想知道哪些指令类型(例如,mov、cmp、jmp、lea)需要为 64 位相对寻址重写,以及如何重写。它是否仅涉及对数据部分中定义的变量的访问(例如,mov rcx,[abc],其中 abc 在数据部分中定义为 abc:dq 0)。
总而言之,我的问题是:由于 NASM 编译器没有 fPIC 选项,我需要为 64 位 NASM 的位置无关代码做哪些更改?我当然不是说逐行,而是需要添加或重写哪些类型的指令。
非常感谢。
最佳答案
Unfortunately, the NASM compiler does not have a -fPIC option.
当然不是;这是编译器 的代码生成选项。 NASM 是一个汇编器,而不是编译器;它汇编的指令由源文件设置,而不是命令行选项。 (错误消息假设人们在编译器输出上使用 ld
,而不是手写的 asm。)
重新编译 = 重新生成 asm 指令,而不是使用不同的选项重新组装相同的 asm。编译器就是您的大脑。
- Do all mov instructions need to be rewritten with the "rel" keyword
不,您可以像普通人一样在文件顶部使用 default rel
而不是修改每个寻址模式以显式使用 [rel foo ]
。
- and 3. Are there any other instruction types that need special handling?
与mov
指令无关,与寻址方式有关。所有指令(包括 LEA)都使用相同的 ModR/M + 可选 SIB + disp0/8/32 编码来寻址模式。 (除了 mov
的一种形式,它在加载/存储 AL/AX/EAX/RAX 时可以使用 64 位绝对地址。但你也不希望这样。)
您还需要避免将地址用作 32 位绝对立即操作数。因此,如果您要将地址放入寄存器,则需要一个 RIP 相关的 LEA,而不是可以在位置相关代码中使用的更高效的 5 字节 mov-immediate。
;; putting a label address into a register
default rel
mov edi, my_string ; optimal in position-dependent executables on Linux
lea rdi, [my_string] ; optimal otherwise, best you can do for PIC/PIE
mov rdi, my_string ; Never use: 64-bit absolute is inefficient
唯一应该使用 64 位绝对地址的情况是在 .data
或 .rodata
中用于跳转表的内容或其他指向静态地址的指针。不在代码中;改用 RIP-relative。
显然,您必须避免 [disp32 + reg]
寻址模式,例如 [array + rdi]
或 [array + rdx*4]
. 唯一的 RIP 相对寻址模式是 [RIP + rel32]
;其他模式仍然使用 32 位位移作为符号扩展的 32 位绝对值(因此它可以是一个常量偏移量,如 1024,根本不是地址)。
Mach-O 64-bit format does not support 32-bit absolute addresses. NASM Accessing Array (MachO64 从不允许 32 位绝对值,因此它与 Linux/ELF PIC 对象的限制相同)
I know that R_X86_64_32S indicates 32-bit addressing, but I don't have any 32-bit addressing in my code.
[abs foo]
是一个 disp32 符号扩展到 64。这就是重定位类型为 32S 的原因。相比之下,mov edi, foo
使用 R_X86_64_32
。
它不是 32 位地址大小,但绝对地址仍必须编码为 32 位有符号整数。这在 PIE/PIC 对象中是不允许的,它必须可重定位在 64 位地址空间中的任何位置。
相关:
gcc -fno-plt
风格从 NASM PIE/PIC 代码调用 libc 函数,使用 call [rel printf wrt ..got]
PIC 库应该支持符号插入。参见 Sorry state of dynamic libraries on Linux了解更多。
如果您想要有效地内部访问您自己的global
符号(无需通过 GOT),您可能需要为它们定义弱别名,这些别名具有“隐藏的”ELF能见度。或者简单地将 2 个标签放在同一个地方,一个全局的一个隐藏。参见 7.9.5 elf Extensions to the GLOBAL Directive在 NASM 手册中:
global hashlookup:function hidden
此外,NASM 手册说明:
Declaring the type and size of global symbols is necessary when writing shared library code. For more information, see section 9.2.4.
关于linux - NASM Linux 共享对象错误 : Relocation R_X86_64_32S against '.data' ,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58106310/
这个问题在这里已经有了答案: Why don't Java's +=, -=, *=, /= compound assignment operators require casting? (11 个
我搜索了很多,但没有一个链接能帮助我解决这个问题。我得到了 ORA-21500: internal error code, arguments: [%s], [%s], [%s], [%s], [%s
我正在做 RegexOne 正则表达式教程,它有一个 question关于编写正则表达式以删除不必要的空格。 教程中提供的解决方案是 We can just skip all the starting
([\s\S]+|\s?) 中 |\s? 的目的或作用是什么?如果没有它,表达式会不会与 ([\s\S]+) 相同? 最佳答案 这不是完全相同的。 ([\s\S]+|\s?) 会匹配空字符串,而 ([
这个正则表达式有一组还是两组? 我正在尝试使用第二组访问 bookTitle 但出现错误: Pattern pattern = Pattern.compile("^\\s*(.*?)\\s+-\\s+
在 C 中给定一个字符串指针 s,下面的迭代会做什么?即它以什么方式遍历字符串? for (++s ; *s; ++s); 最佳答案 for (++s ; *s;++s) 表示 将指针 s 递增到字符
我正在用一个 node.js 应用程序解析一个大列表并有这段代码 sizeCode = dbfr.CN_DESC.split('\s+-\s*|\s*-\s+') 这似乎不起作用,因为它返回了 [ '
我正在编写一个简单的字符串连接程序。 该程序按照我发布的方式运行。但是,我首先使用以下代码编写它来查找字符串的结尾: while (*s++) ; 但是,这个方法并没有奏效。我传递给它的字符串
这个问题已经有答案了: What does (?和aramchand来自Mohandas Karamchand G 因此,在使用这些匹配来分割字符串后,您最终会得到 {"M", "K", "G"} 注
我正在尝试转换 Map到 List使用 lambda。 本质上,我想将键和值与 '=' 连接起来之间。这看起来微不足道,但我找不到如何去做。 例如 Map map = new HashMap<>();
我正在经历 K & R,并且在递增指针时遇到困难。练习 5.3(第 107 页)要求您使用指针编写一个 strcat 函数。 在伪代码中,该函数执行以下操作: 将 2 个字符串作为输入。 找到字符串
在下面的代码中,pS 和 s.pS 在最后一行是否保证相等?也就是说,在语句S s = S();中,是否可以确定不会构造一个临时的S? #include using namespace std; s
演示示例代码: public void ReverseString(char[] s) { for(int i = 0, j = s.Length-1; i < j; i++, j--){
我一直在寻找类似于 .NET examples 中的示例的 PowerShell 脚本.取一个 New-TimeSpan 并显示为 1 天 2 小时 3 分钟 4 秒。排除其零的地方,在需要的地方添加
def func(s): s = s + " is corrected" return s string_list = ["She", "He"] for s in string_li
我是 python 的新手。当我在互联网上搜索 lambda 时。我在 lambda_functions 中找到了这个声明. processFunc = collapse and (lambda s:
我最近开始学习正则表达式,并试图为上面的问题写一个正则表达式。如果限制只放在一个字母上(例如不超过 2 个“b”),这并不困难。 那么答案就是:a* c*(b|ε)a* c*(b|ε)a* c* 但是
当我运行 npm install 时出现以下错误,但我无法修复它。 我试过:npm install -g windows-build-tools 也没有修复这个错误 ERR! configure
有很多有趣的haskell网上可以找到片段。 This post可以在 this (awesome) Stack Overflow question 下找到. The author写道: discou
我知道以下三行代码旨在将字符串提取到$ value中并将其存储在$ header中。但是我不知道$value =~ s/^\s+//;和$value =~ s/\s+$//;之间有什么区别。 $val
我是一名优秀的程序员,十分优秀!