WriteCompilerDetectionHeader.cmake 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665
  1. #.rst:
  2. # WriteCompilerDetectionHeader
  3. # ----------------------------
  4. #
  5. # This module provides the function write_compiler_detection_header().
  6. #
  7. # The ``WRITE_COMPILER_DETECTION_HEADER`` function can be used to generate
  8. # a file suitable for preprocessor inclusion which contains macros to be
  9. # used in source code::
  10. #
  11. # write_compiler_detection_header(
  12. # FILE <file>
  13. # PREFIX <prefix>
  14. # [OUTPUT_FILES_VAR <output_files_var> OUTPUT_DIR <output_dir>]
  15. # COMPILERS <compiler> [...]
  16. # FEATURES <feature> [...]
  17. # [VERSION <version>]
  18. # [PROLOG <prolog>]
  19. # [EPILOG <epilog>]
  20. # )
  21. #
  22. # The ``write_compiler_detection_header`` function generates the
  23. # file ``<file>`` with macros which all have the prefix ``<prefix>``.
  24. #
  25. # By default, all content is written directly to the ``<file>``. The
  26. # ``OUTPUT_FILES_VAR`` may be specified to cause the compiler-specific
  27. # content to be written to separate files. The separate files are then
  28. # available in the ``<output_files_var>`` and may be consumed by the caller
  29. # for installation for example. The ``OUTPUT_DIR`` specifies a relative
  30. # path from the main ``<file>`` to the compiler-specific files. For example:
  31. #
  32. # .. code-block:: cmake
  33. #
  34. # write_compiler_detection_header(
  35. # FILE climbingstats_compiler_detection.h
  36. # PREFIX ClimbingStats
  37. # OUTPUT_FILES_VAR support_files
  38. # OUTPUT_DIR compilers
  39. # COMPILERS GNU Clang MSVC Intel
  40. # FEATURES cxx_variadic_templates
  41. # )
  42. # install(FILES
  43. # ${CMAKE_CURRENT_BINARY_DIR}/climbingstats_compiler_detection.h
  44. # DESTINATION include
  45. # )
  46. # install(FILES
  47. # ${support_files}
  48. # DESTINATION include/compilers
  49. # )
  50. #
  51. #
  52. # ``VERSION`` may be used to specify the API version to be generated.
  53. # Future versions of CMake may introduce alternative APIs. A given
  54. # API is selected by any ``<version>`` value greater than or equal
  55. # to the version of CMake that introduced the given API and less
  56. # than the version of CMake that introduced its succeeding API.
  57. # The value of the :variable:`CMAKE_MINIMUM_REQUIRED_VERSION`
  58. # variable is used if no explicit version is specified.
  59. # (As of CMake version |release| there is only one API version.)
  60. #
  61. # ``PROLOG`` may be specified as text content to write at the start of the
  62. # header. ``EPILOG`` may be specified as text content to write at the end
  63. # of the header
  64. #
  65. # At least one ``<compiler>`` and one ``<feature>`` must be listed. Compilers
  66. # which are known to CMake, but not specified are detected and a preprocessor
  67. # ``#error`` is generated for them. A preprocessor macro matching
  68. # ``<PREFIX>_COMPILER_IS_<compiler>`` is generated for each compiler
  69. # known to CMake to contain the value ``0`` or ``1``.
  70. #
  71. # Possible compiler identifiers are documented with the
  72. # :variable:`CMAKE_<LANG>_COMPILER_ID` variable.
  73. # Available features in this version of CMake are listed in the
  74. # :prop_gbl:`CMAKE_C_KNOWN_FEATURES` and
  75. # :prop_gbl:`CMAKE_CXX_KNOWN_FEATURES` global properties.
  76. #
  77. # See the :manual:`cmake-compile-features(7)` manual for information on
  78. # compile features.
  79. #
  80. # Feature Test Macros
  81. # ===================
  82. #
  83. # For each compiler, a preprocessor macro is generated matching
  84. # ``<PREFIX>_COMPILER_IS_<compiler>`` which has the content either ``0``
  85. # or ``1``, depending on the compiler in use. Preprocessor macros for
  86. # compiler version components are generated matching
  87. # ``<PREFIX>_COMPILER_VERSION_MAJOR`` ``<PREFIX>_COMPILER_VERSION_MINOR``
  88. # and ``<PREFIX>_COMPILER_VERSION_PATCH`` containing decimal values
  89. # for the corresponding compiler version components, if defined.
  90. #
  91. # A preprocessor test is generated based on the compiler version
  92. # denoting whether each feature is enabled. A preprocessor macro
  93. # matching ``<PREFIX>_COMPILER_<FEATURE>``, where ``<FEATURE>`` is the
  94. # upper-case ``<feature>`` name, is generated to contain the value
  95. # ``0`` or ``1`` depending on whether the compiler in use supports the
  96. # feature:
  97. #
  98. # .. code-block:: cmake
  99. #
  100. # write_compiler_detection_header(
  101. # FILE climbingstats_compiler_detection.h
  102. # PREFIX ClimbingStats
  103. # COMPILERS GNU Clang AppleClang MSVC Intel
  104. # FEATURES cxx_variadic_templates
  105. # )
  106. #
  107. # .. code-block:: c++
  108. #
  109. # #if ClimbingStats_COMPILER_CXX_VARIADIC_TEMPLATES
  110. # template<typename... T>
  111. # void someInterface(T t...) { /* ... */ }
  112. # #else
  113. # // Compatibility versions
  114. # template<typename T1>
  115. # void someInterface(T1 t1) { /* ... */ }
  116. # template<typename T1, typename T2>
  117. # void someInterface(T1 t1, T2 t2) { /* ... */ }
  118. # template<typename T1, typename T2, typename T3>
  119. # void someInterface(T1 t1, T2 t2, T3 t3) { /* ... */ }
  120. # #endif
  121. #
  122. # Symbol Macros
  123. # =============
  124. #
  125. # Some additional symbol-defines are created for particular features for
  126. # use as symbols which may be conditionally defined empty:
  127. #
  128. # .. code-block:: c++
  129. #
  130. # class MyClass ClimbingStats_FINAL
  131. # {
  132. # ClimbingStats_CONSTEXPR int someInterface() { return 42; }
  133. # };
  134. #
  135. # The ``ClimbingStats_FINAL`` macro will expand to ``final`` if the
  136. # compiler (and its flags) support the ``cxx_final`` feature, and the
  137. # ``ClimbingStats_CONSTEXPR`` macro will expand to ``constexpr``
  138. # if ``cxx_constexpr`` is supported.
  139. #
  140. # The following features generate corresponding symbol defines:
  141. #
  142. # ========================== =================================== =================
  143. # Feature Define Symbol
  144. # ========================== =================================== =================
  145. # ``c_restrict`` ``<PREFIX>_RESTRICT`` ``restrict``
  146. # ``cxx_constexpr`` ``<PREFIX>_CONSTEXPR`` ``constexpr``
  147. # ``cxx_deleted_functions`` ``<PREFIX>_DELETED_FUNCTION`` ``= delete``
  148. # ``cxx_extern_templates`` ``<PREFIX>_EXTERN_TEMPLATE`` ``extern``
  149. # ``cxx_final`` ``<PREFIX>_FINAL`` ``final``
  150. # ``cxx_noexcept`` ``<PREFIX>_NOEXCEPT`` ``noexcept``
  151. # ``cxx_noexcept`` ``<PREFIX>_NOEXCEPT_EXPR(X)`` ``noexcept(X)``
  152. # ``cxx_override`` ``<PREFIX>_OVERRIDE`` ``override``
  153. # ========================== =================================== =================
  154. #
  155. # Compatibility Implementation Macros
  156. # ===================================
  157. #
  158. # Some features are suitable for wrapping in a macro with a backward
  159. # compatibility implementation if the compiler does not support the feature.
  160. #
  161. # When the ``cxx_static_assert`` feature is not provided by the compiler,
  162. # a compatibility implementation is available via the
  163. # ``<PREFIX>_STATIC_ASSERT(COND)`` and
  164. # ``<PREFIX>_STATIC_ASSERT_MSG(COND, MSG)`` function-like macros. The macros
  165. # expand to ``static_assert`` where that compiler feature is available, and
  166. # to a compatibility implementation otherwise. In the first form, the
  167. # condition is stringified in the message field of ``static_assert``. In
  168. # the second form, the message ``MSG`` is passed to the message field of
  169. # ``static_assert``, or ignored if using the backward compatibility
  170. # implementation.
  171. #
  172. # The ``cxx_attribute_deprecated`` feature provides a macro definition
  173. # ``<PREFIX>_DEPRECATED``, which expands to either the standard
  174. # ``[[deprecated]]`` attribute or a compiler-specific decorator such
  175. # as ``__attribute__((__deprecated__))`` used by GNU compilers.
  176. #
  177. # The ``cxx_alignas`` feature provides a macro definition
  178. # ``<PREFIX>_ALIGNAS`` which expands to either the standard ``alignas``
  179. # decorator or a compiler-specific decorator such as
  180. # ``__attribute__ ((__aligned__))`` used by GNU compilers.
  181. #
  182. # The ``cxx_alignof`` feature provides a macro definition
  183. # ``<PREFIX>_ALIGNOF`` which expands to either the standard ``alignof``
  184. # decorator or a compiler-specific decorator such as ``__alignof__``
  185. # used by GNU compilers.
  186. #
  187. # ============================= ================================ =====================
  188. # Feature Define Symbol
  189. # ============================= ================================ =====================
  190. # ``cxx_alignas`` ``<PREFIX>_ALIGNAS`` ``alignas``
  191. # ``cxx_alignof`` ``<PREFIX>_ALIGNOF`` ``alignof``
  192. # ``cxx_nullptr`` ``<PREFIX>_NULLPTR`` ``nullptr``
  193. # ``cxx_static_assert`` ``<PREFIX>_STATIC_ASSERT`` ``static_assert``
  194. # ``cxx_static_assert`` ``<PREFIX>_STATIC_ASSERT_MSG`` ``static_assert``
  195. # ``cxx_attribute_deprecated`` ``<PREFIX>_DEPRECATED`` ``[[deprecated]]``
  196. # ``cxx_attribute_deprecated`` ``<PREFIX>_DEPRECATED_MSG`` ``[[deprecated]]``
  197. # ``cxx_thread_local`` ``<PREFIX>_THREAD_LOCAL`` ``thread_local``
  198. # ============================= ================================ =====================
  199. #
  200. # A use-case which arises with such deprecation macros is the deprecation
  201. # of an entire library. In that case, all public API in the library may
  202. # be decorated with the ``<PREFIX>_DEPRECATED`` macro. This results in
  203. # very noisy build output when building the library itself, so the macro
  204. # may be may be defined to empty in that case when building the deprecated
  205. # library:
  206. #
  207. # .. code-block:: cmake
  208. #
  209. # add_library(compat_support ${srcs})
  210. # target_compile_definitions(compat_support
  211. # PRIVATE
  212. # CompatSupport_DEPRECATED=
  213. # )
  214. #=============================================================================
  215. # Copyright 2014 Stephen Kelly <steveire@gmail.com>
  216. #
  217. # Distributed under the OSI-approved BSD License (the "License");
  218. # see accompanying file Copyright.txt for details.
  219. #
  220. # This software is distributed WITHOUT ANY WARRANTY; without even the
  221. # implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  222. # See the License for more information.
  223. #=============================================================================
  224. # (To distribute this file outside of CMake, substitute the full
  225. # License text for the above reference.)
  226. include(${CMAKE_CURRENT_LIST_DIR}/CMakeParseArguments.cmake)
  227. include(${CMAKE_CURRENT_LIST_DIR}/CMakeCompilerIdDetection.cmake)
  228. function(_load_compiler_variables CompilerId lang)
  229. include("${CMAKE_ROOT}/Modules/Compiler/${CompilerId}-${lang}-FeatureTests.cmake" OPTIONAL)
  230. set(_cmake_oldestSupported_${CompilerId} ${_cmake_oldestSupported} PARENT_SCOPE)
  231. foreach(feature ${ARGN})
  232. set(_cmake_feature_test_${CompilerId}_${feature} ${_cmake_feature_test_${feature}} PARENT_SCOPE)
  233. endforeach()
  234. include("${CMAKE_ROOT}/Modules/Compiler/${CompilerId}-${lang}-DetermineCompiler.cmake" OPTIONAL
  235. RESULT_VARIABLE determinedCompiler)
  236. if (NOT determinedCompiler)
  237. include("${CMAKE_ROOT}/Modules/Compiler/${CompilerId}-DetermineCompiler.cmake" OPTIONAL)
  238. endif()
  239. set(_compiler_id_version_compute_${CompilerId} ${_compiler_id_version_compute} PARENT_SCOPE)
  240. endfunction()
  241. function(write_compiler_detection_header
  242. file_keyword file_arg
  243. prefix_keyword prefix_arg
  244. )
  245. if (NOT "x${file_keyword}" STREQUAL "xFILE")
  246. message(FATAL_ERROR "write_compiler_detection_header: FILE parameter missing.")
  247. endif()
  248. if (NOT "x${prefix_keyword}" STREQUAL "xPREFIX")
  249. message(FATAL_ERROR "write_compiler_detection_header: PREFIX parameter missing.")
  250. endif()
  251. set(options)
  252. set(oneValueArgs VERSION EPILOG PROLOG OUTPUT_FILES_VAR OUTPUT_DIR)
  253. set(multiValueArgs COMPILERS FEATURES)
  254. cmake_parse_arguments(_WCD "${options}" "${oneValueArgs}" "${multiValueArgs}" ${ARGN})
  255. if (NOT _WCD_COMPILERS)
  256. message(FATAL_ERROR "Invalid arguments. write_compiler_detection_header requires at least one compiler.")
  257. endif()
  258. if (NOT _WCD_FEATURES)
  259. message(FATAL_ERROR "Invalid arguments. write_compiler_detection_header requires at least one feature.")
  260. endif()
  261. if(_WCD_UNPARSED_ARGUMENTS)
  262. message(FATAL_ERROR "Unparsed arguments: ${_WCD_UNPARSED_ARGUMENTS}")
  263. endif()
  264. if (prefix_arg STREQUAL "")
  265. message(FATAL_ERROR "A prefix must be specified")
  266. endif()
  267. string(MAKE_C_IDENTIFIER ${prefix_arg} cleaned_prefix)
  268. if (NOT prefix_arg STREQUAL cleaned_prefix)
  269. message(FATAL_ERROR "The prefix must be a valid C identifier.")
  270. endif()
  271. if(NOT _WCD_VERSION)
  272. set(_WCD_VERSION ${CMAKE_MINIMUM_REQUIRED_VERSION})
  273. endif()
  274. set(_min_version 3.1.0) # Version which introduced this function
  275. if (_WCD_VERSION VERSION_LESS _min_version)
  276. set(err "VERSION compatibility for write_compiler_detection_header is set to ${_WCD_VERSION}, which is too low.")
  277. set(err "${err} It must be set to at least ${_min_version}. ")
  278. set(err "${err} Either set the VERSION parameter to the write_compiler_detection_header function, or update")
  279. set(err "${err} your minimum required CMake version with the cmake_minimum_required command.")
  280. message(FATAL_ERROR "${err}")
  281. endif()
  282. if(_WCD_OUTPUT_FILES_VAR)
  283. if(NOT _WCD_OUTPUT_DIR)
  284. message(FATAL_ERROR "If OUTPUT_FILES_VAR is specified, then OUTPUT_DIR must also be specified.")
  285. endif()
  286. endif()
  287. if(_WCD_OUTPUT_DIR)
  288. if(NOT _WCD_OUTPUT_FILES_VAR)
  289. message(FATAL_ERROR "If OUTPUT_DIR is specified, then OUTPUT_FILES_VAR must also be specified.")
  290. endif()
  291. get_filename_component(main_file_dir ${file_arg} DIRECTORY)
  292. if (NOT IS_ABSOLUTE ${main_file_dir})
  293. set(main_file_dir "${CMAKE_CURRENT_BINARY_DIR}/${main_file_dir}")
  294. endif()
  295. if (NOT IS_ABSOLUTE ${_WCD_OUTPUT_DIR})
  296. set(_WCD_OUTPUT_DIR "${CMAKE_CURRENT_BINARY_DIR}/${_WCD_OUTPUT_DIR}")
  297. endif()
  298. get_filename_component(out_file_dir ${_WCD_OUTPUT_DIR} ABSOLUTE)
  299. string(FIND ${out_file_dir} ${main_file_dir} idx)
  300. if (NOT idx EQUAL 0)
  301. message(FATAL_ERROR "The compiler-specific output directory must be within the same directory as the main file.")
  302. endif()
  303. if (main_file_dir STREQUAL out_file_dir)
  304. unset(_WCD_OUTPUT_DIR)
  305. else()
  306. string(REPLACE "${main_file_dir}/" "" _WCD_OUTPUT_DIR "${out_file_dir}/")
  307. endif()
  308. endif()
  309. set(compilers
  310. GNU
  311. Clang
  312. AppleClang
  313. MSVC
  314. SunPro
  315. Intel
  316. )
  317. set(_hex_compilers ADSP Borland Embarcadero SunPro)
  318. foreach(_comp ${_WCD_COMPILERS})
  319. list(FIND compilers ${_comp} idx)
  320. if (idx EQUAL -1)
  321. message(FATAL_ERROR "Unsupported compiler ${_comp}.")
  322. endif()
  323. if (NOT _need_hex_conversion)
  324. list(FIND _hex_compilers ${_comp} idx)
  325. if (NOT idx EQUAL -1)
  326. set(_need_hex_conversion TRUE)
  327. endif()
  328. endif()
  329. endforeach()
  330. set(file_content "
  331. // This is a generated file. Do not edit!
  332. #ifndef ${prefix_arg}_COMPILER_DETECTION_H
  333. #define ${prefix_arg}_COMPILER_DETECTION_H
  334. ")
  335. if (_WCD_PROLOG)
  336. set(file_content "${file_content}\n${_WCD_PROLOG}\n")
  337. endif()
  338. if (_need_hex_conversion)
  339. set(file_content "${file_content}
  340. #define ${prefix_arg}_DEC(X) (X)
  341. #define ${prefix_arg}_HEX(X) ( \\
  342. ((X)>>28 & 0xF) * 10000000 + \\
  343. ((X)>>24 & 0xF) * 1000000 + \\
  344. ((X)>>20 & 0xF) * 100000 + \\
  345. ((X)>>16 & 0xF) * 10000 + \\
  346. ((X)>>12 & 0xF) * 1000 + \\
  347. ((X)>>8 & 0xF) * 100 + \\
  348. ((X)>>4 & 0xF) * 10 + \\
  349. ((X) & 0xF) \\
  350. )\n")
  351. endif()
  352. foreach(feature ${_WCD_FEATURES})
  353. if (feature MATCHES "^cxx_")
  354. list(APPEND _langs CXX)
  355. list(APPEND CXX_features ${feature})
  356. elseif (feature MATCHES "^c_")
  357. list(APPEND _langs C)
  358. list(APPEND C_features ${feature})
  359. else()
  360. message(FATAL_ERROR "Unsupported feature ${feature}.")
  361. endif()
  362. endforeach()
  363. list(REMOVE_DUPLICATES _langs)
  364. if(_WCD_OUTPUT_FILES_VAR)
  365. get_filename_component(main_file_name ${file_arg} NAME)
  366. set(compiler_file_content_
  367. "#ifndef ${prefix_arg}_COMPILER_DETECTION_H
  368. # error This file may only be included from ${main_file_name}
  369. #endif\n")
  370. endif()
  371. foreach(_lang ${_langs})
  372. set(target_compilers)
  373. foreach(compiler ${_WCD_COMPILERS})
  374. _load_compiler_variables(${compiler} ${_lang} ${${_lang}_features})
  375. if(_cmake_oldestSupported_${compiler})
  376. list(APPEND target_compilers ${compiler})
  377. endif()
  378. endforeach()
  379. get_property(known_features GLOBAL PROPERTY CMAKE_${_lang}_KNOWN_FEATURES)
  380. foreach(feature ${${_lang}_features})
  381. list(FIND known_features ${feature} idx)
  382. if (idx EQUAL -1)
  383. message(FATAL_ERROR "Unsupported feature ${feature}.")
  384. endif()
  385. endforeach()
  386. if(_lang STREQUAL CXX)
  387. set(file_content "${file_content}\n#ifdef __cplusplus\n")
  388. else()
  389. set(file_content "${file_content}\n#ifndef __cplusplus\n")
  390. endif()
  391. compiler_id_detection(ID_CONTENT ${_lang} PREFIX ${prefix_arg}_
  392. ID_DEFINE
  393. )
  394. set(file_content "${file_content}${ID_CONTENT}\n")
  395. set(pp_if "if")
  396. foreach(compiler ${target_compilers})
  397. set(file_content "${file_content}\n# ${pp_if} ${prefix_arg}_COMPILER_IS_${compiler}\n")
  398. if(_WCD_OUTPUT_FILES_VAR)
  399. set(compile_file_name "${_WCD_OUTPUT_DIR}${prefix_arg}_COMPILER_INFO_${compiler}_${_lang}.h")
  400. set(file_content "${file_content}\n# include \"${compile_file_name}\"\n")
  401. endif()
  402. if(_WCD_OUTPUT_FILES_VAR)
  403. set(compiler_file_content compiler_file_content_${compiler}_${_lang})
  404. else()
  405. set(compiler_file_content file_content)
  406. endif()
  407. set(${compiler_file_content} "${${compiler_file_content}}
  408. # if !(${_cmake_oldestSupported_${compiler}})
  409. # error Unsupported compiler version
  410. # endif\n")
  411. set(PREFIX ${prefix_arg}_)
  412. if (_need_hex_conversion)
  413. set(MACRO_DEC ${prefix_arg}_DEC)
  414. set(MACRO_HEX ${prefix_arg}_HEX)
  415. else()
  416. set(MACRO_DEC)
  417. set(MACRO_HEX)
  418. endif()
  419. string(CONFIGURE "${_compiler_id_version_compute_${compiler}}" VERSION_BLOCK @ONLY)
  420. set(${compiler_file_content} "${${compiler_file_content}}${VERSION_BLOCK}\n")
  421. set(PREFIX)
  422. set(MACRO_DEC)
  423. set(MACRO_HEX)
  424. set(pp_if "elif")
  425. foreach(feature ${${_lang}_features})
  426. string(TOUPPER ${feature} feature_upper)
  427. set(feature_PP "COMPILER_${feature_upper}")
  428. set(_define_item "\n# define ${prefix_arg}_${feature_PP} 0\n")
  429. if (_cmake_feature_test_${compiler}_${feature} STREQUAL "1")
  430. set(_define_item "\n# define ${prefix_arg}_${feature_PP} 1\n")
  431. elseif (_cmake_feature_test_${compiler}_${feature})
  432. set(_define_item "\n# define ${prefix_arg}_${feature_PP} 0\n")
  433. set(_define_item "\n# if ${_cmake_feature_test_${compiler}_${feature}}\n# define ${prefix_arg}_${feature_PP} 1\n# else${_define_item}# endif\n")
  434. endif()
  435. set(${compiler_file_content} "${${compiler_file_content}}${_define_item}")
  436. endforeach()
  437. endforeach()
  438. if(pp_if STREQUAL "elif")
  439. set(file_content "${file_content}
  440. # else
  441. # error Unsupported compiler
  442. # endif\n")
  443. endif()
  444. foreach(feature ${${_lang}_features})
  445. string(TOUPPER ${feature} feature_upper)
  446. set(feature_PP "COMPILER_${feature_upper}")
  447. set(def_name ${prefix_arg}_${feature_PP})
  448. if (feature STREQUAL c_restrict)
  449. set(def_value "${prefix_arg}_RESTRICT")
  450. set(file_content "${file_content}
  451. # if ${def_name}
  452. # define ${def_value} restrict
  453. # else
  454. # define ${def_value}
  455. # endif
  456. \n")
  457. endif()
  458. if (feature STREQUAL cxx_constexpr)
  459. set(def_value "${prefix_arg}_CONSTEXPR")
  460. set(file_content "${file_content}
  461. # if ${def_name}
  462. # define ${def_value} constexpr
  463. # else
  464. # define ${def_value}
  465. # endif
  466. \n")
  467. endif()
  468. if (feature STREQUAL cxx_final)
  469. set(def_value "${prefix_arg}_FINAL")
  470. set(file_content "${file_content}
  471. # if ${def_name}
  472. # define ${def_value} final
  473. # else
  474. # define ${def_value}
  475. # endif
  476. \n")
  477. endif()
  478. if (feature STREQUAL cxx_override)
  479. set(def_value "${prefix_arg}_OVERRIDE")
  480. set(file_content "${file_content}
  481. # if ${def_name}
  482. # define ${def_value} override
  483. # else
  484. # define ${def_value}
  485. # endif
  486. \n")
  487. endif()
  488. if (feature STREQUAL cxx_static_assert)
  489. set(def_value "${prefix_arg}_STATIC_ASSERT(X)")
  490. set(def_value_msg "${prefix_arg}_STATIC_ASSERT_MSG(X, MSG)")
  491. set(static_assert_struct "template<bool> struct ${prefix_arg}StaticAssert;\ntemplate<> struct ${prefix_arg}StaticAssert<true>{};\n")
  492. set(def_standard "# define ${def_value} static_assert(X, #X)\n# define ${def_value_msg} static_assert(X, MSG)")
  493. set(def_alternative "${static_assert_struct}# define ${def_value} sizeof(${prefix_arg}StaticAssert<X>)\n# define ${def_value_msg} sizeof(${prefix_arg}StaticAssert<X>)")
  494. set(file_content "${file_content}# if ${def_name}\n${def_standard}\n# else\n${def_alternative}\n# endif\n\n")
  495. endif()
  496. if (feature STREQUAL cxx_alignas)
  497. set(def_value "${prefix_arg}_ALIGNAS(X)")
  498. set(file_content "${file_content}
  499. # if ${def_name}
  500. # define ${def_value} alignas(X)
  501. # elif ${prefix_arg}_COMPILER_IS_GNU || ${prefix_arg}_COMPILER_IS_Clang || ${prefix_arg}_COMPILER_IS_AppleClang
  502. # define ${def_value} __attribute__ ((__aligned__(X)))
  503. # elif ${prefix_arg}_COMPILER_IS_MSVC
  504. # define ${def_value} __declspec(align(X))
  505. # else
  506. # define ${def_value}
  507. # endif
  508. \n")
  509. endif()
  510. if (feature STREQUAL cxx_alignof)
  511. set(def_value "${prefix_arg}_ALIGNOF(X)")
  512. set(file_content "${file_content}
  513. # if ${def_name}
  514. # define ${def_value} alignof(X)
  515. # elif ${prefix_arg}_COMPILER_IS_GNU || ${prefix_arg}_COMPILER_IS_Clang || ${prefix_arg}_COMPILER_IS_AppleClang
  516. # define ${def_value} __alignof__(X)
  517. # elif ${prefix_arg}_COMPILER_IS_MSVC
  518. # define ${def_value} __alignof(X)
  519. # endif
  520. \n")
  521. endif()
  522. if (feature STREQUAL cxx_deleted_functions)
  523. set(def_value "${prefix_arg}_DELETED_FUNCTION")
  524. set(file_content "${file_content}
  525. # if ${def_name}
  526. # define ${def_value} = delete
  527. # else
  528. # define ${def_value}
  529. # endif
  530. \n")
  531. endif()
  532. if (feature STREQUAL cxx_extern_templates)
  533. set(def_value "${prefix_arg}_EXTERN_TEMPLATE")
  534. set(file_content "${file_content}
  535. # if ${def_name}
  536. # define ${def_value} extern
  537. # else
  538. # define ${def_value}
  539. # endif
  540. \n")
  541. endif()
  542. if (feature STREQUAL cxx_noexcept)
  543. set(def_value "${prefix_arg}_NOEXCEPT")
  544. set(file_content "${file_content}
  545. # if ${def_name}
  546. # define ${def_value} noexcept
  547. # define ${def_value}_EXPR(X) noexcept(X)
  548. # else
  549. # define ${def_value}
  550. # define ${def_value}_EXPR(X)
  551. # endif
  552. \n")
  553. endif()
  554. if (feature STREQUAL cxx_nullptr)
  555. set(def_value "${prefix_arg}_NULLPTR")
  556. set(file_content "${file_content}
  557. # if ${def_name}
  558. # define ${def_value} nullptr
  559. # else
  560. # define ${def_value} 0
  561. # endif
  562. \n")
  563. endif()
  564. if (feature STREQUAL cxx_thread_local)
  565. set(def_value "${prefix_arg}_THREAD_LOCAL")
  566. set(file_content "${file_content}
  567. # if ${def_name}
  568. # define ${def_value} thread_local
  569. # elif ${prefix_arg}_COMPILER_IS_GNU || ${prefix_arg}_COMPILER_IS_Clang || ${prefix_arg}_COMPILER_IS_AppleClang
  570. # define ${def_value} __thread
  571. # elif ${prefix_arg}_COMPILER_IS_MSVC
  572. # define ${def_value} __declspec(thread)
  573. # else
  574. // ${def_value} not defined for this configuration.
  575. # endif
  576. \n")
  577. endif()
  578. if (feature STREQUAL cxx_attribute_deprecated)
  579. set(def_name ${prefix_arg}_${feature_PP})
  580. set(def_value "${prefix_arg}_DEPRECATED")
  581. set(file_content "${file_content}
  582. # ifndef ${def_value}
  583. # if ${def_name}
  584. # define ${def_value} [[deprecated]]
  585. # define ${def_value}_MSG(MSG) [[deprecated(MSG)]]
  586. # elif ${prefix_arg}_COMPILER_IS_GNU || ${prefix_arg}_COMPILER_IS_Clang
  587. # define ${def_value} __attribute__((__deprecated__))
  588. # define ${def_value}_MSG(MSG) __attribute__((__deprecated__(MSG)))
  589. # elif ${prefix_arg}_COMPILER_IS_MSVC
  590. # define ${def_value} __declspec(deprecated)
  591. # define ${def_value}_MSG(MSG) __declspec(deprecated(MSG))
  592. # else
  593. # define ${def_value}
  594. # define ${def_value}_MSG(MSG)
  595. # endif
  596. # endif
  597. \n")
  598. endif()
  599. endforeach()
  600. set(file_content "${file_content}#endif\n")
  601. endforeach()
  602. if(_WCD_OUTPUT_FILES_VAR)
  603. foreach(compiler ${_WCD_COMPILERS})
  604. foreach(_lang ${_langs})
  605. if(compiler_file_content_${compiler}_${_lang})
  606. set(CMAKE_CONFIGURABLE_FILE_CONTENT "${compiler_file_content_}")
  607. set(CMAKE_CONFIGURABLE_FILE_CONTENT "${CMAKE_CONFIGURABLE_FILE_CONTENT}${compiler_file_content_${compiler}_${_lang}}")
  608. set(compile_file_name "${_WCD_OUTPUT_DIR}${prefix_arg}_COMPILER_INFO_${compiler}_${_lang}.h")
  609. set(full_path "${main_file_dir}/${compile_file_name}")
  610. list(APPEND ${_WCD_OUTPUT_FILES_VAR} ${full_path})
  611. configure_file("${CMAKE_ROOT}/Modules/CMakeConfigurableFile.in"
  612. "${full_path}"
  613. @ONLY
  614. )
  615. endif()
  616. endforeach()
  617. endforeach()
  618. set(${_WCD_OUTPUT_FILES_VAR} ${${_WCD_OUTPUT_FILES_VAR}} PARENT_SCOPE)
  619. endif()
  620. if (_WCD_EPILOG)
  621. set(file_content "${file_content}\n${_WCD_EPILOG}\n")
  622. endif()
  623. set(file_content "${file_content}\n#endif")
  624. set(CMAKE_CONFIGURABLE_FILE_CONTENT ${file_content})
  625. configure_file("${CMAKE_ROOT}/Modules/CMakeConfigurableFile.in"
  626. "${file_arg}"
  627. @ONLY
  628. )
  629. endfunction()