2012-03-13 23 views
43

我刚刚下载了googletest,用CMake生成了它的makefile并构建它。现在,我需要在我的测试项目中使用它。CMake + GoogleTest

对于CMake,我被建议不要直接指向gtest库(使用include _directorieslink_directories),而是使用find_package()来代替。

问题是,gtest makefile没有生成安装目标。我无法理解find_package(GTest REQUIRED)如何在没有某种安装的情况下工作。另外,将gtest文件夹作为子文件夹放在我的项目中是不可能的。

感谢您的任何帮助。

回答

62

这是一个不寻常的案例;大多数项目指定安装规则。

CMake的ExternalProject_Add模块也许是这份工作的最佳工具。这允许您从项目中下载,配置和构建gtest,然后链接到gtest库。

我已经测试了以下的CMakeLists.txt在Windows上使用Visual Studio 10和11,以及锵3.2 Ubuntu的使用GCC 4.8和 - 它可能需要调整其他平台/编译器:

cmake_minimum_required(VERSION 2.8.7 FATAL_ERROR) 
project(Test) 

# Create main.cpp which uses gtest 
file(WRITE src/main.cpp "#include \"gtest/gtest.h\"\n\n") 
file(APPEND src/main.cpp "TEST(A, B) { SUCCEED(); }\n") 
file(APPEND src/main.cpp "int main(int argc, char **argv) {\n") 
file(APPEND src/main.cpp " testing::InitGoogleTest(&argc, argv);\n") 
file(APPEND src/main.cpp " return RUN_ALL_TESTS();\n") 
file(APPEND src/main.cpp "}\n") 

# Create patch file for gtest with MSVC 2012 
if(MSVC_VERSION EQUAL 1700) 
    file(WRITE gtest.patch "Index: cmake/internal_utils.cmake\n") 
    file(APPEND gtest.patch "===================================================================\n") 
    file(APPEND gtest.patch "--- cmake/internal_utils.cmake (revision 660)\n") 
    file(APPEND gtest.patch "+++ cmake/internal_utils.cmake (working copy)\n") 
    file(APPEND gtest.patch "@@ -66,6 +66,9 @@\n") 
    file(APPEND gtest.patch "  # Resolved overload was found by argument-dependent lookup.\n") 
    file(APPEND gtest.patch "  set(cxx_base_flags \"\${cxx_base_flags} -wd4675\")\n") 
    file(APPEND gtest.patch "  endif()\n") 
    file(APPEND gtest.patch "+ if (MSVC_VERSION EQUAL 1700)\n") 
    file(APPEND gtest.patch "+  set(cxx_base_flags \"\${cxx_base_flags} -D_VARIADIC_MAX=10\")\n") 
    file(APPEND gtest.patch "+ endif()\n") 
    file(APPEND gtest.patch "  set(cxx_base_flags \"\${cxx_base_flags} -D_UNICODE -DUNICODE -DWIN32 -D_WIN32\")\n") 
    file(APPEND gtest.patch "  set(cxx_base_flags \"\${cxx_base_flags} -DSTRICT -DWIN32_LEAN_AND_MEAN\")\n") 
    file(APPEND gtest.patch "  set(cxx_exception_flags \"-EHsc -D_HAS_EXCEPTIONS=1\")\n") 
else() 
    file(WRITE gtest.patch "") 
endif() 

# Enable ExternalProject CMake module 
include(ExternalProject) 

# Set the build type if it isn't already 
if(NOT CMAKE_BUILD_TYPE) 
    set(CMAKE_BUILD_TYPE Release) 
endif() 

# Set default ExternalProject root directory 
set_directory_properties(PROPERTIES EP_PREFIX ${CMAKE_BINARY_DIR}/ThirdParty) 

# Add gtest 
ExternalProject_Add(
    googletest 
    SVN_REPOSITORY http://googletest.googlecode.com/svn/trunk/ 
    SVN_REVISION -r 660 
    TIMEOUT 10 
    PATCH_COMMAND svn patch ${CMAKE_SOURCE_DIR}/gtest.patch ${CMAKE_BINARY_DIR}/ThirdParty/src/googletest 
    # Force separate output paths for debug and release builds to allow easy 
    # identification of correct lib in subsequent TARGET_LINK_LIBRARIES commands 
    CMAKE_ARGS -DCMAKE_BUILD_TYPE=${CMAKE_BUILD_TYPE} 
       -DCMAKE_ARCHIVE_OUTPUT_DIRECTORY_DEBUG:PATH=DebugLibs 
       -DCMAKE_ARCHIVE_OUTPUT_DIRECTORY_RELEASE:PATH=ReleaseLibs 
       -Dgtest_force_shared_crt=ON 
    # Disable install step 
    INSTALL_COMMAND "" 
    # Wrap download, configure and build steps in a script to log output 
    LOG_DOWNLOAD ON 
    LOG_CONFIGURE ON 
    LOG_BUILD ON) 

