使用CMake

时间:2016-06-30 13:02:18

标签: c++ build cmake

我的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不清除?

2 个答案:

答案 0 :(得分:4)

正如@Tsyvarev所说,在你的情况下,ExternalProject_Add优于add_subdirectory。当您希望项目成为构建系统的重要组成部分时,add_subdirectory很好,因为它创建的目标可以在target_link_libraries()命令的右侧使用,同时由{{1}创建目标不能。

这是我在其中一个项目中使用的方法。您尝试查找所需的库并仅在未找到它时构建它。我使用INTERFACE库将FOO_EXTERNAL转换为ExternalProject_Add可接受的目标。

target_link_libraries()

答案 1 :(得分:3)

根据@Hikke的优秀答案,我写了两个宏来简化外部项目的使用。

<强>代码

include(ExternalProject)

#
#   Add external project.
#
#   \param name             Name of external project
#   \param path             Path to source directory
#   \param external         Name of the external target
#
macro(add_external_project name path)
    # Create external project
    set(${name}_SOURCE_DIR ${CMAKE_CURRENT_SOURCE_DIR}/${path})
    set(${name}_BINARY_DIR ${CMAKE_CURRENT_BINARY_DIR}/${path})
    ExternalProject_Add(${name}
        SOURCE_DIR "${${name}_SOURCE_DIR}"
        BINARY_DIR "${${name}_BINARY_DIR}"
        CMAKE_ARGS "-DCMAKE_C_FLAGS=${CMAKE_C_FLAGS}"
                   "-DCMAKE_CXX_FLAGS=${CMAKE_CXX_FLAGS}"
                   "-DCMAKE_BUILD_TYPE=${CMAKE_BUILD_TYPE}"
                   # These are only useful if you're cross-compiling.
                   # They, however, will not hurt regardless.
                   "-DCMAKE_SYSTEM_NAME=${CMAKE_SYSTEM_NAME}"
                   "-DCMAKE_SYSTEM_PROCESSOR=${CMAKE_SYSTEM_PROCESSOR}"
                   "-DCMAKE_AR=${CMAKE_AR}"
                   "-DCMAKE_C_COMPILER=${CMAKE_C_COMPILER}"
                   "-DCMAKE_CXX_COMPILER=${CMAKE_CXX_COMPILER}"
                   "-DCMAKE_RC_COMPILER=${CMAKE_RC_COMPILER}"
                   "-DCMAKE_COMPILER_PREFIX=${CMAKE_COMPILER_PREFIX}"
                   "-DCMAKE_FIND_ROOT_PATH=${CMAKE_FIND_ROOT_PATH}"
       INSTALL_COMMAND ""
    )

endmacro(add_external_project)

#
#   Add external target to external project.
#
#   \param name             Name of external project
#   \param includedir       Path to include directory
#   \param libdir           Path to library directory
#   \param build_type       Build type {STATIC, SHARED}
#   \param external         Name of the external target
#
macro(add_external_target name includedir libdir build_type external)
    # Configurations
    set(${name}_BINARY_DIR ${CMAKE_CURRENT_BINARY_DIR}/${libdir})

    # Create external library
    add_library(${name} ${build_type} IMPORTED)
    set(${name}_LIBRARY "${${name}_BINARY_DIR}/${CMAKE_CFG_INTDIR}/${CMAKE_${build_type}_LIBRARY_PREFIX}${name}${CMAKE_${build_type}_LIBRARY_SUFFIX}")

    # Find paths and set dependencies
    add_dependencies(${name} ${external})
    set(${name}_INCLUDE_DIR "${CMAKE_CURRENT_SOURCE_DIR}/${includedir}")

    # Set interface properties
    set_target_properties(${name} PROPERTIES IMPORTED_LOCATION ${${name}_LIBRARY})
    set_target_properties(${name} PROPERTIES INCLUDE_DIRECTORIES ${${name}_INCLUDE_DIR})
endmacro(add_external_target)

<强>解释

第一个宏创建外部项目,它执行整个外部构建步骤,而第二个步骤设置必要的依赖项并定义接口。将两者分开很重要,因为大多数项目都有多个接口/库。

示例

假设我的GoogleTest作为项目中的子模块,位于googletest子文件夹中。我可以使用以下界面来定义gtestgtest_main宏,非常类似于Googletest本身的工作方式。

add_external_project(googletest_external googletest)
add_external_target(gtest googletest/googletest/include googletest/googlemock/gtest STATIC googletest_external)
add_external_target(gtest_main googletest/googletest/include googletest/googlemock/gtest STATIC googletest_external)

然后我可以像以前一样将我的目标链接到googletest:

target_link_libraries(target_tests
    gtest
    gtest_main
    # The CMAKE_THREAD_LIBS_INIT can be found from `find_package(Threads)`
    # and is required for all but MinGW builds.
    ${CMAKE_THREAD_LIBS_INIT}
)

这应该提供足够的样板来简化实际的外部构建过程,即使使用CMake驱动的项目也是如此。