LLVMExternalProjectUtils.cmake 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214
  1. include(ExternalProject)
  2. # llvm_ExternalProject_BuildCmd(out_var target)
  3. # Utility function for constructing command lines for external project targets
  4. function(llvm_ExternalProject_BuildCmd out_var target bin_dir)
  5. cmake_parse_arguments(ARG "" "CONFIGURATION" "" ${ARGN})
  6. if(NOT ARG_CONFIGURATION)
  7. set(ARG_CONFIGURATION "$<CONFIGURATION>")
  8. endif()
  9. if (CMAKE_GENERATOR MATCHES "Make")
  10. # Use special command for Makefiles to support parallelism.
  11. set(${out_var} "$(MAKE)" "-C" "${bin_dir}" "${target}" PARENT_SCOPE)
  12. else()
  13. set(${out_var} ${CMAKE_COMMAND} --build ${bin_dir} --target ${target}
  14. --config ${ARG_CONFIGURATION} PARENT_SCOPE)
  15. endif()
  16. endfunction()
  17. # llvm_ExternalProject_Add(name source_dir ...
  18. # USE_TOOLCHAIN
  19. # Use just-built tools (see TOOLCHAIN_TOOLS)
  20. # EXCLUDE_FROM_ALL
  21. # Exclude this project from the all target
  22. # NO_INSTALL
  23. # Don't generate install targets for this project
  24. # ALWAYS_CLEAN
  25. # Always clean the sub-project before building
  26. # CMAKE_ARGS arguments...
  27. # Optional cmake arguments to pass when configuring the project
  28. # TOOLCHAIN_TOOLS targets...
  29. # Targets for toolchain tools (defaults to clang;lld)
  30. # DEPENDS targets...
  31. # Targets that this project depends on
  32. # EXTRA_TARGETS targets...
  33. # Extra targets in the subproject to generate targets for
  34. # PASSTHROUGH_PREFIXES prefix...
  35. # Extra variable prefixes (name is always included) to pass down
  36. # )
  37. function(llvm_ExternalProject_Add name source_dir)
  38. cmake_parse_arguments(ARG
  39. "USE_TOOLCHAIN;EXCLUDE_FROM_ALL;NO_INSTALL;ALWAYS_CLEAN"
  40. "SOURCE_DIR"
  41. "CMAKE_ARGS;TOOLCHAIN_TOOLS;RUNTIME_LIBRARIES;DEPENDS;EXTRA_TARGETS;PASSTHROUGH_PREFIXES"
  42. ${ARGN})
  43. canonicalize_tool_name(${name} nameCanon)
  44. if(NOT ARG_TOOLCHAIN_TOOLS)
  45. set(ARG_TOOLCHAIN_TOOLS clang lld)
  46. if(NOT APPLE AND NOT WIN32)
  47. list(APPEND ARG_TOOLCHAIN_TOOLS llvm-ar llvm-ranlib)
  48. endif()
  49. endif()
  50. foreach(tool ${ARG_TOOLCHAIN_TOOLS})
  51. if(TARGET ${tool})
  52. list(APPEND TOOLCHAIN_TOOLS ${tool})
  53. list(APPEND TOOLCHAIN_BINS $<TARGET_FILE:${tool}>)
  54. endif()
  55. endforeach()
  56. if(NOT ARG_RUNTIME_LIBRARIES)
  57. set(ARG_RUNTIME_LIBRARIES compiler-rt libcxx)
  58. endif()
  59. foreach(lib ${ARG_RUNTIME_LIBRARIES})
  60. if(TARGET ${lib})
  61. list(APPEND RUNTIME_LIBRARIES ${lib})
  62. endif()
  63. endforeach()
  64. if(ARG_ALWAYS_CLEAN)
  65. set(always_clean clean)
  66. endif()
  67. list(FIND TOOLCHAIN_TOOLS clang FOUND_CLANG)
  68. if(FOUND_CLANG GREATER -1)
  69. set(CLANG_IN_TOOLCHAIN On)
  70. endif()
  71. if(RUNTIME_LIBRARIES AND CLANG_IN_TOOLCHAIN)
  72. list(APPEND TOOLCHAIN_BINS ${RUNTIME_LIBRARIES})
  73. endif()
  74. set(STAMP_DIR ${CMAKE_CURRENT_BINARY_DIR}/${name}-stamps/)
  75. set(BINARY_DIR ${CMAKE_CURRENT_BINARY_DIR}/${name}-bins/)
  76. add_custom_target(${name}-clear
  77. COMMAND ${CMAKE_COMMAND} -E remove_directory ${BINARY_DIR}
  78. COMMAND ${CMAKE_COMMAND} -E remove_directory ${STAMP_DIR}
  79. COMMENT "Clobbering ${name} build and stamp directories"
  80. USES_TERMINAL
  81. )
  82. # Find all variables that start with a prefix and propagate them through
  83. get_cmake_property(variableNames VARIABLES)
  84. list(APPEND ARG_PASSTHROUGH_PREFIXES ${nameCanon})
  85. foreach(prefix ${ARG_PASSTHROUGH_PREFIXES})
  86. foreach(variableName ${variableNames})
  87. if(variableName MATCHES "^${prefix}")
  88. string(REPLACE ";" "\;" value "${${variableName}}")
  89. list(APPEND PASSTHROUGH_VARIABLES
  90. -D${variableName}=${value})
  91. endif()
  92. endforeach()
  93. endforeach()
  94. if(ARG_USE_TOOLCHAIN)
  95. if(CLANG_IN_TOOLCHAIN)
  96. set(compiler_args -DCMAKE_C_COMPILER=${LLVM_RUNTIME_OUTPUT_INTDIR}/clang
  97. -DCMAKE_CXX_COMPILER=${LLVM_RUNTIME_OUTPUT_INTDIR}/clang++)
  98. endif()
  99. if(llvm-ar IN_LIST TOOLCHAIN_TOOLS)
  100. list(APPEND compiler_args -DCMAKE_AR=${LLVM_RUNTIME_OUTPUT_INTDIR}/llvm-ar)
  101. endif()
  102. if(llvm-ranlib IN_LIST TOOLCHAIN_TOOLS)
  103. list(APPEND compiler_args -DCMAKE_RANLIB=${LLVM_RUNTIME_OUTPUT_INTDIR}/llvm-ranlib)
  104. endif()
  105. list(APPEND ARG_DEPENDS ${TOOLCHAIN_TOOLS})
  106. endif()
  107. add_custom_command(
  108. OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/${name}-clobber-stamp
  109. DEPENDS ${ARG_DEPENDS}
  110. COMMAND ${CMAKE_COMMAND} -E touch ${BINARY_DIR}/CMakeCache.txt
  111. COMMAND ${CMAKE_COMMAND} -E touch ${STAMP_DIR}/${name}-mkdir
  112. COMMAND ${CMAKE_COMMAND} -E touch ${CMAKE_CURRENT_BINARY_DIR}/${name}-clobber-stamp
  113. COMMENT "Clobbering bootstrap build and stamp directories"
  114. )
  115. add_custom_target(${name}-clobber
  116. DEPENDS ${CMAKE_CURRENT_BINARY_DIR}/${name}-clobber-stamp)
  117. if(ARG_EXCLUDE_FROM_ALL)
  118. set(exclude EXCLUDE_FROM_ALL 1)
  119. endif()
  120. ExternalProject_Add(${name}
  121. DEPENDS ${ARG_DEPENDS} llvm-config
  122. ${name}-clobber
  123. PREFIX ${CMAKE_BINARY_DIR}/projects/${name}
  124. SOURCE_DIR ${source_dir}
  125. STAMP_DIR ${STAMP_DIR}
  126. BINARY_DIR ${BINARY_DIR}
  127. ${exclude}
  128. CMAKE_ARGS ${${nameCanon}_CMAKE_ARGS}
  129. ${compiler_args}
  130. -DCMAKE_INSTALL_PREFIX=${CMAKE_INSTALL_PREFIX}
  131. -DLLVM_BINARY_DIR=${PROJECT_BINARY_DIR}
  132. -DLLVM_CONFIG_PATH=$<TARGET_FILE:llvm-config>
  133. -DLLVM_ENABLE_WERROR=${LLVM_ENABLE_WERROR}
  134. -DPACKAGE_VERSION=${PACKAGE_VERSION}
  135. -DCMAKE_BUILD_TYPE=${CMAKE_BUILD_TYPE}
  136. -DCMAKE_MAKE_PROGRAM=${CMAKE_MAKE_PROGRAM}
  137. ${ARG_CMAKE_ARGS}
  138. ${PASSTHROUGH_VARIABLES}
  139. INSTALL_COMMAND ""
  140. STEP_TARGETS configure build
  141. BUILD_ALWAYS 1
  142. USES_TERMINAL_CONFIGURE 1
  143. USES_TERMINAL_BUILD 1
  144. USES_TERMINAL_INSTALL 1
  145. )
  146. if(ARG_USE_TOOLCHAIN)
  147. set(force_deps DEPENDS ${TOOLCHAIN_BINS})
  148. endif()
  149. llvm_ExternalProject_BuildCmd(run_clean clean ${BINARY_DIR})
  150. ExternalProject_Add_Step(${name} clean
  151. COMMAND ${run_clean}
  152. COMMENT "Cleaning ${name}..."
  153. DEPENDEES configure
  154. ${force_deps}
  155. WORKING_DIRECTORY ${BINARY_DIR}
  156. EXCLUDE_FROM_MAIN 1
  157. USES_TERMINAL 1
  158. )
  159. ExternalProject_Add_StepTargets(${name} clean)
  160. if(ARG_USE_TOOLCHAIN)
  161. add_dependencies(${name}-clean ${name}-clobber)
  162. set_target_properties(${name}-clean PROPERTIES
  163. SOURCES ${CMAKE_CURRENT_BINARY_DIR}/${name}-clobber-stamp)
  164. endif()
  165. if(NOT ARG_NO_INSTALL)
  166. install(CODE "execute_process\(COMMAND \${CMAKE_COMMAND} -DCMAKE_INSTALL_PREFIX=\${CMAKE_INSTALL_PREFIX} -P ${BINARY_DIR}/cmake_install.cmake \)"
  167. COMPONENT ${name})
  168. add_custom_target(install-${name}
  169. DEPENDS ${name}
  170. COMMAND "${CMAKE_COMMAND}"
  171. -DCMAKE_INSTALL_COMPONENT=${name}
  172. -P "${CMAKE_BINARY_DIR}/cmake_install.cmake"
  173. USES_TERMINAL)
  174. endif()
  175. # Add top-level targets
  176. foreach(target ${ARG_EXTRA_TARGETS})
  177. string(REPLACE ":" ";" target_list ${target})
  178. list(GET target_list 0 target)
  179. list(LENGTH target_list target_list_len)
  180. if(${target_list_len} GREATER 1)
  181. list(GET target_list 1 target_name)
  182. else()
  183. set(target_name "${target}")
  184. endif()
  185. llvm_ExternalProject_BuildCmd(build_runtime_cmd ${target} ${BINARY_DIR})
  186. add_custom_target(${target_name}
  187. COMMAND ${build_runtime_cmd}
  188. DEPENDS ${name}-configure
  189. WORKING_DIRECTORY ${BINARY_DIR}
  190. VERBATIM
  191. USES_TERMINAL)
  192. endforeach()
  193. endfunction()