提问者:小点点

Cmake对符号'dlsym@@glibc2.2.5的未定义引用,即使我链接到-ldl


我得到了对dlsym@@glibc2.2.5的未定义引用,即使将其链接在库前后。 然而,在链接输出中,它似乎是在链接之前,但链接之前所有的库应该工作,我猜。

/bin/g++-9     CMakeFiles/http_downloader.dir/http_downloader_cli.cpp.o CMakeFiles/http_downloader.dir/SimpleOpenVPNSocket.cpp.o  -o http_downloader  -lpthread /usr/lib/x86_64-linux-gnu/libssl.so /usr/lib/x86_64-linux-gnu/liblz4.so -ldl downloader/libhttp_downloader_cli.a downloader/libhttp_downloader_lib.a ../../libopenvpn/libopenvpn_lib.a ../../_smoltcp_cpp_interface/libsmoltcp_cpp.a ../../_libtins/lib/libtins.a -ldl /usr/lib/x86_64-linux-gnu/libcrypto.so -lpthread /usr/lib/x86_64-linux-gnu/liblz4.so /usr/lib/x86_64-linux-gnu/libssl.so /usr/lib/x86_64-linux-gnu/libcrypto.so ../../../../smoltcp_cpp_interface/target/debug/libsmoltcp_cpp_interface_rust.a /usr/lib/x86_64-linux-gnu/libssl.so /usr/lib/x86_64-linux-gnu/libcrypto.so 
/usr/bin/ld: ../../../../smoltcp_cpp_interface/target/debug/libsmoltcp_cpp_interface_rust.a(std-6640d3868fa846e8.std.1mk5kra7-cgu.0.rcgu.o): undefined reference to symbol 'dlsym@@GLIBC_2.2.5'
/usr/bin/ld: /usr/lib/gcc/x86_64-linux-gnu/9/../../../x86_64-linux-gnu/libdl.so: error adding symbols: DSO missing from command line
collect2: error: ld returned 1 exit status
make[2]: *** [src/examples/http_downloader/CMakeFiles/http_downloader.dir/build.make:113: src/examples/http_downloader/http_downloader] Error 1
make[2]: Leaving directory '/workspaces/libopenvpnclient/build'
make[1]: *** [CMakeFiles/Makefile2:519: src/examples/http_downloader/CMakeFiles/http_downloader.dir/all] Error 2
make[1]: Leaving directory '/workspaces/libopenvpnclient/build'
make: *** [Makefile:130: all] Error 2

这就是我在cmake中链接它的方式:

add_executable(http_downloader http_downloader_cli.cpp SimpleOpenVPNSocket.cpp)
add_core_dependencies(http_downloader)

target_include_directories(http_downloader PUBLIC ${CMAKE_CURRENT_SOURCE_DIR}/downloader/include)
add_dependencies(http_downloader http_downloader_cli http_downloader_lib openvpn_lib smoltcp_cpp tins)
set_property(TARGET http_downloader PROPERTY CXX_STANDARD 17)
target_link_libraries(http_downloader dl http_downloader_cli http_downloader_lib openvpn_lib smoltcp_cpp tins dl)

正如您所看到的,我将dl放在所有东西的前面和后面,只是为了确保。

如果我理解正确,.a库没有依赖项,它们只是缺少符号,我们必须填充它们。 似乎我是通过链接-ldl来实现这一点的


共1个答案

匿名用户

如果仔细检查用于链接的实际命令行,您会发现在libsmoltcp_cpp_interface_rust.a(缺少符号)后面没有-ldl参数。

在您的示例中,smoltcp_cpp似乎是一个导入的库目标,它将libsmoltcp_cpp_interface_rust.a作为链接依赖项(而不是imported_location属性)。

虽然CMake保留链接到单个二进制文件(一个或另一个库)中的库的顺序,但这些库的依赖项之间的顺序没有定义。

您需要将-LDL添加为smoltcp_cpp目标本身的链接依赖项:

target_link_libraries(smoltcp_cpp INTERFACE -ldl)

但是,只有当libsmoltcp_cpp_interface_rust.asmoltcp_cpp目标的直接链接依赖项(指定为

target_link_libraries(smoltcp_cpp INTERFACE libsmoltcp_cpp_interface_rust.a)

在间接依赖的情况下,例如

target_link_libraries(smoltcp_cpp INTERFACE <intermediate-target>)
target_link_libraries(<intermediate-target> INTERFACE libsmoltcp_cpp_interface_rust.a)

您需要将-ldl添加为该的依赖项。

理想情况下,每个导入的目标都应该是自包含的,因此您可以安全地与该目标链接,而无需了解其内部结构。