为什么我使用gcc收到“未定义的引用”错误?
我正在尝试创建一个导出一个函数的共享对象(.so),“external()”。然后我尝试链接.so但得到“未定义的引用'外部'”。我在这里做错了什么?
文件:external.c
int external() {
return 5;
}
文件:program.c
int external();
int main(char** argv, int* argc) {
return external();
}
命令:
$ gcc -fPIC -c external.c
$ gcc -shared -o libexternal.so external.o
$ gcc -L. -lexternal -o program program.c
/tmp/cc3MmhAE.o: In function `main':
program.c:(.text+0x7): undefined reference to `external'
collect2: ld returned 1 exit status
我甚至可以运行nm并看到.so正在定义'external':
命令:
$ nm libexternal.so | grep external
0000040c T external
我在这里缺少什么?
答案 0 :(得分:36)
gcc / ld的最新版本默认链接--as-needed
。
这意味着如果您在C文件之前写-lexternal
,库将自动被排除(测试时,如果“需要”这样的话,这个顺序很重要)
您可以使用以下任意一种解决此问题:
gcc -L. -o program program.c -lexternal
gcc -L. -Wl,--no-as-needed -lexternal -o program program.c
后者将--no-as-needed
传递给链接器,这会导致库仍然被链接,即使您没有从中调用external()
。
注意:-Wl,--no-as-needed
未全局应用于链接的所有内容,它仅适用于命令行顺序中的所有内容。所以-lexternal -Wl,--no-as-needed
也行不通。这确实意味着您可以混合和匹配行为,例如,gcc -L. -Wl,--no-as-needed -lexternal -Wl,--as-needed -o program program.c -lmightneed
将始终链接到外部,但仅链接到mightneed 如果 program.c / libexternal.so中的一个或两个它需要。