从使用 dlopen 加载的共享库引用全局符号

tom*_*KPZ 3 c shared-libraries dlopen dlsym

我有一个共享库,我想从主程序访问符号。例如:

main.c

#include <stdio.h>

void bar(void) { puts("bar"); }

extern void foo(void);

int main(void) {
    foo();
    return 0;
}
Run Code Online (Sandbox Code Playgroud)

foo.c

#include <stdio.h>

extern void bar(void);

void foo(void) {
    puts("foo");
    bar();
}
Run Code Online (Sandbox Code Playgroud)

我编译并运行如下:

gcc -c -fpic foo.c
gcc -shared -o libfoo.so foo.o
gcc -L$(pwd) -o test main.c -lfoo
./test
Run Code Online (Sandbox Code Playgroud)

我得到了我期望的输出:

foo
bar
Run Code Online (Sandbox Code Playgroud)

但是,我必须使用dlopen()anddlsym()因为我想控制库的加载时间。更改的文件是:

main.c

#include <stdio.h>
#include <dlfcn.h>

void bar(void) { puts("bar"); }

int main(void) {
    void *handle = dlopen("./libfoo.so", RTLD_LAZY);
    void (*foo)(void) = (void(*)(void))dlsym(handle,"foo");
    foo();
    return 0;
}
Run Code Online (Sandbox Code Playgroud)

foo.c

#include <stdio.h>
#include <dlfcn.h>

extern void bar(void);

void foo(void) {
    puts("foo");
    bar();
}
Run Code Online (Sandbox Code Playgroud)

我改为编译并运行:

gcc -c -fpic foo.c
gcc -shared -o libfoo.so foo.o
gcc -o test main.c -ldl
./test
Run Code Online (Sandbox Code Playgroud)

但是,这次我得到了输出

foo
./test: symbol lookup error: ./libfoo.so: undefined symbol: bar
Run Code Online (Sandbox Code Playgroud)

如何从 libfoo 引用主程序中的符号?

456*_*976 5

您必须-rdynamic在链接时添加选项test

gcc -o test main.c -ldl -rdynamic
Run Code Online (Sandbox Code Playgroud)

这里

-rdynamic 在支持它的目标上将标志 -export-dynamic 传递给 ELF 链接器。这指示链接器将所有符号(而不仅仅是使用过的符号)添加到动态符号表中。对于 dlopen 的某些用途或允许从程序内获取回溯,需要此选项。