我想让 dlerror() 告诉我加载失败期间丢失的所有 10 个符号。
你不能:加载器在发现库无法加载时立即报告错误(即,一旦它发现第一个丢失的符号)。装载机没有必要再往前走,所以它没有。
ldd -r
但是,您可以通过使用LD_PRELOAD
和来模拟什么LD_TRACE_LOADED_OBJECTS
,以获得完整的答案。例子:
$ cat main.c
#include <dlfcn.h>
#include <stdio.h>
int main()
{
void *p = dlopen("./foo.so", RTLD_NOW);
if (p == NULL) {
printf("%s\n", dlerror());
return 1;
}
return 0;
}
$ cat foo.c
int bar(), baz(); // not defined anywhere
int foo() {
return bar() + baz();
}
$ gcc main.c -ldl; gcc -fPIC -shared -o foo.so foo.c
$ ./a.out
./foo.so: undefined symbol: baz # only the first symbol is reported
$ ldd -r ./a.out
linux-vdso.so.1 => (0x00007fff52ddc000)
libdl.so.2 => /lib/x86_64-linux-gnu/libdl.so.2 (0x00007f158e48d000)
libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f158e0ce000)
/lib64/ld-linux-x86-64.so.2 (0x00007f158e6b2000)
那是行不通的,因为您的程序从未执行过,因此从未加载过foo.so
。但LD_PRELOAD
为了救援:
$ LD_BIND_NOW=1 LD_WARN=1 LD_TRACE_LOADED_OBJECTS=1 LD_PRELOAD=./foo.so ./a.out
linux-vdso.so.1 => (0x00007fff3c1b6000)
./foo.so (0x00007ffd33212000)
libdl.so.2 => /lib/x86_64-linux-gnu/libdl.so.2 (0x00007ffd3300e000)
libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007ffd32c4f000)
/lib64/ld-linux-x86-64.so.2 (0x00007ffd33414000)
undefined symbol: baz (./foo.so)
undefined symbol: bar (./foo.so)
瞧:现在报告了所有未解析的符号。