# Specify include dir 
ExternalProject_Get_Property(googletest source_dir) 
include_directories(${source_dir}/include) 

# Add compiler flag for MSVC 2012 
if(MSVC_VERSION EQUAL 1700) 
    add_definitions(-D_VARIADIC_MAX=10) 
endif() 

# Add test executable target 
add_executable(MainTest ${PROJECT_SOURCE_DIR}/src/main.cpp) 

# Create dependency of MainTest on googletest 
add_dependencies(MainTest googletest) 

# Specify MainTest's link libraries 
ExternalProject_Get_Property(googletest binary_dir) 
if(MSVC) 
    set(Suffix ".lib") 
else() 
    set(Suffix ".a") 
    set(Pthread "-pthread") 
endif() 
target_link_libraries(
    MainTest 
    debug ${binary_dir}/DebugLibs/${CMAKE_FIND_LIBRARY_PREFIXES}gtest${Suffix} 
    optimized ${binary_dir}/ReleaseLibs/${CMAKE_FIND_LIBRARY_PREFIXES}gtest${Suffix} 
    ${Pthread}) 

如果在空目录中创建这个作为的CMakeLists.txt(比如MyTest),则:

cd MyTest 
mkdir build 
cd build 
cmake .. 

这应该MyTest/src创建一个基本的main.cpp和(在Windows MyTest/build/Test.sln)创建一个项目文件

当您构建项目时,它应该将gtest源下载到MyTest/build/ThirdParty/src/googletest,并在MyTest/build/ThirdParty/src/googletest-build中构建它们。然后,您应该能够成功运行MainTest目标。

+3

对我来说似乎很复杂。有没有办法直接引用谷歌测试CMakeLists.txt文件?或者,我应该将googletest库作为子目录添加吗? – Korchkidu 2012-03-14 07:03:59

+3

我不认为它比将googletest作为子目录添加得复杂得多,但如果您有选择,那么请确保 - 将googletest作为源代码树的子目录添加并通过“ADD_SUBDIRECTORY”将其拉入。 (你确实指出这不是你原来的问题中的一个选项)。 – Fraser 2012-03-14 10:31:44

+0

是的,确实如此。我想使用更好的解决方案。但是一个子目录似乎比你提供的解决方案更好,更清洁......但是我只是从CMake世界开始......; – Korchkidu 2012-03-14 12:08:10

8

使用ExternalProject模块和导入的库功能cmake有一个不那么复杂的解决方案。它从存储库检出代码,构建它并从构建的静态库(它们在我的系统上为libgtest.alibgtest_main.a)创建目标。

find_package(Threads REQUIRED) 
include(ExternalProject) 

set(GTEST_PREFIX "${CMAKE_CURRENT_BINARY_DIR}/gtest") 
ExternalProject_Add(GTestExternal 
    SVN_REPOSITORY http://googletest.googlecode.com/svn/trunk 
    SVN_REVISION -r HEAD 
    TIMEOUT 10 
    PREFIX "${GTEST_PREFIX}" 
    INSTALL_COMMAND "") 

set(LIBPREFIX "${CMAKE_STATIC_LIBRARY_PREFIX}") 
set(LIBSUFFIX "${CMAKE_STATIC_LIBRARY_SUFFIX}") 
set(GTEST_LOCATION "${GTEST_PREFIX}/src/GTestExternal-build") 
set(GTEST_INCLUDES "${GTEST_PREFIX}/src/GTestExternal/include") 
set(GTEST_LIBRARY "${GTEST_LOCATION}/${LIBPREFIX}gtest${LIBSUFFIX}") 
set(GTEST_MAINLIB "${GTEST_LOCATION}/${LIBPREFIX}gtest_main${LIBSUFFIX}") 

add_library(GTest IMPORTED STATIC GLOBAL) 
set_target_properties(GTest PROPERTIES 
    IMPORTED_LOCATION     "${GTEST_LIBRARY}" 
    INTERFACE_INCLUDE_DIRECTORIES  "${GTEST_INCLUDES}" 
    IMPORTED_LINK_INTERFACE_LIBRARIES "${CMAKE_THREAD_LIBS_INIT}") 

