CMakeFindBinUtils.cmake 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287
  1. # Distributed under the OSI-approved BSD 3-Clause License. See accompanying
  2. # file Copyright.txt or https://cmake.org/licensing for details.
  3. # search for additional tools required for C/C++ (and other languages ?)
  4. #
  5. # If the internal cmake variable _CMAKE_TOOLCHAIN_PREFIX is set, this is used
  6. # as prefix for the tools (e.g. arm-elf-gcc etc.)
  7. # If the cmake variable _CMAKE_TOOLCHAIN_LOCATION is set, the compiler is
  8. # searched only there. The other tools are at first searched there, then
  9. # also in the default locations.
  10. #
  11. # Sets the following variables:
  12. # CMAKE_AR
  13. # CMAKE_RANLIB
  14. # CMAKE_LINKER
  15. # CMAKE_MT
  16. # CMAKE_STRIP
  17. # CMAKE_INSTALL_NAME_TOOL
  18. # on UNIX, cygwin and mingw
  19. cmake_policy(PUSH)
  20. cmake_policy(SET CMP0057 NEW) # if IN_LIST
  21. # Resolve full path of CMAKE_TOOL from user-defined name and SEARCH_PATH.
  22. function(__resolve_tool_path CMAKE_TOOL SEARCH_PATH DOCSTRING)
  23. if(${CMAKE_TOOL})
  24. # We only get here if CMAKE_TOOL was
  25. # specified using -D or a pre-made CMakeCache.txt (e.g. via ctest)
  26. # or set in CMAKE_TOOLCHAIN_FILE.
  27. get_filename_component(_CMAKE_USER_TOOL_PATH "${${CMAKE_TOOL}}" DIRECTORY)
  28. # Is CMAKE_TOOL a user-defined name instead of a full path?
  29. if(NOT _CMAKE_USER_TOOL_PATH)
  30. # Find CMAKE_TOOL in the SEARCH_PATH directory by user-defined name.
  31. find_program(_CMAKE_TOOL_WITH_PATH NAMES ${${CMAKE_TOOL}} HINTS ${SEARCH_PATH} NO_CMAKE_PATH NO_CMAKE_ENVIRONMENT_PATH)
  32. if(_CMAKE_TOOL_WITH_PATH)
  33. # Overwrite CMAKE_TOOL with full path found in SEARCH_PATH.
  34. set(${CMAKE_TOOL} ${_CMAKE_TOOL_WITH_PATH} PARENT_SCOPE)
  35. get_property(_CMAKE_TOOL_CACHED CACHE ${CMAKE_TOOL} PROPERTY TYPE)
  36. # If CMAKE_TOOL is present in the CMake Cache, then overwrit it as well.
  37. if(_CMAKE_TOOL_CACHED)
  38. set(${CMAKE_TOOL} "${_CMAKE_TOOL_WITH_PATH}" CACHE STRING ${DOCSTRING} FORCE)
  39. endif()
  40. endif()
  41. unset(_CMAKE_TOOL_WITH_PATH CACHE)
  42. endif()
  43. endif()
  44. endfunction()
  45. __resolve_tool_path(CMAKE_LINKER "${_CMAKE_TOOLCHAIN_LOCATION}" "Default Linker")
  46. __resolve_tool_path(CMAKE_MT "${_CMAKE_TOOLCHAIN_LOCATION}" "Default Manifest Tool")
  47. set(_CMAKE_TOOL_VARS "")
  48. # if it's the MS C/CXX compiler, search for link
  49. if(("x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_SIMULATE_ID}" STREQUAL "xMSVC" AND
  50. ("x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_FRONTEND_VARIANT}" STREQUAL "xMSVC"
  51. OR NOT "x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" STREQUAL "xClang"))
  52. OR "x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" STREQUAL "xMSVC"
  53. OR (CMAKE_HOST_WIN32 AND "x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" STREQUAL "xPGI")
  54. OR (CMAKE_HOST_WIN32 AND "x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" STREQUAL "xNVIDIA")
  55. OR (CMAKE_HOST_WIN32 AND "x${_CMAKE_PROCESSING_LANGUAGE}" STREQUAL "xISPC")
  56. OR (CMAKE_GENERATOR MATCHES "Visual Studio"
  57. AND NOT CMAKE_VS_PLATFORM_NAME STREQUAL "Tegra-Android"))
  58. # Start with the canonical names.
  59. set(_CMAKE_LINKER_NAMES "link")
  60. set(_CMAKE_AR_NAMES "lib")
  61. set(_CMAKE_MT_NAMES "mt")
  62. # Prepend toolchain-specific names.
  63. if("x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" STREQUAL "xClang")
  64. set(_CMAKE_NM_NAMES "llvm-nm" "nm")
  65. list(PREPEND _CMAKE_AR_NAMES "llvm-lib")
  66. # llvm-mt does not support all flags we need in vs_link_exe
  67. # list(PREPEND _CMAKE_MT_NAMES "llvm-mt")
  68. list(PREPEND _CMAKE_LINKER_NAMES "lld-link")
  69. list(APPEND _CMAKE_TOOL_VARS NM)
  70. elseif("x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" STREQUAL "xIntel")
  71. list(PREPEND _CMAKE_AR_NAMES "xilib")
  72. list(PREPEND _CMAKE_LINKER_NAMES "xilink")
  73. endif()
  74. list(APPEND _CMAKE_TOOL_VARS LINKER MT AR)
  75. elseif("x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" MATCHES "^x(Open)?Watcom$")
  76. set(_CMAKE_LINKER_NAMES "wlink")
  77. set(_CMAKE_AR_NAMES "wlib")
  78. list(APPEND _CMAKE_TOOL_VARS LINKER AR)
  79. elseif("x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" MATCHES "^xIAR$")
  80. # Small helper declaring an IAR tool (e.g. linker) to avoid repeating the same idiom every time
  81. macro(__append_IAR_tool TOOL_VAR NAME)
  82. set(_CMAKE_${TOOL_VAR}_NAMES "${NAME}" "${NAME}.exe")
  83. list(APPEND _CMAKE_TOOL_VARS ${TOOL_VAR})
  84. endmacro()
  85. # Resolve hint path from an IAR compiler
  86. function(__resolve_IAR_hints COMPILER RESULT)
  87. get_filename_component(_CMAKE_IAR_HINT "${COMPILER}" REALPATH)
  88. get_filename_component(_CMAKE_IAR_HINT "${_CMAKE_IAR_HINT}" DIRECTORY)
  89. list(APPEND _IAR_HINTS "${_CMAKE_IAR_HINT}")
  90. get_filename_component(_CMAKE_IAR_HINT "${COMPILER}" DIRECTORY)
  91. list(APPEND _IAR_HINTS "${_CMAKE_IAR_HINT}")
  92. set(${RESULT} "${_IAR_HINTS}" PARENT_SCOPE)
  93. endfunction()
  94. __resolve_IAR_hints("${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER}" _CMAKE_TOOLCHAIN_LOCATION)
  95. set(_CMAKE_IAR_ITOOLS "ARM" "RX" "RH850" "RL78" "RISCV" "RISC-V" "STM8")
  96. set(_CMAKE_IAR_XTOOLS "AVR" "MSP430" "V850" "8051")
  97. string(TOLOWER "${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ARCHITECTURE_ID}" _CMAKE_IAR_LOWER_ARCHITECTURE_ID)
  98. if("${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ARCHITECTURE_ID}" IN_LIST _CMAKE_IAR_ITOOLS)
  99. __append_IAR_tool(AR "iarchive")
  100. __append_IAR_tool(LINKER "ilink${_CMAKE_IAR_LOWER_ARCHITECTURE_ID}")
  101. __append_IAR_tool(IAR_ELFDUMP "ielfdump${_CMAKE_IAR_LOWER_ARCHITECTURE_ID}")
  102. __append_IAR_tool(IAR_ELFTOOL "ielftool")
  103. __append_IAR_tool(IAR_OBJMANIP "iobjmanip")
  104. __append_IAR_tool(IAR_SYMEXPORT "isymexport")
  105. elseif("${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ARCHITECTURE_ID}" IN_LIST _CMAKE_IAR_XTOOLS)
  106. __append_IAR_tool(AR "xar")
  107. if("${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ARCHITECTURE_ID}" STREQUAL "AVR" AND
  108. (CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_VERSION VERSION_GREATER_EQUAL 8))
  109. # IAR UBROF Linker V8.10+ for Microchip AVR is `xlinkavr`
  110. __append_IAR_tool(LINKER "xlink${_CMAKE_IAR_LOWER_ARCHITECTURE_ID}")
  111. else()
  112. __append_IAR_tool(LINKER "xlink")
  113. endif()
  114. else()
  115. message(FATAL_ERROR "Failed to find linker and librarian for ${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID} on ${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ARCHITECTURE_ID}.")
  116. endif()
  117. unset(_CMAKE_IAR_LOWER_ARCHITECTURE_ID)
  118. unset(_CMAKE_IAR_ITOOLS)
  119. unset(_CMAKE_IAR_XTOOLS)
  120. # in all other cases search for ar, ranlib, etc.
  121. else()
  122. if(CMAKE_C_COMPILER_EXTERNAL_TOOLCHAIN)
  123. set(_CMAKE_TOOLCHAIN_LOCATION ${_CMAKE_TOOLCHAIN_LOCATION} ${CMAKE_C_COMPILER_EXTERNAL_TOOLCHAIN}/bin)
  124. endif()
  125. if(CMAKE_CXX_COMPILER_EXTERNAL_TOOLCHAIN)
  126. set(_CMAKE_TOOLCHAIN_LOCATION ${_CMAKE_TOOLCHAIN_LOCATION} ${CMAKE_CXX_COMPILER_EXTERNAL_TOOLCHAIN}/bin)
  127. endif()
  128. # Start with the canonical names.
  129. set(_CMAKE_AR_NAMES "ar")
  130. set(_CMAKE_RANLIB_NAMES "ranlib")
  131. set(_CMAKE_STRIP_NAMES "strip")
  132. set(_CMAKE_LINKER_NAMES "ld")
  133. set(_CMAKE_NM_NAMES "nm")
  134. set(_CMAKE_OBJDUMP_NAMES "objdump")
  135. set(_CMAKE_OBJCOPY_NAMES "objcopy")
  136. set(_CMAKE_READELF_NAMES "readelf")
  137. set(_CMAKE_DLLTOOL_NAMES "dlltool")
  138. set(_CMAKE_ADDR2LINE_NAMES "addr2line")
  139. set(_CMAKE_TAPI_NAMES "tapi")
  140. # Prepend toolchain-specific names.
  141. if("${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" STREQUAL Clang)
  142. if("x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_SIMULATE_ID}" STREQUAL "xMSVC")
  143. list(PREPEND _CMAKE_LINKER_NAMES "lld-link")
  144. elseif(NOT APPLE)
  145. list(PREPEND _CMAKE_LINKER_NAMES "ld.lld")
  146. endif()
  147. if(APPLE)
  148. # llvm-ar does not generate a symbol table that the Apple ld64 linker accepts.
  149. # FIXME(#23333): We still need to consider 'llvm-ar' as a fallback because
  150. # the 'APPLE' definition may be based on the host in this context, and a
  151. # cross-compiling toolchain may not have 'ar'.
  152. list(APPEND _CMAKE_AR_NAMES "llvm-ar")
  153. else()
  154. list(PREPEND _CMAKE_AR_NAMES "llvm-ar")
  155. endif()
  156. list(PREPEND _CMAKE_RANLIB_NAMES "llvm-ranlib")
  157. if("${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_VERSION}" VERSION_GREATER_EQUAL 11)
  158. # llvm-strip versions prior to 11 require additional flags we do not yet add.
  159. if(APPLE)
  160. # llvm-strip does not seem to support chained fixup format correctly.
  161. # FIXME(#23333): We still need to consider 'llvm-strip' as a fallback
  162. # because the 'APPLE' definition may be based on the host in this context,
  163. # and a cross-compiling toolchain may not have 'strip'.
  164. list(APPEND _CMAKE_STRIP_NAMES "llvm-strip")
  165. else()
  166. list(PREPEND _CMAKE_STRIP_NAMES "llvm-strip")
  167. endif()
  168. endif()
  169. list(PREPEND _CMAKE_NM_NAMES "llvm-nm")
  170. if("${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_VERSION}" VERSION_GREATER_EQUAL 9)
  171. # llvm-objdump versions prior to 9 did not support everything we need.
  172. list(PREPEND _CMAKE_OBJDUMP_NAMES "llvm-objdump")
  173. endif()
  174. list(PREPEND _CMAKE_OBJCOPY_NAMES "llvm-objcopy")
  175. list(PREPEND _CMAKE_READELF_NAMES "llvm-readelf")
  176. list(PREPEND _CMAKE_DLLTOOL_NAMES "llvm-dlltool")
  177. list(PREPEND _CMAKE_ADDR2LINE_NAMES "llvm-addr2line")
  178. elseif("${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" STREQUAL ARMClang)
  179. list(PREPEND _CMAKE_AR_NAMES "armar")
  180. list(PREPEND _CMAKE_LINKER_NAMES "armlink")
  181. endif()
  182. list(APPEND _CMAKE_TOOL_VARS AR RANLIB STRIP LINKER NM OBJDUMP OBJCOPY READELF DLLTOOL ADDR2LINE TAPI)
  183. endif()
  184. foreach(_CMAKE_TOOL IN LISTS _CMAKE_TOOL_VARS)
  185. # Build the final list of prefixed/suffixed names.
  186. set(_CMAKE_${_CMAKE_TOOL}_FIND_NAMES "")
  187. foreach(_CMAKE_TOOL_NAME IN LISTS _CMAKE_${_CMAKE_TOOL}_NAMES)
  188. list(APPEND _CMAKE_${_CMAKE_TOOL}_FIND_NAMES
  189. ${_CMAKE_TOOLCHAIN_PREFIX}${_CMAKE_TOOL_NAME}${_CMAKE_TOOLCHAIN_SUFFIX}
  190. ${_CMAKE_TOOLCHAIN_PREFIX}${_CMAKE_TOOL_NAME}
  191. ${_CMAKE_TOOL_NAME}${_CMAKE_TOOLCHAIN_SUFFIX}
  192. ${_CMAKE_TOOL_NAME}
  193. )
  194. endforeach()
  195. list(REMOVE_DUPLICATES _CMAKE_${_CMAKE_TOOL}_FIND_NAMES)
  196. find_program(CMAKE_${_CMAKE_TOOL} NAMES ${_CMAKE_${_CMAKE_TOOL}_FIND_NAMES} HINTS ${_CMAKE_TOOLCHAIN_LOCATION} NO_CMAKE_PATH NO_CMAKE_ENVIRONMENT_PATH)
  197. unset(_CMAKE_${_CMAKE_TOOL}_FIND_NAMES)
  198. endforeach()
  199. if(NOT CMAKE_RANLIB)
  200. set(CMAKE_RANLIB : CACHE INTERNAL "noop for ranlib")
  201. endif()
  202. if(NOT CMAKE_TAPI)
  203. # try to pick-up from Apple toolchain
  204. execute_process(COMMAND xcrun --find tapi
  205. OUTPUT_VARIABLE _xcrun_out
  206. OUTPUT_STRIP_TRAILING_WHITESPACE
  207. ERROR_QUIET
  208. RESULT_VARIABLE _xcrun_failed)
  209. if(NOT _xcrun_failed AND EXISTS "${_xcrun_out}")
  210. set_property(CACHE CMAKE_TAPI PROPERTY VALUE "${_xcrun_out}")
  211. endif()
  212. unset(_xcrun_out)
  213. unset(_xcrun_failed)
  214. endif()
  215. if(CMAKE_PLATFORM_HAS_INSTALLNAME)
  216. find_program(CMAKE_INSTALL_NAME_TOOL NAMES ${_CMAKE_TOOLCHAIN_PREFIX}install_name_tool HINTS ${_CMAKE_TOOLCHAIN_LOCATION} NO_CMAKE_PATH NO_CMAKE_ENVIRONMENT_PATH)
  217. if(NOT CMAKE_INSTALL_NAME_TOOL)
  218. message(FATAL_ERROR "Could not find install_name_tool, please check your installation.")
  219. endif()
  220. list(APPEND _CMAKE_TOOL_VARS INSTALL_NAME_TOOL)
  221. endif()
  222. # Mark any tool cache entries as advanced.
  223. foreach(_CMAKE_TOOL IN LISTS _CMAKE_TOOL_VARS)
  224. get_property(_CMAKE_TOOL_CACHED CACHE CMAKE_${_CMAKE_TOOL} PROPERTY TYPE)
  225. if(_CMAKE_TOOL_CACHED)
  226. mark_as_advanced(CMAKE_${_CMAKE_TOOL})
  227. endif()
  228. unset(_CMAKE_${_CMAKE_TOOL}_NAMES)
  229. endforeach()
  230. unset(_CMAKE_TOOL_VARS)
  231. unset(_CMAKE_TOOL_CACHED)
  232. unset(_CMAKE_TOOL_NAME)
  233. unset(_CMAKE_TOOL)
  234. if("x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" MATCHES "^xIAR$")
  235. # Set for backwards compatibility
  236. set(CMAKE_IAR_ARCHIVE "${CMAKE_AR}" CACHE FILEPATH "The IAR archiver")
  237. set(CMAKE_IAR_LINKER "${CMAKE_LINKER}" CACHE FILEPATH "The IAR ILINK linker")
  238. mark_as_advanced(CMAKE_IAR_LINKER CMAKE_IAR_AR)
  239. endif()
  240. cmake_policy(POP)