我正在尝试创建一个使用静态库和共享库中的代码的可执行文件:
静态库有几个提升.a,pthread
和libbus.a
。共享库是libwrap.so
。
请注意,libwrap使用libbus
中的代码,libbus
使用pthread
中的代码。最后,可执行文件使用libwrap
和boost
中的代码。
由于链接器中包含的库的顺序很重要,我试图找到“获胜”序列。
链接阶段如下(为方便起见,粘贴在多行中):
$ /usr/bin/c++
-Wall -Wextra
-fPIC
-fvisibility=hidden -fno-strict-aliasing -Wno-long-long
-m64
-rdynamic
-D_UNICODE -DUNICODE
CMakeFiles/Wrapper_Test.dir/test.cpp.o
/usr/local/lib/libboost_log.a
/usr/local/lib/libboost_system.a
/usr/local/lib/libboost_filesystem.a
/usr/local/lib/libboost_date_time.a
/usr/local/lib/libboost_thread.a
/usr/local/lib/libboost_log_setup.a
/usr/local/lib/libboost_chrono.a
-pthread
/home/nass/dev/Data_Parser/trunk/external/lib/linux64_gcc_release/libbus.a
-L/home/nass/dev/Data_Parser_build/lib #this is where the libwrap.so is located
-Wl,-rpath,/home/nass/dev/Data_Parser_build/lib
-lwrap #the shared lib
-o ../../../bin/Wrapper_Test
链接错误是
CMakeFiles/Wrapper_Test.dir/test.cpp.o: In function `main':
test.cpp:(.text+0x2e): undefined reference to `wrapperNamespace::GetWrapper()'
collect2: error: ld returned 1 exit status
GetWrapper()
当然位于libwrap.so
,我可以验证它是可以在那里找到的符号:
$ nm -Ca ../../../lib/libwrap.so | grep GetWrapper
00000000000423d6 t wrapperNamespace::GetWrapper()
但是,链接器找不到它。我在这做错了什么?
上面的链接命令由以下CMakeLists.txt文件生成:
set(TARGET_NAME Wrapper_Test)
#set(CMAKE_BINARY_DIR ${CMAKE_SOURCE_DIR}/bin)
#set(EXECUTABLE_OUTPUT_PATH ${CMAKE_BINARY_DIR})
#set(LIBRARY_OUTPUT_PATH ${CMAKE_BINARY_DIR})
# INCLUDE INTERNAL FOLDER
include_directories(${CMAKE_SOURCE_DIR}/include/Wrapper)
add_executable(${TARGET_NAME} test.cpp)
add_boost_lib(${TARGET_NAME} log system filesystem date_time thread log_setup chrono)
setup_libbus(${TARGET_NAME}) #the libbus.a
target_link_libraries(${TARGET_NAME} -L../../../lib -lwrap)
set_property(TARGET ${TARGET_NAME} PROPERTY FOLDER test)
答案 0 :(得分:1)
我首先看一下生成这些行的cmake
文件。
将共享库添加到路径中应该很简单,例如:
find_library(
LIBWrapper
NAMES wrap
PATHS /home/nass/dev/Data_Parser_build/lib
)
然后将它们链接到您的测试文件,例如
add_executable(test src/test.cpp)
target_link_libraries(test ${LIBWrapper})
类似的应该适用于静态库。这样做的好处是,您不必处理CMake据称为您处理的所有编译器/平台特定细节,这些细节可能相当复杂。晦涩。
如果您的库是动态生成的,即在cmake配置时间之前,您可以将相应的链接标记传递给target_link_libraries
:
target_link_libraries(test -L/home/nass/dev/Data_Parser_build/lib -lwrap)
我在一些动态生成库,然后链接它的项目(例如https://github.com/caskorg/cask/blob/master/CMakeLists.txt)中使用了这个建议。如果这不起作用,我怀疑其他问题。
答案 1 :(得分:0)
您应该在静态库前使用-Wl,-Bstatic
,在动态库中使用-Wl,-Bdynamic
。您还需要对库使用-l
,对库路径使用-L
。
类似的东西:
$ /usr/bin/c++ test.cpp.o \
-L/usr/local/lib \
-Wl,-Bstatic \
-lboost_log \
-lsystem \
-L/home/nass/dev/Data_Parser_build/lib \
-Wl-Bdynamic \
-Wl,-rpath,/home/nass/dev/Data_Parser_build/lib \
-lwrap \
-o ../../../bin/Wrapper_Test
看起来更好。许多事情取决于编译器/链接器/操作系统的版本。