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

Gue*_*OCs 2 c++ dependencies cmake shared-libraries

dlsym@@GLIBC_2.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
Run Code Online (Sandbox Code Playgroud)

这就是我在 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)
Run Code Online (Sandbox Code Playgroud)

正如你所看到的,我dl在所有内容之前和之后都做了确认。

如果我理解正确的话,.alibs 没有依赖关系,它们只有缺失的符号,我们必须填充它们。好像我是通过链接来做到这一点的-ldl

Tsy*_*rev 5

您是否仔细检查用于链接的实际命令行,您会发现 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)
Run Code Online (Sandbox Code Playgroud)

但是,这仅适用于目标libsmoltcp_cpp_interface_rust.a直接链接依赖项smoltcp_cpp,即指定为

target_link_libraries(smoltcp_cpp INTERFACE libsmoltcp_cpp_interface_rust.a)
Run Code Online (Sandbox Code Playgroud)

间接依赖的情况下,例如

target_link_libraries(smoltcp_cpp INTERFACE <intermediate-target>)
target_link_libraries(<intermediate-target> INTERFACE libsmoltcp_cpp_interface_rust.a)
Run Code Online (Sandbox Code Playgroud)

您需要为此添加-ldl一个依赖项<intermediate-target>

理想情况下,每个 IMPORTED 目标都应该是self-contained,因此您可以安全地与该目标链接,而无需了解其内部结构。