我的C项目包括第三方库的源代码(目前作为git子模块).
我们的主要CMakelists通过使用add_subdirectory将该库添加到项目中,然后将库与主目标链接.
这是我当前的Cmake文件的简化版本:
- add_subdirectory(foo)
- set(FOO_LIBRARY ${CMAKE_CURRENT_SOURCE_DIR}/libfoo/libfoo.so)
- add_executable(target main.cpp)
- add_dependencies(target foo)
- target_link_libraries(target ${FOO_LIBRARY})
这个库需要很长时间才能构建,因为我没有更改它的代码,所以我只需要构建一次(每个构建配置).但是当我清理并重建我的代码时,它还会清理库文件并重新编译它们.
我试图在库的目录中设置属性CLEAN_NO_CUSTOM
,但根据文档,它只适用于自定义命令目标.
在CMake中是否有一种机制可以指定此库目标只需生成一次,或者不通过make clean清除?
解决方法
正如@Tsyvarev所说,在你的情况下,ExternalProject_Add比add_subdirectory更好.当你希望项目成为构建系统的重要组成部分时,add_subdirectory很好,因为它创建的目标可以在target_link_libraries()命令的右侧使用,而由ExternalProject_Add创建的目标则不能.
这是我在其中一个项目中使用的方法.您尝试查找所需的库并仅在未找到它时构建它.我使用INTERFACE库将FOO_EXTERNAL转换为target_link_libraries()可接受的目标.
- add_library(foo INTERFACE)
- find_package(foo ${FOO_VER})
- if(NOT foo_FOUND)
- include(ExternalProject)
- include(GNUInstallDirs)
- ExternalProject_Add(FOO_EXTERNAL
- SOURCE_DIR "${FOO_SOURCE_DIR}"
- BINARY_DIR "${FOO_BINARY_DIR}"
- INSTALL_DIR "${FOO_INSTALL_DIR}"
- CMAKE_ARGS "-DCMAKE_C_FLAGS=${CMAKE_C_FLAGS}"
- "-DCMAKE_CXX_FLAGS=${CMAKE_CXX_FLAGS}"
- "-DCMAKE_BUILD_TYPE=${CMAKE_BUILD_TYPE}"
- "-DCMAKE_INSTALL_PREFIX=${FOO_INSTALL_DIR}"
- )
- add_dependencies(foo FOO_EXTERNAL)
- set(foo_LIBRARY
- "${FOO_INSTALL_DIR}/${CMAKE_INSTALL_LIBDIR}/${CMAKE_STATIC_LIBRARY_PREFIX}foo${CMAKE_STATIC_LIBRARY_SUFFIX}")
- set(foo_INCLUDE_DIR "${FOO_INSTALL_DIR}/include")
- endif()
- target_link_libraries(foo INTERFACE ${foo_LIBRARY})
- target_include_directories(foo INTERFACE ${foo_INCLUDE_DIR})