add_library(GTestMain IMPORTED STATIC GLOBAL) 
set_target_properties(GTestMain PROPERTIES 
    IMPORTED_LOCATION "${GTEST_MAINLIB}" 
    IMPORTED_LINK_INTERFACE_LIBRARIES 
     "${GTEST_LIBRARY};${CMAKE_THREAD_LIBS_INIT}") 

add_dependencies(GTest GTestExternal) 

您可能需要更换SVN_REVISION或在此处添加LOG_CONFIGURELOG_BUILD选项。创建GTestGTestMain目标后,他们可以像这样使用:

add_executable(Test 
    test1.cc 
    test2.cc) 
target_link_libraries(Test GTestMain) 

,或者,如果你有自己的main()功能:

add_executable(Test 
    main.cc 
    test1.cc 
    test2.cc) 
target_link_libraries(Test GTest) 
+2

如何添加包含目录? – Jeff 2014-07-12 05:02:36

+1

@Jeff:更新了答案,对不起半年的延迟。请参阅目标的“INTERFACE_INCLUDE_DIRECTORIES”属性。 – firegurafiku 2015-02-25 09:09:40

+0

我不得不删除行INTERFACE_INCLUDE_DIRECTORIES $ {GTEST_INCLUDES}''为了使这个工作与我自己的'main()'功能 – Twonky 2015-05-20 09:03:51

7

我的答案是基于firegurafiku答案。我修改了它在以下几个方面:

  1. 添加CMAKE_ARGSExternalProject_Add调用,这样它用MSVC工作。
  2. 会从一个文件位置GTEST源,而不是下载
  3. 增加便携式(对于MSVC和非MSVC)定义和IMPORTED_LOCATION
  4. 使用解决的问题与调用set_target_properties在配置时,当不工作INTERFACE_INCLUDE_DIRECTORIES尚不存在,因为外部项目尚未建立。

我更喜欢让gtest作为外部项目,而不是直接将其源代码添加到我的项目中。其中一个原因是因为当我搜索我的代码时,我不喜欢包含gtest源代码。由我的代码构建GTEST时,可加入到呼叫的CMAKE_ARGS定义ExternalProject_Add

这里应该也时需要使用任何特别打造的标志是我修改的方法:

include(ExternalProject) 

# variables to help keep track of gtest paths 
set(GTEST_PREFIX "${CMAKE_CURRENT_BINARY_DIR}/gtest") 
set(GTEST_LOCATION "${GTEST_PREFIX}/src/GTestExternal-build") 
set(GTEST_INCLUDES "${GTEST_PREFIX}/src/GTestExternal/include") 

# external project download and build (no install for gtest) 
ExternalProject_Add(GTestExternal 
    URL ${CMAKE_CURRENT_SOURCE_DIR}/../googletest 
    PREFIX "${GTEST_PREFIX}" 

    # cmake arguments 
    CMAKE_ARGS -Dgtest_force_shared_crt=ON 

    # Disable install step 
    INSTALL_COMMAND "" 

    # Wrap download, configure and build steps in a script to log output 
    LOG_DOWNLOAD ON 
    LOG_CONFIGURE ON 
    LOG_BUILD ON 
    ) 

# variables defining the import location properties for the generated gtest and 
# gtestmain libraries 
if (MSVC) 
    set(GTEST_IMPORTED_LOCATION 
     IMPORTED_LOCATION_DEBUG   "${GTEST_LOCATION}/Debug/${CMAKE_STATIC_LIBRARY_PREFIX}gtest${CMAKE_STATIC_LIBRARY_SUFFIX}" 
     IMPORTED_LOCATION_RELEASE   "${GTEST_LOCATION}/Release/${CMAKE_STATIC_LIBRARY_PREFIX}gtest${CMAKE_STATIC_LIBRARY_SUFFIX}" 
     ) 
    set(GTESTMAIN_IMPORTED_LOCATION 
     IMPORTED_LOCATION_DEBUG   "${GTEST_LOCATION}/Debug/${CMAKE_STATIC_LIBRARY_PREFIX}gtest_main${CMAKE_STATIC_LIBRARY_SUFFIX}" 
     IMPORTED_LOCATION_RELEASE   "${GTEST_LOCATION}/Release/${CMAKE_STATIC_LIBRARY_PREFIX}gtest_main${CMAKE_STATIC_LIBRARY_SUFFIX}" 
     ) 
