在共享库中不使用 PLT 调用另一个目标文件中的函数?

标签 c assembly linker shared-libraries dynamic-linking

我有两个汇编代码,code1.scode2.s,我想从这两个代码构建一个可重定位(使用 -fPIC 开关)共享库。

我想要code2.s调用一个名为myfun1的函数,它在code1.s中定义。

当我在 code2.s 中使用 call myfun1@PLT 时,它找到了该函数并且它像一个魅力一样工作但是它使用 PLT 部分来调用这个函数同一个共享库。我想在不遵守 PLT 部分的情况下执行此操作。当我删除 @PLT 时,我得到 myfun1relocation R_X86_64_PC32 against symbol 错误。

如何在不使用 PLT 部分的情况下执行此操作?有什么办法吗?我认为这应该是可行的,因为共享库应该是可重定位的,但不需要它的每个目标文件,因此为什么调用同一个库中的函数应该通过 PLT 部分。

这是我的编译命令:

对于 codeX.s:

gcc -c codeX.s -fPIC -DPIC -o codeX.o

gcc -c codeX.s -o codeX.o

对于名为 libcodes.so 的共享库:

gcc -shared -fPIC -DPIC -o libcodes.so code1.o code2.o

正如您可能好奇我为什么这样做一样,我有很多目标文件,每个目标文件都想调用 myfun1。在这里,我只是让询问技术部分变得更简单。即使我尝试将 myfun1 放入所有 codeX.s 文件中,但我收到多次定义 myfun1 的错误。我不太在意空间,也不太在意是否要将 myfun1 放入所有文件中。

最佳答案

在一个源文件中,您可以只使用两个标签 ( Building .so with recursive function in it ),一个带有 .globl,另一个没有。但这还不足以跨共享库中的源文件。

结合以下针对也导出的函数的答案仍然有用:一个 .hidden 和一个不是,因此您可以在库中高效地调用。


使用 .globl.hidden创建一个可以在当前目标文件之外看到的符号,但不能在共享库之外。因此它不受符号插入的影响,也不受同一共享中其他文件的调用图书馆可以直接调用它,而不是通过 PLT 或 GOT。

测试和工作示例:

## foo.S
.globl myfunc
.hidden myfunc
myfunc:
   #.globl myfunc_external    # optional, a non-hidden symbol at the same addr
   #myfunc_external:
    ret

## bar.S
.globl bar
bar:
    call myfunc
    ret

使用 gcc -shared foo.S bar.S -o foo.soobjdump -drwC -Mintel foo.so 构建:

Disassembly of section .text:

000000000000024d <myfunc>:
 24d:   c3                      ret    

000000000000024e <bar>:
 24e:   e8 fa ff ff ff          call   24d <myfunc>     # a direct near call
 253:   c3                      ret    

(我实际上也使用 -nostdlib 构建,为了示例目的,通过省略其他函数(如 __do_global_dtors_auxregister_tm_clones 来保持反汇编输出干净> 和 .init 部分。)


我认为 Glibc 为此使用 strong 或 weak_alias ( what does the weak_alias function do and where is it defined ),因此来自共享库的调用可以使用普通名称。 Where are syscalls located in glibc source ,例如__chdirchdir

例如glibc's printf.c定义 __printf 并使 printf 成为它的 strong 别名。

io/chdir.c定义 __chdir 并使 chdir 成为它的 weak 别名。

其中一个 x86-64 memchr asm 实现也使用 strong_alias 宏(在文件底部)。


相关的 GAS 指令是:

没有强别名 GAS 指令。这可能等效于简单的 foo = foo_internal 或等效的 .set foo, foo_internal

(TODO:完整的示例和关于强/弱究竟做什么的更多细节。我目前不知道,所以如果我自己没有时间阅读文档,欢迎编辑。 我知道这个东西存在并解决了这个问题,但我不知 Prop 体如何。)

关于在共享库中不使用 PLT 调用另一个目标文件中的函数?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51573771/

相关文章:

c - 即使进程未刷新,cmd.exe 如何显示标准输出?

从命令行执行时 CGI C 输出正确,但从 web 请求执行时不正确

iphone - 如何在 iPhone 上使用 ARMv6 优化的 OpenMAX DL 库

windows - 从文本创建 COM 文件

linker - GNU 链接器和链接器脚本 : Linker does not generate correct LMA

c - CodeBlock 中的结构大小错误

C:运行系统命令并获取输出?

c - 如何修复: "Disk Read Error" in my bootloader

c++ - Visual Studio 2010 中的外部库

c - C 源代码中使用的链接器变量的“ undefined reference ”