CMakeParseImplicitLinkInfo.cmake 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391
  1. # Distributed under the OSI-approved BSD 3-Clause License. See accompanying
  2. # file Copyright.txt or https://cmake.org/licensing for details.
  3. cmake_policy(PUSH)
  4. cmake_policy(SET CMP0053 NEW)
  5. cmake_policy(SET CMP0054 NEW)
  6. # Function to parse implicit linker options.
  7. #
  8. # This is used internally by CMake and should not be included by user
  9. # code.
  10. #
  11. # Note: this function is leaked/exposed by FindOpenMP and therefore needs
  12. # to have a stable API so projects that copied `FindOpenMP` for backwards
  13. # compatibility don't break.
  14. #
  15. function(CMAKE_PARSE_IMPLICIT_LINK_INFO text lib_var dir_var fwk_var log_var obj_regex)
  16. set(keywordArgs)
  17. set(oneValueArgs LANGUAGE COMPUTE_IMPLICIT_OBJECTS)
  18. set(multiValueArgs )
  19. cmake_parse_arguments(EXTRA_PARSE "${keywordArgs}" "${oneValueArgs}" "${multiValueArgs}" ${ARGN})
  20. cmake_parse_implicit_link_info2("${text}" "${log_var}" "${obj_regex}"
  21. COMPUTE_IMPLICIT_LIBS "${lib_var}" COMPUTE_IMPLICIT_DIRS "${dir_var}"
  22. COMPUTE_IMPLICIT_FWKS "${fwk_var}" ${ARGN})
  23. set(${lib_var} "${${lib_var}}" PARENT_SCOPE)
  24. set(${dir_var} "${${dir_var}}" PARENT_SCOPE)
  25. set(${fwk_var} "${${fwk_var}}" PARENT_SCOPE)
  26. set(${log_var} "${${log_var}}" PARENT_SCOPE)
  27. if(EXTRA_PARSE_COMPUTE_IMPLICIT_OBJECTS)
  28. set(${EXTRA_PARSE_COMPUTE_IMPLICIT_OBJECTS} "${${EXTRA_PARSE_COMPUTE_IMPLICIT_OBJECTS}}" PARENT_SCOPE)
  29. endif()
  30. endfunction()
  31. function(cmake_parse_implicit_link_info2 text log_var obj_regex)
  32. set(implicit_libs_tmp "")
  33. set(implicit_objs_tmp "")
  34. set(implicit_dirs_tmp)
  35. set(implicit_fwks_tmp)
  36. set(log "")
  37. set(keywordArgs)
  38. set(oneValueArgs LANGUAGE
  39. COMPUTE_IMPLICIT_LIBS COMPUTE_IMPLICIT_DIRS COMPUTE_IMPLICIT_FWKS
  40. COMPUTE_IMPLICIT_OBJECTS COMPUTE_LINKER)
  41. set(multiValueArgs )
  42. cmake_parse_arguments(EXTRA_PARSE "${keywordArgs}" "${oneValueArgs}" "${multiValueArgs}" ${ARGN})
  43. set(is_msvc 0)
  44. if(EXTRA_PARSE_LANGUAGE AND
  45. ("x${CMAKE_${EXTRA_PARSE_LANGUAGE}_COMPILER_ID}" STREQUAL "xMSVC" OR
  46. "x${CMAKE_${EXTRA_PARSE_LANGUAGE}_SIMULATE_ID}" STREQUAL "xMSVC"))
  47. set(is_msvc 1)
  48. endif()
  49. # Parse implicit linker arguments.
  50. set(linker "ld[0-9]*(\\.[a-z]+)?")
  51. if(is_msvc)
  52. string(APPEND linker "|link\\.exe|lld-link(\\.exe)?")
  53. endif()
  54. if(CMAKE_LINKER)
  55. get_filename_component(default_linker ${CMAKE_LINKER} NAME)
  56. if (NOT default_linker MATCHES "(${linker})")
  57. string(REGEX REPLACE "([][+.*?()^$])" "\\\\\\1" default_linker "${default_linker}")
  58. list(PREPEND linker "${default_linker}|")
  59. endif()
  60. endif()
  61. set(startfile "CMAKE_LINK_STARTFILE-NOTFOUND")
  62. if(CMAKE_LINK_STARTFILE)
  63. set(startfile "${CMAKE_LINK_STARTFILE}")
  64. endif()
  65. # Construct a regex to match linker lines. It must match both the
  66. # whole line and just the command (argv[0]).
  67. set(linker_regex "^( *|.*[/\\])(${linker}|${startfile}|([^/\\]+-)?ld|collect2)[^/\\]*( |$)")
  68. set(linker_exclude_regex "collect2 version |^[A-Za-z0-9_]+=|/ldfe ")
  69. set(linker_tool_regex "^[ \t]*(->|\")?[ \t]*(([^\"]*[/\\])?(${linker}))(\"|,| |$)")
  70. set(linker_tool_exclude_regex "cuda-fake-ld|-fuse-ld=|^ExecuteExternalTool ")
  71. set(linker_tool "NOTFOUND")
  72. set(linker_tool_fallback "")
  73. set(link_line_parsed 0)
  74. string(APPEND log " link line regex: [${linker_regex}]\n")
  75. if(EXTRA_PARSE_COMPUTE_LINKER)
  76. string(APPEND log " linker tool regex: [${linker_tool_regex}]\n")
  77. endif()
  78. string(REGEX REPLACE "\r?\n" ";" output_lines "${text}")
  79. foreach(line IN LISTS output_lines)
  80. if(EXTRA_PARSE_COMPUTE_LINKER AND
  81. NOT linker_tool AND NOT "${line}" MATCHES "${linker_tool_exclude_regex}")
  82. if("${line}" MATCHES "exec: ([^()]*/(${linker}))") # IBM XL as nvcc host compiler
  83. set(linker_tool "${CMAKE_MATCH_1}")
  84. elseif("${line}" MATCHES "^export XL_LINKER=(.*/${linker})[ \t]*$") # IBM XL
  85. set(linker_tool "${CMAKE_MATCH_1}")
  86. elseif("${line}" MATCHES "--with-ld=") # GNU
  87. # The GNU compiler reports how it was configured.
  88. # This does not account for -fuse-ld= so use it only as a fallback.
  89. if("${line}" MATCHES " --with-ld=([^ ]+/${linker})( |$)")
  90. set(linker_tool_fallback "${CMAKE_MATCH_1}")
  91. endif()
  92. elseif("${line}" MATCHES "vs_link.*-- +([^\"]*[/\\](${linker})) ") # cmake -E vs_link_exe
  93. set(linker_tool "${CMAKE_MATCH_1}")
  94. elseif("${line}" MATCHES "${linker_tool_regex}")
  95. set(linker_tool "${CMAKE_MATCH_2}")
  96. endif()
  97. endif()
  98. if(NOT (EXTRA_PARSE_COMPUTE_IMPLICIT_LIBS OR EXTRA_PARSE_COMPUTE_IMPLICIT_DIRS
  99. OR EXTRA_PARSE_COMPUTE_IMPLICIT_FWKS OR EXTRA_PARSE_COMPUTE_IMPLICIT_OBJECTS))
  100. if(linker_tool)
  101. break()
  102. else()
  103. continue()
  104. endif()
  105. endif()
  106. set(cmd)
  107. if("${line}" MATCHES "${linker_regex}" AND
  108. NOT "${line}" MATCHES "${linker_exclude_regex}")
  109. if(XCODE)
  110. # Xcode unconditionally adds a path under the project build tree and
  111. # on older versions it is not reported with proper quotes. Remove it.
  112. string(REGEX REPLACE "([][+.*()^])" "\\\\\\1" _dir_regex "${CMAKE_BINARY_DIR}")
  113. string(REGEX REPLACE " -[FL]${_dir_regex}/([^ ]| [^-])+( |$)" " " xline "${line}")
  114. if(NOT "x${xline}" STREQUAL "x${line}")
  115. string(APPEND log " reduced line: [${line}]\n to: [${xline}]\n")
  116. set(line "${xline}")
  117. endif()
  118. endif()
  119. separate_arguments(args NATIVE_COMMAND "${line}")
  120. list(GET args 0 cmd)
  121. if("${cmd}" MATCHES "->")
  122. # LCC has '-> ' in-front of the linker
  123. list(GET args 1 cmd)
  124. endif()
  125. else()
  126. #check to see if the link line is comma-separated instead of space separated
  127. string(REGEX REPLACE "," " " line "${line}")
  128. if("${line}" MATCHES "${linker_regex}" AND
  129. NOT "${line}" MATCHES "${linker_exclude_regex}")
  130. separate_arguments(args NATIVE_COMMAND "${line}")
  131. list(GET args 0 cmd)
  132. if("${cmd}" MATCHES "exec:")
  133. # ibm xl sometimes has 'exec: ' in-front of the linker
  134. list(GET args 1 cmd)
  135. endif()
  136. endif()
  137. endif()
  138. set(search_static 0)
  139. if(NOT link_line_parsed AND "${cmd}" MATCHES "${linker_regex}")
  140. set(link_line_parsed 1)
  141. string(APPEND log " link line: [${line}]\n")
  142. string(REGEX REPLACE ";-([LYz]);" ";-\\1" args "${args}")
  143. set(skip_value_of "")
  144. foreach(arg IN LISTS args)
  145. if(skip_value_of)
  146. string(APPEND log " arg [${arg}] ==> skip value of ${skip_value_of}\n")
  147. set(skip_value_of "")
  148. elseif("${arg}" MATCHES "^-L(.:)?[/\\]")
  149. if(EXTRA_PARSE_COMPUTE_IMPLICIT_DIRS)
  150. # Unix search path.
  151. string(REGEX REPLACE "^-L" "" dir "${arg}")
  152. list(APPEND implicit_dirs_tmp ${dir})
  153. string(APPEND log " arg [${arg}] ==> dir [${dir}]\n")
  154. endif()
  155. elseif("${arg}" MATCHES "^[-/](LIBPATH|libpath):(.+)")
  156. if(EXTRA_PARSE_COMPUTE_IMPLICIT_DIRS)
  157. # MSVC search path.
  158. set(dir "${CMAKE_MATCH_2}")
  159. list(APPEND implicit_dirs_tmp ${dir})
  160. string(APPEND log " arg [${arg}] ==> dir [${dir}]\n")
  161. endif()
  162. elseif(is_msvc AND "${arg}" STREQUAL "-link")
  163. string(APPEND log " arg [${arg}] ==> ignore MSVC cl option\n")
  164. elseif(is_msvc AND "${arg}" MATCHES "^[-/][Ii][Mm][Pp][Ll][Ii][Bb]:")
  165. string(APPEND log " arg [${arg}] ==> ignore MSVC link option\n")
  166. elseif(is_msvc AND "${arg}" MATCHES "^[-/][Ww][Hh][Oo][Ll][Ee][Aa][Rr][Cc][Hh][Ii][Vv][Ee]:Fortran_main")
  167. string(APPEND log " arg [${arg}] ==> ignore LLVMFlang program entry point\n")
  168. elseif(is_msvc AND "${arg}" MATCHES "^(.*\\.[Ll][Ii][Bb])$")
  169. if(EXTRA_PARSE_COMPUTE_IMPLICIT_LIBS)
  170. set(lib "${CMAKE_MATCH_1}")
  171. list(APPEND implicit_libs_tmp ${lib})
  172. string(APPEND log " arg [${arg}] ==> lib [${lib}]\n")
  173. endif()
  174. elseif("${arg}" STREQUAL "-lto_library")
  175. # ld argument "-lto_library <path>"
  176. set(skip_value_of "${arg}")
  177. string(APPEND log " arg [${arg}] ==> ignore, skip following value\n")
  178. elseif("${arg}" MATCHES "^-l([^:].*)$")
  179. if(EXTRA_PARSE_COMPUTE_IMPLICIT_LIBS)
  180. # Unix library.
  181. set(lib "${CMAKE_MATCH_1}")
  182. if(search_static AND lib MATCHES "^(gfortran|quadmath|stdc\\+\\+)$")
  183. # Search for the static library later, once all link dirs are known.
  184. set(lib "SEARCH_STATIC:${lib}")
  185. endif()
  186. list(APPEND implicit_libs_tmp ${lib})
  187. string(APPEND log " arg [${arg}] ==> lib [${lib}]\n")
  188. endif()
  189. elseif("${arg}" MATCHES "^(.:)?[/\\].*\\.a$")
  190. if(EXTRA_PARSE_COMPUTE_IMPLICIT_LIBS)
  191. # Unix library full path.
  192. list(APPEND implicit_libs_tmp ${arg})
  193. string(APPEND log " arg [${arg}] ==> lib [${arg}]\n")
  194. endif()
  195. elseif("${arg}" MATCHES "^[-/](DEFAULTLIB|defaultlib):(.+)")
  196. if(EXTRA_PARSE_COMPUTE_IMPLICIT_LIBS)
  197. # Windows library.
  198. set(lib "${CMAKE_MATCH_2}")
  199. list(APPEND implicit_libs_tmp ${lib})
  200. string(APPEND log " arg [${arg}] ==> lib [${lib}]\n")
  201. endif()
  202. elseif("${arg}" MATCHES "^(.:)?[/\\].*\\.o$")
  203. if(EXTRA_PARSE_COMPUTE_IMPLICIT_OBJECTS)
  204. list(APPEND implicit_objs_tmp ${arg})
  205. string(APPEND log " arg [${arg}] ==> obj [${arg}]\n")
  206. endif()
  207. if(EXTRA_PARSE_COMPUTE_IMPLICIT_LIBS)
  208. if(obj_regex AND "${arg}" MATCHES "${obj_regex}")
  209. # Object file full path.
  210. list(APPEND implicit_libs_tmp ${arg})
  211. endif()
  212. endif()
  213. elseif("${arg}" MATCHES "^-Y(P,)?[^0-9]")
  214. if(EXTRA_PARSE_COMPUTE_IMPLICIT_DIRS)
  215. # Sun search path ([^0-9] avoids conflict with Mac -Y<num>).
  216. string(REGEX REPLACE "^-Y(P,)?" "" dirs "${arg}")
  217. string(REPLACE ":" ";" dirs "${dirs}")
  218. list(APPEND implicit_dirs_tmp ${dirs})
  219. string(APPEND log " arg [${arg}] ==> dirs [${dirs}]\n")
  220. endif()
  221. elseif("${arg}" STREQUAL "-Bstatic")
  222. set(search_static 1)
  223. string(APPEND log " arg [${arg}] ==> search static\n" )
  224. elseif("${arg}" STREQUAL "-Bdynamic")
  225. set(search_static 0)
  226. string(APPEND log " arg [${arg}] ==> search dynamic\n" )
  227. elseif("${arg}" MATCHES "^-l:")
  228. if(EXTRA_PARSE_COMPUTE_IMPLICIT_LIBS)
  229. # HP named library.
  230. list(APPEND implicit_libs_tmp ${arg})
  231. string(APPEND log " arg [${arg}] ==> lib [${arg}]\n")
  232. endif()
  233. elseif("${arg}" MATCHES "^-z(all|default|weak)extract")
  234. if(EXTRA_PARSE_COMPUTE_IMPLICIT_LIBS)
  235. # Link editor option.
  236. list(APPEND implicit_libs_tmp ${arg})
  237. string(APPEND log " arg [${arg}] ==> opt [${arg}]\n")
  238. endif()
  239. elseif("${arg}" STREQUAL "cl.exe")
  240. string(APPEND log " arg [${arg}] ==> recognize MSVC cl\n")
  241. set(is_msvc 1)
  242. else()
  243. string(APPEND log " arg [${arg}] ==> ignore\n")
  244. endif()
  245. endforeach()
  246. elseif("${line}" MATCHES "LPATH(=| is:? *)(.*)$")
  247. if(EXTRA_PARSE_COMPUTE_IMPLICIT_DIRS)
  248. string(APPEND log " LPATH line: [${line}]\n")
  249. # HP search path.
  250. string(REPLACE ":" ";" paths "${CMAKE_MATCH_2}")
  251. list(APPEND implicit_dirs_tmp ${paths})
  252. string(APPEND log " dirs [${paths}]\n")
  253. endif()
  254. else()
  255. string(APPEND log " ignore line: [${line}]\n")
  256. endif()
  257. if((NOT EXTRA_PARSE_COMPUTE_LINKER OR linker_tool) AND link_line_parsed)
  258. break()
  259. endif()
  260. endforeach()
  261. if(NOT linker_tool AND linker_tool_fallback)
  262. set(linker_tool "${linker_tool_fallback}")
  263. endif()
  264. if(linker_tool)
  265. if(CMAKE_HOST_SYSTEM_NAME STREQUAL "Windows")
  266. cmake_path(NORMAL_PATH linker_tool)
  267. endif()
  268. string(APPEND log " linker tool for '${EXTRA_PARSE_LANGUAGE}': ${linker_tool}\n")
  269. endif()
  270. # Look for library search paths reported by linker.
  271. if(EXTRA_PARSE_COMPUTE_IMPLICIT_DIRS AND "${output_lines}" MATCHES ";Library search paths:((;\t[^;]+)+)")
  272. string(REPLACE ";\t" ";" implicit_dirs_match "${CMAKE_MATCH_1}")
  273. string(APPEND log " Library search paths: [${implicit_dirs_match}]\n")
  274. list(APPEND implicit_dirs_tmp ${implicit_dirs_match})
  275. endif()
  276. if(EXTRA_PARSE_COMPUTE_IMPLICIT_FWKS AND "${output_lines}" MATCHES ";Framework search paths:((;\t[^;]+)+)")
  277. string(REPLACE ";\t" ";" implicit_fwks_match "${CMAKE_MATCH_1}")
  278. string(APPEND log " Framework search paths: [${implicit_fwks_match}]\n")
  279. list(APPEND implicit_fwks_tmp ${implicit_fwks_match})
  280. endif()
  281. # Cleanup list of libraries and flags.
  282. # We remove items that are not language-specific.
  283. set(implicit_libs "")
  284. foreach(lib IN LISTS implicit_libs_tmp)
  285. if("x${lib}" MATCHES "^xSEARCH_STATIC:(.*)")
  286. set(search_static 1)
  287. set(lib "${CMAKE_MATCH_1}")
  288. else()
  289. set(search_static 0)
  290. endif()
  291. if("x${lib}" MATCHES "^x(crt.*\\.o|gcc_eh.*|.*libgcc_eh.*|System.*|.*libclang_rt.*|msvcrt.*|libvcruntime.*|libucrt.*|libcmt.*)$")
  292. string(APPEND log " remove lib [${lib}]\n")
  293. elseif(search_static)
  294. # This library appears after a -Bstatic flag. Due to ordering
  295. # and filtering for mixed-language link lines, we do not preserve
  296. # the -Bstatic flag itself. Instead, use an absolute path.
  297. # Search using a temporary variable with a distinct name
  298. # so that our test suite does not depend on disk content.
  299. find_library("CMAKE_${lang}_IMPLICIT_LINK_LIBRARY_${lib}" NO_CACHE NAMES "lib${lib}.a" NO_DEFAULT_PATH PATHS ${implicit_dirs_tmp})
  300. set(_lib_static "${CMAKE_${lang}_IMPLICIT_LINK_LIBRARY_${lib}}")
  301. if(_lib_static)
  302. string(APPEND log " search lib [SEARCH_STATIC:${lib}] ==> [${_lib_static}]\n")
  303. list(APPEND implicit_libs "${_lib_static}")
  304. else()
  305. string(APPEND log " search lib [SEARCH_STATIC:${lib}] ==> [${lib}]\n")
  306. list(APPEND implicit_libs "${lib}")
  307. endif()
  308. elseif(IS_ABSOLUTE "${lib}")
  309. get_filename_component(abs "${lib}" ABSOLUTE)
  310. if(NOT "x${lib}" STREQUAL "x${abs}")
  311. string(APPEND log " collapse lib [${lib}] ==> [${abs}]\n")
  312. endif()
  313. list(APPEND implicit_libs "${abs}")
  314. else()
  315. list(APPEND implicit_libs "${lib}")
  316. endif()
  317. endforeach()
  318. if(EXTRA_PARSE_COMPUTE_IMPLICIT_OBJECTS)
  319. set(implicit_objs "")
  320. foreach(obj IN LISTS implicit_objs_tmp)
  321. if(IS_ABSOLUTE "${obj}")
  322. get_filename_component(abs "${obj}" ABSOLUTE)
  323. if(NOT "x${obj}" STREQUAL "x${abs}")
  324. string(APPEND log " collapse obj [${obj}] ==> [${abs}]\n")
  325. endif()
  326. list(APPEND implicit_objs "${abs}")
  327. else()
  328. list(APPEND implicit_objs "${obj}")
  329. endif()
  330. endforeach()
  331. endif()
  332. # Cleanup list of library and framework directories.
  333. set(desc_dirs "library")
  334. set(desc_fwks "framework")
  335. foreach(t dirs fwks)
  336. set(implicit_${t} "")
  337. foreach(d IN LISTS implicit_${t}_tmp)
  338. get_filename_component(dir "${d}" ABSOLUTE)
  339. string(FIND "${dir}" "${CMAKE_FILES_DIRECTORY}/" pos)
  340. if(NOT pos LESS 0)
  341. set(msg ", skipping non-system directory")
  342. else()
  343. set(msg "")
  344. list(APPEND implicit_${t} "${dir}")
  345. endif()
  346. string(APPEND log " collapse ${desc_${t}} dir [${d}] ==> [${dir}]${msg}\n")
  347. endforeach()
  348. list(REMOVE_DUPLICATES implicit_${t})
  349. endforeach()
  350. # Log results.
  351. string(APPEND log " implicit libs: [${implicit_libs}]\n")
  352. string(APPEND log " implicit objs: [${implicit_objs}]\n")
  353. string(APPEND log " implicit dirs: [${implicit_dirs}]\n")
  354. string(APPEND log " implicit fwks: [${implicit_fwks}]\n")
  355. # Return results.
  356. if(EXTRA_PARSE_COMPUTE_LINKER)
  357. set(${EXTRA_PARSE_COMPUTE_LINKER} "${linker_tool}" PARENT_SCOPE)
  358. endif()
  359. set(${log_var} "${log}" PARENT_SCOPE)
  360. if(EXTRA_PARSE_COMPUTE_IMPLICIT_LIBS)
  361. set(${EXTRA_PARSE_COMPUTE_IMPLICIT_LIBS} "${implicit_libs}" PARENT_SCOPE)
  362. endif()
  363. if(EXTRA_PARSE_COMPUTE_IMPLICIT_DIRS)
  364. set(${EXTRA_PARSE_COMPUTE_IMPLICIT_DIRS} "${implicit_dirs}" PARENT_SCOPE)
  365. endif()
  366. if(EXTRA_PARSE_COMPUTE_IMPLICIT_FWKS)
  367. set(${EXTRA_PARSE_COMPUTE_IMPLICIT_FWKS} "${implicit_fwks}" PARENT_SCOPE)
  368. endif()
  369. if(EXTRA_PARSE_COMPUTE_IMPLICIT_OBJECTS)
  370. set(${EXTRA_PARSE_COMPUTE_IMPLICIT_OBJECTS} "${implicit_objs}" PARENT_SCOPE)
  371. endif()
  372. endfunction()
  373. cmake_policy(POP)