CMakeDetermineHIPCompiler.cmake 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335
  1. # Distributed under the OSI-approved BSD 3-Clause License. See accompanying
  2. # file LICENSE.rst or https://cmake.org/licensing for details.
  3. include(${CMAKE_ROOT}/Modules/CMakeDetermineCompiler.cmake)
  4. include(${CMAKE_ROOT}/Modules/CMakeParseImplicitLinkInfo.cmake)
  5. include(${CMAKE_ROOT}/Modules/CMakeParseLibraryArchitecture.cmake)
  6. if(NOT ((CMAKE_GENERATOR MATCHES "Make") OR
  7. (CMAKE_GENERATOR MATCHES "Ninja")))
  8. message(FATAL_ERROR "HIP language not currently supported by \"${CMAKE_GENERATOR}\" generator")
  9. endif()
  10. if(NOT CMAKE_HIP_PLATFORM)
  11. execute_process(COMMAND hipconfig --platform
  12. OUTPUT_VARIABLE _CMAKE_HIPCONFIG_PLATFORM OUTPUT_STRIP_TRAILING_WHITESPACE
  13. RESULT_VARIABLE _CMAKE_HIPCONFIG_RESULT
  14. )
  15. if(_CMAKE_HIPCONFIG_RESULT EQUAL 0 AND _CMAKE_HIPCONFIG_PLATFORM MATCHES "^(nvidia|nvcc)$")
  16. set(CMAKE_HIP_PLATFORM "nvidia" CACHE STRING "HIP platform" FORCE)
  17. else()
  18. set(CMAKE_HIP_PLATFORM "amd" CACHE STRING "HIP platform" FORCE)
  19. endif()
  20. endif()
  21. if(NOT CMAKE_HIP_PLATFORM MATCHES "^(amd|nvidia)$")
  22. message(FATAL_ERROR
  23. "The CMAKE_HIP_PLATFORM has unsupported value:\n"
  24. " '${CMAKE_HIP_PLATFORM}'\n"
  25. "It must be 'amd' or 'nvidia'."
  26. )
  27. endif()
  28. if(NOT CMAKE_HIP_COMPILER)
  29. set(CMAKE_HIP_COMPILER_INIT NOTFOUND)
  30. # prefer the environment variable HIPCXX
  31. if(NOT $ENV{HIPCXX} STREQUAL "")
  32. if("$ENV{HIPCXX}" MATCHES "hipcc")
  33. message(FATAL_ERROR
  34. "The HIPCXX environment variable is set to the hipcc wrapper:\n"
  35. " $ENV{HIPCXX}\n"
  36. "This is not supported. Use Clang directly, or let CMake pick a default."
  37. )
  38. endif()
  39. get_filename_component(CMAKE_HIP_COMPILER_INIT $ENV{HIPCXX} PROGRAM PROGRAM_ARGS CMAKE_HIP_FLAGS_ENV_INIT)
  40. if(CMAKE_HIP_FLAGS_ENV_INIT)
  41. set(CMAKE_HIP_COMPILER_ARG1 "${CMAKE_HIP_FLAGS_ENV_INIT}" CACHE STRING "Arguments to CXX compiler")
  42. endif()
  43. if(NOT EXISTS ${CMAKE_HIP_COMPILER_INIT})
  44. message(FATAL_ERROR "Could not find compiler set in environment variable HIPCXX:\n$ENV{HIPCXX}.\n${CMAKE_HIP_COMPILER_INIT}")
  45. endif()
  46. endif()
  47. # finally list compilers to try
  48. if(NOT CMAKE_HIP_COMPILER_INIT)
  49. if(CMAKE_HIP_PLATFORM STREQUAL "nvidia")
  50. set(CMAKE_HIP_COMPILER_LIST nvcc)
  51. elseif(CMAKE_HIP_PLATFORM STREQUAL "amd")
  52. set(CMAKE_HIP_COMPILER_LIST clang++)
  53. # Look for the Clang coming with ROCm to support HIP.
  54. execute_process(COMMAND hipconfig --hipclangpath
  55. OUTPUT_VARIABLE _CMAKE_HIPCONFIG_CLANGPATH
  56. RESULT_VARIABLE _CMAKE_HIPCONFIG_RESULT
  57. )
  58. if(_CMAKE_HIPCONFIG_RESULT EQUAL 0 AND EXISTS "${_CMAKE_HIPCONFIG_CLANGPATH}")
  59. set(CMAKE_HIP_COMPILER_HINTS "${_CMAKE_HIPCONFIG_CLANGPATH}")
  60. endif()
  61. endif()
  62. endif()
  63. _cmake_find_compiler(HIP)
  64. elseif(CMAKE_HIP_COMPILER MATCHES "hipcc")
  65. message(FATAL_ERROR
  66. "CMAKE_HIP_COMPILER is set to the hipcc wrapper:\n"
  67. " ${CMAKE_HIP_COMPILER}\n"
  68. "This is not supported. Use Clang directly, or let CMake pick a default."
  69. )
  70. else()
  71. _cmake_find_compiler_path(HIP)
  72. endif()
  73. mark_as_advanced(CMAKE_HIP_COMPILER)
  74. # Build a small source file to identify the compiler.
  75. if(NOT CMAKE_HIP_COMPILER_ID_RUN)
  76. set(CMAKE_HIP_COMPILER_ID_RUN 1)
  77. include(${CMAKE_ROOT}/Modules/CMakeDetermineCompilerId.cmake)
  78. # We determine the vendor to use the right flags for detection right away.
  79. # The main compiler identification is still needed below to extract other information.
  80. list(APPEND CMAKE_HIP_COMPILER_ID_VENDORS NVIDIA Clang)
  81. set(CMAKE_HIP_COMPILER_ID_VENDOR_REGEX_NVIDIA "nvcc: [^\n]+ Cuda compiler driver")
  82. set(CMAKE_HIP_COMPILER_ID_VENDOR_REGEX_Clang "(clang version)")
  83. CMAKE_DETERMINE_COMPILER_ID_VENDOR(HIP "--version")
  84. if(CMAKE_HIP_COMPILER_ID STREQUAL "NVIDIA")
  85. # Find the CUDA toolkit to get:
  86. # - CMAKE_HIP_COMPILER_CUDA_TOOLKIT_VERSION
  87. # - CMAKE_HIP_COMPILER_CUDA_TOOLKIT_ROOT
  88. # - CMAKE_HIP_COMPILER_CUDA_LIBRARY_ROOT
  89. # We save them in CMakeHIPCompiler.cmake.
  90. # Match arguments with cmake_cuda_architectures_all call.
  91. include(Internal/CMakeCUDAFindToolkit)
  92. cmake_cuda_find_toolkit(HIP CMAKE_HIP_COMPILER_CUDA_)
  93. # If the user set CMAKE_HIP_ARCHITECTURES, validate its value.
  94. include(Internal/CMakeCUDAArchitecturesValidate)
  95. cmake_cuda_architectures_validate(HIP)
  96. if(NOT CMAKE_HIP_HOST_COMPILER AND NOT $ENV{HIPHOSTCXX} STREQUAL "")
  97. get_filename_component(CMAKE_HIP_HOST_COMPILER $ENV{HIPHOSTCXX} PROGRAM)
  98. if(NOT EXISTS "${CMAKE_HIP_HOST_COMPILER}")
  99. message(FATAL_ERROR "Could not find compiler set in environment variable HIPHOSTCXX:\n$ENV{HIPHOSTCXX}.\n${CMAKE_HIP_HOST_COMPILER}")
  100. endif()
  101. endif()
  102. endif()
  103. if(CMAKE_HIP_COMPILER_ID STREQUAL "Clang")
  104. list(APPEND CMAKE_HIP_COMPILER_ID_TEST_FLAGS_FIRST "-v")
  105. elseif(CMAKE_HIP_COMPILER_ID STREQUAL "NVIDIA")
  106. # Tell nvcc to treat .hip files as CUDA sources.
  107. list(APPEND CMAKE_HIP_COMPILER_ID_TEST_FLAGS_FIRST "-x cu -v")
  108. if(CMAKE_HIP_HOST_COMPILER)
  109. string(APPEND CMAKE_HIP_COMPILER_ID_TEST_FLAGS_FIRST " -ccbin=\"${CMAKE_HIP_HOST_COMPILER}\"")
  110. endif()
  111. endif()
  112. # We perform compiler identification for a second time to extract implicit linking info.
  113. # We need to unset the compiler ID otherwise CMAKE_DETERMINE_COMPILER_ID() doesn't work.
  114. set(CMAKE_HIP_COMPILER_ID)
  115. set(CMAKE_HIP_PLATFORM_ID)
  116. file(READ ${CMAKE_ROOT}/Modules/CMakePlatformId.h.in
  117. CMAKE_HIP_COMPILER_ID_PLATFORM_CONTENT)
  118. CMAKE_DETERMINE_COMPILER_ID(HIP HIPFLAGS CMakeHIPCompilerId.hip)
  119. if(CMAKE_HIP_COMPILER_ID STREQUAL "NVIDIA")
  120. include(Internal/CMakeCUDAArchitecturesAll)
  121. # From CMAKE_HIP_COMPILER_CUDA_TOOLKIT_VERSION and CMAKE_HIP_COMPILER_{ID,VERSION}, get:
  122. # - CMAKE_HIP_ARCHITECTURES_ALL
  123. # - CMAKE_HIP_ARCHITECTURES_ALL_MAJOR
  124. # Match arguments with cmake_cuda_find_toolkit call.
  125. cmake_cuda_architectures_all(HIP CMAKE_HIP_COMPILER_CUDA_)
  126. endif()
  127. _cmake_find_compiler_sysroot(HIP)
  128. endif()
  129. if(NOT CMAKE_HIP_COMPILER_ROCM_ROOT AND CMAKE_HIP_COMPILER_ID STREQUAL "Clang")
  130. execute_process(COMMAND "${CMAKE_HIP_COMPILER}" -v -print-targets
  131. OUTPUT_STRIP_TRAILING_WHITESPACE
  132. RESULT_VARIABLE _CMAKE_HIP_COMPILER_RESULT
  133. OUTPUT_VARIABLE _CMAKE_HIP_COMPILER_STDOUT
  134. ERROR_VARIABLE _CMAKE_HIP_COMPILER_STDERR
  135. )
  136. if(_CMAKE_HIP_COMPILER_RESULT EQUAL 0 AND _CMAKE_HIP_COMPILER_STDERR MATCHES "Found HIP installation: *([^,]*)[,\n]")
  137. set(CMAKE_HIP_COMPILER_ROCM_ROOT "${CMAKE_MATCH_1}")
  138. file(TO_CMAKE_PATH "${CMAKE_HIP_COMPILER_ROCM_ROOT}" CMAKE_HIP_COMPILER_ROCM_ROOT)
  139. endif()
  140. endif()
  141. if(NOT CMAKE_HIP_COMPILER_ROCM_ROOT)
  142. execute_process(
  143. COMMAND hipconfig --rocmpath
  144. OUTPUT_VARIABLE _CMAKE_HIPCONFIG_ROCMPATH
  145. RESULT_VARIABLE _CMAKE_HIPCONFIG_RESULT
  146. )
  147. if(_CMAKE_HIPCONFIG_RESULT EQUAL 0 AND EXISTS "${_CMAKE_HIPCONFIG_ROCMPATH}")
  148. set(CMAKE_HIP_COMPILER_ROCM_ROOT "${_CMAKE_HIPCONFIG_ROCMPATH}")
  149. endif()
  150. endif()
  151. if(NOT CMAKE_HIP_COMPILER_ROCM_ROOT)
  152. message(FATAL_ERROR "Failed to find ROCm root directory.")
  153. endif()
  154. if(CMAKE_HIP_PLATFORM STREQUAL "amd")
  155. # For this platform we need the hip-lang cmake package.
  156. # Normally implicit link information is not detected until ABI detection,
  157. # but we need to populate CMAKE_HIP_LIBRARY_ARCHITECTURE to find hip-lang.
  158. cmake_parse_implicit_link_info("${CMAKE_HIP_COMPILER_PRODUCED_OUTPUT}"
  159. _CMAKE_HIP_COMPILER_ID_IMPLICIT_LIBS
  160. _CMAKE_HIP_COMPILER_ID_IMPLICIT_DIRS
  161. _CMAKE_HIP_COMPILER_ID_IMPLICIT_FWKS
  162. _CMAKE_HIP_COMPILER_ID_IMPLICIT_LOG
  163. "" LANGUAGE HIP)
  164. message(CONFIGURE_LOG
  165. "Parsed HIP implicit link information from compiler id output:\n${_CMAKE_HIP_COMPILER_ID_IMPLICIT_LOG}\n\n")
  166. cmake_parse_library_architecture(HIP "${_CMAKE_HIP_COMPILER_ID_IMPLICIT_DIRS}" "" CMAKE_HIP_LIBRARY_ARCHITECTURE)
  167. if(CMAKE_HIP_LIBRARY_ARCHITECTURE)
  168. message(CONFIGURE_LOG
  169. "Parsed HIP library architecture from compiler id output: ${CMAKE_HIP_LIBRARY_ARCHITECTURE}\n")
  170. endif()
  171. unset(_CMAKE_HIP_COMPILER_ID_IMPLICIT_LIBS)
  172. unset(_CMAKE_HIP_COMPILER_ID_IMPLICIT_DIRS)
  173. unset(_CMAKE_HIP_COMPILER_ID_IMPLICIT_FWKS)
  174. unset(_CMAKE_HIP_COMPILER_ID_IMPLICIT_LOG)
  175. if(NOT CMAKE_HIP_COMPILER_ROCM_LIB)
  176. set(_CMAKE_HIP_COMPILER_ROCM_LIB_DIRS
  177. "${CMAKE_HIP_COMPILER_ROCM_ROOT}/lib"
  178. "${CMAKE_HIP_COMPILER_ROCM_ROOT}/lib64"
  179. )
  180. if(CMAKE_HIP_LIBRARY_ARCHITECTURE)
  181. list(APPEND _CMAKE_HIP_COMPILER_ROCM_LIB_DIRS "${CMAKE_HIP_COMPILER_ROCM_ROOT}/lib/${CMAKE_HIP_LIBRARY_ARCHITECTURE}")
  182. endif()
  183. foreach(dir IN LISTS _CMAKE_HIP_COMPILER_ROCM_LIB_DIRS)
  184. if(EXISTS "${dir}/cmake/hip-lang/hip-lang-config.cmake")
  185. set(CMAKE_HIP_COMPILER_ROCM_LIB "${dir}")
  186. break()
  187. endif()
  188. endforeach()
  189. if(NOT CMAKE_HIP_COMPILER_ROCM_LIB)
  190. list(TRANSFORM _CMAKE_HIP_COMPILER_ROCM_LIB_DIRS APPEND "/cmake/hip-lang/hip-lang-config.cmake")
  191. string(REPLACE ";" "\n " _CMAKE_HIP_COMPILER_ROCM_LIB_DIRS "${_CMAKE_HIP_COMPILER_ROCM_LIB_DIRS}")
  192. message(FATAL_ERROR
  193. "The ROCm root directory:\n"
  194. " ${CMAKE_HIP_COMPILER_ROCM_ROOT}\n"
  195. "does not contain the HIP runtime CMake package, expected at one of:\n"
  196. " ${_CMAKE_HIP_COMPILER_ROCM_LIB_DIRS}\n"
  197. )
  198. endif()
  199. unset(_CMAKE_HIP_COMPILER_ROCM_LIB_DIRS)
  200. endif()
  201. if(NOT DEFINED CMAKE_SIZEOF_VOID_P)
  202. # We have not yet determined the target ABI but we need 'find_package' to
  203. # search lib64 directories to find hip-lang CMake package dependencies.
  204. # This will be replaced by ABI detection later.
  205. set(CMAKE_HIP_SIZEOF_DATA_PTR 8)
  206. endif()
  207. endif()
  208. if (NOT _CMAKE_TOOLCHAIN_LOCATION)
  209. get_filename_component(_CMAKE_TOOLCHAIN_LOCATION "${CMAKE_HIP_COMPILER}" PATH)
  210. endif ()
  211. set(_CMAKE_PROCESSING_LANGUAGE "HIP")
  212. include(CMakeFindBinUtils)
  213. include(Compiler/${CMAKE_HIP_COMPILER_ID}-FindBinUtils OPTIONAL)
  214. unset(_CMAKE_PROCESSING_LANGUAGE)
  215. if(CMAKE_HIP_COMPILER_ID STREQUAL "Clang")
  216. set(CMAKE_HIP_RUNTIME_LIBRARY_DEFAULT "SHARED")
  217. elseif(CMAKE_HIP_COMPILER_ID STREQUAL "NVIDIA")
  218. include(Internal/CMakeNVCCParseImplicitInfo)
  219. # Parse CMAKE_HIP_COMPILER_PRODUCED_OUTPUT to get:
  220. # - CMAKE_HIP_ARCHITECTURES_DEFAULT
  221. # - CMAKE_HIP_HOST_IMPLICIT_LINK_DIRECTORIES
  222. # - CMAKE_HIP_HOST_IMPLICIT_LINK_FRAMEWORK_DIRECTORIES
  223. # - CMAKE_HIP_HOST_IMPLICIT_LINK_LIBRARIES
  224. # - CMAKE_HIP_HOST_LINK_LAUNCHER
  225. # - CMAKE_HIP_RUNTIME_LIBRARY_DEFAULT
  226. # - CMAKE_HIP_CUDA_TOOLKIT_INCLUDE_DIRECTORIES
  227. # Match arguments with cmake_nvcc_filter_implicit_info call in CMakeTestHIPCompiler.
  228. cmake_nvcc_parse_implicit_info(HIP CMAKE_HIP_CUDA_)
  229. include(Internal/CMakeCUDAFilterImplicitLibs)
  230. # Filter out implicit link libraries that should not be passed unconditionally.
  231. cmake_cuda_filter_implicit_libs(CMAKE_HIP_HOST_IMPLICIT_LINK_LIBRARIES)
  232. endif()
  233. if(CMAKE_HIP_COMPILER_SYSROOT)
  234. string(CONCAT _SET_CMAKE_HIP_COMPILER_SYSROOT
  235. "set(CMAKE_HIP_COMPILER_SYSROOT \"${CMAKE_HIP_COMPILER_SYSROOT}\")\n"
  236. "set(CMAKE_COMPILER_SYSROOT \"${CMAKE_HIP_COMPILER_SYSROOT}\")")
  237. else()
  238. set(_SET_CMAKE_HIP_COMPILER_SYSROOT "")
  239. endif()
  240. if(MSVC_HIP_ARCHITECTURE_ID)
  241. set(SET_MSVC_HIP_ARCHITECTURE_ID
  242. "set(MSVC_HIP_ARCHITECTURE_ID ${MSVC_HIP_ARCHITECTURE_ID})")
  243. endif()
  244. if(CMAKE_HIP_COMPILER_ID STREQUAL "NVIDIA")
  245. if(NOT "$ENV{CUDAARCHS}" STREQUAL "")
  246. set(CMAKE_HIP_ARCHITECTURES "$ENV{CUDAARCHS}" CACHE STRING "CUDA architectures")
  247. endif()
  248. # If the user did not set CMAKE_HIP_ARCHITECTURES, use the compiler's default.
  249. if("${CMAKE_HIP_ARCHITECTURES}" STREQUAL "")
  250. set(CMAKE_HIP_ARCHITECTURES "${CMAKE_HIP_ARCHITECTURES_DEFAULT}" CACHE STRING "HIP architectures" FORCE)
  251. if(NOT CMAKE_HIP_ARCHITECTURES)
  252. message(FATAL_ERROR "Failed to detect a default HIP architecture.\n\nCompiler output:\n${CMAKE_HIP_COMPILER_PRODUCED_OUTPUT}")
  253. endif()
  254. endif()
  255. unset(CMAKE_HIP_ARCHITECTURES_DEFAULT)
  256. elseif(NOT DEFINED CMAKE_HIP_ARCHITECTURES)
  257. # Use 'rocm_agent_enumerator' to get the current GPU architecture.
  258. set(_CMAKE_HIP_ARCHITECTURES)
  259. find_program(_CMAKE_HIP_ROCM_AGENT_ENUMERATOR
  260. NAMES rocm_agent_enumerator
  261. HINTS "${CMAKE_HIP_COMPILER_ROCM_ROOT}/bin"
  262. NO_CACHE)
  263. if(_CMAKE_HIP_ROCM_AGENT_ENUMERATOR)
  264. execute_process(COMMAND "${_CMAKE_HIP_ROCM_AGENT_ENUMERATOR}" -t GPU
  265. RESULT_VARIABLE _CMAKE_ROCM_AGENT_ENUMERATOR_RESULT
  266. OUTPUT_VARIABLE _CMAKE_ROCM_AGENT_ENUMERATOR_STDOUT
  267. ERROR_VARIABLE _CMAKE_ROCM_AGENT_ENUMERATOR_STDERR
  268. )
  269. if(_CMAKE_ROCM_AGENT_ENUMERATOR_RESULT EQUAL 0)
  270. separate_arguments(_hip_archs NATIVE_COMMAND "${_CMAKE_ROCM_AGENT_ENUMERATOR_STDOUT}")
  271. foreach(_hip_arch ${_hip_archs})
  272. if(_hip_arch STREQUAL "gfx000")
  273. continue()
  274. endif()
  275. string(FIND ${_hip_arch} ":" pos)
  276. if(NOT pos STREQUAL "-1")
  277. string(SUBSTRING ${_hip_arch} 0 ${pos} _hip_arch)
  278. endif()
  279. list(APPEND _CMAKE_HIP_ARCHITECTURES "${_hip_arch}")
  280. endforeach()
  281. endif()
  282. unset(_CMAKE_ROCM_AGENT_ENUMERATOR_RESULT)
  283. unset(_CMAKE_ROCM_AGENT_ENUMERATOR_STDOUT)
  284. unset(_CMAKE_ROCM_AGENT_ENUMERATOR_STDERR)
  285. endif()
  286. unset(_CMAKE_HIP_ROCM_AGENT_ENUMERATOR)
  287. if(_CMAKE_HIP_ARCHITECTURES)
  288. set(CMAKE_HIP_ARCHITECTURES "${_CMAKE_HIP_ARCHITECTURES}" CACHE STRING "HIP architectures")
  289. elseif(CMAKE_HIP_COMPILER_PRODUCED_OUTPUT MATCHES " -target-cpu ([a-z0-9]+) ")
  290. set(CMAKE_HIP_ARCHITECTURES "${CMAKE_MATCH_1}" CACHE STRING "HIP architectures")
  291. else()
  292. message(FATAL_ERROR "Failed to find a default HIP architecture.")
  293. endif()
  294. unset(_CMAKE_HIP_ARCHITECTURES)
  295. endif()
  296. # configure variables set in this file for fast reload later on
  297. configure_file(${CMAKE_ROOT}/Modules/CMakeHIPCompiler.cmake.in
  298. ${CMAKE_PLATFORM_INFO_DIR}/CMakeHIPCompiler.cmake
  299. @ONLY
  300. )
  301. set(CMAKE_HIP_COMPILER_ENV_VAR "HIPCXX")