CMake第三方库未定义引用
CMake third party library undefined reference
我已经阅读和搜索了很多(例如 2 、CMake 的几个文档、类似的项目等,以找到解决方案,但我一直无法解决我的问题。我比较陌生Cmake 和 Linux (Ubuntu 14.04).
我想使用 libsbp (https://github.com/swift-nav/libsbp) 用 C++ 编写一个程序来与 GPS 模块通信。我克隆了存储库并安装了 C-Library。所以现在 /usr/local/lib 中有两个文件:libsbp.so 和 libsbp-static。 a 和 headers 在 /usr/local/include/libsbp
在我自己的项目中,我将 headers 和 #include "libsbp/sbp.h"
包括在内,这也有效。
现在的问题是:如果我想使用 libsbp 中的方法,例如sbp_state_init(&s);
我得到 对 "sbp_state_init(sbp_state_t*)"
的未定义引用
我自己项目的Cmake相关部分:
link_directories(/usr/local/lib)
add_executable(main ${QT_SOURCES} ${QT_HEADER_HPP})
target_link_libraries(main ${QT_LIBRARIES} ${catkin_LIBRARIES} sbp)
正如我之前所说,我尝试了一些东西:
find_library(SBP_LIB sbp /usr/local/lib)
-> 同样的错误
- 同样适用于在 target_link_libraries 中使用 libsbp 或搜索它
link_directory(/usr/local/lib)
- 尝试不同的路径,甚至将 libsbp.so 移动到项目目录中,然后 "finding" 使用
${CMAKE_CURRENT_SOURCE_DIR}
也许你能帮帮我!
编辑:
这是 libsbp/c/src 目录中的 CMakeList.txt
if (NOT DEFINED BUILD_SHARED_LIBS)
set(BUILD_SHARED_LIBS ON)
endif (NOT DEFINED BUILD_SHARED_LIBS)
file(GLOB libsbp_HEADERS "${PROJECT_SOURCE_DIR}/include/libsbp/*.h")
include_directories("${PROJECT_SOURCE_DIR}/CBLAS/include")
include_directories("${PROJECT_SOURCE_DIR}/clapack-3.2.1-CMAKE/INCLUDE")
include_directories("${PROJECT_SOURCE_DIR}/lapacke/include")
include_directories("${PROJECT_SOURCE_DIR}/include/libsbp")
set(libsbp_SRCS
edc.c
sbp.c
)
add_library(sbp-static STATIC ${libsbp_SRCS})
install(TARGETS sbp-static DESTINATION lib${LIB_SUFFIX})
if(BUILD_SHARED_LIBS)
add_library(sbp SHARED ${libsbp_SRCS})
install(TARGETS sbp DESTINATION lib${LIB_SUFFIX})
else(BUILD_SHARED_LIBS)
message(STATUS "Not building shared libraries")
endif(BUILD_SHARED_LIBS)
install(FILES ${libsbp_HEADERS} DESTINATION include/libsbp)
这是来自 /libsbp/c/
的 CMakeList.txt
cmake_minimum_required(VERSION 2.8.9)
project(libsbp)
# Setup flags for Code Coverage build mode
set(CMAKE_CXX_FLAGS_COVERAGE "${CMAKE_CXX_FLAGS_DEBUG} --coverage" CACHE STRING
"Flags used by the C++ compiler for building with code coverage."
FORCE )
set(CMAKE_C_FLAGS_COVERAGE "${CMAKE_C_FLAGS_DEBUG} --coverage" CACHE STRING
"Flags used by the C compiler for building with code coverage."
FORCE )
SET(CMAKE_EXE_LINKER_FLAGS_COVERAGE
"${CMAKE_EXE_LINKER_FLAGS_DEBUG} --coverage" CACHE STRING
"Flags used for linking binaries with code coverage."
FORCE )
set(CMAKE_SHARED_LINKER_FLAGS_COVERAGE
"${CMAKE_SHARED_LINKER_FLAGS_DEBUG} --coverage" CACHE STRING
"Flags used by the shared libraries linker during builds with code coverage."
FORCE )
mark_as_advanced(
CMAKE_CXX_FLAGS_COVERAGE
CMAKE_C_FLAGS_COVERAGE
CMAKE_EXE_LINKER_FLAGS_COVERAGE
CMAKE_SHARED_LINKER_FLAGS_COVERAGE )
# Update the documentation string of CMAKE_BUILD_TYPE for GUIs
set(CMAKE_BUILD_TYPE "${CMAKE_BUILD_TYPE}" CACHE STRING
"Choose the type of build, options are: None Debug Release RelWithDebInfo MinSizeRel Coverage."
FORCE )
# Set project version using Git tag and hash.
execute_process(
COMMAND git describe --dirty --tags --always
WORKING_DIRECTORY ${PROJECT_SOURCE_DIR}
RESULT_VARIABLE GIT_VERSION_FOUND
ERROR_QUIET
OUTPUT_VARIABLE GIT_VERSION
OUTPUT_STRIP_TRAILING_WHITESPACE
)
if (GIT_VERSION_FOUND)
set(VERSION "unknown")
else (GIT_VERSION_FOUND)
set(VERSION ${GIT_VERSION})
endif (GIT_VERSION_FOUND)
# Set project version explicitly for release tarballs.
#set(VERSION foo)
message(STATUS "libsbp version: ${VERSION}")
cmake_minimum_required(VERSION 2.8)
set(CMAKE_MODULE_PATH "${CMAKE_SOURCE_DIR}/cmake")
# Some compiler options used globally
set(CMAKE_C_FLAGS "-Wall -Wextra -Wno-strict-prototypes -Wno-unknown-warning-option -Werror -std=gnu99 ${CMAKE_C_FLAGS}")
add_subdirectory(src)
add_subdirectory(docs)
add_subdirectory(test)
你有(至少)两种可能:
- 正在安装库(这就是您所做的)
- 将库集成到您的 CMake 项目中
安装库时,target_link_libraries
命令需要稍做修改:
find_library(SBP_LIB sbp /usr/local/lib)
target_link_libraries(main ${QT_LIBRARIES} ${catkin_LIBRARIES} ${SBP_LIB})
当你在你的CMake项目中集成库时,你可以直接使用下面的命令而不用find_library
。这是可行的,因为该库是 CMake 已知的,因为它是在当前项目中构建的。
target_link_libraries(main ${QT_LIBRARIES} ${catkin_LIBRARIES} sbp)
看来你的程序用的是C++,库是用C写的
C 和 C++ 中的符号编码不同(损坏)。当从 C++ 中包含 C headers 时,您需要告诉编译器。这可以通过声明符号 extern "C"
.
来完成
extern "C" {
#include <libsbp/sbp.h>
}
有些图书馆已经将其包含在它们的 headers 中,但 sbp 中没有。
我已经阅读和搜索了很多(例如
我想使用 libsbp (https://github.com/swift-nav/libsbp) 用 C++ 编写一个程序来与 GPS 模块通信。我克隆了存储库并安装了 C-Library。所以现在 /usr/local/lib 中有两个文件:libsbp.so 和 libsbp-static。 a 和 headers 在 /usr/local/include/libsbp
在我自己的项目中,我将 headers 和 #include "libsbp/sbp.h"
包括在内,这也有效。
现在的问题是:如果我想使用 libsbp 中的方法,例如sbp_state_init(&s);
我得到 对 "sbp_state_init(sbp_state_t*)"
我自己项目的Cmake相关部分:
link_directories(/usr/local/lib)
add_executable(main ${QT_SOURCES} ${QT_HEADER_HPP})
target_link_libraries(main ${QT_LIBRARIES} ${catkin_LIBRARIES} sbp)
正如我之前所说,我尝试了一些东西:
find_library(SBP_LIB sbp /usr/local/lib)
-> 同样的错误- 同样适用于在 target_link_libraries 中使用 libsbp 或搜索它
link_directory(/usr/local/lib)
- 尝试不同的路径,甚至将 libsbp.so 移动到项目目录中,然后 "finding" 使用
${CMAKE_CURRENT_SOURCE_DIR}
也许你能帮帮我!
编辑:
这是 libsbp/c/src 目录中的 CMakeList.txt
if (NOT DEFINED BUILD_SHARED_LIBS)
set(BUILD_SHARED_LIBS ON)
endif (NOT DEFINED BUILD_SHARED_LIBS)
file(GLOB libsbp_HEADERS "${PROJECT_SOURCE_DIR}/include/libsbp/*.h")
include_directories("${PROJECT_SOURCE_DIR}/CBLAS/include")
include_directories("${PROJECT_SOURCE_DIR}/clapack-3.2.1-CMAKE/INCLUDE")
include_directories("${PROJECT_SOURCE_DIR}/lapacke/include")
include_directories("${PROJECT_SOURCE_DIR}/include/libsbp")
set(libsbp_SRCS
edc.c
sbp.c
)
add_library(sbp-static STATIC ${libsbp_SRCS})
install(TARGETS sbp-static DESTINATION lib${LIB_SUFFIX})
if(BUILD_SHARED_LIBS)
add_library(sbp SHARED ${libsbp_SRCS})
install(TARGETS sbp DESTINATION lib${LIB_SUFFIX})
else(BUILD_SHARED_LIBS)
message(STATUS "Not building shared libraries")
endif(BUILD_SHARED_LIBS)
install(FILES ${libsbp_HEADERS} DESTINATION include/libsbp)
这是来自 /libsbp/c/
的 CMakeList.txtcmake_minimum_required(VERSION 2.8.9)
project(libsbp)
# Setup flags for Code Coverage build mode
set(CMAKE_CXX_FLAGS_COVERAGE "${CMAKE_CXX_FLAGS_DEBUG} --coverage" CACHE STRING
"Flags used by the C++ compiler for building with code coverage."
FORCE )
set(CMAKE_C_FLAGS_COVERAGE "${CMAKE_C_FLAGS_DEBUG} --coverage" CACHE STRING
"Flags used by the C compiler for building with code coverage."
FORCE )
SET(CMAKE_EXE_LINKER_FLAGS_COVERAGE
"${CMAKE_EXE_LINKER_FLAGS_DEBUG} --coverage" CACHE STRING
"Flags used for linking binaries with code coverage."
FORCE )
set(CMAKE_SHARED_LINKER_FLAGS_COVERAGE
"${CMAKE_SHARED_LINKER_FLAGS_DEBUG} --coverage" CACHE STRING
"Flags used by the shared libraries linker during builds with code coverage."
FORCE )
mark_as_advanced(
CMAKE_CXX_FLAGS_COVERAGE
CMAKE_C_FLAGS_COVERAGE
CMAKE_EXE_LINKER_FLAGS_COVERAGE
CMAKE_SHARED_LINKER_FLAGS_COVERAGE )
# Update the documentation string of CMAKE_BUILD_TYPE for GUIs
set(CMAKE_BUILD_TYPE "${CMAKE_BUILD_TYPE}" CACHE STRING
"Choose the type of build, options are: None Debug Release RelWithDebInfo MinSizeRel Coverage."
FORCE )
# Set project version using Git tag and hash.
execute_process(
COMMAND git describe --dirty --tags --always
WORKING_DIRECTORY ${PROJECT_SOURCE_DIR}
RESULT_VARIABLE GIT_VERSION_FOUND
ERROR_QUIET
OUTPUT_VARIABLE GIT_VERSION
OUTPUT_STRIP_TRAILING_WHITESPACE
)
if (GIT_VERSION_FOUND)
set(VERSION "unknown")
else (GIT_VERSION_FOUND)
set(VERSION ${GIT_VERSION})
endif (GIT_VERSION_FOUND)
# Set project version explicitly for release tarballs.
#set(VERSION foo)
message(STATUS "libsbp version: ${VERSION}")
cmake_minimum_required(VERSION 2.8)
set(CMAKE_MODULE_PATH "${CMAKE_SOURCE_DIR}/cmake")
# Some compiler options used globally
set(CMAKE_C_FLAGS "-Wall -Wextra -Wno-strict-prototypes -Wno-unknown-warning-option -Werror -std=gnu99 ${CMAKE_C_FLAGS}")
add_subdirectory(src)
add_subdirectory(docs)
add_subdirectory(test)
你有(至少)两种可能:
- 正在安装库(这就是您所做的)
- 将库集成到您的 CMake 项目中
安装库时,target_link_libraries
命令需要稍做修改:
find_library(SBP_LIB sbp /usr/local/lib)
target_link_libraries(main ${QT_LIBRARIES} ${catkin_LIBRARIES} ${SBP_LIB})
当你在你的CMake项目中集成库时,你可以直接使用下面的命令而不用find_library
。这是可行的,因为该库是 CMake 已知的,因为它是在当前项目中构建的。
target_link_libraries(main ${QT_LIBRARIES} ${catkin_LIBRARIES} sbp)
看来你的程序用的是C++,库是用C写的
C 和 C++ 中的符号编码不同(损坏)。当从 C++ 中包含 C headers 时,您需要告诉编译器。这可以通过声明符号 extern "C"
.
extern "C" {
#include <libsbp/sbp.h>
}
有些图书馆已经将其包含在它们的 headers 中,但 sbp 中没有。