У меня довольно большое приложение, настроенное в CMake. Недавно я добавил пару классов, которые используют функциональность С++ 0x, и это нарушает сборку, поскольку CMake не настроен для компиляции с поддержкой С++ 0x. Как мне добавить это в качестве опции для CMake?
Добавить поддержку С++ 0x в CMake
Ответ 1
Вам нужно добавить флаг в CMAKE_CXX_FLAGS
:
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++0x")
Ответ 2
Я использую этот фрагмент для GCC, но это более сложный способ:
if(CMAKE_COMPILER_IS_GNUCXX)
SET(ENABLE_CXX11 "-std=c++11")
EXECUTE_PROCESS(COMMAND "${CMAKE_CXX_COMPILER} -dumpversion" OUTPUT_VARIABLE GCC_VERSION)
if (GCC_VERSION VERSION_LESS 4.7)
SET(ENABLE_CXX11 "-std=c++0x")
endif()
SET(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${ENABLE_CXX11}")
endif()
Ответ 3
У меня есть следующий модуль:
$ cat FindCXX11.cmake
# - Finds if the compiler has C++11 support
# This module can be used to detect compiler flags for using C++11, and checks
# a small subset of the language.
#
# The following variables are set:
# CXX11_FLAGS - flags to add to the CXX compiler for C++11 support
# CXX11_FOUND - true if the compiler supports C++11
#
# TODO: When compilers starts implementing the whole C++11, check the full set
include(CheckCXXSourceCompiles)
include(FindPackageHandleStandardArgs)
set(CXX11_FLAG_CANDIDATES
# GNU and Intel Linux
"-std=c++0x"
# Microsoft Visual Studio, and everything that automatically accepts C++11
" "
#Intel Windows
"/Qstd=c++0x"
)
set(CXX11_TEST_SOURCE
"
class Matrix
{
public:
Matrix(int a, int b, int c, int d)
: data {a, b, c, d}
{}
private:
int data[4];
};
int main()
{
int n[] {4,7,6,1,2};
for (auto i : n)
Matrix mat (3,5,1,2);
return 0;
}
")
foreach(FLAG ${CXX11_FLAG_CANDIDATES})
set(SAFE_CMAKE_REQUIRED_FLAGS "${CMAKE_REQUIRED_FLAGS}")
set(CMAKE_REQUIRED_FLAGS "${FLAG}")
unset(CXX11_FLAG_DETECTED CACHE)
message(STATUS "Try C++11 flag = [${FLAG}]")
check_cxx_source_compiles("${CXX11_TEST_SOURCE}" CXX11_FLAG_DETECTED)
set(CMAKE_REQUIRED_FLAGS "${SAFE_CMAKE_REQUIRED_FLAGS}")
if(CXX11_FLAG_DETECTED)
set(CXX11_FLAGS_INTERNAL "${FLAG}")
break()
endif(CXX11_FLAG_DETECTED)
endforeach(FLAG ${CXX11_FLAG_CANDIDATES})
set(CXX11_FLAGS "${CXX11_FLAGS_INTERNAL}")
find_package_handle_standard_args(CXX11 DEFAULT_MSG CXX11_FLAGS)
mark_as_advanced(CXX11_FLAGS)
Я использую это, чтобы проверить, поддерживается ли компилятором подмножество С++ 11, которое я использую (на самом деле это сейчас избыточно, так как я использую намного больше). Это легко построить дальше, добавить больше возможностей и добавить другие ключи компилятора. Фактически, единственные компиляторы, которые я нашел до сих пор, которые проходят этот тест, это GCC> = 4.6 и Clang> = 3.1, если я правильно помню.
Ответ 4
Как показано в этом ответе, лучше использовать target_compile_features
, чтобы потребовать поддержку определенных функций, которые вам нужны.
Это объясняет/документирует, какая функциональность требует ваш код, совместима с кросс-платформой и даст пользователю более полезное сообщение об ошибке, если требуемая функция недоступна в их компиляторе.
На странице справочная страница для этой опции:
target_compile_features
Добавьте ожидаемые функции компилятора в цель.
target_compile_features(<target> <PRIVATE|PUBLIC|INTERFACE> <feature> [...])
Укажите параметры компилятора, необходимые при компиляции заданной цели. Если эта функция не указана в переменной CMAKE_C_COMPILE_FEATURES или CMAKE_CXX_COMPILE_FEATURES, то сообщение об ошибке будет сообщено CMake. Если для использования функции требуется дополнительный флаг компилятора, например
-std=gnu++11
, флаг будет добавлен автоматически.
Список функций, которые можно протестировать с помощью этого макроса, можно найти в здесь.