CMakeFindBinUtils.cmake 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288
  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. macro(__resolve_linker_path __linker_type __name __search_path __doc)
  48. if(NOT CMAKE_LINKER_${__linker_type})
  49. set( CMAKE_LINKER_${__linker_type} "${__name}")
  50. endif()
  51. __resolve_tool_path(CMAKE_LINKER_${__linker_type} "${__search_path}" "${__doc}")
  52. endmacro()
  53. set(_CMAKE_TOOL_VARS "")
  54. # if it's the MS C/CXX compiler, search for link
  55. if(("x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_SIMULATE_ID}" STREQUAL "xMSVC" AND
  56. ("x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_FRONTEND_VARIANT}" STREQUAL "xMSVC"
  57. OR NOT "x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" STREQUAL "xClang"))
  58. OR "x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" STREQUAL "xMSVC"
  59. OR (CMAKE_HOST_WIN32 AND "x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" STREQUAL "xPGI")
  60. OR (CMAKE_HOST_WIN32 AND "x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" STREQUAL "xNVIDIA")
  61. OR (CMAKE_HOST_WIN32 AND "x${_CMAKE_PROCESSING_LANGUAGE}" STREQUAL "xISPC")
  62. OR (CMAKE_GENERATOR MATCHES "Visual Studio"
  63. AND NOT CMAKE_VS_PLATFORM_NAME STREQUAL "Tegra-Android"))
  64. # Start with the canonical names.
  65. set(_CMAKE_LINKER_NAMES "link")
  66. set(_CMAKE_AR_NAMES "lib")
  67. set(_CMAKE_MT_NAMES "mt")
  68. # Prepend toolchain-specific names.
  69. if("x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" MATCHES "^x(Clang|LLVMFlang)$")
  70. set(_CMAKE_NM_NAMES "llvm-nm" "nm")
  71. list(PREPEND _CMAKE_AR_NAMES "llvm-lib")
  72. # llvm-mt is not ready to be used as a replacement for mt.exe
  73. # list(PREPEND _CMAKE_MT_NAMES "llvm-mt")
  74. list(PREPEND _CMAKE_LINKER_NAMES "lld-link")
  75. list(APPEND _CMAKE_TOOL_VARS NM)
  76. elseif("x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" STREQUAL "xIntel")
  77. list(PREPEND _CMAKE_AR_NAMES "xilib")
  78. list(PREPEND _CMAKE_LINKER_NAMES "xilink")
  79. endif()
  80. list(APPEND _CMAKE_TOOL_VARS LINKER MT AR)
  81. # look-up for possible usable linker
  82. __resolve_linker_path(LINK "link" "${_CMAKE_TOOLCHAIN_LOCATION}" "link Linker")
  83. __resolve_linker_path(LLD "lld-link" "${_CMAKE_TOOLCHAIN_LOCATION}" "lld-link Linker")
  84. elseif("x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" MATCHES "^x(Open)?Watcom$")
  85. set(_CMAKE_LINKER_NAMES "wlink")
  86. set(_CMAKE_AR_NAMES "wlib")
  87. list(APPEND _CMAKE_TOOL_VARS LINKER AR)
  88. elseif("x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" MATCHES "^xIAR$")
  89. # Small helper declaring an IAR tool (e.g. linker) to avoid repeating the same idiom every time
  90. macro(__append_IAR_tool TOOL_VAR NAME)
  91. set(_CMAKE_${TOOL_VAR}_NAMES "${NAME}" "${NAME}.exe")
  92. list(APPEND _CMAKE_TOOL_VARS ${TOOL_VAR})
  93. endmacro()
  94. # Resolve hint path from an IAR compiler
  95. function(__resolve_IAR_hints COMPILER RESULT)
  96. get_filename_component(_CMAKE_IAR_HINT "${COMPILER}" REALPATH)
  97. get_filename_component(_CMAKE_IAR_HINT "${_CMAKE_IAR_HINT}" DIRECTORY)
  98. list(APPEND _IAR_HINTS "${_CMAKE_IAR_HINT}")
  99. get_filename_component(_CMAKE_IAR_HINT "${COMPILER}" DIRECTORY)
  100. list(APPEND _IAR_HINTS "${_CMAKE_IAR_HINT}")
  101. set(${RESULT} "${_IAR_HINTS}" PARENT_SCOPE)
  102. endfunction()
  103. __resolve_IAR_hints("${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER}" _CMAKE_TOOLCHAIN_LOCATION)
  104. set(_CMAKE_IAR_ITOOLS "ARM" "RX" "RH850" "RL78" "RISCV" "RISC-V" "STM8")
  105. set(_CMAKE_IAR_XTOOLS "AVR" "MSP430" "V850" "8051")
  106. string(TOLOWER "${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ARCHITECTURE_ID}" _CMAKE_IAR_LOWER_ARCHITECTURE_ID)
  107. if("${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ARCHITECTURE_ID}" IN_LIST _CMAKE_IAR_ITOOLS)
  108. __append_IAR_tool(AR "iarchive")
  109. __append_IAR_tool(LINKER "ilink${_CMAKE_IAR_LOWER_ARCHITECTURE_ID}")
  110. __append_IAR_tool(IAR_ELFDUMP "ielfdump${_CMAKE_IAR_LOWER_ARCHITECTURE_ID}")
  111. __append_IAR_tool(IAR_ELFTOOL "ielftool")
  112. __append_IAR_tool(IAR_OBJMANIP "iobjmanip")
  113. __append_IAR_tool(IAR_SYMEXPORT "isymexport")
  114. elseif("${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ARCHITECTURE_ID}" IN_LIST _CMAKE_IAR_XTOOLS)
  115. __append_IAR_tool(AR "xar")
  116. if("${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ARCHITECTURE_ID}" STREQUAL "AVR" AND
  117. (CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_VERSION VERSION_GREATER_EQUAL 8))
  118. # IAR UBROF Linker V8.10+ for Microchip AVR is `xlinkavr`
  119. __append_IAR_tool(LINKER "xlink${_CMAKE_IAR_LOWER_ARCHITECTURE_ID}")
  120. else()
  121. __append_IAR_tool(LINKER "xlink")
  122. endif()
  123. else()
  124. message(FATAL_ERROR "Failed to find linker and librarian for ${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID} on ${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ARCHITECTURE_ID}.")
  125. endif()
  126. unset(_CMAKE_IAR_LOWER_ARCHITECTURE_ID)
  127. unset(_CMAKE_IAR_ITOOLS)
  128. unset(_CMAKE_IAR_XTOOLS)
  129. # in all other cases search for ar, ranlib, etc.
  130. else()
  131. if(CMAKE_C_COMPILER_EXTERNAL_TOOLCHAIN)
  132. set(_CMAKE_TOOLCHAIN_LOCATION ${_CMAKE_TOOLCHAIN_LOCATION} ${CMAKE_C_COMPILER_EXTERNAL_TOOLCHAIN}/bin)
  133. endif()
  134. if(CMAKE_CXX_COMPILER_EXTERNAL_TOOLCHAIN)
  135. set(_CMAKE_TOOLCHAIN_LOCATION ${_CMAKE_TOOLCHAIN_LOCATION} ${CMAKE_CXX_COMPILER_EXTERNAL_TOOLCHAIN}/bin)
  136. endif()
  137. # Start with the canonical names.
  138. set(_CMAKE_AR_NAMES "ar")
  139. set(_CMAKE_RANLIB_NAMES "ranlib")
  140. set(_CMAKE_STRIP_NAMES "strip")
  141. set(_CMAKE_LINKER_NAMES "ld")
  142. set(_CMAKE_NM_NAMES "nm")
  143. set(_CMAKE_OBJDUMP_NAMES "objdump")
  144. set(_CMAKE_OBJCOPY_NAMES "objcopy")
  145. set(_CMAKE_READELF_NAMES "readelf")
  146. set(_CMAKE_DLLTOOL_NAMES "dlltool")
  147. set(_CMAKE_ADDR2LINE_NAMES "addr2line")
  148. set(_CMAKE_TAPI_NAMES "tapi")
  149. # Prepend toolchain-specific names.
  150. if("${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" STREQUAL Clang)
  151. if("x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_SIMULATE_ID}" STREQUAL "xMSVC")
  152. list(PREPEND _CMAKE_LINKER_NAMES "lld-link")
  153. elseif(NOT APPLE)
  154. list(PREPEND _CMAKE_LINKER_NAMES "ld.lld")
  155. endif()
  156. # llvm-ar does not generate a symbol table that the Apple ld64 linker accepts.
  157. if(NOT APPLE)
  158. list(PREPEND _CMAKE_AR_NAMES "llvm-ar")
  159. endif()
  160. list(PREPEND _CMAKE_RANLIB_NAMES "llvm-ranlib")
  161. # llvm-strip versions prior to 11 require additional flags we do not yet add.
  162. if("${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_VERSION}" VERSION_GREATER_EQUAL 11)
  163. # llvm-strip does not seem to support chained fixup format on macOS correctly.
  164. if(NOT APPLE)
  165. list(PREPEND _CMAKE_STRIP_NAMES "llvm-strip")
  166. endif()
  167. endif()
  168. list(PREPEND _CMAKE_NM_NAMES "llvm-nm")
  169. if("${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_VERSION}" VERSION_GREATER_EQUAL 9)
  170. # llvm-objdump versions prior to 9 did not support everything we need.
  171. list(PREPEND _CMAKE_OBJDUMP_NAMES "llvm-objdump")
  172. endif()
  173. list(PREPEND _CMAKE_OBJCOPY_NAMES "llvm-objcopy")
  174. list(PREPEND _CMAKE_READELF_NAMES "llvm-readelf")
  175. list(PREPEND _CMAKE_DLLTOOL_NAMES "llvm-dlltool")
  176. list(PREPEND _CMAKE_ADDR2LINE_NAMES "llvm-addr2line")
  177. elseif("${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" STREQUAL ARMClang)
  178. list(PREPEND _CMAKE_AR_NAMES "armar")
  179. list(PREPEND _CMAKE_LINKER_NAMES "armlink")
  180. endif()
  181. list(APPEND _CMAKE_TOOL_VARS AR RANLIB STRIP LINKER NM OBJDUMP OBJCOPY READELF DLLTOOL ADDR2LINE TAPI)
  182. endif()
  183. foreach(_CMAKE_TOOL IN LISTS _CMAKE_TOOL_VARS)
  184. # Build the final list of prefixed/suffixed names.
  185. set(_CMAKE_${_CMAKE_TOOL}_FIND_NAMES "")
  186. foreach(_CMAKE_TOOL_NAME IN LISTS _CMAKE_${_CMAKE_TOOL}_NAMES)
  187. list(APPEND _CMAKE_${_CMAKE_TOOL}_FIND_NAMES
  188. ${_CMAKE_TOOLCHAIN_PREFIX}${_CMAKE_TOOL_NAME}${_CMAKE_TOOLCHAIN_SUFFIX}
  189. ${_CMAKE_TOOLCHAIN_PREFIX}${_CMAKE_TOOL_NAME}
  190. ${_CMAKE_TOOL_NAME}${_CMAKE_TOOLCHAIN_SUFFIX}
  191. ${_CMAKE_TOOL_NAME}
  192. )
  193. endforeach()
  194. list(REMOVE_DUPLICATES _CMAKE_${_CMAKE_TOOL}_FIND_NAMES)
  195. find_program(CMAKE_${_CMAKE_TOOL} NAMES ${_CMAKE_${_CMAKE_TOOL}_FIND_NAMES} HINTS ${_CMAKE_TOOLCHAIN_LOCATION} NO_CMAKE_PATH NO_CMAKE_ENVIRONMENT_PATH)
  196. unset(_CMAKE_${_CMAKE_TOOL}_FIND_NAMES)
  197. endforeach()
  198. if(NOT CMAKE_RANLIB)
  199. set(CMAKE_RANLIB : CACHE INTERNAL "noop for ranlib")
  200. endif()
  201. if(APPLE AND "TAPI" IN_LIST _CMAKE_TOOL_VARS AND NOT CMAKE_TAPI)
  202. # try to pick-up from Apple toolchain
  203. execute_process(COMMAND xcrun --find tapi
  204. OUTPUT_VARIABLE _xcrun_out
  205. OUTPUT_STRIP_TRAILING_WHITESPACE
  206. ERROR_QUIET
  207. RESULT_VARIABLE _xcrun_failed)
  208. if(NOT _xcrun_failed AND EXISTS "${_xcrun_out}")
  209. set_property(CACHE CMAKE_TAPI PROPERTY VALUE "${_xcrun_out}")
  210. endif()
  211. unset(_xcrun_out)
  212. unset(_xcrun_failed)
  213. endif()
  214. if(CMAKE_PLATFORM_HAS_INSTALLNAME)
  215. find_program(CMAKE_INSTALL_NAME_TOOL NAMES ${_CMAKE_TOOLCHAIN_PREFIX}install_name_tool HINTS ${_CMAKE_TOOLCHAIN_LOCATION} NO_CMAKE_PATH NO_CMAKE_ENVIRONMENT_PATH)
  216. if(NOT CMAKE_INSTALL_NAME_TOOL)
  217. message(FATAL_ERROR "Could not find install_name_tool, please check your installation.")
  218. endif()
  219. list(APPEND _CMAKE_TOOL_VARS INSTALL_NAME_TOOL)
  220. endif()
  221. # Mark any tool cache entries as advanced.
  222. foreach(_CMAKE_TOOL IN LISTS _CMAKE_TOOL_VARS)
  223. get_property(_CMAKE_TOOL_CACHED CACHE CMAKE_${_CMAKE_TOOL} PROPERTY TYPE)
  224. if(_CMAKE_TOOL_CACHED)
  225. mark_as_advanced(CMAKE_${_CMAKE_TOOL})
  226. endif()
  227. unset(_CMAKE_${_CMAKE_TOOL}_NAMES)
  228. endforeach()
  229. unset(_CMAKE_TOOL_VARS)
  230. unset(_CMAKE_TOOL_CACHED)
  231. unset(_CMAKE_TOOL_NAME)
  232. unset(_CMAKE_TOOL)
  233. if("x${CMAKE_${_CMAKE_PROCESSING_LANGUAGE}_COMPILER_ID}" MATCHES "^xIAR$")
  234. # Set for backwards compatibility
  235. set(CMAKE_IAR_ARCHIVE "${CMAKE_AR}" CACHE FILEPATH "The IAR archiver")
  236. set(CMAKE_IAR_LINKER "${CMAKE_LINKER}" CACHE FILEPATH "The IAR ILINK linker")
  237. mark_as_advanced(CMAKE_IAR_LINKER CMAKE_IAR_AR)
  238. endif()
  239. cmake_policy(POP)