21

我将开始说明我在 Cmake 问题上几乎完全是哑巴。

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

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_executable应该target_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如果我手动将其添加到 Makefile中/usr/lib/libqserialdeviced.so.1.0.0,则可以正确地链接到/usr/lib/libqserialdeviced.so并轻松找到它。

我显然尝试过:

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的子目录分开,现在一切对我来说都是有意义的。

谢谢大家!

4

2 回答 2

15

对于原始 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})
于 2011-03-22T17:52:31.650 回答
0

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

于 2021-03-17T15:28:22.987 回答