我尝试在x86-64上创建一个共享库但是失败了。问题归结为以下代码(请不要介意,它没有多大意义):
.section .data
newline:
.ascii "\n"
.section .text
.globl write_newline
.type write_newline, @function
write_newline:
mov $newline, %rax
ret
建设如下:
as minimal.s -o minimal.o
ld -shared minimal.o -o libmin.so
导致以下错误:
ld: minimal.o: relocation R_X86_64_32 against `.data' can not be used when making a shared object; recompile with -fPIC
minimal.o: error adding symbols: Bad value
但是气体并不知道选项-fPIC,所以我不能用它重新编译:
as -fPIC minimal.s -o minimal.o
as: unrecognized option '-PIC'
可以做些什么呢?
答案 0 :(得分:4)
正如Marc B在评论中所说,错误消息假设您正在使用GCC或其他一些编译器。由于您已在汇编中编写代码,因此您有责任确保代码与位置无关。
在x86-64目标上,您可以使用RIP相对寻址来解决此错误:
.section .data
newline:
.ascii "\n"
.section .text
.globl write_newline
.type write_newline, @function
write_newline:
lea newline(%rip), %rax
ret
请注意,您不需要在此处使用GOT,因为newline
是本地符号。如果它是全局的,并且您希望代码中的newline
引用能够引用某些其他运行时组件中的newline
的其他定义,那么您需要使用@GOTPCREL
搬迁。这将允许动态链接器更改GOT中的条目以指向newline
的不同定义。