当我编译openvswitch-1.5.0时,我遇到了以下编译错误:
gcc -Wstrict-prototypes -Wall -Wno-sign-compare -Wpointer-arith
-Wdeclaration-after-statement -Wformat-security -Wswitch-enum -Wunused-parameter -Wstrict-aliasing -Wbad-function-cast -Wcast-align -Wstrict-prototypes -Wold-style-definition -Wmissing-prototypes -Wmissing-field-initializers -Wno-override-init -g -O2 -export-dynamic ***-lpthread*** -o utilities/ovs-dpctl utilities/ovs-dpctl.o lib/libopenvswitch.a
/home/jyyoo/src/dpdk/build/lib/librte_eal.a
/home/jyyoo/src/dpdk/build/lib/libethdev.a
/home/jyyoo/src/dpdk/build/lib/librte_cmdline.a
/home/jyyoo/src/dpdk/build/lib/librte_hash.a
/home/jyyoo/src/dpdk/build/lib/librte_lpm.a
/home/jyyoo/src/dpdk/build/lib/librte_mbuf.a
/home/jyyoo/src/dpdk/build/lib/librte_ring.a
/home/jyyoo/src/dpdk/build/lib/librte_mempool.a
/home/jyyoo/src/dpdk/build/lib/librte_malloc.a -lrt -lm
/usr/bin/ld: /home/jyyoo/src/dpdk/build/lib/librte_eal.a(eal.o): undefined reference
to symbol 'pthread_create@@GLIBC_2.2.5'
/lib/x86_64-linux-gnu/libpthread.so.0: error adding symbols: DSO missing from
command line
如果我尝试查看libpthread的符号,看起来没有问题。
$ readelf -s /lib/x86_64-linux-gnu/libpthread.so.0 | grep pthread_create
199: 0000000000008220 2814 FUNC GLOBAL DEFAULT 13 pthread_create@@GLIBC_2.2.5
173: 0000000000008220 2814 FUNC LOCAL DEFAULT 13 __pthread_create_2_1
462: 0000000000008220 2814 FUNC GLOBAL DEFAULT 13 pthread_create@@GLIBC_2.2
你能给我一些提示或指点吗?
我发现了另一个案子,所以我认为你们都错了。
这是我所拥有的:
/usr/lib64/gcc/x86_64-suse-linux/4.8/../../../../x86_64-suse-linux/bin/ld: eggtrayicon.o: undefined reference to symbol 'XFlush'
/usr/lib64/libX11.so.6: error adding symbols: DSO missing from command line
问题是,命令行中没有包含- lx11 -尽管libX11. exe文件。所以应该作为依赖项添加,因为在参数中还有GTK和GNOME库。
所以,对我来说,唯一的解释是,这条信息可能是为了帮助你,但它没有正确地做到这一点。这可能很简单:提供该符号的库没有添加到命令行。
请注意POSIX中有关链接的三个重要规则:
Dynamic libraries have defined dependencies, so only libraries from the top-dependency should be supplied in whatever order (although after the static libraries)
Static libraries have just undefined symbols - it's up to you to know their dependencies and supply all of them in the command line
The order in static libraries is always: requester first, provider follows. Otherwise you'll get undefined symbol message, just like when you forgot to add the library to the command line
When you specify the library with -l<name>, you never know whether it will take lib<name>.so or lib<name>.a. The dynamic library is preferred, if found, and static libraries only can be enforced by compiler option - that's all. And whether you have any problems as above, it depends on whether you had static or dynamic libraries
Well, sometimes the dependencies may be lacking in dynamic libraries :D
背景
当链接器在正常搜索中没有找到所需的符号,但该符号在直接指定的动态库的依赖项之一中可用时,命令行消息将显示DSO missing。
在过去,链接器认为具有特定语言依赖关系的符号是可用的。但在后来的一些版本中,这种情况发生了变化,现在链接器对可用内容执行了更严格的视图。因此,这一信息旨在帮助实现这一转变。
怎么办呢?
如果您是软件的维护者
您应该通过确保在链接器命令行上直接指定满足所需符号所需的所有库来解决这个问题。还要记住,顺序很重要。
如果你只是想编译软件
作为一种变通方法,可以使用选项-Wl,——copy-dt-needed-entries切换回更宽松的视图来查看可用的符号。
将此注入构建的常用方法是在运行configure或类似于此之前导出LDFLAGS:
export LDFLAGS="-Wl,--copy-dt-needed-entries"
有时直接传递LDFLAGS="-Wl,——copy-dt-needed-entries"给make也可以。
你应该在目标文件被编译后在命令行中提到这个库:
gcc -Wstrict-prototypes -Wall -Wno-sign-compare -Wpointer-arith -Wdeclaration-after-statement -Wformat-security -Wswitch-enum -Wunused-parameter -Wstrict-aliasing -Wbad-function-cast -Wcast-align -Wstrict-prototypes -Wold-style-definition -Wmissing-prototypes -Wmissing-field-initializers -Wno-override-init \
-g -O2 -export-dynamic -o utilities/ovs-dpctl utilities/ovs-dpctl.o \
lib/libopenvswitch.a \
/home/jyyoo/src/dpdk/build/lib/librte_eal.a /home/jyyoo/src/dpdk/build/lib/libethdev.a /home/jyyoo/src/dpdk/build/lib/librte_cmdline.a /home/jyyoo/src/dpdk/build/lib/librte_hash.a /home/jyyoo/src/dpdk/build/lib/librte_lpm.a /home/jyyoo/src/dpdk/build/lib/librte_mbuf.a /home/jyyoo/src/dpdk/build/lib/librte_ring.a /home/jyyoo/src/dpdk/build/lib/librte_mempool.a /home/jyyoo/src/dpdk/build/lib/librte_malloc.a \
-lrt -lm -lpthread
说明:链接取决于模块的顺序。首先请求符号,然后从包含它们的库中链接它们。因此,您必须指定先使用库的模块,然后再指定库。是这样的:
gcc x.o y.o z.o -la -lb -lc
此外,如果存在循环依赖关系,您应该在命令行上多次指定相同的库。因此,如果libb需要来自libc的符号,libc需要来自libb的符号,命令行应该是:
gcc x.o y.o z.o -la -lb -lc -lb