Cmake target_link_libraries没有链接我的库

时间:2011-03-22 17:44:04

标签: c++ linux linker cmake kdevelop

我会开始说我在Cmake问题上几乎完全愚蠢。

对于Kdevelop 4.1项目,我有以下CMakeLists.txt

project(uart)

find_package(KDE4 REQUIRED)
include (KDE4Defaults)

include_directories( ${KDE4_INCLUDES} ${QT_INCLUDES} src/include src/include/QSerialDevce )

add_subdirectory(doc)
add_subdirectory(src)
add_subdirectory(icons)

link_directories(/usr/lib)

find_library(SERIALDEVICE_LIB qserialdeviced)

add_executable(uart ${uart_SRCS})
target_link_libraries(uart ${SERIALDEVICE_LIB})

当我尝试构建我的项目时,我看到:

uart/build> make -j2
-- Found Qt-Version 4.6.3 (using /usr/bin/qmake-qt4)
-- Found X11: /usr/lib64/libX11.so
-- Found KDE 4.5 include dir: /usr/include/kde4
-- Found KDE 4.5 library dir: /usr/lib64/kde4/devel
-- Found the KDE4 kconfig_compiler4 preprocessor: /usr/bin/kconfig_compiler4
-- Found automoc4: /usr/bin/automoc4
CMake Error at CMakeLists.txt:16 (add_executable):
  add_executable called with incorrect number of arguments


CMake Error: Attempt to add link library "/usr/lib/libqserialdeviced.so" to target "uart" which is not built by this project.
-- Configuring incomplete, errors occurred!
make: *** [cmake_check_build_system] Error 1
*** Failed ***

我读到的所有内容都表明add_executabletarget_link_libraries应该看起来像我文件的最后两行:

add_executable(uart ${uart_SRCS})
target_link_libraries(uart ${SERIALDEVICE_LIB})

如果我将这两行CMakeLists.txt更改为:

project(uart)

find_package(KDE4 REQUIRED)
include (KDE4Defaults)

include_directories( ${KDE4_INCLUDES} ${QT_INCLUDES} src/include src/include/QSerialDevce )

add_subdirectory(doc)
add_subdirectory(src)
add_subdirectory(icons)

link_directories(/usr/lib)

find_library(SERIALDEVICE_LIB qserialdeviced)

target_link_libraries(${SERIALDEVICE_LIB})

我明白了:

uart/build> make -j2
-- Found Qt-Version 4.6.3 (using /usr/bin/qmake-qt4)
-- Found X11: /usr/lib64/libX11.so
-- Found KDE 4.5 include dir: /usr/include/kde4
-- Found KDE 4.5 library dir: /usr/lib64/kde4/devel
-- Found the KDE4 kconfig_compiler4 preprocessor: /usr/bin/kconfig_compiler4
-- Found automoc4: /usr/bin/automoc4
-- Configuring done
-- Generating done
-- Build files have been written to: uart/build
[ 11%] Built target doc-handbook
[ 11%] Built target uart_automoc
Linking CXX executable uart
CMakeFiles/uart.dir/uart.o: In function `uart::setupSerial()':
uart/src/uart.cpp:126: undefined reference to `AbstractSerial::AbstractSerial(QObject*)'
CMakeFiles/uart.dir/uart.o: In function `uart::setupEnumerator()':
uart/src/uart.cpp:108: undefined reference to `SerialDeviceEnumerator::SerialDeviceEnumerator(QObject*)'
CMakeFiles/uart.dir/uart.o: In function `uart::setupSerial()':
uart_/uart/src/uart.cpp:136: undefined reference to `AbstractSerial::enableEmitStatus(bool)'
CMakeFiles/uart.dir/uart.o: In function `uart::setupEnumerator()':
uart_/uart/src/uart.cpp:112: undefined reference to `SerialDeviceEnumerator::setEnabled(bool)'
collect2: ld returned 1 exit status
make[2]: *** [src/uart] Error 1
make[1]: *** [src/CMakeFiles/uart.dir/all] Error 2
make: *** [all] Error 2
*** Failed ***

这清楚地表明target_link_libraries没有关联我的qserialdeviced

qserialdeviced位于/usr/lib/libqserialdeviced.so.1.0.0,正确地链接到/usr/lib/libqserialdeviced.so,如果我手动将其添加到Makefile中,则很容易找到。

我显然试过了:

target_link_libraries(-lqserialdeviced)

没有变化。

我也尝试过:

if ("${SERIALDEVICE_LIB}" STREQUAL "SERIALDEVICE_LIB-NOTFOUND")
    message(FATAL_ERROR "'qserialdeviced' wasn't found!")
else()
    message("'qserialdeviced' found: " ${SERIALDEVICE_LIB})
endif ()

但是这个测试成功了。找到了图书馆:

'qserialdeviced' found: /usr/lib/libqserialdeviced.so

有人可以帮我理解这里发生了什么吗?

我使用的是Linux Fedora 13,cmake版本2.8.0,gcc(GCC)4.4.5 20101112(Red Hat 4.4.5-2)和kdevelop-4.1.0-1.fc13.x86_64。

谢谢我提前。


编辑:

根据@DatChu的建议,我将CMakeLists.txt分割到我的子目录中,现在一切对我都有意义。

非常感谢!

2 个答案:

答案 0 :(得分:13)

对于原始CMakeLists.txt文件,问题不在于target_link_libraries,而在于add_executable

add_executable(uart ${uart_SRCS})

你在哪里设置你的uart_SRCS变量?你有吗

set(uart_SRCS src/blahblah.cpp src/somethingblahblah.cpp)

我想你可能会误解add_subdirectory的作用。它不会在里面添加源文件。它告诉CMake进入该文件夹并查找另一个CMakeLists.txt。当项目文件夹中有子项目时,通常使用它。

如果你有许多你不想手动设置的源文件,你也可以

file(GLOB uart_SRCS src/*.cpp src/*.c)

缺点是你需要手动重新运行CMake才能检测到新文件。请参阅杰克关于为什么这可能不是您想要使用的评论。

你的CMakeLists.txt很可能是

project(uart)

find_package(Qt4 REQUIRED)
include (${QT_USE_FILE})
find_package(KDE4 REQUIRED)
include (KDE4Defaults)

include_directories( ${KDE4_INCLUDES} ${QT_INCLUDES} src/include src/include/QSerialDevice )
link_directories(/usr/lib) 

file(GLOB uart_SRCS src/*.cpp src/*.h)
file(GLOB uart_HDRS include/*.h include/QSerialDevice/*.h)

find_library(SERIALDEVICE_LIB qserialdeviced)

add_executable(uart ${uart_SRCS} ${uart_HDRS})
target_link_libraries(uart ${SERIALDEVICE_LIB} ${QT_LIBRARIES})

答案 1 :(得分:0)

这不是一个真正的直接解决方案,但是我遇到了“未定义引用”错误(以前通过链接适当的库解决,但在这种情况下没有解决)的困难,直到我发现了一些东西 - 与 c 不兼容vs cpp 不知何故。定义这些引用函数的文件位于 .c 文件中(默认 cmake 使用 C 编译器进行编译。)而我引用这些函数的文件是一个 .cpp 文件(使用 g++ 编译器或任何您的环境 c++ 编译器)。一旦我将 .c 文件更改为 .cpp,“未定义引用”错误就消失了。上面看起来你的uart文件是.cpp,但也许检查其他文件是什么并尝试这个方法。这可能不是合适的解决方案,甚至根本不是一个解决方案,但这可能会让您度过一天并继续前进。