2

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

main.c

#include <stdio.h>

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

extern void foo(void);

int main(void) {
    foo();
    return 0;
}

foo.c

#include <stdio.h>

extern void bar(void);

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

我编译并运行如下:

gcc -c -fpic foo.c
gcc -shared -o libfoo.so foo.o
gcc -L$(pwd) -o test main.c -lfoo
./test

我得到了我期望的输出:

foo
bar

但是,我必须使用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;
}

foo.c

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

extern void bar(void);

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

我改为编译并运行:

gcc -c -fpic foo.c
gcc -shared -o libfoo.so foo.o
gcc -o test main.c -ldl
./test

但是,这次我得到了输出

foo
./test: symbol lookup error: ./libfoo.so: undefined symbol: bar

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

4

1 回答 1

3

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

gcc -o test main.c -ldl -rdynamic

这里

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

于 2015-07-02T19:43:58.373 回答