1
0

CMakeLists.txt 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. cmake_minimum_required(VERSION 3.1.0)
  2. if(POLICY CMP0129)
  3. cmake_policy(SET CMP0129 NEW)
  4. endif()
  5. project(WriteCompilerDetectionHeader)
  6. set(CMAKE_INCLUDE_CURRENT_DIR ON)
  7. include(WriteCompilerDetectionHeader)
  8. include(CheckCXXSourceCompiles)
  9. get_property(cxx_known_features GLOBAL PROPERTY CMAKE_CXX_KNOWN_FEATURES)
  10. get_property(c_known_features GLOBAL PROPERTY CMAKE_C_KNOWN_FEATURES)
  11. write_compiler_detection_header(
  12. FILE "${CMAKE_CURRENT_BINARY_DIR}/test_compiler_detection.h"
  13. PREFIX TEST
  14. COMPILERS GNU Clang AppleClang MSVC SunPro Intel
  15. VERSION 3.1
  16. PROLOG "// something"
  17. EPILOG "// more"
  18. FEATURES
  19. ${cxx_known_features} ${c_known_features}
  20. )
  21. write_compiler_detection_header(
  22. FILE "${CMAKE_CURRENT_BINARY_DIR}/compiler_multi_files/multi_file_compiler_detection.h"
  23. PREFIX MULTI
  24. OUTPUT_FILES_VAR multi_files
  25. OUTPUT_DIR "${CMAKE_CURRENT_BINARY_DIR}/compiler_multi_files/compiler_support"
  26. COMPILERS GNU Clang AppleClang MSVC SunPro Intel
  27. VERSION 3.1
  28. FEATURES
  29. ${cxx_known_features} ${c_known_features}
  30. )
  31. macro(set_defines target true_defs false_defs)
  32. set(defines)
  33. foreach(def ${true_defs})
  34. list(APPEND defines ${def}=1)
  35. endforeach()
  36. foreach(def ${false_defs})
  37. list(APPEND defines ${def}=0)
  38. endforeach()
  39. target_compile_definitions(${target}
  40. PRIVATE
  41. ${defines}
  42. EXPECTED_COMPILER_VERSION_MAJOR=${COMPILER_VERSION_MAJOR}
  43. EXPECTED_COMPILER_VERSION_MINOR=${COMPILER_VERSION_MINOR}
  44. EXPECTED_COMPILER_VERSION_PATCH=${COMPILER_VERSION_PATCH}
  45. )
  46. endmacro()
  47. # Only run the compiler detection header test for compilers with
  48. # detailed features tables, not just meta-features
  49. if (CMAKE_C_COMPILE_FEATURES)
  50. if (NOT CMAKE_C_COMPILER_ID MATCHES "^(LCC|Cray|PGI|NVHPC|XL|XLClang|IBMClang|IntelLLVM|Fujitsu|FujitsuClang)$")
  51. set(C_expected_features ${CMAKE_C_COMPILE_FEATURES})
  52. list(FILTER C_expected_features EXCLUDE REGEX "^c_std_[0-9][0-9]")
  53. endif()
  54. endif()
  55. if (C_expected_features)
  56. string(REGEX REPLACE "^([0-9]+)\\.[0-9]+\\.[0-9]+.*" "\\1" COMPILER_VERSION_MAJOR "${CMAKE_C_COMPILER_VERSION}")
  57. string(REGEX REPLACE "^[0-9]+\\.([0-9]+)\\.[0-9]+.*" "\\1" COMPILER_VERSION_MINOR "${CMAKE_C_COMPILER_VERSION}")
  58. string(REGEX REPLACE "^[0-9]+\\.[0-9]+\\.([0-9]+).*" "\\1" COMPILER_VERSION_PATCH "${CMAKE_C_COMPILER_VERSION}")
  59. if (CMAKE_C_COMPILER_ID STREQUAL "GNU"
  60. OR (CMAKE_C_COMPILER_ID STREQUAL "Clang" AND NOT "x${CMAKE_CXX_SIMULATE_ID}" STREQUAL "xMSVC")
  61. OR CMAKE_C_COMPILER_ID STREQUAL "AppleClang"
  62. OR CMAKE_C_COMPILER_ID STREQUAL "Intel")
  63. add_executable(WriteCompilerDetectionHeader_C11 main.c)
  64. set_property(TARGET WriteCompilerDetectionHeader_C11 PROPERTY C_STANDARD 11)
  65. set_defines(WriteCompilerDetectionHeader_C11 "EXPECTED_COMPILER_C_FUNCTION_PROTOTYPES;EXPECTED_COMPILER_C_RESTRICT" "")
  66. add_executable(WriteCompilerDetectionHeader_C11_multi main_multi.c)
  67. set_property(TARGET WriteCompilerDetectionHeader_C11_multi PROPERTY C_STANDARD 11)
  68. set_defines(WriteCompilerDetectionHeader_C11_multi "EXPECTED_COMPILER_C_FUNCTION_PROTOTYPES;EXPECTED_COMPILER_C_RESTRICT" "")
  69. target_include_directories(WriteCompilerDetectionHeader_C11_multi PRIVATE ${CMAKE_CURRENT_BINARY_DIR}/compiler_multi_files)
  70. add_executable(C_undefined c_undefined.c)
  71. set_property(TARGET C_undefined PROPERTY C_STANDARD 90)
  72. include(CheckCCompilerFlag)
  73. check_c_compiler_flag(-Werror=undef use_error_undef)
  74. if (use_error_undef)
  75. target_compile_options(C_undefined PRIVATE -Werror=undef)
  76. endif()
  77. add_executable(WriteCompilerDetectionHeader_C main.c)
  78. set_property(TARGET WriteCompilerDetectionHeader_C PROPERTY C_STANDARD 90)
  79. set_defines(WriteCompilerDetectionHeader_C "EXPECTED_COMPILER_C_FUNCTION_PROTOTYPES" "EXPECTED_COMPILER_C_RESTRICT")
  80. add_executable(WriteCompilerDetectionHeader_C_multi main_multi.c)
  81. set_property(TARGET WriteCompilerDetectionHeader_C_multi PROPERTY C_STANDARD 90)
  82. set_defines(WriteCompilerDetectionHeader_C_multi "EXPECTED_COMPILER_C_FUNCTION_PROTOTYPES" "EXPECTED_COMPILER_C_RESTRICT")
  83. target_include_directories(WriteCompilerDetectionHeader_C_multi PRIVATE ${CMAKE_CURRENT_BINARY_DIR}/compiler_multi_files)
  84. endif()
  85. endif()
  86. if (CMAKE_CXX_COMPILE_FEATURES)
  87. if (NOT CMAKE_CXX_COMPILER_ID MATCHES "^(LCC|Cray|PGI|NVHPC|XL|XLClang|IBMClang|IntelLLVM|Fujitsu|FujitsuClang)$")
  88. set(CXX_expected_features ${CMAKE_CXX_COMPILE_FEATURES})
  89. list(FILTER CXX_expected_features EXCLUDE REGEX "^cxx_std_[0-9][0-9]")
  90. endif()
  91. endif()
  92. if (NOT CXX_expected_features)
  93. file(WRITE "${CMAKE_CURRENT_BINARY_DIR}/dummy.cpp"
  94. "int main(int,char**) { return 0; }\n"
  95. )
  96. add_executable(WriteCompilerDetectionHeader "${CMAKE_CURRENT_BINARY_DIR}/dummy.cpp")
  97. if(UNIX OR NOT CMAKE_CXX_COMPILER_ID STREQUAL "GNU")
  98. check_cxx_source_compiles("#include \"${CMAKE_CURRENT_BINARY_DIR}/test_compiler_detection.h\"\nint main() { return 0; }\n"
  99. file_include_works
  100. )
  101. if (file_include_works)
  102. message(SEND_ERROR "Inclusion of ${CMAKE_CURRENT_BINARY_DIR}/test_compiler_detection.h was expected to cause an error, but did not.")
  103. endif()
  104. endif()
  105. return()
  106. endif()
  107. string(REGEX REPLACE "^([0-9]+)\\.[0-9]+\\.[0-9]+.*" "\\1" COMPILER_VERSION_MAJOR "${CMAKE_CXX_COMPILER_VERSION}")
  108. string(REGEX REPLACE "^[0-9]+\\.([0-9]+)\\.[0-9]+.*" "\\1" COMPILER_VERSION_MINOR "${CMAKE_CXX_COMPILER_VERSION}")
  109. string(REGEX REPLACE "^[0-9]+\\.[0-9]+\\.([0-9]+).*" "\\1" COMPILER_VERSION_PATCH "${CMAKE_CXX_COMPILER_VERSION}")
  110. if (CMAKE_CXX_COMPILER_ID STREQUAL "GNU"
  111. OR (CMAKE_CXX_COMPILER_ID STREQUAL "Clang" AND NOT "x${CMAKE_CXX_SIMULATE_ID}" STREQUAL "xMSVC")
  112. OR CMAKE_CXX_COMPILER_ID STREQUAL "AppleClang"
  113. OR CMAKE_CXX_COMPILER_ID STREQUAL "SunPro"
  114. OR CMAKE_CXX_COMPILER_ID STREQUAL "Intel")
  115. # False for C++98 mode.
  116. list(APPEND false_defs EXPECTED_COMPILER_CXX_DELEGATING_CONSTRUCTORS)
  117. list(APPEND false_defs EXPECTED_COMPILER_CXX_VARIADIC_TEMPLATES)
  118. endif()
  119. # for msvc the compiler version determines which c++11 features are available.
  120. if (CMAKE_CXX_COMPILER_ID STREQUAL "MSVC"
  121. OR (CMAKE_CXX_COMPILER_ID STREQUAL "Clang" AND "x${CMAKE_CXX_SIMULATE_ID}" STREQUAL "xMSVC"))
  122. if(";${CMAKE_CXX_COMPILE_FEATURES};" MATCHES ";cxx_delegating_constructors;")
  123. list(APPEND true_defs EXPECTED_COMPILER_CXX_DELEGATING_CONSTRUCTORS)
  124. list(APPEND true_defs EXPECTED_COMPILER_CXX_VARIADIC_TEMPLATES)
  125. else()
  126. list(APPEND false_defs EXPECTED_COMPILER_CXX_DELEGATING_CONSTRUCTORS)
  127. list(APPEND false_defs EXPECTED_COMPILER_CXX_VARIADIC_TEMPLATES)
  128. endif()
  129. endif()
  130. add_executable(WriteCompilerDetectionHeader main.cpp)
  131. set_property(TARGET WriteCompilerDetectionHeader PROPERTY CXX_STANDARD 98)
  132. set_defines(WriteCompilerDetectionHeader "${true_defs}" "${false_defs}")
  133. add_executable(multi_files multi_files.cpp)
  134. set_property(TARGET multi_files PROPERTY CXX_STANDARD 98)
  135. target_include_directories(multi_files PRIVATE ${CMAKE_CURRENT_BINARY_DIR}/compiler_multi_files)
  136. set_defines(multi_files "${true_defs}" "${false_defs}")
  137. if(MSVC)
  138. return() # MSVC has only one mode.
  139. endif()
  140. # Since GNU 4.7
  141. if (";${CMAKE_CXX_COMPILE_FEATURES};" MATCHES ";cxx_delegating_constructors;")
  142. list(APPEND true_defs EXPECTED_COMPILER_CXX_DELEGATING_CONSTRUCTORS)
  143. list(REMOVE_ITEM false_defs EXPECTED_COMPILER_CXX_DELEGATING_CONSTRUCTORS)
  144. endif()
  145. # Since GNU 4.4
  146. if (";${CMAKE_CXX_COMPILE_FEATURES};" MATCHES ";cxx_variadic_templates;")
  147. list(APPEND true_defs EXPECTED_COMPILER_CXX_VARIADIC_TEMPLATES)
  148. list(REMOVE_ITEM false_defs EXPECTED_COMPILER_CXX_VARIADIC_TEMPLATES)
  149. endif()
  150. add_executable(WriteCompilerDetectionHeader_11 main.cpp)
  151. set_property(TARGET WriteCompilerDetectionHeader_11 PROPERTY CXX_STANDARD 11)
  152. set_defines(WriteCompilerDetectionHeader_11 "${true_defs}" "${false_defs}")
  153. add_executable(multi_files_11 multi_files.cpp)
  154. set_property(TARGET multi_files_11 PROPERTY CXX_STANDARD 11)
  155. target_include_directories(multi_files_11 PRIVATE ${CMAKE_CURRENT_BINARY_DIR}/compiler_multi_files)
  156. set_defines(multi_files_11 "${true_defs}" "${false_defs}")
  157. # test for ALLOW_UNKNOWN_COMPILERS
  158. # use a compiler does not match the current one,
  159. # so one always hits the fallback code
  160. if (CMAKE_CXX_COMPILER_ID STREQUAL "SunPro")
  161. set(OTHER_CXX "Intel")
  162. else()
  163. set(OTHER_CXX "SunPro")
  164. endif()
  165. write_compiler_detection_header(
  166. FILE "${CMAKE_CURRENT_BINARY_DIR}/test_compiler_detection_allow_unknown.h"
  167. PREFIX TEST
  168. COMPILERS ${OTHER_CXX}
  169. FEATURES cxx_nullptr
  170. ALLOW_UNKNOWN_COMPILERS
  171. )
  172. # intentionally abuse the TEST_NULLPTR variable: this will only work
  173. # with the fallback code.
  174. check_cxx_source_compiles("#include \"${CMAKE_CURRENT_BINARY_DIR}/test_compiler_detection_allow_unknown.h\"
  175. int main() {\n int i = TEST_NULLPTR;\n return 0; }\n"
  176. file_include_works_allow_unknown
  177. )
  178. if (NOT file_include_works_allow_unknown)
  179. message(SEND_ERROR "Inclusion of ${CMAKE_CURRENT_BINARY_DIR}/test_compiler_detection_allow_unknown.h was expected to work, but did not.")
  180. endif()
  181. # test for BARE_FEATURES
  182. write_compiler_detection_header(
  183. FILE "${CMAKE_CURRENT_BINARY_DIR}/test_compiler_detection_bare_features.h"
  184. PREFIX TEST
  185. COMPILERS GNU Clang AppleClang MSVC SunPro Intel
  186. VERSION 3.1
  187. BARE_FEATURES cxx_nullptr cxx_override cxx_noexcept cxx_final
  188. )
  189. add_executable(WriteCompilerDetectionHeaderBareFeatures main_bare.cpp)
  190. set_property(TARGET WriteCompilerDetectionHeaderBareFeatures PROPERTY CXX_STANDARD 11)