CMakeLists.txt 8.3 KB

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