CMakeDetermineCompilerId.cmake 46 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089
  1. # Distributed under the OSI-approved BSD 3-Clause License. See accompanying
  2. # file Copyright.txt or https://cmake.org/licensing for details.
  3. macro(__determine_compiler_id_test testflags_in userflags)
  4. separate_arguments(testflags UNIX_COMMAND "${testflags_in}")
  5. CMAKE_DETERMINE_COMPILER_ID_BUILD("${lang}" "${testflags}" "${userflags}" "${src}")
  6. CMAKE_DETERMINE_COMPILER_ID_MATCH_VENDOR("${lang}" "${COMPILER_${lang}_PRODUCED_OUTPUT}")
  7. if(NOT CMAKE_${lang}_COMPILER_ID)
  8. foreach(file ${COMPILER_${lang}_PRODUCED_FILES})
  9. CMAKE_DETERMINE_COMPILER_ID_CHECK("${lang}" "${CMAKE_${lang}_COMPILER_ID_DIR}/${file}" "${src}")
  10. endforeach()
  11. endif()
  12. endmacro()
  13. # Function to compile a source file to identify the compiler. This is
  14. # used internally by CMake and should not be included by user code.
  15. # If successful, sets CMAKE_<lang>_COMPILER_ID and CMAKE_<lang>_PLATFORM_ID
  16. function(CMAKE_DETERMINE_COMPILER_ID lang flagvar src)
  17. # Make sure the compiler arguments are clean.
  18. string(STRIP "${CMAKE_${lang}_COMPILER_ARG1}" CMAKE_${lang}_COMPILER_ID_ARG1)
  19. string(REGEX REPLACE " +" ";" CMAKE_${lang}_COMPILER_ID_ARG1 "${CMAKE_${lang}_COMPILER_ID_ARG1}")
  20. # Make sure user-specified compiler flags are used.
  21. if(CMAKE_${lang}_FLAGS)
  22. set(CMAKE_${lang}_COMPILER_ID_FLAGS ${CMAKE_${lang}_FLAGS})
  23. elseif(DEFINED ENV{${flagvar}})
  24. set(CMAKE_${lang}_COMPILER_ID_FLAGS $ENV{${flagvar}})
  25. else(CMAKE_${lang}_FLAGS_INIT)
  26. set(CMAKE_${lang}_COMPILER_ID_FLAGS ${CMAKE_${lang}_FLAGS_INIT})
  27. endif()
  28. string(REPLACE " " ";" CMAKE_${lang}_COMPILER_ID_FLAGS_LIST "${CMAKE_${lang}_COMPILER_ID_FLAGS}")
  29. # Compute the directory in which to run the test.
  30. set(CMAKE_${lang}_COMPILER_ID_DIR ${CMAKE_PLATFORM_INFO_DIR}/CompilerId${lang})
  31. # If we REQUIRE_SUCCESS, i.e. TEST_FLAGS_FIRST has the correct flags, we still need to
  32. # try two combinations: with COMPILER_ID_FLAGS (from user) and without (see issue #21869).
  33. if(CMAKE_${lang}_COMPILER_ID_REQUIRE_SUCCESS)
  34. # If there COMPILER_ID_FLAGS is empty we can error for the first invocation.
  35. if("${CMAKE_${lang}_COMPILER_ID_FLAGS_LIST}" STREQUAL "")
  36. set(__compiler_id_require_success TRUE)
  37. endif()
  38. foreach(userflags "${CMAKE_${lang}_COMPILER_ID_FLAGS_LIST}" "")
  39. __determine_compiler_id_test("${CMAKE_${lang}_COMPILER_ID_TEST_FLAGS_FIRST}" "${userflags}")
  40. if(CMAKE_${lang}_COMPILER_ID)
  41. break()
  42. endif()
  43. set(__compiler_id_require_success TRUE)
  44. endforeach()
  45. else()
  46. # Try building with no extra flags and then try each set
  47. # of helper flags. Stop when the compiler is identified.
  48. foreach(userflags "${CMAKE_${lang}_COMPILER_ID_FLAGS_LIST}" "")
  49. foreach(testflags ${CMAKE_${lang}_COMPILER_ID_TEST_FLAGS_FIRST} "" ${CMAKE_${lang}_COMPILER_ID_TEST_FLAGS})
  50. __determine_compiler_id_test("${testflags}" "${userflags}")
  51. if(CMAKE_${lang}_COMPILER_ID)
  52. break()
  53. endif()
  54. endforeach()
  55. if(CMAKE_${lang}_COMPILER_ID)
  56. break()
  57. endif()
  58. endforeach()
  59. endif()
  60. # Check if compiler id detection gave us the compiler tool.
  61. if(CMAKE_${lang}_COMPILER_ID_TOOL)
  62. set(CMAKE_${lang}_COMPILER "${CMAKE_${lang}_COMPILER_ID_TOOL}")
  63. set(CMAKE_${lang}_COMPILER "${CMAKE_${lang}_COMPILER_ID_TOOL}" PARENT_SCOPE)
  64. elseif(NOT CMAKE_${lang}_COMPILER)
  65. set(CMAKE_${lang}_COMPILER "CMAKE_${lang}_COMPILER-NOTFOUND" PARENT_SCOPE)
  66. endif()
  67. # If the compiler is still unknown, try to query its vendor.
  68. if(CMAKE_${lang}_COMPILER AND NOT CMAKE_${lang}_COMPILER_ID)
  69. foreach(userflags "${CMAKE_${lang}_COMPILER_ID_FLAGS_LIST}" "")
  70. CMAKE_DETERMINE_COMPILER_ID_VENDOR(${lang} "${userflags}")
  71. endforeach()
  72. endif()
  73. # If the compiler is still unknown, fallback to GHS
  74. if(NOT CMAKE_${lang}_COMPILER_ID AND "${CMAKE_GENERATOR}" MATCHES "Green Hills MULTI")
  75. set(CMAKE_${lang}_COMPILER_ID GHS)
  76. file(APPEND ${CMAKE_BINARY_DIR}${CMAKE_FILES_DIRECTORY}/CMakeOutput.log
  77. "The ${lang} compiler identification is falling back to GHS.\n\n")
  78. endif()
  79. # CUDA < 7.5 is missing version macros
  80. if(lang STREQUAL "CUDA"
  81. AND CMAKE_${lang}_COMPILER_ID STREQUAL "NVIDIA"
  82. AND NOT CMAKE_${lang}_COMPILER_VERSION)
  83. execute_process(
  84. COMMAND "${CMAKE_${lang}_COMPILER}"
  85. --version
  86. OUTPUT_VARIABLE output ERROR_VARIABLE output
  87. RESULT_VARIABLE result
  88. TIMEOUT 10
  89. )
  90. if(output MATCHES [=[ V([0-9]+)\.([0-9]+)\.([0-9]+)]=])
  91. set(CMAKE_${lang}_COMPILER_VERSION "${CMAKE_MATCH_1}.${CMAKE_MATCH_2}.${CMAKE_MATCH_3}")
  92. endif()
  93. endif()
  94. # For Swift we need to explicitly query the version.
  95. if(lang STREQUAL "Swift"
  96. AND CMAKE_${lang}_COMPILER
  97. AND NOT CMAKE_${lang}_COMPILER_VERSION)
  98. execute_process(
  99. COMMAND "${CMAKE_${lang}_COMPILER}"
  100. -version
  101. OUTPUT_VARIABLE output ERROR_VARIABLE output
  102. RESULT_VARIABLE result
  103. TIMEOUT 10
  104. )
  105. file(APPEND ${CMAKE_BINARY_DIR}${CMAKE_FILES_DIRECTORY}/CMakeOutput.log
  106. "Running the ${lang} compiler: \"${CMAKE_${lang}_COMPILER}\" -version\n"
  107. "${output}\n"
  108. )
  109. if(output MATCHES [[Swift version ([0-9]+\.[0-9]+(\.[0-9]+)?)]])
  110. set(CMAKE_${lang}_COMPILER_VERSION "${CMAKE_MATCH_1}")
  111. if(NOT CMAKE_${lang}_COMPILER_ID)
  112. set(CMAKE_Swift_COMPILER_ID "Apple")
  113. endif()
  114. endif()
  115. endif()
  116. # For ISPC we need to explicitly query the version.
  117. if(lang STREQUAL "ISPC"
  118. AND CMAKE_${lang}_COMPILER
  119. AND NOT CMAKE_${lang}_COMPILER_VERSION)
  120. execute_process(
  121. COMMAND "${CMAKE_${lang}_COMPILER}"
  122. --version
  123. OUTPUT_VARIABLE output ERROR_VARIABLE output
  124. RESULT_VARIABLE result
  125. TIMEOUT 10
  126. )
  127. file(APPEND ${CMAKE_BINARY_DIR}${CMAKE_FILES_DIRECTORY}/CMakeOutput.log
  128. "Running the ${lang} compiler: \"${CMAKE_${lang}_COMPILER}\" -version\n"
  129. "${output}\n"
  130. )
  131. if(output MATCHES [[ISPC\), ([0-9]+\.[0-9]+(\.[0-9]+)?)]])
  132. set(CMAKE_${lang}_COMPILER_VERSION "${CMAKE_MATCH_1}")
  133. endif()
  134. endif()
  135. if (COMPILER_QNXNTO AND CMAKE_${lang}_COMPILER_ID STREQUAL "GNU")
  136. execute_process(
  137. COMMAND "${CMAKE_${lang}_COMPILER}"
  138. -V
  139. OUTPUT_VARIABLE output ERROR_VARIABLE output
  140. RESULT_VARIABLE result
  141. TIMEOUT 10
  142. )
  143. if (output MATCHES "targets available")
  144. set(CMAKE_${lang}_COMPILER_ID QCC)
  145. # http://community.qnx.com/sf/discussion/do/listPosts/projects.community/discussion.qnx_momentics_community_support.topc3555?_pagenum=2
  146. # The qcc driver does not itself have a version.
  147. endif()
  148. endif()
  149. # The Fujitsu compiler does not always convey version information through
  150. # preprocessor symbols so we extract through command line info
  151. if (CMAKE_${lang}_COMPILER_ID STREQUAL "Fujitsu")
  152. if(NOT CMAKE_${lang}_COMPILER_VERSION)
  153. execute_process(
  154. COMMAND "${CMAKE_${lang}_COMPILER}" -V
  155. OUTPUT_VARIABLE output
  156. ERROR_VARIABLE output
  157. RESULT_VARIABLE result
  158. TIMEOUT 10
  159. )
  160. if (result EQUAL 0)
  161. if (output MATCHES [[Fujitsu [^ ]* Compiler ([0-9]+\.[0-9]+\.[0-9]+)]])
  162. set(CMAKE_${lang}_COMPILER_VERSION "${CMAKE_MATCH_1}")
  163. endif()
  164. endif()
  165. endif()
  166. endif()
  167. # if the format is unknown after all files have been checked, put "Unknown" in the cache
  168. if(NOT CMAKE_EXECUTABLE_FORMAT)
  169. set(CMAKE_EXECUTABLE_FORMAT "Unknown" CACHE INTERNAL "Executable file format")
  170. endif()
  171. if((CMAKE_GENERATOR MATCHES "^Ninja"
  172. OR ((NOT DEFINED CMAKE_DEPENDS_USE_COMPILER OR CMAKE_DEPENDS_USE_COMPILER)
  173. AND CMAKE_GENERATOR MATCHES "Makefiles|WMake"))
  174. AND MSVC_${lang}_ARCHITECTURE_ID)
  175. foreach(userflags "${CMAKE_${lang}_COMPILER_ID_FLAGS_LIST}" "")
  176. CMAKE_DETERMINE_MSVC_SHOWINCLUDES_PREFIX(${lang} "${userflags}")
  177. endforeach()
  178. else()
  179. set(CMAKE_${lang}_CL_SHOWINCLUDES_PREFIX "")
  180. endif()
  181. set(_variant "")
  182. if("x${CMAKE_${lang}_COMPILER_ID}" STREQUAL "xClang"
  183. OR "x${CMAKE_${lang}_COMPILER_ID}" STREQUAL "xIntelLLVM")
  184. if("x${CMAKE_${lang}_SIMULATE_ID}" STREQUAL "xMSVC")
  185. if(CMAKE_GENERATOR MATCHES "Visual Studio")
  186. set(CMAKE_${lang}_COMPILER_FRONTEND_VARIANT "MSVC")
  187. else()
  188. # Test whether an MSVC-like command-line option works.
  189. execute_process(COMMAND "${CMAKE_${lang}_COMPILER}" -?
  190. RESULT_VARIABLE _clang_result
  191. OUTPUT_VARIABLE _clang_stdout
  192. ERROR_VARIABLE _clang_stderr)
  193. if(_clang_result EQUAL 0)
  194. set(CMAKE_${lang}_COMPILER_FRONTEND_VARIANT "MSVC")
  195. else()
  196. set(CMAKE_${lang}_COMPILER_FRONTEND_VARIANT "GNU")
  197. endif()
  198. endif()
  199. set(_variant " with ${CMAKE_${lang}_COMPILER_FRONTEND_VARIANT}-like command-line")
  200. else()
  201. set(CMAKE_${lang}_COMPILER_FRONTEND_VARIANT "GNU")
  202. endif()
  203. else()
  204. set(CMAKE_${lang}_COMPILER_FRONTEND_VARIANT "")
  205. endif()
  206. # Display the final identification result.
  207. if(CMAKE_${lang}_COMPILER_ID)
  208. if(CMAKE_${lang}_COMPILER_VERSION)
  209. set(_version " ${CMAKE_${lang}_COMPILER_VERSION}")
  210. else()
  211. set(_version "")
  212. endif()
  213. if(CMAKE_${lang}_COMPILER_ARCHITECTURE_ID AND "x${CMAKE_${lang}_COMPILER_ID}" STREQUAL "xIAR")
  214. set(_archid " ${CMAKE_${lang}_COMPILER_ARCHITECTURE_ID}")
  215. else()
  216. set(_archid "")
  217. endif()
  218. message(STATUS "The ${lang} compiler identification is "
  219. "${CMAKE_${lang}_COMPILER_ID}${_archid}${_version}${_variant}")
  220. unset(_archid)
  221. unset(_version)
  222. unset(_variant)
  223. else()
  224. message(STATUS "The ${lang} compiler identification is unknown")
  225. endif()
  226. if(lang STREQUAL "Fortran" AND CMAKE_${lang}_COMPILER_ID STREQUAL "XL")
  227. set(CMAKE_${lang}_XL_CPP "${CMAKE_${lang}_COMPILER_ID_CPP}" PARENT_SCOPE)
  228. endif()
  229. set(CMAKE_${lang}_COMPILER_ID "${CMAKE_${lang}_COMPILER_ID}" PARENT_SCOPE)
  230. set(CMAKE_${lang}_PLATFORM_ID "${CMAKE_${lang}_PLATFORM_ID}" PARENT_SCOPE)
  231. set(CMAKE_${lang}_COMPILER_ARCHITECTURE_ID "${CMAKE_${lang}_COMPILER_ARCHITECTURE_ID}" PARENT_SCOPE)
  232. set(MSVC_${lang}_ARCHITECTURE_ID "${MSVC_${lang}_ARCHITECTURE_ID}"
  233. PARENT_SCOPE)
  234. set(CMAKE_${lang}_XCODE_ARCHS "${CMAKE_${lang}_XCODE_ARCHS}" PARENT_SCOPE)
  235. set(CMAKE_${lang}_CL_SHOWINCLUDES_PREFIX "${CMAKE_${lang}_CL_SHOWINCLUDES_PREFIX}" PARENT_SCOPE)
  236. set(CMAKE_${lang}_COMPILER_FRONTEND_VARIANT "${CMAKE_${lang}_COMPILER_FRONTEND_VARIANT}" PARENT_SCOPE)
  237. set(CMAKE_${lang}_COMPILER_VERSION "${CMAKE_${lang}_COMPILER_VERSION}" PARENT_SCOPE)
  238. set(CMAKE_${lang}_COMPILER_VERSION_INTERNAL "${CMAKE_${lang}_COMPILER_VERSION_INTERNAL}" PARENT_SCOPE)
  239. set(CMAKE_${lang}_COMPILER_WRAPPER "${CMAKE_${lang}_COMPILER_WRAPPER}" PARENT_SCOPE)
  240. set(CMAKE_${lang}_SIMULATE_ID "${CMAKE_${lang}_SIMULATE_ID}" PARENT_SCOPE)
  241. set(CMAKE_${lang}_SIMULATE_VERSION "${CMAKE_${lang}_SIMULATE_VERSION}" PARENT_SCOPE)
  242. set(CMAKE_${lang}_STANDARD_COMPUTED_DEFAULT "${CMAKE_${lang}_STANDARD_COMPUTED_DEFAULT}" PARENT_SCOPE)
  243. set(CMAKE_${lang}_COMPILER_PRODUCED_OUTPUT "${COMPILER_${lang}_PRODUCED_OUTPUT}" PARENT_SCOPE)
  244. set(CMAKE_${lang}_COMPILER_PRODUCED_FILES "${COMPILER_${lang}_PRODUCED_FILES}" PARENT_SCOPE)
  245. endfunction()
  246. include(CMakeCompilerIdDetection)
  247. #-----------------------------------------------------------------------------
  248. # Function to write the compiler id source file.
  249. function(CMAKE_DETERMINE_COMPILER_ID_WRITE lang src)
  250. find_file(src_in ${src}.in PATHS ${CMAKE_ROOT}/Modules ${CMAKE_MODULE_PATH} NO_DEFAULT_PATH NO_CMAKE_FIND_ROOT_PATH)
  251. file(READ ${src_in} ID_CONTENT_IN)
  252. compiler_id_detection(CMAKE_${lang}_COMPILER_ID_CONTENT ${lang}
  253. ID_STRING
  254. VERSION_STRINGS
  255. PLATFORM_DEFAULT_COMPILER
  256. )
  257. unset(src_in CACHE)
  258. string(CONFIGURE "${ID_CONTENT_IN}" ID_CONTENT_OUT @ONLY)
  259. file(WRITE ${CMAKE_${lang}_COMPILER_ID_DIR}/${src} "${ID_CONTENT_OUT}")
  260. endfunction()
  261. #-----------------------------------------------------------------------------
  262. # Function to build the compiler id source file and look for output
  263. # files.
  264. function(CMAKE_DETERMINE_COMPILER_ID_BUILD lang testflags userflags src)
  265. # Create a clean working directory.
  266. file(REMOVE_RECURSE ${CMAKE_${lang}_COMPILER_ID_DIR})
  267. file(MAKE_DIRECTORY ${CMAKE_${lang}_COMPILER_ID_DIR})
  268. file(MAKE_DIRECTORY ${CMAKE_${lang}_COMPILER_ID_DIR}/tmp)
  269. CMAKE_DETERMINE_COMPILER_ID_WRITE("${lang}" "${src}")
  270. # Construct a description of this test case.
  271. set(COMPILER_DESCRIPTION
  272. "Compiler: ${CMAKE_${lang}_COMPILER} ${CMAKE_${lang}_COMPILER_ID_ARG1}
  273. Build flags: ${userflags}
  274. Id flags: ${testflags} ${CMAKE_${lang}_COMPILER_ID_FLAGS_ALWAYS}
  275. ")
  276. # Compile the compiler identification source.
  277. if("${CMAKE_GENERATOR}" MATCHES "Visual Studio ([0-9]+)")
  278. set(vs_version ${CMAKE_MATCH_1})
  279. set(id_platform ${CMAKE_VS_PLATFORM_NAME})
  280. set(id_lang "${lang}")
  281. set(id_PostBuildEvent_Command "")
  282. if(CMAKE_VS_PLATFORM_TOOLSET MATCHES "^[Ll][Ll][Vv][Mm](_v[0-9]+(_xp)?)?$")
  283. set(id_cl_var "ClangClExecutable")
  284. elseif(CMAKE_VS_PLATFORM_TOOLSET MATCHES "^[Cc][Ll][Aa][Nn][Gg]([Cc][Ll]$|_[0-9])")
  285. set(id_cl "$(CLToolExe)")
  286. elseif(CMAKE_VS_PLATFORM_TOOLSET MATCHES "v[0-9]+_clang_.*")
  287. set(id_cl clang.exe)
  288. # Executable names have choosen according documentation
  289. # URL: (https://software.intel.com/content/www/us/en/develop/documentation/get-started-with-dpcpp-compiler/top.html#top_GUID-A9B4C91D-97AC-450D-9742-9D895BC8AEE1)
  290. elseif(CMAKE_VS_PLATFORM_TOOLSET MATCHES "Intel")
  291. if(CMAKE_VS_PLATFORM_TOOLSET MATCHES "DPC\\+\\+ Compiler")
  292. set(id_cl dpcpp.exe)
  293. elseif(CMAKE_VS_PLATFORM_TOOLSET MATCHES "C\\+\\+ Compiler 2021")
  294. set(id_cl icx.exe)
  295. elseif(CMAKE_VS_PLATFORM_TOOLSET MATCHES "C\\+\\+ Compiler")
  296. set(id_cl icl.exe)
  297. endif()
  298. else()
  299. set(id_cl cl.exe)
  300. endif()
  301. if(CMAKE_VS_PLATFORM_NAME STREQUAL "Tegra-Android")
  302. set(v NsightTegra)
  303. set(ext vcxproj)
  304. if(lang STREQUAL CXX)
  305. set(id_gcc g++)
  306. set(id_clang clang++)
  307. else()
  308. set(id_gcc gcc)
  309. set(id_clang clang)
  310. endif()
  311. elseif(lang STREQUAL Fortran)
  312. set(v Intel)
  313. set(ext vfproj)
  314. set(id_cl ifort.exe)
  315. elseif(lang STREQUAL CSharp)
  316. set(v 10)
  317. set(ext csproj)
  318. set(id_cl csc.exe)
  319. elseif(NOT "${vs_version}" VERSION_LESS 10)
  320. set(v 10)
  321. set(ext vcxproj)
  322. else()
  323. set(id_version ${vs_version}.00)
  324. set(v 7)
  325. set(ext vcproj)
  326. endif()
  327. if(CMAKE_VS_PLATFORM_TOOLSET)
  328. if(CMAKE_VS_PLATFORM_NAME STREQUAL "Tegra-Android")
  329. set(id_toolset "<NdkToolchainVersion>${CMAKE_VS_PLATFORM_TOOLSET}</NdkToolchainVersion>")
  330. else()
  331. set(id_toolset "<PlatformToolset>${CMAKE_VS_PLATFORM_TOOLSET}</PlatformToolset>")
  332. if(CMAKE_VS_PLATFORM_TOOLSET_VERSION)
  333. set(id_sep "\\")
  334. if(CMAKE_VS_PLATFORM_TOOLSET_VERSION VERSION_GREATER_EQUAL "14.20")
  335. if(EXISTS "${CMAKE_GENERATOR_INSTANCE}/VC/Auxiliary/Build.${CMAKE_VS_PLATFORM_TOOLSET_VERSION}/Microsoft.VCToolsVersion.${CMAKE_VS_PLATFORM_TOOLSET_VERSION}.props")
  336. set(id_sep ".")
  337. endif()
  338. endif()
  339. set(id_toolset_version_props "<Import Project=\"${CMAKE_GENERATOR_INSTANCE}\\VC\\Auxiliary\\Build${id_sep}${CMAKE_VS_PLATFORM_TOOLSET_VERSION}\\Microsoft.VCToolsVersion.${CMAKE_VS_PLATFORM_TOOLSET_VERSION}.props\" />")
  340. unset(id_sep)
  341. endif()
  342. endif()
  343. else()
  344. set(id_toolset "")
  345. set(id_toolset_version_props "")
  346. endif()
  347. if(CMAKE_VS_PLATFORM_TOOLSET_HOST_ARCHITECTURE)
  348. set(id_PreferredToolArchitecture "<PreferredToolArchitecture>${CMAKE_VS_PLATFORM_TOOLSET_HOST_ARCHITECTURE}</PreferredToolArchitecture>")
  349. else()
  350. set(id_PreferredToolArchitecture "")
  351. endif()
  352. if(CMAKE_SYSTEM_NAME STREQUAL "WindowsPhone")
  353. set(id_keyword "Win32Proj")
  354. set(id_system "<ApplicationType>Windows Phone</ApplicationType>")
  355. elseif(CMAKE_SYSTEM_NAME STREQUAL "WindowsStore")
  356. set(id_keyword "Win32Proj")
  357. set(id_system "<ApplicationType>Windows Store</ApplicationType>")
  358. elseif(CMAKE_SYSTEM_NAME STREQUAL "Android")
  359. set(id_keyword "Android")
  360. set(id_system "<ApplicationType>Android</ApplicationType>")
  361. else()
  362. set(id_keyword "Win32Proj")
  363. set(id_system "")
  364. endif()
  365. if(id_keyword STREQUAL "Android")
  366. if(CMAKE_GENERATOR MATCHES "Visual Studio 14")
  367. set(id_system_version "<ApplicationTypeRevision>2.0</ApplicationTypeRevision>")
  368. elseif(CMAKE_GENERATOR MATCHES "Visual Studio 1[56]")
  369. set(id_system_version "<ApplicationTypeRevision>3.0</ApplicationTypeRevision>")
  370. else()
  371. set(id_system_version "")
  372. endif()
  373. elseif(id_system AND CMAKE_SYSTEM_VERSION MATCHES "^([0-9]+\\.[0-9]+)")
  374. set(id_system_version "<ApplicationTypeRevision>${CMAKE_MATCH_1}</ApplicationTypeRevision>")
  375. else()
  376. set(id_system_version "")
  377. endif()
  378. if(id_keyword STREQUAL "Android")
  379. set(id_config_type "DynamicLibrary")
  380. else()
  381. set(id_config_type "Application")
  382. endif()
  383. if(CMAKE_VS_WINDOWS_TARGET_PLATFORM_VERSION)
  384. set(id_WindowsTargetPlatformVersion "<WindowsTargetPlatformVersion>${CMAKE_VS_WINDOWS_TARGET_PLATFORM_VERSION}</WindowsTargetPlatformVersion>")
  385. endif()
  386. if(CMAKE_VS_PLATFORM_TOOLSET_VCTARGETS_CUSTOM_DIR)
  387. set(id_ToolsetVCTargetsDir "<VCTargetsPath>${CMAKE_VS_PLATFORM_TOOLSET_VCTARGETS_CUSTOM_DIR}</VCTargetsPath>")
  388. endif()
  389. set(id_CustomGlobals "")
  390. foreach(pair IN LISTS CMAKE_VS_GLOBALS)
  391. if("${pair}" MATCHES "([^=]+)=(.*)$")
  392. string(APPEND id_CustomGlobals "<${CMAKE_MATCH_1}>${CMAKE_MATCH_2}</${CMAKE_MATCH_1}>\n ")
  393. endif()
  394. endforeach()
  395. if(id_keyword STREQUAL "Android")
  396. set(id_WindowsSDKDesktopARMSupport "")
  397. elseif(id_platform STREQUAL "ARM64")
  398. set(id_WindowsSDKDesktopARMSupport "<WindowsSDKDesktopARM64Support>true</WindowsSDKDesktopARM64Support>")
  399. elseif(id_platform STREQUAL "ARM")
  400. set(id_WindowsSDKDesktopARMSupport "<WindowsSDKDesktopARMSupport>true</WindowsSDKDesktopARMSupport>")
  401. else()
  402. set(id_WindowsSDKDesktopARMSupport "")
  403. endif()
  404. if(CMAKE_VS_WINCE_VERSION)
  405. set(id_entrypoint "mainACRTStartup")
  406. if("${vs_version}" VERSION_LESS 9)
  407. set(id_subsystem 9)
  408. else()
  409. set(id_subsystem 8)
  410. endif()
  411. else()
  412. set(id_subsystem 1)
  413. endif()
  414. set(id_dir ${CMAKE_${lang}_COMPILER_ID_DIR})
  415. set(id_src "${src}")
  416. set(id_compile "ClCompile")
  417. if(id_cl_var)
  418. set(id_PostBuildEvent_Command "echo CMAKE_${lang}_COMPILER=$(${id_cl_var})")
  419. else()
  420. set(id_PostBuildEvent_Command "for %%i in (${id_cl}) do %40echo CMAKE_${lang}_COMPILER=%%~$PATH:i")
  421. endif()
  422. set(id_Import_props "")
  423. set(id_Import_targets "")
  424. set(id_ItemDefinitionGroup_entry "")
  425. set(id_Link_AdditionalDependencies "")
  426. if(lang STREQUAL CUDA)
  427. if(NOT CMAKE_VS_PLATFORM_TOOLSET_CUDA)
  428. message(FATAL_ERROR "No CUDA toolset found.")
  429. endif()
  430. set(cuda_tools "CUDA ${CMAKE_VS_PLATFORM_TOOLSET_CUDA}")
  431. set(id_compile "CudaCompile")
  432. if(CMAKE_VS_PLATFORM_NAME STREQUAL x64)
  433. set(cuda_target "<TargetMachinePlatform>64</TargetMachinePlatform>")
  434. endif()
  435. foreach(arch ${CMAKE_CUDA_ARCHITECTURES})
  436. string(REGEX MATCH "[0-9]+" arch_name "${arch}")
  437. string(APPEND cuda_codegen "compute_${arch_name},sm_${arch_name};")
  438. endforeach()
  439. set(id_ItemDefinitionGroup_entry "<CudaCompile>${cuda_target}<AdditionalOptions>%(AdditionalOptions)-v</AdditionalOptions><CodeGeneration>${cuda_codegen}</CodeGeneration></CudaCompile>")
  440. set(id_PostBuildEvent_Command [[echo CMAKE_CUDA_COMPILER=$(CudaToolkitBinDir)\nvcc.exe]])
  441. if(CMAKE_VS_PLATFORM_TOOLSET_CUDA_CUSTOM_DIR)
  442. # check for legacy cuda custom toolkit folder structure
  443. if(EXISTS ${CMAKE_VS_PLATFORM_TOOLSET_CUDA_CUSTOM_DIR}nvcc)
  444. set(id_CudaToolkitCustomDir "<CudaToolkitCustomDir>${CMAKE_VS_PLATFORM_TOOLSET_CUDA_CUSTOM_DIR}nvcc</CudaToolkitCustomDir>")
  445. else()
  446. set(id_CudaToolkitCustomDir "<CudaToolkitCustomDir>${CMAKE_VS_PLATFORM_TOOLSET_CUDA_CUSTOM_DIR}</CudaToolkitCustomDir>")
  447. endif()
  448. if(EXISTS ${CMAKE_VS_PLATFORM_TOOLSET_CUDA_CUSTOM_DIR}CUDAVisualStudioIntegration)
  449. string(CONCAT id_Import_props "<Import Project=\"${CMAKE_VS_PLATFORM_TOOLSET_CUDA_CUSTOM_DIR}CUDAVisualStudioIntegration\\extras\\visual_studio_integration\\MSBuildExtensions\\${cuda_tools}.props\" />")
  450. string(CONCAT id_Import_targets "<Import Project=\"${CMAKE_VS_PLATFORM_TOOLSET_CUDA_CUSTOM_DIR}CUDAVisualStudioIntegration\\extras\\visual_studio_integration\\MSBuildExtensions\\${cuda_tools}.targets\" />")
  451. else()
  452. string(CONCAT id_Import_props "<Import Project=\"${CMAKE_VS_PLATFORM_TOOLSET_CUDA_CUSTOM_DIR}\\extras\\visual_studio_integration\\MSBuildExtensions\\${cuda_tools}.props\" />")
  453. string(CONCAT id_Import_targets "<Import Project=\"${CMAKE_VS_PLATFORM_TOOLSET_CUDA_CUSTOM_DIR}\\extras\\visual_studio_integration\\MSBuildExtensions\\${cuda_tools}.targets\" />")
  454. endif()
  455. else()
  456. string(CONCAT id_Import_props [[<Import Project="$(VCTargetsPath)\BuildCustomizations\]] "${cuda_tools}" [[.props" />]])
  457. string(CONCAT id_Import_targets [[<Import Project="$(VCTargetsPath)\BuildCustomizations\]] "${cuda_tools}" [[.targets" />]])
  458. endif()
  459. if(CMAKE_CUDA_FLAGS MATCHES "(^| )-cudart +shared( |$)")
  460. set(id_Link_AdditionalDependencies "<AdditionalDependencies>cudart.lib</AdditionalDependencies>")
  461. else()
  462. set(id_Link_AdditionalDependencies "<AdditionalDependencies>cudart_static.lib</AdditionalDependencies>")
  463. endif()
  464. endif()
  465. configure_file(${CMAKE_ROOT}/Modules/CompilerId/VS-${v}.${ext}.in
  466. ${id_dir}/CompilerId${lang}.${ext} @ONLY)
  467. if(CMAKE_VS_MSBUILD_COMMAND AND NOT lang STREQUAL "Fortran")
  468. set(command "${CMAKE_VS_MSBUILD_COMMAND}" "CompilerId${lang}.${ext}"
  469. "/p:Configuration=Debug" "/p:Platform=${id_platform}" "/p:VisualStudioVersion=${vs_version}.0"
  470. )
  471. elseif(CMAKE_VS_DEVENV_COMMAND)
  472. set(command "${CMAKE_VS_DEVENV_COMMAND}" "CompilerId${lang}.${ext}" "/build" "Debug")
  473. else()
  474. set(command "")
  475. endif()
  476. if(command)
  477. execute_process(
  478. COMMAND ${command}
  479. WORKING_DIRECTORY ${CMAKE_${lang}_COMPILER_ID_DIR}
  480. OUTPUT_VARIABLE CMAKE_${lang}_COMPILER_ID_OUTPUT
  481. ERROR_VARIABLE CMAKE_${lang}_COMPILER_ID_OUTPUT
  482. RESULT_VARIABLE CMAKE_${lang}_COMPILER_ID_RESULT
  483. )
  484. else()
  485. set(CMAKE_${lang}_COMPILER_ID_RESULT 1)
  486. set(CMAKE_${lang}_COMPILER_ID_OUTPUT "VS environment not known to support ${lang}")
  487. endif()
  488. # Match the compiler location line printed out.
  489. if("${CMAKE_${lang}_COMPILER_ID_OUTPUT}" MATCHES "CMAKE_${lang}_COMPILER=([^%\r\n]+)[\r\n]")
  490. # Strip VS diagnostic output from the end of the line.
  491. string(REGEX REPLACE " \\(TaskId:[0-9]*\\)$" "" _comp "${CMAKE_MATCH_1}")
  492. if(EXISTS "${_comp}")
  493. file(TO_CMAKE_PATH "${_comp}" _comp)
  494. set(CMAKE_${lang}_COMPILER_ID_TOOL "${_comp}" PARENT_SCOPE)
  495. endif()
  496. endif()
  497. elseif("${CMAKE_GENERATOR}" MATCHES "Xcode")
  498. set(id_lang "${lang}")
  499. set(id_type ${CMAKE_${lang}_COMPILER_XCODE_TYPE})
  500. set(id_dir ${CMAKE_${lang}_COMPILER_ID_DIR})
  501. set(id_src "${src}")
  502. if(CMAKE_XCODE_PLATFORM_TOOLSET)
  503. set(id_toolset "GCC_VERSION = ${CMAKE_XCODE_PLATFORM_TOOLSET};")
  504. else()
  505. set(id_toolset "")
  506. endif()
  507. if("${lang}" STREQUAL "Swift")
  508. if(CMAKE_Swift_LANGUAGE_VERSION)
  509. set(id_lang_version "SWIFT_VERSION = ${CMAKE_Swift_LANGUAGE_VERSION};")
  510. elseif(XCODE_VERSION VERSION_GREATER_EQUAL 10.2)
  511. set(id_lang_version "SWIFT_VERSION = 4.0;")
  512. elseif(XCODE_VERSION VERSION_GREATER_EQUAL 8.3)
  513. set(id_lang_version "SWIFT_VERSION = 3.0;")
  514. else()
  515. set(id_lang_version "SWIFT_VERSION = 2.3;")
  516. endif()
  517. else()
  518. set(id_lang_version "")
  519. endif()
  520. if(CMAKE_OSX_DEPLOYMENT_TARGET)
  521. set(id_deployment_target
  522. "MACOSX_DEPLOYMENT_TARGET = \"${CMAKE_OSX_DEPLOYMENT_TARGET}\";")
  523. else()
  524. set(id_deployment_target "")
  525. endif()
  526. set(id_product_type "com.apple.product-type.tool")
  527. if(CMAKE_OSX_SYSROOT)
  528. set(id_sdkroot "SDKROOT = \"${CMAKE_OSX_SYSROOT}\";")
  529. if(CMAKE_OSX_SYSROOT MATCHES "(^|/)[Ii][Pp][Hh][Oo][Nn][Ee]" OR
  530. CMAKE_OSX_SYSROOT MATCHES "(^|/)[Aa][Pp][Pp][Ll][Ee][Tt][Vv]")
  531. set(id_product_type "com.apple.product-type.bundle.unit-test")
  532. elseif(CMAKE_OSX_SYSROOT MATCHES "(^|/)[Ww][Aa][Tt][Cc][Hh]")
  533. set(id_product_type "com.apple.product-type.framework")
  534. endif()
  535. else()
  536. set(id_sdkroot "")
  537. endif()
  538. set(id_clang_cxx_library "")
  539. set(stdlib_regex "(^| )(-stdlib=)([^ ]+)( |$)")
  540. string(REGEX MATCHALL "${stdlib_regex}" all_stdlib_matches "${CMAKE_CXX_FLAGS}")
  541. if(all_stdlib_matches)
  542. list(GET all_stdlib_matches "-1" last_stdlib_match)
  543. if(last_stdlib_match MATCHES "${stdlib_regex}")
  544. set(id_clang_cxx_library "CLANG_CXX_LIBRARY = \"${CMAKE_MATCH_3}\";")
  545. endif()
  546. endif()
  547. if(CMAKE_SYSTEM_NAME STREQUAL "Darwin" AND CMAKE_OSX_SYSROOT MATCHES "^$|[Mm][Aa][Cc][Oo][Ss]")
  548. # When targeting macOS, use only the host architecture.
  549. if (_CMAKE_APPLE_ARCHS_DEFAULT)
  550. set(id_archs "ARCHS = \"${_CMAKE_APPLE_ARCHS_DEFAULT}\";")
  551. set(id_arch_active "ONLY_ACTIVE_ARCH = NO;")
  552. else()
  553. set(id_archs [[ARCHS = "$(NATIVE_ARCH_ACTUAL)";]])
  554. set(id_arch_active "ONLY_ACTIVE_ARCH = YES;")
  555. endif()
  556. else()
  557. set(id_archs "")
  558. set(id_arch_active "ONLY_ACTIVE_ARCH = YES;")
  559. endif()
  560. configure_file(${CMAKE_ROOT}/Modules/CompilerId/Xcode-3.pbxproj.in
  561. ${id_dir}/CompilerId${lang}.xcodeproj/project.pbxproj @ONLY)
  562. unset(_ENV_MACOSX_DEPLOYMENT_TARGET)
  563. if(DEFINED ENV{MACOSX_DEPLOYMENT_TARGET})
  564. set(_ENV_MACOSX_DEPLOYMENT_TARGET "$ENV{MACOSX_DEPLOYMENT_TARGET}")
  565. set(ENV{MACOSX_DEPLOYMENT_TARGET} "")
  566. endif()
  567. execute_process(COMMAND xcodebuild
  568. WORKING_DIRECTORY ${CMAKE_${lang}_COMPILER_ID_DIR}
  569. OUTPUT_VARIABLE CMAKE_${lang}_COMPILER_ID_OUTPUT
  570. ERROR_VARIABLE CMAKE_${lang}_COMPILER_ID_OUTPUT
  571. RESULT_VARIABLE CMAKE_${lang}_COMPILER_ID_RESULT
  572. )
  573. if(DEFINED _ENV_MACOSX_DEPLOYMENT_TARGET)
  574. set(ENV{MACOSX_DEPLOYMENT_TARGET} "${_ENV_MACOSX_DEPLOYMENT_TARGET}")
  575. endif()
  576. if(DEFINED CMAKE_${lang}_COMPILER_ID_TOOL_MATCH_REGEX)
  577. if("${CMAKE_${lang}_COMPILER_ID_OUTPUT}" MATCHES "${CMAKE_${lang}_COMPILER_ID_TOOL_MATCH_REGEX}")
  578. set(_comp "${CMAKE_MATCH_${CMAKE_${lang}_COMPILER_ID_TOOL_MATCH_INDEX}}")
  579. if(EXISTS "${_comp}")
  580. set(CMAKE_${lang}_COMPILER_ID_TOOL "${_comp}" PARENT_SCOPE)
  581. endif()
  582. endif()
  583. endif()
  584. if("${CMAKE_${lang}_COMPILER_ID_OUTPUT}" MATCHES "ARCHS=([^%\r\n]+)[\r\n]")
  585. set(CMAKE_${lang}_XCODE_ARCHS "${CMAKE_MATCH_1}")
  586. separate_arguments(CMAKE_${lang}_XCODE_ARCHS)
  587. set(CMAKE_${lang}_XCODE_ARCHS "${CMAKE_${lang}_XCODE_ARCHS}" PARENT_SCOPE)
  588. endif()
  589. elseif("${CMAKE_GENERATOR}" MATCHES "Green Hills MULTI")
  590. set(id_dir ${CMAKE_${lang}_COMPILER_ID_DIR})
  591. set(id_src "${src}")
  592. if (GHS_PRIMARY_TARGET)
  593. set(ghs_primary_target "${GHS_PRIMARY_TARGET}")
  594. else()
  595. set(ghs_primary_target "${CMAKE_GENERATOR_PLATFORM}_${GHS_TARGET_PLATFORM}.tgt")
  596. endif()
  597. if ("${GHS_TARGET_PLATFORM}" MATCHES "integrity")
  598. set(bsp_name "macro GHS_BSP=${GHS_BSP_NAME}")
  599. set(os_dir "macro GHS_OS=${GHS_OS_DIR}")
  600. endif()
  601. set(command "${CMAKE_MAKE_PROGRAM}" "-commands" "-top" "GHS_default.gpj")
  602. configure_file(${CMAKE_ROOT}/Modules/CompilerId/GHS_default.gpj.in
  603. ${id_dir}/GHS_default.gpj @ONLY)
  604. configure_file(${CMAKE_ROOT}/Modules/CompilerId/GHS_lib.gpj.in
  605. ${id_dir}/GHS_lib.gpj @ONLY)
  606. execute_process(COMMAND ${command}
  607. WORKING_DIRECTORY ${id_dir}
  608. OUTPUT_VARIABLE CMAKE_${lang}_COMPILER_ID_OUTPUT
  609. ERROR_VARIABLE CMAKE_${lang}_COMPILER_ID_OUTPUT
  610. RESULT_VARIABLE CMAKE_${lang}_COMPILER_ID_RESULT
  611. )
  612. # Match the compiler location line printed out.
  613. set(ghs_toolpath "${CMAKE_MAKE_PROGRAM}")
  614. if(CMAKE_HOST_UNIX)
  615. string(REPLACE "/gbuild" "/" ghs_toolpath ${ghs_toolpath})
  616. else()
  617. string(REPLACE "/gbuild.exe" "/" ghs_toolpath ${ghs_toolpath})
  618. string(REPLACE / "\\\\" ghs_toolpath ${ghs_toolpath})
  619. endif()
  620. if("${CMAKE_${lang}_COMPILER_ID_OUTPUT}" MATCHES "(${ghs_toolpath}[^ ]*)")
  621. if(CMAKE_HOST_UNIX)
  622. set(_comp "${CMAKE_MATCH_1}")
  623. else()
  624. set(_comp "${CMAKE_MATCH_1}.exe")
  625. endif()
  626. if(EXISTS "${_comp}")
  627. file(TO_CMAKE_PATH "${_comp}" _comp)
  628. set(CMAKE_${lang}_COMPILER_ID_TOOL "${_comp}" PARENT_SCOPE)
  629. endif()
  630. endif()
  631. else()
  632. execute_process(
  633. COMMAND "${CMAKE_${lang}_COMPILER}"
  634. ${CMAKE_${lang}_COMPILER_ID_ARG1}
  635. ${userflags}
  636. ${testflags}
  637. ${CMAKE_${lang}_COMPILER_ID_FLAGS_ALWAYS}
  638. "${src}"
  639. WORKING_DIRECTORY ${CMAKE_${lang}_COMPILER_ID_DIR}
  640. OUTPUT_VARIABLE CMAKE_${lang}_COMPILER_ID_OUTPUT
  641. ERROR_VARIABLE CMAKE_${lang}_COMPILER_ID_OUTPUT
  642. RESULT_VARIABLE CMAKE_${lang}_COMPILER_ID_RESULT
  643. )
  644. if("${CMAKE_${lang}_COMPILER_ID_OUTPUT}" MATCHES "exec: [^\n]*\\((/[^,\n]*/cpp),CMakeFortranCompilerId.F")
  645. set(_cpp "${CMAKE_MATCH_1}")
  646. if(EXISTS "${_cpp}")
  647. set(CMAKE_${lang}_COMPILER_ID_CPP "${_cpp}" PARENT_SCOPE)
  648. endif()
  649. endif()
  650. endif()
  651. # Check the result of compilation.
  652. if(CMAKE_${lang}_COMPILER_ID_RESULT
  653. # Intel Fortran warns and ignores preprocessor lines without /fpp
  654. OR CMAKE_${lang}_COMPILER_ID_OUTPUT MATCHES "Bad # preprocessor line"
  655. )
  656. # Compilation failed.
  657. set(MSG
  658. "Compiling the ${lang} compiler identification source file \"${src}\" failed.
  659. ${COMPILER_DESCRIPTION}
  660. The output was:
  661. ${CMAKE_${lang}_COMPILER_ID_RESULT}
  662. ${CMAKE_${lang}_COMPILER_ID_OUTPUT}
  663. ")
  664. file(APPEND ${CMAKE_BINARY_DIR}${CMAKE_FILES_DIRECTORY}/CMakeError.log "${MSG}")
  665. # Some languages may know the correct/desired set of flags and want to fail right away if they don't work.
  666. # This is currently only used by CUDA.
  667. if(__compiler_id_require_success)
  668. message(FATAL_ERROR "${MSG}")
  669. endif()
  670. # No output files should be inspected.
  671. set(COMPILER_${lang}_PRODUCED_FILES)
  672. set(COMPILER_${lang}_PRODUCED_OUTPUT)
  673. else()
  674. # Compilation succeeded.
  675. file(APPEND ${CMAKE_BINARY_DIR}${CMAKE_FILES_DIRECTORY}/CMakeOutput.log
  676. "Compiling the ${lang} compiler identification source file \"${src}\" succeeded.
  677. ${COMPILER_DESCRIPTION}
  678. The output was:
  679. ${CMAKE_${lang}_COMPILER_ID_RESULT}
  680. ${CMAKE_${lang}_COMPILER_ID_OUTPUT}
  681. ")
  682. # Find the executable produced by the compiler, try all files in the
  683. # binary dir.
  684. string(REGEX REPLACE "([][])" "[\\1]" _glob_id_dir "${CMAKE_${lang}_COMPILER_ID_DIR}")
  685. file(GLOB files
  686. RELATIVE ${CMAKE_${lang}_COMPILER_ID_DIR}
  687. # normal case
  688. ${_glob_id_dir}/*
  689. # com.apple.package-type.bundle.unit-test
  690. ${_glob_id_dir}/*.xctest/*
  691. # com.apple.product-type.framework
  692. ${_glob_id_dir}/*.framework/*
  693. )
  694. list(REMOVE_ITEM files "${src}")
  695. set(COMPILER_${lang}_PRODUCED_FILES "")
  696. foreach(file ${files})
  697. if(NOT IS_DIRECTORY ${CMAKE_${lang}_COMPILER_ID_DIR}/${file})
  698. list(APPEND COMPILER_${lang}_PRODUCED_FILES ${file})
  699. file(APPEND ${CMAKE_BINARY_DIR}${CMAKE_FILES_DIRECTORY}/CMakeOutput.log
  700. "Compilation of the ${lang} compiler identification source \""
  701. "${src}\" produced \"${file}\"\n\n")
  702. endif()
  703. endforeach()
  704. if(NOT COMPILER_${lang}_PRODUCED_FILES)
  705. # No executable was found.
  706. file(APPEND ${CMAKE_BINARY_DIR}${CMAKE_FILES_DIRECTORY}/CMakeError.log
  707. "Compilation of the ${lang} compiler identification source \""
  708. "${src}\" did not produce an executable in \""
  709. "${CMAKE_${lang}_COMPILER_ID_DIR}\".\n\n")
  710. endif()
  711. set(COMPILER_${lang}_PRODUCED_OUTPUT "${CMAKE_${lang}_COMPILER_ID_OUTPUT}")
  712. endif()
  713. # Return the files produced by the compilation.
  714. set(COMPILER_${lang}_PRODUCED_FILES "${COMPILER_${lang}_PRODUCED_FILES}" PARENT_SCOPE)
  715. set(COMPILER_${lang}_PRODUCED_OUTPUT "${COMPILER_${lang}_PRODUCED_OUTPUT}" PARENT_SCOPE)
  716. endfunction()
  717. #-----------------------------------------------------------------------------
  718. # Function to extract the compiler id from compiler output.
  719. function(CMAKE_DETERMINE_COMPILER_ID_MATCH_VENDOR lang output)
  720. foreach(vendor ${CMAKE_${lang}_COMPILER_ID_MATCH_VENDORS})
  721. if(output MATCHES "${CMAKE_${lang}_COMPILER_ID_MATCH_VENDOR_REGEX_${vendor}}")
  722. set(CMAKE_${lang}_COMPILER_ID "${vendor}")
  723. endif()
  724. endforeach()
  725. set(CMAKE_${lang}_COMPILER_ID "${CMAKE_${lang}_COMPILER_ID}" PARENT_SCOPE)
  726. endfunction()
  727. #-----------------------------------------------------------------------------
  728. # Function to extract the compiler id from an executable.
  729. function(CMAKE_DETERMINE_COMPILER_ID_CHECK lang file)
  730. # Look for a compiler id if not yet known.
  731. if(NOT CMAKE_${lang}_COMPILER_ID)
  732. # Read the compiler identification string from the executable file.
  733. set(COMPILER_ID)
  734. set(COMPILER_VERSION)
  735. set(COMPILER_VERSION_MAJOR 0)
  736. set(COMPILER_VERSION_MINOR 0)
  737. set(COMPILER_VERSION_PATCH 0)
  738. set(COMPILER_VERSION_TWEAK 0)
  739. set(COMPILER_VERSION_INTERNAL "")
  740. set(HAVE_COMPILER_VERSION_MAJOR 0)
  741. set(HAVE_COMPILER_VERSION_MINOR 0)
  742. set(HAVE_COMPILER_VERSION_PATCH 0)
  743. set(HAVE_COMPILER_VERSION_TWEAK 0)
  744. set(COMPILER_WRAPPER)
  745. set(DIGIT_VALUE_1 1)
  746. set(DIGIT_VALUE_2 10)
  747. set(DIGIT_VALUE_3 100)
  748. set(DIGIT_VALUE_4 1000)
  749. set(DIGIT_VALUE_5 10000)
  750. set(DIGIT_VALUE_6 100000)
  751. set(DIGIT_VALUE_7 1000000)
  752. set(DIGIT_VALUE_8 10000000)
  753. set(PLATFORM_ID)
  754. set(ARCHITECTURE_ID)
  755. set(SIMULATE_ID)
  756. set(SIMULATE_VERSION)
  757. foreach(encoding "" "ENCODING;UTF-16LE" "ENCODING;UTF-16BE")
  758. file(STRINGS "${file}" CMAKE_${lang}_COMPILER_ID_STRINGS
  759. LIMIT_COUNT 38 ${encoding}
  760. REGEX ".?I.?N.?F.?O.?:.?[A-Za-z0-9_]+\\[[^]]*\\]")
  761. if(NOT CMAKE_${lang}_COMPILER_ID_STRINGS STREQUAL "")
  762. break()
  763. endif()
  764. endforeach()
  765. # With the IAR Compiler, some strings are found twice, first time as incomplete
  766. # list like "?<Constant "INFO:compiler[IAR]">". Remove the incomplete copies.
  767. list(FILTER CMAKE_${lang}_COMPILER_ID_STRINGS EXCLUDE REGEX "\\?<Constant \\\"")
  768. # The IAR-AVR compiler uses a binary format that places a '6'
  769. # character (0x34) before each character in the string. Strip
  770. # out these characters without removing any legitimate characters.
  771. if(CMAKE_${lang}_COMPILER_ID_STRINGS MATCHES "(.)I.N.F.O.:.")
  772. string(REGEX REPLACE "${CMAKE_MATCH_1}([^;])" "\\1"
  773. CMAKE_${lang}_COMPILER_ID_STRINGS "${CMAKE_${lang}_COMPILER_ID_STRINGS}")
  774. endif()
  775. # Remove arbitrary text that may appear before or after each INFO string.
  776. string(REGEX MATCHALL "INFO:[A-Za-z0-9_]+\\[([^]\"]*)\\]"
  777. CMAKE_${lang}_COMPILER_ID_STRINGS "${CMAKE_${lang}_COMPILER_ID_STRINGS}")
  778. # In C# binaries, some strings are found more than once.
  779. list(REMOVE_DUPLICATES CMAKE_${lang}_COMPILER_ID_STRINGS)
  780. set(COMPILER_ID_TWICE)
  781. foreach(info ${CMAKE_${lang}_COMPILER_ID_STRINGS})
  782. if("${info}" MATCHES "INFO:compiler\\[([^]\"]*)\\]")
  783. if(COMPILER_ID)
  784. set(COMPILER_ID_TWICE 1)
  785. endif()
  786. set(COMPILER_ID "${CMAKE_MATCH_1}")
  787. endif()
  788. if("${info}" MATCHES "INFO:platform\\[([^]\"]*)\\]")
  789. set(PLATFORM_ID "${CMAKE_MATCH_1}")
  790. endif()
  791. if("${info}" MATCHES "INFO:arch\\[([^]\"]*)\\]")
  792. set(ARCHITECTURE_ID "${CMAKE_MATCH_1}")
  793. endif()
  794. if("${info}" MATCHES "INFO:compiler_version\\[([^]\"]*)\\]")
  795. string(REGEX REPLACE "^0+([0-9]+)" "\\1" COMPILER_VERSION "${CMAKE_MATCH_1}")
  796. string(REGEX REPLACE "\\.0+([0-9])" ".\\1" COMPILER_VERSION "${COMPILER_VERSION}")
  797. endif()
  798. if("${info}" MATCHES "INFO:compiler_version_internal\\[([^]\"]*)\\]")
  799. set(COMPILER_VERSION_INTERNAL "${CMAKE_MATCH_1}")
  800. string(REGEX REPLACE "^0+([0-9]+)" "\\1" COMPILER_VERSION_INTERNAL "${COMPILER_VERSION_INTERNAL}")
  801. string(REGEX REPLACE "\\.0+([0-9]+)" ".\\1" COMPILER_VERSION_INTERNAL "${COMPILER_VERSION_INTERNAL}")
  802. string(STRIP "${COMPILER_VERSION_INTERNAL}" COMPILER_VERSION_INTERNAL)
  803. endif()
  804. foreach(comp MAJOR MINOR PATCH TWEAK)
  805. foreach(digit 1 2 3 4 5 6 7 8 9)
  806. if("${info}" MATCHES "INFO:compiler_version_${comp}_digit_${digit}\\[([0-9])\\]")
  807. set(value ${CMAKE_MATCH_1})
  808. math(EXPR COMPILER_VERSION_${comp} "${COMPILER_VERSION_${comp}} + ${value} * ${DIGIT_VALUE_${digit}}")
  809. set(HAVE_COMPILER_VERSION_${comp} 1)
  810. endif()
  811. endforeach()
  812. endforeach()
  813. if("${info}" MATCHES "INFO:compiler_wrapper\\[([^]\"]*)\\]")
  814. set(COMPILER_WRAPPER "${CMAKE_MATCH_1}")
  815. endif()
  816. if("${info}" MATCHES "INFO:simulate\\[([^]\"]*)\\]")
  817. set(SIMULATE_ID "${CMAKE_MATCH_1}")
  818. endif()
  819. if("${info}" MATCHES "INFO:simulate_version\\[([^]\"]*)\\]")
  820. string(REGEX REPLACE "^0+([0-9])" "\\1" SIMULATE_VERSION "${CMAKE_MATCH_1}")
  821. string(REGEX REPLACE "\\.0+([0-9])" ".\\1" SIMULATE_VERSION "${SIMULATE_VERSION}")
  822. endif()
  823. if("${info}" MATCHES "INFO:qnxnto\\[\\]")
  824. set(COMPILER_QNXNTO 1)
  825. endif()
  826. if("${info}" MATCHES "INFO:dialect_default\\[([^]\"]*)\\]")
  827. set(CMAKE_${lang}_STANDARD_COMPUTED_DEFAULT "${CMAKE_MATCH_1}")
  828. endif()
  829. endforeach()
  830. # Construct compiler version from components if needed.
  831. if(NOT DEFINED COMPILER_VERSION AND HAVE_COMPILER_VERSION_MAJOR)
  832. set(COMPILER_VERSION "${COMPILER_VERSION_MAJOR}")
  833. if(HAVE_COMPILER_VERSION_MINOR)
  834. string(APPEND COMPILER_VERSION ".${COMPILER_VERSION_MINOR}")
  835. if(HAVE_COMPILER_VERSION_PATCH)
  836. string(APPEND COMPILER_VERSION ".${COMPILER_VERSION_PATCH}")
  837. if(HAVE_COMPILER_VERSION_TWEAK)
  838. string(APPEND COMPILER_VERSION ".${COMPILER_VERSION_TWEAK}")
  839. endif()
  840. endif()
  841. endif()
  842. endif()
  843. # Detect the exact architecture from the PE header.
  844. if(WIN32)
  845. # The offset to the PE signature is stored at 0x3c.
  846. file(READ ${file} peoffsethex LIMIT 1 OFFSET 60 HEX)
  847. if(NOT peoffsethex STREQUAL "")
  848. string(SUBSTRING "${peoffsethex}" 0 1 peoffsethex1)
  849. string(SUBSTRING "${peoffsethex}" 1 1 peoffsethex2)
  850. set(peoffsetexpression "${peoffsethex1} * 16 + ${peoffsethex2}")
  851. string(REPLACE "a" "10" peoffsetexpression "${peoffsetexpression}")
  852. string(REPLACE "b" "11" peoffsetexpression "${peoffsetexpression}")
  853. string(REPLACE "c" "12" peoffsetexpression "${peoffsetexpression}")
  854. string(REPLACE "d" "13" peoffsetexpression "${peoffsetexpression}")
  855. string(REPLACE "e" "14" peoffsetexpression "${peoffsetexpression}")
  856. string(REPLACE "f" "15" peoffsetexpression "${peoffsetexpression}")
  857. math(EXPR peoffset "${peoffsetexpression}")
  858. file(READ ${file} peheader LIMIT 6 OFFSET ${peoffset} HEX)
  859. if(peheader STREQUAL "50450000a201")
  860. set(ARCHITECTURE_ID "SH3")
  861. elseif(peheader STREQUAL "50450000a301")
  862. set(ARCHITECTURE_ID "SH3DSP")
  863. elseif(peheader STREQUAL "50450000a601")
  864. set(ARCHITECTURE_ID "SH4")
  865. elseif(peheader STREQUAL "50450000a801")
  866. set(ARCHITECTURE_ID "SH5")
  867. endif()
  868. endif()
  869. endif()
  870. # Check if a valid compiler and platform were found.
  871. if(COMPILER_ID AND NOT COMPILER_ID_TWICE)
  872. set(CMAKE_${lang}_COMPILER_ID "${COMPILER_ID}")
  873. set(CMAKE_${lang}_PLATFORM_ID "${PLATFORM_ID}")
  874. set(CMAKE_${lang}_COMPILER_ARCHITECTURE_ID "${ARCHITECTURE_ID}")
  875. set(MSVC_${lang}_ARCHITECTURE_ID "${ARCHITECTURE_ID}")
  876. set(CMAKE_${lang}_COMPILER_VERSION "${COMPILER_VERSION}")
  877. set(CMAKE_${lang}_COMPILER_VERSION_INTERNAL "${COMPILER_VERSION_INTERNAL}")
  878. set(CMAKE_${lang}_SIMULATE_ID "${SIMULATE_ID}")
  879. set(CMAKE_${lang}_SIMULATE_VERSION "${SIMULATE_VERSION}")
  880. endif()
  881. # Check the compiler identification string.
  882. if(CMAKE_${lang}_COMPILER_ID)
  883. # The compiler identification was found.
  884. file(APPEND ${CMAKE_BINARY_DIR}${CMAKE_FILES_DIRECTORY}/CMakeOutput.log
  885. "The ${lang} compiler identification is ${CMAKE_${lang}_COMPILER_ID}, found in \""
  886. "${file}\"\n\n")
  887. else()
  888. # The compiler identification could not be found.
  889. file(APPEND ${CMAKE_BINARY_DIR}${CMAKE_FILES_DIRECTORY}/CMakeError.log
  890. "The ${lang} compiler identification could not be found in \""
  891. "${file}\"\n\n")
  892. endif()
  893. endif()
  894. # try to figure out the executable format: ELF, COFF, Mach-O
  895. if(NOT CMAKE_EXECUTABLE_FORMAT)
  896. file(READ ${file} CMAKE_EXECUTABLE_MAGIC LIMIT 4 HEX)
  897. # ELF files start with 0x7f"ELF"
  898. if("${CMAKE_EXECUTABLE_MAGIC}" STREQUAL "7f454c46")
  899. set(CMAKE_EXECUTABLE_FORMAT "ELF" CACHE INTERNAL "Executable file format")
  900. endif()
  901. # # COFF (.exe) files start with "MZ"
  902. # if("${CMAKE_EXECUTABLE_MAGIC}" MATCHES "4d5a....")
  903. # set(CMAKE_EXECUTABLE_FORMAT "COFF" CACHE INTERNAL "Executable file format")
  904. # endif()
  905. #
  906. # Mach-O files start with MH_MAGIC or MH_CIGAM
  907. if("${CMAKE_EXECUTABLE_MAGIC}" MATCHES "feedface|cefaedfe|feedfacf|cffaedfe")
  908. set(CMAKE_EXECUTABLE_FORMAT "MACHO" CACHE INTERNAL "Executable file format")
  909. endif()
  910. # XCOFF files start with 0x01 followed by 0xDF (32-bit) or 0xF7 (64-bit).
  911. if("${CMAKE_EXECUTABLE_MAGIC}" MATCHES "^01(df|f7)")
  912. set(CMAKE_EXECUTABLE_FORMAT "XCOFF" CACHE INTERNAL "Executable file format")
  913. endif()
  914. endif()
  915. if(NOT DEFINED CMAKE_EXECUTABLE_FORMAT)
  916. set(CMAKE_EXECUTABLE_FORMAT)
  917. endif()
  918. # Return the information extracted.
  919. set(CMAKE_${lang}_COMPILER_ID "${CMAKE_${lang}_COMPILER_ID}" PARENT_SCOPE)
  920. set(CMAKE_${lang}_PLATFORM_ID "${CMAKE_${lang}_PLATFORM_ID}" PARENT_SCOPE)
  921. set(CMAKE_${lang}_COMPILER_ARCHITECTURE_ID "${CMAKE_${lang}_COMPILER_ARCHITECTURE_ID}" PARENT_SCOPE)
  922. set(MSVC_${lang}_ARCHITECTURE_ID "${MSVC_${lang}_ARCHITECTURE_ID}"
  923. PARENT_SCOPE)
  924. set(CMAKE_${lang}_COMPILER_VERSION "${CMAKE_${lang}_COMPILER_VERSION}" PARENT_SCOPE)
  925. set(CMAKE_${lang}_COMPILER_VERSION_INTERNAL "${CMAKE_${lang}_COMPILER_VERSION_INTERNAL}" PARENT_SCOPE)
  926. set(CMAKE_${lang}_COMPILER_WRAPPER "${COMPILER_WRAPPER}" PARENT_SCOPE)
  927. set(CMAKE_${lang}_SIMULATE_ID "${CMAKE_${lang}_SIMULATE_ID}" PARENT_SCOPE)
  928. set(CMAKE_${lang}_SIMULATE_VERSION "${CMAKE_${lang}_SIMULATE_VERSION}" PARENT_SCOPE)
  929. set(CMAKE_EXECUTABLE_FORMAT "${CMAKE_EXECUTABLE_FORMAT}" PARENT_SCOPE)
  930. set(COMPILER_QNXNTO "${COMPILER_QNXNTO}" PARENT_SCOPE)
  931. set(CMAKE_${lang}_STANDARD_COMPUTED_DEFAULT "${CMAKE_${lang}_STANDARD_COMPUTED_DEFAULT}" PARENT_SCOPE)
  932. endfunction()
  933. #-----------------------------------------------------------------------------
  934. # Function to query the compiler vendor.
  935. # This uses a table with entries of the form
  936. # list(APPEND CMAKE_${lang}_COMPILER_ID_VENDORS ${vendor})
  937. # set(CMAKE_${lang}_COMPILER_ID_VENDOR_FLAGS_${vendor} -some-vendor-flag)
  938. # set(CMAKE_${lang}_COMPILER_ID_VENDOR_REGEX_${vendor} "Some Vendor Output")
  939. # We try running the compiler with the flag for each vendor and
  940. # matching its regular expression in the output.
  941. function(CMAKE_DETERMINE_COMPILER_ID_VENDOR lang userflags)
  942. if(NOT CMAKE_${lang}_COMPILER_ID_DIR)
  943. # We get here when this function is called not from within CMAKE_DETERMINE_COMPILER_ID()
  944. # This is done e.g. for detecting the compiler ID for assemblers.
  945. # Compute the directory in which to run the test and Create a clean working directory.
  946. set(CMAKE_${lang}_COMPILER_ID_DIR ${CMAKE_PLATFORM_INFO_DIR}/CompilerId${lang})
  947. file(REMOVE_RECURSE ${CMAKE_${lang}_COMPILER_ID_DIR})
  948. file(MAKE_DIRECTORY ${CMAKE_${lang}_COMPILER_ID_DIR})
  949. endif()
  950. # Save the current LC_ALL, LC_MESSAGES, and LANG environment variables
  951. # and set them to "C" so we get the expected output to match.
  952. set(_orig_lc_all $ENV{LC_ALL})
  953. set(_orig_lc_messages $ENV{LC_MESSAGES})
  954. set(_orig_lang $ENV{LANG})
  955. set(ENV{LC_ALL} C)
  956. set(ENV{LC_MESSAGES} C)
  957. set(ENV{LANG} C)
  958. foreach(vendor ${CMAKE_${lang}_COMPILER_ID_VENDORS})
  959. set(flags ${CMAKE_${lang}_COMPILER_ID_VENDOR_FLAGS_${vendor}})
  960. set(regex ${CMAKE_${lang}_COMPILER_ID_VENDOR_REGEX_${vendor}})
  961. execute_process(
  962. COMMAND "${CMAKE_${lang}_COMPILER}"
  963. ${CMAKE_${lang}_COMPILER_ID_ARG1}
  964. ${userflags}
  965. ${flags}
  966. WORKING_DIRECTORY ${CMAKE_${lang}_COMPILER_ID_DIR}
  967. OUTPUT_VARIABLE output ERROR_VARIABLE output
  968. RESULT_VARIABLE result
  969. TIMEOUT 10
  970. )
  971. if("${output}" MATCHES "${regex}")
  972. file(APPEND ${CMAKE_BINARY_DIR}${CMAKE_FILES_DIRECTORY}/CMakeOutput.log
  973. "Checking whether the ${lang} compiler is ${vendor} using \"${flags}\" "
  974. "matched \"${regex}\":\n${output}")
  975. set(CMAKE_${lang}_COMPILER_ID "${vendor}" PARENT_SCOPE)
  976. set(CMAKE_${lang}_COMPILER_ID_OUTPUT "${output}" PARENT_SCOPE)
  977. set(CMAKE_${lang}_COMPILER_ID_VENDOR_MATCH "${CMAKE_MATCH_1}" PARENT_SCOPE)
  978. break()
  979. else()
  980. if("${result}" MATCHES "timeout")
  981. file(APPEND ${CMAKE_BINARY_DIR}${CMAKE_FILES_DIRECTORY}/CMakeError.log
  982. "Checking whether the ${lang} compiler is ${vendor} using \"${flags}\" "
  983. "terminated after 10 s due to timeout.")
  984. else()
  985. file(APPEND ${CMAKE_BINARY_DIR}${CMAKE_FILES_DIRECTORY}/CMakeError.log
  986. "Checking whether the ${lang} compiler is ${vendor} using \"${flags}\" "
  987. "did not match \"${regex}\":\n${output}")
  988. endif()
  989. endif()
  990. endforeach()
  991. # Restore original LC_ALL, LC_MESSAGES, and LANG
  992. set(ENV{LC_ALL} ${_orig_lc_all})
  993. set(ENV{LC_MESSAGES} ${_orig_lc_messages})
  994. set(ENV{LANG} ${_orig_lang})
  995. endfunction()
  996. function(CMAKE_DETERMINE_MSVC_SHOWINCLUDES_PREFIX lang userflags)
  997. # Run this MSVC-compatible compiler to detect what the /showIncludes
  998. # option displays. We can use a C source even with the C++ compiler
  999. # because MSVC-compatible compilers handle both and show the same output.
  1000. set(showdir ${CMAKE_BINARY_DIR}/CMakeFiles/ShowIncludes)
  1001. file(WRITE ${showdir}/foo.h "\n")
  1002. file(WRITE ${showdir}/main.c "#include \"foo.h\" \nint main(){}\n")
  1003. execute_process(
  1004. COMMAND "${CMAKE_${lang}_COMPILER}"
  1005. ${CMAKE_${lang}_COMPILER_ID_ARG1}
  1006. ${userflags}
  1007. /nologo /showIncludes /c main.c
  1008. WORKING_DIRECTORY ${showdir}
  1009. OUTPUT_VARIABLE out
  1010. ERROR_VARIABLE err
  1011. RESULT_VARIABLE res
  1012. ENCODING AUTO # cl prints in current code page
  1013. )
  1014. if(res EQUAL 0 AND "${out}" MATCHES "(^|\n)([^:\n]*:[^:\n]*:[ \t]*)")
  1015. set(CMAKE_${lang}_CL_SHOWINCLUDES_PREFIX "${CMAKE_MATCH_2}" PARENT_SCOPE)
  1016. else()
  1017. set(CMAKE_${lang}_CL_SHOWINCLUDES_PREFIX "" PARENT_SCOPE)
  1018. endif()
  1019. endfunction()