
Previously compilers that only supported the meta-level flags would not have any of the granular features listed. Now we presume that they have full support and enable all the features. Update granular feature tests to skip the actual compilation checks for the presumed features.
220 lines
9.1 KiB
CMake
220 lines
9.1 KiB
CMake
cmake_minimum_required(VERSION 3.1.0)
|
|
project(WriteCompilerDetectionHeader)
|
|
|
|
set(CMAKE_INCLUDE_CURRENT_DIR ON)
|
|
|
|
include(WriteCompilerDetectionHeader)
|
|
include(CheckCXXSourceCompiles)
|
|
|
|
get_property(cxx_known_features GLOBAL PROPERTY CMAKE_CXX_KNOWN_FEATURES)
|
|
get_property(c_known_features GLOBAL PROPERTY CMAKE_C_KNOWN_FEATURES)
|
|
|
|
write_compiler_detection_header(
|
|
FILE "${CMAKE_CURRENT_BINARY_DIR}/test_compiler_detection.h"
|
|
PREFIX TEST
|
|
COMPILERS GNU Clang AppleClang MSVC SunPro Intel
|
|
VERSION 3.1
|
|
PROLOG "// something"
|
|
EPILOG "// more"
|
|
FEATURES
|
|
${cxx_known_features} ${c_known_features}
|
|
)
|
|
|
|
write_compiler_detection_header(
|
|
FILE "${CMAKE_CURRENT_BINARY_DIR}/compiler_multi_files/multi_file_compiler_detection.h"
|
|
PREFIX MULTI
|
|
OUTPUT_FILES_VAR multi_files
|
|
OUTPUT_DIR "${CMAKE_CURRENT_BINARY_DIR}/compiler_multi_files/compiler_support"
|
|
COMPILERS GNU Clang AppleClang MSVC SunPro Intel
|
|
VERSION 3.1
|
|
FEATURES
|
|
${cxx_known_features} ${c_known_features}
|
|
)
|
|
|
|
macro(set_defines target true_defs false_defs)
|
|
set(defines)
|
|
foreach(def ${true_defs})
|
|
list(APPEND defines ${def}=1)
|
|
endforeach()
|
|
foreach(def ${false_defs})
|
|
list(APPEND defines ${def}=0)
|
|
endforeach()
|
|
target_compile_definitions(${target}
|
|
PRIVATE
|
|
${defines}
|
|
EXPECTED_COMPILER_VERSION_MAJOR=${COMPILER_VERSION_MAJOR}
|
|
EXPECTED_COMPILER_VERSION_MINOR=${COMPILER_VERSION_MINOR}
|
|
EXPECTED_COMPILER_VERSION_PATCH=${COMPILER_VERSION_PATCH}
|
|
)
|
|
endmacro()
|
|
|
|
# Only run the compiler detection header test for compilers with
|
|
# detailed features tables, not just meta-features
|
|
|
|
if (CMAKE_C_COMPILE_FEATURES)
|
|
if(NOT CMAKE_C_COMPILER_ID MATCHES "^(Cray|PGI|XL|XLClang)$")
|
|
set(C_expected_features ${CMAKE_C_COMPILE_FEATURES})
|
|
list(FILTER C_expected_features EXCLUDE REGEX "^c_std_[0-9][0-9]")
|
|
endif()
|
|
endif()
|
|
if (C_expected_features)
|
|
string(REGEX REPLACE "^([0-9]+)\\.[0-9]+\\.[0-9]+.*" "\\1" COMPILER_VERSION_MAJOR "${CMAKE_C_COMPILER_VERSION}")
|
|
string(REGEX REPLACE "^[0-9]+\\.([0-9]+)\\.[0-9]+.*" "\\1" COMPILER_VERSION_MINOR "${CMAKE_C_COMPILER_VERSION}")
|
|
string(REGEX REPLACE "^[0-9]+\\.[0-9]+\\.([0-9]+).*" "\\1" COMPILER_VERSION_PATCH "${CMAKE_C_COMPILER_VERSION}")
|
|
|
|
if (CMAKE_C_COMPILER_ID STREQUAL "GNU"
|
|
OR (CMAKE_C_COMPILER_ID STREQUAL "Clang" AND NOT "x${CMAKE_CXX_SIMULATE_ID}" STREQUAL "xMSVC")
|
|
OR CMAKE_C_COMPILER_ID STREQUAL "AppleClang"
|
|
OR CMAKE_C_COMPILER_ID STREQUAL "Intel")
|
|
add_executable(WriteCompilerDetectionHeader_C11 main.c)
|
|
set_property(TARGET WriteCompilerDetectionHeader_C11 PROPERTY C_STANDARD 11)
|
|
set_defines(WriteCompilerDetectionHeader_C11 "EXPECTED_COMPILER_C_FUNCTION_PROTOTYPES;EXPECTED_COMPILER_C_RESTRICT" "")
|
|
|
|
add_executable(WriteCompilerDetectionHeader_C11_multi main_multi.c)
|
|
set_property(TARGET WriteCompilerDetectionHeader_C11_multi PROPERTY C_STANDARD 11)
|
|
set_defines(WriteCompilerDetectionHeader_C11_multi "EXPECTED_COMPILER_C_FUNCTION_PROTOTYPES;EXPECTED_COMPILER_C_RESTRICT" "")
|
|
target_include_directories(WriteCompilerDetectionHeader_C11_multi PRIVATE ${CMAKE_CURRENT_BINARY_DIR}/compiler_multi_files)
|
|
|
|
add_executable(C_undefined c_undefined.c)
|
|
set_property(TARGET C_undefined PROPERTY C_STANDARD 90)
|
|
include(CheckCCompilerFlag)
|
|
check_c_compiler_flag(-Werror=undef use_error_undef)
|
|
if (use_error_undef)
|
|
target_compile_options(C_undefined PRIVATE -Werror=undef)
|
|
endif()
|
|
|
|
add_executable(WriteCompilerDetectionHeader_C main.c)
|
|
set_property(TARGET WriteCompilerDetectionHeader_C PROPERTY C_STANDARD 90)
|
|
set_defines(WriteCompilerDetectionHeader_C "EXPECTED_COMPILER_C_FUNCTION_PROTOTYPES" "EXPECTED_COMPILER_C_RESTRICT")
|
|
|
|
add_executable(WriteCompilerDetectionHeader_C_multi main_multi.c)
|
|
set_property(TARGET WriteCompilerDetectionHeader_C_multi PROPERTY C_STANDARD 90)
|
|
set_defines(WriteCompilerDetectionHeader_C_multi "EXPECTED_COMPILER_C_FUNCTION_PROTOTYPES" "EXPECTED_COMPILER_C_RESTRICT")
|
|
target_include_directories(WriteCompilerDetectionHeader_C_multi PRIVATE ${CMAKE_CURRENT_BINARY_DIR}/compiler_multi_files)
|
|
endif()
|
|
endif()
|
|
|
|
if (CMAKE_CXX_COMPILE_FEATURES)
|
|
if(NOT CMAKE_CXX_COMPILER_ID MATCHES "^(Cray|PGI|XL|XLClang)$")
|
|
set(CXX_expected_features ${CMAKE_CXX_COMPILE_FEATURES})
|
|
list(FILTER CXX_expected_features EXCLUDE REGEX "^cxx_std_[0-9][0-9]")
|
|
endif()
|
|
endif()
|
|
if (NOT CXX_expected_features)
|
|
file(WRITE "${CMAKE_CURRENT_BINARY_DIR}/dummy.cpp"
|
|
"int main(int,char**) { return 0; }\n"
|
|
)
|
|
add_executable(WriteCompilerDetectionHeader "${CMAKE_CURRENT_BINARY_DIR}/dummy.cpp")
|
|
|
|
if(UNIX OR NOT CMAKE_CXX_COMPILER_ID STREQUAL "GNU")
|
|
check_cxx_source_compiles("#include \"${CMAKE_CURRENT_BINARY_DIR}/test_compiler_detection.h\"\nint main() { return 0; }\n"
|
|
file_include_works
|
|
)
|
|
if (file_include_works)
|
|
message(SEND_ERROR "Inclusion of ${CMAKE_CURRENT_BINARY_DIR}/test_compiler_detection.h was expected to cause an error, but did not.")
|
|
endif()
|
|
endif()
|
|
return()
|
|
endif()
|
|
|
|
string(REGEX REPLACE "^([0-9]+)\\.[0-9]+\\.[0-9]+.*" "\\1" COMPILER_VERSION_MAJOR "${CMAKE_CXX_COMPILER_VERSION}")
|
|
string(REGEX REPLACE "^[0-9]+\\.([0-9]+)\\.[0-9]+.*" "\\1" COMPILER_VERSION_MINOR "${CMAKE_CXX_COMPILER_VERSION}")
|
|
string(REGEX REPLACE "^[0-9]+\\.[0-9]+\\.([0-9]+).*" "\\1" COMPILER_VERSION_PATCH "${CMAKE_CXX_COMPILER_VERSION}")
|
|
|
|
if (CMAKE_CXX_COMPILER_ID STREQUAL "GNU"
|
|
OR (CMAKE_CXX_COMPILER_ID STREQUAL "Clang" AND NOT "x${CMAKE_CXX_SIMULATE_ID}" STREQUAL "xMSVC")
|
|
OR CMAKE_CXX_COMPILER_ID STREQUAL "AppleClang"
|
|
OR CMAKE_CXX_COMPILER_ID STREQUAL "SunPro"
|
|
OR CMAKE_CXX_COMPILER_ID STREQUAL "Intel")
|
|
# False for C++98 mode.
|
|
list(APPEND false_defs EXPECTED_COMPILER_CXX_DELEGATING_CONSTRUCTORS)
|
|
list(APPEND false_defs EXPECTED_COMPILER_CXX_VARIADIC_TEMPLATES)
|
|
endif()
|
|
|
|
# for msvc the compiler version determines which c++11 features are available.
|
|
if (CMAKE_CXX_COMPILER_ID STREQUAL "MSVC"
|
|
OR (CMAKE_CXX_COMPILER_ID STREQUAL "Clang" AND "x${CMAKE_CXX_SIMULATE_ID}" STREQUAL "xMSVC"))
|
|
if(";${CMAKE_CXX_COMPILE_FEATURES};" MATCHES ";cxx_delegating_constructors;")
|
|
list(APPEND true_defs EXPECTED_COMPILER_CXX_DELEGATING_CONSTRUCTORS)
|
|
list(APPEND true_defs EXPECTED_COMPILER_CXX_VARIADIC_TEMPLATES)
|
|
else()
|
|
list(APPEND false_defs EXPECTED_COMPILER_CXX_DELEGATING_CONSTRUCTORS)
|
|
list(APPEND false_defs EXPECTED_COMPILER_CXX_VARIADIC_TEMPLATES)
|
|
endif()
|
|
endif()
|
|
|
|
add_executable(WriteCompilerDetectionHeader main.cpp)
|
|
set_property(TARGET WriteCompilerDetectionHeader PROPERTY CXX_STANDARD 98)
|
|
set_defines(WriteCompilerDetectionHeader "${true_defs}" "${false_defs}")
|
|
|
|
add_executable(multi_files multi_files.cpp)
|
|
set_property(TARGET multi_files PROPERTY CXX_STANDARD 98)
|
|
target_include_directories(multi_files PRIVATE ${CMAKE_CURRENT_BINARY_DIR}/compiler_multi_files)
|
|
set_defines(multi_files "${true_defs}" "${false_defs}")
|
|
|
|
if(MSVC)
|
|
return() # MSVC has only one mode.
|
|
endif()
|
|
|
|
# Since GNU 4.7
|
|
if (";${CMAKE_CXX_COMPILE_FEATURES};" MATCHES ";cxx_delegating_constructors;")
|
|
list(APPEND true_defs EXPECTED_COMPILER_CXX_DELEGATING_CONSTRUCTORS)
|
|
list(REMOVE_ITEM false_defs EXPECTED_COMPILER_CXX_DELEGATING_CONSTRUCTORS)
|
|
endif()
|
|
|
|
# Since GNU 4.4
|
|
if (";${CMAKE_CXX_COMPILE_FEATURES};" MATCHES ";cxx_variadic_templates;")
|
|
list(APPEND true_defs EXPECTED_COMPILER_CXX_VARIADIC_TEMPLATES)
|
|
list(REMOVE_ITEM false_defs EXPECTED_COMPILER_CXX_VARIADIC_TEMPLATES)
|
|
endif()
|
|
|
|
add_executable(WriteCompilerDetectionHeader_11 main.cpp)
|
|
set_property(TARGET WriteCompilerDetectionHeader_11 PROPERTY CXX_STANDARD 11)
|
|
set_defines(WriteCompilerDetectionHeader_11 "${true_defs}" "${false_defs}")
|
|
|
|
add_executable(multi_files_11 multi_files.cpp)
|
|
set_property(TARGET multi_files_11 PROPERTY CXX_STANDARD 11)
|
|
target_include_directories(multi_files_11 PRIVATE ${CMAKE_CURRENT_BINARY_DIR}/compiler_multi_files)
|
|
set_defines(multi_files_11 "${true_defs}" "${false_defs}")
|
|
|
|
# test for ALLOW_UNKNOWN_COMPILERS
|
|
|
|
# use a compiler does not match the current one,
|
|
# so one always hits the fallback code
|
|
if (CMAKE_CXX_COMPILER_ID STREQUAL "SunPro")
|
|
set(OTHER_CXX "Intel")
|
|
else()
|
|
set(OTHER_CXX "SunPro")
|
|
endif()
|
|
|
|
write_compiler_detection_header(
|
|
FILE "${CMAKE_CURRENT_BINARY_DIR}/test_compiler_detection_allow_unknown.h"
|
|
PREFIX TEST
|
|
COMPILERS ${OTHER_CXX}
|
|
FEATURES cxx_nullptr
|
|
ALLOW_UNKNOWN_COMPILERS
|
|
)
|
|
|
|
# intentionally abuse the TEST_NULLPTR variable: this will only work
|
|
# with the fallback code.
|
|
check_cxx_source_compiles("#include \"${CMAKE_CURRENT_BINARY_DIR}/test_compiler_detection_allow_unknown.h\"
|
|
int main() {\n int i = TEST_NULLPTR;\n return 0; }\n"
|
|
file_include_works_allow_unknown
|
|
)
|
|
if (NOT file_include_works_allow_unknown)
|
|
message(SEND_ERROR "Inclusion of ${CMAKE_CURRENT_BINARY_DIR}/test_compiler_detection_allow_unknown.h was expected to work, but did not.")
|
|
endif()
|
|
|
|
# test for BARE_FEATURES
|
|
|
|
write_compiler_detection_header(
|
|
FILE "${CMAKE_CURRENT_BINARY_DIR}/test_compiler_detection_bare_features.h"
|
|
PREFIX TEST
|
|
COMPILERS GNU Clang AppleClang MSVC SunPro Intel
|
|
VERSION 3.1
|
|
BARE_FEATURES cxx_nullptr cxx_override cxx_noexcept cxx_final
|
|
)
|
|
|
|
add_executable(WriteCompilerDetectionHeaderBareFeatures main_bare.cpp)
|
|
set_property(TARGET WriteCompilerDetectionHeaderBareFeatures PROPERTY CXX_STANDARD 11)
|