4

我是 cmake (2.8.12.1) 的新手,我在 Windows 上使用它来生成项目文件以使用 Visual Studio 2012 构建 cpp-netlib。

默认情况下,它使用 /MDd 编译器开关进行编译。我想更改它以使其使用 /MTd。

我遵循了这里给出的建议https://stackoverflow.com/a/14172871但它对我不起作用。

具体来说,我在 CmakeLists.txt 的 if 语句中添加了如下所示的第二行。

if (MSVC)
    set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} /bigobj")
    set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} /MTd")
endif()

当我打开 Visual Studio sln 文件时,我可以看到 /MDd 选项仍然设置。此外,我在 CMakeCache.txt 中看到以下内容:

//Flags used by the compiler during debug builds.
CMAKE_CXX_FLAGS_DEBUG:STRING=/D_DEBUG /MDd /Zi /Ob0 /Od /RTC1

我也尝试过像这样从头开始设置标志:

set(CMAKE_CXX_FLAGS_DEBUG "/MTd")

但这也不起作用。

如果我像这样通过命令行传递选项:

-DCMAKE_CXX_FLAGS_DEBUG="/MTd"

该选项已在 Visual Studio 项目中成功设置。

谁能告诉我我做错了什么?

如果有人能告诉我缓存中的值来自我未在命令行中指定或不在 CmakeLists.txt 中的值,我也将不胜感激。

按要求添加 CMakeList.txt。如果我没有做对,我以前从未发布过这样的道歉。

# Original from cpp-netlib.org with my edits

cmake_minimum_required(VERSION 2.8)
project(CPP-NETLIB)
set(Boost_USE_STATIC_LIBS ON)
set(Boost_USE_MULTI_THREADED ON)
find_package( Boost 1.45.0 REQUIRED unit_test_framework system regex date_time thread filesystem program_options chrono )
find_package( OpenSSL )
find_package( Threads )
set(CMAKE_VERBOSE_MAKEFILE true)

if (CMAKE_BUILD_TYPE MATCHES Debug)
    add_definitions(-DBOOST_NETWORK_DEBUG)
endif()

if (OPENSSL_FOUND)
    add_definitions(-DBOOST_NETWORK_ENABLE_HTTPS)
    include_directories(${OPENSSL_INCLUDE_DIR})
endif()

if (${CMAKE_CXX_COMPILER_ID} MATCHES GNU)
  set (CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Wall")
endif()

if (Boost_FOUND)
    ################# added #################
    add_definitions(-DBOOST_ALL_NO_LIB)
    #########################################
    if (MSVC)
      add_definitions(-D_SCL_SECURE_NO_WARNINGS)
    endif(MSVC)
    if (WIN32)
      add_definitions(-D_WIN32_WINNT=0x0501)
    endif(WIN32)
    include_directories(${Boost_INCLUDE_DIRS})
    enable_testing()
    add_subdirectory(libs/network/src)
    add_subdirectory(libs/network/test)
    if (NOT MSVC)
      add_subdirectory(libs/mime/test)
    endif(NOT MSVC)
    add_subdirectory(libs/network/example)
endif(Boost_FOUND)

if (MSVC)
    set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} /bigobj")
    ################# added #################
    set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} /MTd")
    #########################################
endif()

enable_testing()
4

1 回答 1

8

if块中添加代码应该可以正常工作。但是,由于它不会在您的 .sln 中产生所需的效果,因此if块内的代码没有被执行,或者标志稍后在 CMakeLists.txt 中被替换。

至于这些标志的缓存值来自哪里;CMake 应用并缓存被认为对每个默认构建类型有用的默认值。

您是否在通话project添加了对标志的更改? project设置了很多东西,包括编译器标志。如果您设置这些标志而不缓存它们(即set如您所示调用),project将覆盖它们。如果您设置标志缓存它们(例如使用set(... CACHE),或通过命令行 arg 添加它们,-D如您所示),则project 不会覆盖它们。

此外,由于调用set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} /bigobj")不使用CACHE选项set,因此对变量的此更改CMAKE_CXX_FLAGS不会出现在您的 CMakeCache.txt 文件中。

为了将其放入缓存中,您还必须使用FORCE选项set,因为您会告诉 CMake 替换已缓存的值(默认情况下,CMake 本身会缓存这些标志变量)。但是我不推荐这个 - 我不明白你为什么需要这样做,你最好寻找一种比附加你想要的标志更好的方法,因为每次 CMake 运行附加的标志将被重新附加。

如果这不允许您解决问题,也许您可​​以显示一个完整的(小)CMakeLists.txt 来显示问题?

更新:

好的 - 问题是范围问题。对编译器标志的更改将应用​​于该 CMakeLists.txt 中定义的所有目标(exes/libs),但仅应用于在修改标志后添加的子目录目标。

add_subdirectory命令引入了一个新的范围(尽管文档中没有提到),它继承了父变量的副本。但是,随后对父作用域中变量的更改不会影响子副本,同样更改子副本也不会影响父副本的值(除非您使用 强制它set(... PARENT_SCOPE))。

因此,要解决您的问题,只需将if(MSVC)块移至所有add_subdirectory呼叫之前。

于 2013-12-27T16:42:13.523 回答