else() 
    set(GTEST_IMPORTED_LOCATION 
     IMPORTED_LOCATION     "${GTEST_LOCATION}/${CMAKE_STATIC_LIBRARY_PREFIX}gtest${CMAKE_STATIC_LIBRARY_SUFFIX}") 
    set(GTESTMAIN_IMPORTED_LOCATION 
     IMPORTED_LOCATION     "${GTEST_LOCATION}/${CMAKE_STATIC_LIBRARY_PREFIX}gtest_main${CMAKE_STATIC_LIBRARY_SUFFIX}") 
endif() 

# the gtest include directory exists only after it is build, but it is used/needed 
# for the set_target_properties call below, so make it to avoid an error 
file(MAKE_DIRECTORY ${GTEST_INCLUDES}) 

# define imported library GTest 
add_library(GTest IMPORTED STATIC GLOBAL) 
set_target_properties(GTest PROPERTIES 
    INTERFACE_INCLUDE_DIRECTORIES  "${GTEST_INCLUDES}" 
    IMPORTED_LINK_INTERFACE_LIBRARIES "${CMAKE_THREAD_LIBS_INIT}" 
    ${GTEST_IMPORTED_LOCATION} 
    ) 

# define imported library GTestMain 
add_library(GTestMain IMPORTED STATIC GLOBAL) 
set_target_properties(GTestMain PROPERTIES 
    IMPORTED_LINK_INTERFACE_LIBRARIES GTest 
    ${GTESTMAIN_IMPORTED_LOCATION} 
    ) 

# make GTest depend on GTestExternal 
add_dependencies(GTest GTestExternal) 

# 
# My targets 
# 

project(test_pipeline) 
add_executable(${PROJECT_NAME} test_pipeline.cpp) 
set_target_properties(${PROJECT_NAME} PROPERTIES DEBUG_POSTFIX ${CMAKE_DEBUG_POSTFIX}) 
target_link_libraries(${PROJECT_NAME} ${TBB_LIBRARIES}) 
target_link_libraries(${PROJECT_NAME} GTest) 
+0

感谢您对以前答案的重大更正。 顺便说一句,当在Visual Studio中按下“重建”我的任何项目,与gtest链接它重新下载并重建gtest。这是一个正常的行为?为什么要重建“外部项目”? – 2016-06-06 21:38:52

+1

有关CMake ExternalProject如何运行的详细信息对我来说仍然有点神秘。我描述的方式,gtest已经是本地的,至少它不会重新下载。通常在Visual Studio中,重建不仅会重建选定的项目,还会重建它所依赖的其他项目。 cmake生成的解决方案/项目中必须有一些东西使ExternalProject添加目标成为依赖项。因此gtest得到重建。为了避免这种情况,您可以执行“仅限项目重建”,或者您可以转至“生成” - >“批量生成”并清理要重建的项目。 – Phil 2016-06-07 12:57:27

+0

感谢您的回答! 此外,如果你只是做了以上的事情,你的应用程序将不会使用默认设置在Visual Studio中生成(你会得到多个符号链接错误),因为gtest库使用/ MT运行时而不是默认/ MD 。以下是关于该问题的FAQ条目:https://github.com/google/googletest/blob/master/googletest/docs/FAQ.md#i-am-building-my-project-with-google-test-in- visual-studio-and-all-im-getting-is-a-bunch-of-linker-errors-or-warnings-help – 2016-06-07 20:51:39

21

这是很久以前,当提出原始问题时,为了别人的利益,可以使用ExternalProject下载gtest源代码,然后使用add_subdirectory()将其添加到您的版本中。这具有以下优点:

  • gtest作为您的主构建的一部分而构建,因此它使用相同的编译器标志等,并且不需要在任何地方安装。
  • 不需要将gtest源代码添加到您自己的源代码树中。

以正常方式使用,ExternalProject不会在配置时(即运行CMake时)下载和解包,但您可以这样做。我写了一篇关于如何做到这一点的博客文章,其中还包括一个适用于任何使用CMake作为构建系统的外部项目的通用实现,而不仅仅是gtest。你可以在这里找到它:

https://crascit.com/2015/07/25/cmake-gtest/

更新:上述方法现在也part of the googletest documentation

+2

这个伟大的答案不应该在最底层! – user3667089 2015-11-20 17:29:35

+0

这确实是一个非常好的解决方案。 – JonesV 2016-04-27 06:48:25

+0

BRAVO! - 为什么这不是CMake内置的?你可以提交吗? – Zak 2017-03-21 04:54:46