ctkMacroCheckExternalProjectDependency.cmake 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853
  1. #.rst:
  2. # ExternalProjectDependency
  3. # -------------------------
  4. ###########################################################################
  5. #
  6. # Library: CTK
  7. #
  8. # Copyright (c) Kitware Inc.
  9. #
  10. # Licensed under the Apache License, Version 2.0 (the "License");
  11. # you may not use this file except in compliance with the License.
  12. # You may obtain a copy of the License at
  13. #
  14. # http://www.apache.org/licenses/LICENSE-2.0.txt
  15. #
  16. # Unless required by applicable law or agreed to in writing, software
  17. # distributed under the License is distributed on an "AS IS" BASIS,
  18. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  19. # See the License for the specific language governing permissions and
  20. # limitations under the License.
  21. #
  22. ###########################################################################
  23. include(CMakeParseArguments)
  24. if(NOT DEFINED EP_LIST_SEPARATOR)
  25. set(EP_LIST_SEPARATOR "^^")
  26. endif()
  27. #.rst:
  28. # .. cmake:variable:: EXTERNAL_PROJECT_DIR
  29. #
  30. if(NOT EXISTS "${EXTERNAL_PROJECT_DIR}")
  31. set(EXTERNAL_PROJECT_DIR ${CMAKE_SOURCE_DIR}/SuperBuild)
  32. endif()
  33. #.rst:
  34. # .. cmake:variable:: EXTERNAL_PROJECT_FILE_PREFIX
  35. #
  36. if(NOT DEFINED EXTERNAL_PROJECT_FILE_PREFIX)
  37. set(EXTERNAL_PROJECT_FILE_PREFIX "External_")
  38. endif()
  39. # Compute -G arg for configuring external projects with the same CMake generator:
  40. if(CMAKE_EXTRA_GENERATOR)
  41. set(EP_CMAKE_GENERATOR "${CMAKE_EXTRA_GENERATOR} - ${CMAKE_GENERATOR}")
  42. else()
  43. set(EP_CMAKE_GENERATOR "${CMAKE_GENERATOR}")
  44. endif()
  45. set(EP_CMAKE_GENERATOR_PLATFORM "${CMAKE_GENERATOR_PLATFORM}")
  46. set(EP_CMAKE_GENERATOR_TOOLSET "${CMAKE_GENERATOR_TOOLSET}")
  47. #.rst:
  48. # .. cmake:function:: mark_as_superbuild
  49. #
  50. # .. code-block:: cmake
  51. #
  52. # mark_as_superbuild(<varname1>[:<vartype1>] [<varname2>[:<vartype2>] [...]])
  53. #
  54. # .. code-block:: cmake
  55. #
  56. # mark_as_superbuild(
  57. # VARS <varname1>[:<vartype1>] [<varname2>[:<vartype2>] [...]]
  58. # [PROJECTS <projectname> [<projectname> [...]] | ALL_PROJECTS]
  59. # [LABELS <label1> [<label2> [...]]]
  60. # )
  61. #
  62. # .. code-block:: cmake
  63. #
  64. # PROJECTS corresponds to a list of <projectname> that will be added using 'ExternalProject_Add' function.
  65. # If not specified and called within a project file, it defaults to the value of 'SUPERBUILD_TOPLEVEL_PROJECT'
  66. # Otherwise, it defaults to 'CMAKE_PROJECT_NAME'.
  67. # If instead 'ALL_PROJECTS' is specified, the variables and labels will be passed to all projects.
  68. #
  69. # VARS is an expected list of variables specified as <varname>:<vartype> to pass to <projectname>
  70. #
  71. #
  72. # LABELS is an optional list of label to associate with the variable names specified using 'VARS' and passed to
  73. # the <projectname> as CMake CACHE args of the form:
  74. # -D<projectname>_EP_LABEL_<label1>=<varname1>;<varname2>[...]
  75. # -D<projectname>_EP_LABEL_<label2>=<varname1>;<varname2>[...]
  76. #
  77. function(mark_as_superbuild)
  78. set(options ALL_PROJECTS CMAKE_CMD)
  79. set(oneValueArgs)
  80. set(multiValueArgs VARS PROJECTS LABELS)
  81. cmake_parse_arguments(_sb "${options}" "${oneValueArgs}" "${multiValueArgs}" ${ARGN})
  82. set(_vars ${_sb_UNPARSED_ARGUMENTS})
  83. set(_named_parameters_expected 0)
  84. if(_sb_PROJECTS OR _sb_ALL_PROJECTS OR _sb_LABELS OR _sb_VARS)
  85. set(_named_parameters_expected 1)
  86. set(_vars ${_sb_VARS})
  87. endif()
  88. if(_named_parameters_expected AND _sb_UNPARSED_ARGUMENTS)
  89. message(FATAL_ERROR "Arguments '${_sb_UNPARSED_ARGUMENTS}' should be associated with VARS parameter !")
  90. endif()
  91. if(_sb_PROJECTS AND _sb_ALL_PROJECTS)
  92. message(FATAL_ERROR "Arguments 'PROJECTS' and 'ALL_PROJECTS' are mutually exclusive !")
  93. endif()
  94. foreach(var ${_vars})
  95. set(_type_specified 0)
  96. if(var MATCHES ":")
  97. set(_type_specified 1)
  98. endif()
  99. # XXX Display warning with variable type is also specified for cache variable.
  100. set(_var ${var})
  101. if(NOT _type_specified)
  102. get_property(_type_set_in_cache CACHE ${_var} PROPERTY TYPE SET)
  103. set(_var_name ${_var})
  104. set(_var_type "STRING")
  105. if(_type_set_in_cache)
  106. get_property(_var_type CACHE ${_var_name} PROPERTY TYPE)
  107. endif()
  108. set(_var ${_var_name}:${_var_type})
  109. endif()
  110. list(APPEND _vars_with_type ${_var})
  111. endforeach()
  112. if(_sb_ALL_PROJECTS)
  113. set(optional_arg_ALL_PROJECTS "ALL_PROJECTS")
  114. else()
  115. set(optional_arg_ALL_PROJECTS PROJECTS ${_sb_PROJECTS})
  116. endif()
  117. _sb_append_to_cmake_args(
  118. VARS ${_vars_with_type} LABELS ${_sb_LABELS} ${optional_arg_ALL_PROJECTS})
  119. endfunction()
  120. #
  121. # _sb_extract_varname_and_vartype(<cmake_varname_and_type> <varname_var> [<vartype_var>])
  122. #
  123. # <cmake_varname_and_type> corresponds to variable name and variable type passed as "<varname>:<vartype>"
  124. #
  125. # <varname_var> will be set to "<varname>"
  126. #
  127. # <vartype_var> is an optional variable name that will be set to "<vartype>"
  128. #
  129. function(_sb_extract_varname_and_vartype cmake_varname_and_type varname_var)
  130. set(_vartype_var "${ARGV2}")
  131. string(REPLACE ":" ";" varname_and_vartype ${cmake_varname_and_type})
  132. list(GET varname_and_vartype 0 _varname)
  133. list(GET varname_and_vartype 1 _vartype)
  134. set(${varname_var} ${_varname} PARENT_SCOPE)
  135. if(_vartype_var MATCHES ".+")
  136. set(${_vartype_var} ${_vartype} PARENT_SCOPE)
  137. endif()
  138. endfunction()
  139. function(_sb_list_to_string separator input_list output_string_var)
  140. set(_string "")
  141. # Get list length
  142. list(LENGTH input_list list_length)
  143. # If the list has 0 or 1 element, there is no need to loop over.
  144. if(list_length LESS 2)
  145. set(_string "${input_list}")
  146. else()
  147. math(EXPR last_element_index "${list_length} - 1")
  148. foreach(index RANGE ${last_element_index})
  149. # Get current item_value
  150. list(GET input_list ${index} item_value)
  151. if(NOT item_value STREQUAL "")
  152. # .. and append non-empty value to output string
  153. set(_string "${_string}${item_value}")
  154. # Append separator if current element is NOT the last one.
  155. if(NOT index EQUAL last_element_index)
  156. set(_string "${_string}${separator}")
  157. endif()
  158. endif()
  159. endforeach()
  160. endif()
  161. set(${output_string_var} ${_string} PARENT_SCOPE)
  162. endfunction()
  163. #
  164. # _sb_cmakevar_to_cmakearg(<cmake_varname_and_type> <cmake_arg_var> <has_cfg_intdir_var> [<varname_var> [<vartype_var>]])
  165. #
  166. # <cmake_varname_and_type> corresponds to variable name and variable type passed as "<varname>:<vartype>"
  167. #
  168. # <cmake_arg_var> is a variable name that will be set to "-D<varname>:<vartype>=${<varname>}"
  169. #
  170. # <has_int_dir_var> is set to either TRUE or FALSE.
  171. # FALSE means that the value does NOT reference ${CMAKE_CFG_INTDIR} and
  172. # the generated cmake argument should be passed to ExternalProject_Add as CMAKE_CACHE_ARGS.
  173. # TRUEmeans that the value does reference ${CMAKE_CFG_INTDIR} and
  174. # the generated cmake argument should be passed to ExternalProject_Add as CMAKE_ARGS.
  175. #
  176. # <varname_var> is an optional variable name that will be set to "<varname>"
  177. #
  178. # <vartype_var> is an optional variable name that will be set to "<vartype>"
  179. #
  180. function(_sb_cmakevar_to_cmakearg cmake_varname_and_type cmake_arg_var has_cfg_intdir_var)
  181. set(_varname_var "${ARGV3}")
  182. set(_vartype_var "${ARGV4}")
  183. _sb_extract_varname_and_vartype(${cmake_varname_and_type} _varname _vartype)
  184. set(_var_value "${${_varname}}")
  185. set(_consider_cache_value 1)
  186. if(DEFINED ${_varname}_SKIPCACHEVALUE AND ${_varname}_SKIPCACHEVALUE)
  187. set(_consider_cache_value 0)
  188. endif()
  189. if(_consider_cache_value)
  190. get_property(_value_set_in_cache CACHE ${_varname} PROPERTY VALUE SET)
  191. if(_value_set_in_cache)
  192. get_property(_var_value CACHE ${_varname} PROPERTY VALUE)
  193. endif()
  194. endif()
  195. set(_has_cfg_intdir FALSE)
  196. if(CMAKE_CONFIGURATION_TYPES)
  197. string(FIND "${_var_value}" ${CMAKE_CFG_INTDIR} _index)
  198. if(NOT _index EQUAL -1)
  199. # Separate list item with <EP_LIST_SEPARATOR>
  200. _sb_list_to_string(${EP_LIST_SEPARATOR} "${_var_value}" _var_value)
  201. set(_has_cfg_intdir TRUE)
  202. endif()
  203. endif()
  204. if(NOT _has_cfg_intdir)
  205. string(REPLACE "\"" "\\\"" _var_value "${_var_value}")
  206. endif()
  207. set(${cmake_arg_var} -D${_varname}:${_vartype}=${_var_value} PARENT_SCOPE)
  208. set(${has_cfg_intdir_var} ${_has_cfg_intdir} PARENT_SCOPE)
  209. if(_varname_var MATCHES ".+")
  210. set(${_varname_var} ${_varname} PARENT_SCOPE)
  211. endif()
  212. if(_vartype_var MATCHES ".+")
  213. set(${_vartype_var} ${_vartype} PARENT_SCOPE)
  214. endif()
  215. endfunction()
  216. set(_ALL_PROJECT_IDENTIFIER "ALLALLALL")
  217. #
  218. # _sb_append_to_cmake_args(
  219. # VARS <varname1>:<vartype1> [<varname2>:<vartype2> [...]]
  220. # [PROJECTS <projectname> [<projectname> [...]] | ALL_PROJECTS]
  221. # [LABELS <label1> [<label2> [...]]]
  222. # )
  223. #
  224. # PROJECTS corresponds to a list of <projectname> that will be added using 'ExternalProject_Add' function.
  225. # If not specified and called within a project file, it defaults to the value of 'SUPERBUILD_TOPLEVEL_PROJECT'
  226. # Otherwise, it defaults to 'CMAKE_PROJECT_NAME'.
  227. # If instead 'ALL_PROJECTS' is specified, the variables and labels will be passed to all projects.
  228. #
  229. # VARS is an expected list of variables specified as <varname>:<vartype> to pass to <projectname>
  230. #
  231. #
  232. # LABELS is an optional list of label to associate with the variable names specified using 'VARS' and passed to
  233. # the <projectname> as CMake CACHE args of the form:
  234. # -D<projectname>_EP_LABEL_<label1>=<varname1>;<varname2>[...]
  235. # -D<projectname>_EP_LABEL_<label2>=<varname1>;<varname2>[...]
  236. #
  237. function(_sb_append_to_cmake_args)
  238. set(options ALL_PROJECTS)
  239. set(oneValueArgs)
  240. set(multiValueArgs VARS PROJECTS LABELS)
  241. cmake_parse_arguments(_sb "${options}" "${oneValueArgs}" "${multiValueArgs}" ${ARGN})
  242. if(NOT _sb_PROJECTS AND NOT _sb_ALL_PROJECTS)
  243. if(SUPERBUILD_TOPLEVEL_PROJECT)
  244. set(_sb_PROJECTS ${SUPERBUILD_TOPLEVEL_PROJECT})
  245. else()
  246. set(_sb_PROJECTS ${CMAKE_PROJECT_NAME})
  247. endif()
  248. endif()
  249. if(_sb_ALL_PROJECTS)
  250. set(_sb_PROJECTS ${_ALL_PROJECT_IDENTIFIER})
  251. endif()
  252. foreach(_sb_PROJECT ${_sb_PROJECTS})
  253. set(_ep_varnames "")
  254. foreach(varname_and_vartype ${_sb_VARS})
  255. if(NOT TARGET ${_sb_PROJECT})
  256. set_property(GLOBAL APPEND PROPERTY ${_sb_PROJECT}_EP_CMAKE_ARGS ${varname_and_vartype})
  257. _sb_extract_varname_and_vartype(${varname_and_vartype} _varname)
  258. else()
  259. message(FATAL_ERROR "Function _sb_append_to_cmake_args not allowed because project '${_sb_PROJECT}' already added !")
  260. endif()
  261. list(APPEND _ep_varnames ${_varname})
  262. endforeach()
  263. if(_sb_LABELS)
  264. set_property(GLOBAL APPEND PROPERTY ${_sb_PROJECT}_EP_LABELS ${_sb_LABELS})
  265. foreach(label ${_sb_LABELS})
  266. set_property(GLOBAL APPEND PROPERTY ${_sb_PROJECT}_EP_LABEL_${label} ${_ep_varnames})
  267. endforeach()
  268. endif()
  269. endforeach()
  270. endfunction()
  271. function(_sb_get_external_project_arguments proj varname)
  272. mark_as_superbuild(${SUPERBUILD_TOPLEVEL_PROJECT}_USE_SYSTEM_${proj}:BOOL)
  273. function(_sb_collect_args proj)
  274. # Set list of CMake args associated with each label
  275. get_property(_labels GLOBAL PROPERTY ${proj}_EP_LABELS)
  276. if(_labels)
  277. list(REMOVE_DUPLICATES _labels)
  278. foreach(label ${_labels})
  279. get_property(${proj}_EP_LABEL_${label} GLOBAL PROPERTY ${proj}_EP_LABEL_${label})
  280. list(REMOVE_DUPLICATES ${proj}_EP_LABEL_${label})
  281. _sb_append_to_cmake_args(PROJECTS ${proj}
  282. VARS ${proj}_EP_LABEL_${label}:STRING)
  283. endforeach()
  284. endif()
  285. get_property(_args GLOBAL PROPERTY ${proj}_EP_CMAKE_ARGS)
  286. foreach(var ${_args})
  287. _sb_cmakevar_to_cmakearg(${var} cmake_arg _has_cfg_intdir)
  288. set(_ep_property "CMAKE_CACHE_ARGS")
  289. if(_has_cfg_intdir)
  290. set(_ep_property "CMAKE_ARGS")
  291. endif()
  292. set_property(GLOBAL APPEND PROPERTY ${proj}_EP_PROPERTY_${_ep_property} ${cmake_arg})
  293. endforeach()
  294. endfunction()
  295. _sb_collect_args(${proj})
  296. _sb_collect_args(${_ALL_PROJECT_IDENTIFIER})
  297. set(_ep_arguments "")
  298. foreach(property CMAKE_ARGS CMAKE_CACHE_ARGS)
  299. get_property(${proj}_EP_PROPERTY_${property} GLOBAL PROPERTY ${proj}_EP_PROPERTY_${property})
  300. get_property(${_ALL_PROJECT_IDENTIFIER}_EP_PROPERTY_${property} GLOBAL PROPERTY ${_ALL_PROJECT_IDENTIFIER}_EP_PROPERTY_${property})
  301. set(_all ${${proj}_EP_PROPERTY_${property}} ${${_ALL_PROJECT_IDENTIFIER}_EP_PROPERTY_${property}})
  302. if(_all)
  303. list(APPEND _ep_arguments ${property} ${_all})
  304. endif()
  305. endforeach()
  306. list(APPEND _ep_arguments LIST_SEPARATOR ${EP_LIST_SEPARATOR})
  307. list(APPEND _ep_arguments CMAKE_GENERATOR ${_sb_CMAKE_GENERATOR})
  308. if(CMAKE_VERSION VERSION_GREATER "3.0")
  309. list(APPEND _ep_arguments CMAKE_GENERATOR_PLATFORM ${_sb_CMAKE_GENERATOR_PLATFORM})
  310. endif()
  311. list(APPEND _ep_arguments CMAKE_GENERATOR_TOOLSET ${_sb_CMAKE_GENERATOR_TOOLSET})
  312. if(CMAKE_VERSION VERSION_EQUAL "3.4" OR CMAKE_VERSION VERSION_GREATER "3.4")
  313. # USES_TERMINAL_* options were introduced in CMake 3.4
  314. foreach(step IN ITEMS DOWNLOAD UPDATE CONFIGURE BUILD TEST INSTALL)
  315. list(APPEND _ep_arguments
  316. USES_TERMINAL_${step}
  317. )
  318. endforeach()
  319. endif()
  320. set(${varname} ${_ep_arguments} PARENT_SCOPE)
  321. endfunction()
  322. function(_sb_update_indent proj)
  323. superbuild_stack_size(SB_PROJECT_STACK _stack_size)
  324. set(_indent "")
  325. if(_stack_size GREATER 0)
  326. foreach(not_used RANGE 1 ${_stack_size})
  327. set(_indent " ${_indent}")
  328. endforeach()
  329. endif()
  330. set_property(GLOBAL PROPERTY SUPERBUILD_${proj}_INDENT ${_indent})
  331. endfunction()
  332. #.rst:
  333. # .. cmake:function:: ExternalProject_Message
  334. #
  335. # .. code-block:: cmake
  336. #
  337. # ExternalProject_Message(<project_name> <msg> [condition])
  338. #
  339. function(ExternalProject_Message proj msg)
  340. set(_display 1)
  341. if(NOT "x${ARGV2}" STREQUAL "x")
  342. set(_display ${ARGN})
  343. endif()
  344. if(${_display})
  345. get_property(_indent GLOBAL PROPERTY SUPERBUILD_${proj}_INDENT)
  346. message(STATUS "SuperBuild - ${_indent}${msg}")
  347. endif()
  348. endfunction()
  349. #
  350. # superbuild_stack_content(<stack_name> <output_var>)
  351. #
  352. # <stack_name> corresponds to the name of stack.
  353. #
  354. # <output_var> is the name of CMake variable that will be set with the content
  355. # of the stack identified by <stack_name>.
  356. #
  357. function(superbuild_stack_content stack_name output_var)
  358. get_property(_stack GLOBAL PROPERTY ${stack_name})
  359. set(${output_var} ${_stack} PARENT_SCOPE)
  360. endfunction()
  361. #
  362. # superbuild_stack_size(<stack_name> <output_var>)
  363. #
  364. # <stack_name> corresponds to the name of stack.
  365. #
  366. # <output_var> is the name of CMake variable that will be set with the size
  367. # of the stack identified by <stack_name>.
  368. #
  369. function(superbuild_stack_size stack_name output_var)
  370. get_property(_stack GLOBAL PROPERTY ${stack_name})
  371. list(LENGTH _stack _stack_size)
  372. set(${output_var} ${_stack_size} PARENT_SCOPE)
  373. endfunction()
  374. #
  375. # superbuild_stack_push(<stack_name> <value>)
  376. #
  377. # <stack_name> corresponds to the name of stack.
  378. #
  379. # <value> is appended to the stack identified by <stack_name>.
  380. #
  381. function(superbuild_stack_push stack_name value)
  382. set_property(GLOBAL APPEND PROPERTY ${stack_name} ${value})
  383. endfunction()
  384. #
  385. # superbuild_stack_pop(<stack_name> <item_var>)
  386. #
  387. # <stack_name> corresponds to the name of stack.
  388. #
  389. # <item_var> names a CMake variable that will be set with the item
  390. # removed from the stack identified by <stack_name>
  391. #
  392. function(superbuild_stack_pop stack_name item_var)
  393. get_property(_stack GLOBAL PROPERTY ${stack_name})
  394. list(LENGTH _stack _stack_size)
  395. if(_stack_size GREATER 0)
  396. math(EXPR _index_to_remove "${_stack_size} - 1")
  397. list(GET _stack ${_index_to_remove} _item)
  398. list(REMOVE_AT _stack ${_index_to_remove})
  399. set_property(GLOBAL PROPERTY ${stack_name} ${_stack})
  400. set(${item_var} ${_item} PARENT_SCOPE)
  401. endif()
  402. endfunction()
  403. function(_sb_is_optional proj output_var)
  404. set(_include_project 1)
  405. if(COMMAND superbuild_is_external_project_includable)
  406. superbuild_is_external_project_includable("${proj}" _include_project)
  407. endif()
  408. set(optional 1)
  409. if(_include_project)
  410. set(optional 0)
  411. endif()
  412. set(${output_var} ${optional} PARENT_SCOPE)
  413. endfunction()
  414. #.rst:
  415. # .. cmake:function:: ExternalProject_Include_Dependencies
  416. #
  417. # .. code-block:: cmake
  418. #
  419. # ExternalProject_Include_Dependencies(<project_name>
  420. # [PROJECT_VAR <project_var>]
  421. # [EP_ARGS_VAR <external_project_args_var>]
  422. # [DEPENDS_VAR <depends_var>]
  423. # [USE_SYSTEM_VAR <use_system_var>]
  424. # [SUPERBUILD_VAR <superbuild_var>]
  425. # [CMAKE_GENERATOR <cmake_generator>]
  426. # [CMAKE_GENERATOR_PLATFORM <cmake_generator_platform>]
  427. # [CMAKE_GENERATOR_TOOLSET <cmake_generator_toolset>]
  428. # )
  429. #
  430. macro(ExternalProject_Include_Dependencies project_name)
  431. set(options)
  432. set(oneValueArgs PROJECT_VAR DEPENDS_VAR EP_ARGS_VAR USE_SYSTEM_VAR SUPERBUILD_VAR
  433. CMAKE_GENERATOR
  434. CMAKE_GENERATOR_PLATFORM
  435. CMAKE_GENERATOR_TOOLSET
  436. )
  437. set(multiValueArgs)
  438. cmake_parse_arguments(_sb "${options}" "${oneValueArgs}" "${multiValueArgs}" ${ARGN})
  439. # Sanity checks
  440. if(x${project_name} STREQUAL xPROJECT_VAR
  441. OR x${project_name} STREQUAL xEP_ARGS_VAR
  442. OR x${project_name} STREQUAL xDEPENDS_VAR
  443. OR x${project_name} STREQUAL xUSE_SYSTEM_VAR
  444. OR x${project_name} STREQUAL xSUPERBUILD_VAR
  445. OR x${project_name} STREQUAL xCMAKE_GENERATOR
  446. OR x${project_name} STREQUAL xCMAKE_GENERATOR_PLATFORM
  447. OR x${project_name} STREQUAL xCMAKE_GENERATOR_TOOLSET
  448. )
  449. message(FATAL_ERROR "Argument <project_name> is missing !")
  450. endif()
  451. if(_sb_UNPARSED_ARGUMENTS)
  452. message(FATAL_ERROR "Invalid arguments: ${_sb_UNPARSED_ARGUMENTS}")
  453. endif()
  454. # Set default for optional PROJECT_VAR parameter
  455. if(NOT _sb_PROJECT_VAR)
  456. set(_sb_PROJECT_VAR proj)
  457. set(${_sb_PROJECT_VAR} ${project_name})
  458. #message("[${project_name}] Setting _sb_PROJECT_VAR with default value '${_sb_PROJECT_VAR}'")
  459. endif()
  460. if(_sb_PROJECT_VAR AND NOT x${project_name} STREQUAL x${${_sb_PROJECT_VAR}})
  461. message(FATAL_ERROR
  462. "Argument <project_name>:${project_name} and PROJECT_VAR:${_sb_PROJECT_VAR}:${${_sb_PROJECT_VAR}} are different !")
  463. endif()
  464. set(_sb_proj ${project_name})
  465. # Skip if project already included
  466. get_property(_is_included GLOBAL PROPERTY SB_${_sb_proj}_FILE_INCLUDED)
  467. if(_is_included)
  468. return()
  469. endif()
  470. # Set default for optional DEPENDS_VAR and EP_ARGS parameters
  471. foreach(param DEPENDS EP_ARGS)
  472. if(NOT _sb_${param}_VAR)
  473. set(_sb_${param}_VAR ${_sb_proj}_${param})
  474. #message("[${project_name}] Setting _sb_${param}_VAR with default value '${_sb_${param}_VAR}'")
  475. endif()
  476. endforeach()
  477. # Set top level project
  478. superbuild_stack_size(SB_PROJECT_STACK _stack_size)
  479. if(_stack_size EQUAL 0)
  480. set(SUPERBUILD_TOPLEVEL_PROJECT ${_sb_proj})
  481. endif()
  482. # Set default for optional USE_SYSTEM_VAR parameter
  483. if(NOT _sb_USE_SYSTEM_VAR)
  484. set(_sb_USE_SYSTEM_VAR ${SUPERBUILD_TOPLEVEL_PROJECT}_USE_SYSTEM_${_sb_proj})
  485. #message("[${project_name}] Setting _sb_USE_SYSTEM_VAR with default value '${_sb_USE_SYSTEM_VAR}'")
  486. endif()
  487. # Set default for optional SUPERBUILD_VAR parameter
  488. if(NOT _sb_SUPERBUILD_VAR)
  489. set(_sb_SUPERBUILD_VAR ${SUPERBUILD_TOPLEVEL_PROJECT}_SUPERBUILD)
  490. #message("[${project_name}] Setting _sb_SUPERBUILD_VAR with default value '${_sb_SUPERBUILD_VAR}'")
  491. endif()
  492. # Set default for optional CMAKE_GENERATOR_* parameters
  493. foreach(varname IN ITEMS
  494. "CMAKE_GENERATOR"
  495. "CMAKE_GENERATOR_PLATFORM"
  496. "CMAKE_GENERATOR_TOOLSET"
  497. )
  498. if(NOT _sb_${varname})
  499. set(_sb_${varname} ${EP_${varname}})
  500. #message("[${project_name}] Setting _sb_${varname} with default value '${_sb_${varname}}'")
  501. else()
  502. #message("[${project_name}] Setting _sb_${varname} to value '${_sb_${varname}}'")
  503. endif()
  504. endforeach()
  505. # Keeping track of variable name independently of the recursion
  506. if(NOT DEFINED _sb_SB_VAR)
  507. set(_sb_SB_VAR ${_sb_SUPERBUILD_VAR})
  508. #message("[${project_name}] Setting _sb_SB_VAR with default value '${_sb_SB_VAR}'")
  509. endif()
  510. # Set local variables
  511. set(_sb_DEPENDS ${${_sb_DEPENDS_VAR}})
  512. set(_sb_USE_SYSTEM ${${_sb_USE_SYSTEM_VAR}})
  513. _sb_update_indent(${_sb_proj})
  514. # Keep track of the projects
  515. list(APPEND SB_${SUPERBUILD_TOPLEVEL_PROJECT}_POSSIBLE_DEPENDS ${_sb_proj})
  516. # Use system ?
  517. get_property(_use_system_set GLOBAL PROPERTY SB_${_sb_proj}_USE_SYSTEM SET)
  518. if(_use_system_set)
  519. get_property(_sb_USE_SYSTEM GLOBAL PROPERTY SB_${_sb_proj}_USE_SYSTEM)
  520. endif()
  521. # Is this the first run ?
  522. if(${_sb_proj} STREQUAL ${SUPERBUILD_TOPLEVEL_PROJECT} AND NOT DEFINED SB_FIRST_PASS)
  523. message(STATUS "SuperBuild - First pass")
  524. set(SB_FIRST_PASS TRUE)
  525. endif()
  526. set(_sb_REQUIRED_DEPENDS)
  527. foreach(dep ${_sb_DEPENDS})
  528. if(NOT ${_sb_proj} STREQUAL ${SUPERBUILD_TOPLEVEL_PROJECT})
  529. # Set "use system" variable if it has NOT already been explicitly set
  530. get_property(_sb_${dep}_USE_SYSTEM_VAR GLOBAL PROPERTY SB_${dep}_USE_SYSTEM_VAR)
  531. if(_sb_USE_SYSTEM AND NOT DEFINED ${_sb_${dep}_USE_SYSTEM_VAR})
  532. set_property(GLOBAL PROPERTY SB_${dep}_USE_SYSTEM ${_sb_USE_SYSTEM})
  533. #message(${_sb_proj} "Property SB_${dep}_USE_SYSTEM set to [${_sb_USE_SYSTEM_VAR}:${_sb_USE_SYSTEM}]")
  534. endif()
  535. endif()
  536. _sb_is_optional(${dep} _optional)
  537. set_property(GLOBAL PROPERTY SB_${dep}_OPTIONAL ${_optional})
  538. #message(${_sb_proj} "[${_sb_proj}] Property SB_${dep}_OPTIONAL set to ${_optional}")
  539. if(NOT _optional)
  540. list(APPEND _sb_REQUIRED_DEPENDS ${dep})
  541. endif()
  542. endforeach()
  543. # Display dependency of project being processed
  544. if(_sb_REQUIRED_DEPENDS AND SB_SECOND_PASS AND ${_sb_SB_VAR})
  545. set(dependency_str "")
  546. foreach(dep ${_sb_REQUIRED_DEPENDS})
  547. get_property(_is_included GLOBAL PROPERTY SB_${dep}_FILE_INCLUDED)
  548. set(_include_status "")
  549. if(_is_included)
  550. set(_include_status "[INCLUDED]")
  551. endif()
  552. set(dependency_str "${dependency_str}${dep}${_include_status}, ")
  553. endforeach()
  554. ExternalProject_Message(${_sb_proj} "${_sb_proj} => Requires ${dependency_str}")
  555. endif()
  556. # Save variables
  557. set_property(GLOBAL PROPERTY SB_${_sb_proj}_REQUIRED_DEPENDS ${_sb_REQUIRED_DEPENDS})
  558. set_property(GLOBAL PROPERTY SB_${_sb_proj}_DEPENDS ${_sb_DEPENDS})
  559. set_property(GLOBAL PROPERTY SB_${_sb_proj}_DEPENDS_VAR ${_sb_DEPENDS_VAR})
  560. set_property(GLOBAL PROPERTY SB_${_sb_proj}_EP_ARGS_VAR ${_sb_EP_ARGS_VAR})
  561. set_property(GLOBAL PROPERTY SB_${_sb_proj}_USE_SYSTEM ${_sb_USE_SYSTEM})
  562. set_property(GLOBAL PROPERTY SB_${_sb_proj}_USE_SYSTEM_VAR ${_sb_USE_SYSTEM_VAR})
  563. set_property(GLOBAL PROPERTY SB_${_sb_proj}_PROJECT_VAR ${_sb_PROJECT_VAR})
  564. foreach(varname IN ITEMS
  565. "CMAKE_GENERATOR"
  566. "CMAKE_GENERATOR_PLATFORM"
  567. "CMAKE_GENERATOR_TOOLSET"
  568. )
  569. set_property(GLOBAL PROPERTY SB_${_sb_proj}_${varname} ${_sb_${varname}})
  570. endforeach()
  571. superbuild_stack_push(SB_PROJECT_STACK ${_sb_proj})
  572. # Include dependencies
  573. foreach(dep ${_sb_DEPENDS})
  574. get_property(_included GLOBAL PROPERTY SB_${dep}_FILE_INCLUDED)
  575. if(NOT _included)
  576. # XXX - Refactor - Add a single variable named 'EXTERNAL_PROJECT_DIRS'
  577. if(EXISTS "${EXTERNAL_PROJECT_DIR}/${EXTERNAL_PROJECT_FILE_PREFIX}${dep}.cmake")
  578. include(${EXTERNAL_PROJECT_DIR}/${EXTERNAL_PROJECT_FILE_PREFIX}${dep}.cmake)
  579. elseif(EXISTS "${${dep}_FILEPATH}")
  580. include(${${dep}_FILEPATH})
  581. elseif(EXISTS "${EXTERNAL_PROJECT_ADDITIONAL_DIR}/${EXTERNAL_PROJECT_FILE_PREFIX}${dep}.cmake")
  582. include(${EXTERNAL_PROJECT_ADDITIONAL_DIR}/${EXTERNAL_PROJECT_FILE_PREFIX}${dep}.cmake)
  583. else()
  584. message(FATAL_ERROR "Can't find ${EXTERNAL_PROJECT_FILE_PREFIX}${dep}.cmake")
  585. endif()
  586. set_property(GLOBAL PROPERTY SB_${dep}_FILE_INCLUDED 1)
  587. endif()
  588. endforeach()
  589. # Restore variables
  590. superbuild_stack_pop(SB_PROJECT_STACK _sb_proj)
  591. foreach(varname IN ITEMS
  592. "CMAKE_GENERATOR"
  593. "CMAKE_GENERATOR_PLATFORM"
  594. "CMAKE_GENERATOR_TOOLSET"
  595. )
  596. get_property(_sb_${varname} GLOBAL PROPERTY SB_${_sb_proj}_${varname})
  597. endforeach()
  598. get_property(_sb_PROJECT_VAR GLOBAL PROPERTY SB_${_sb_proj}_PROJECT_VAR)
  599. get_property(_sb_USE_SYSTEM_VAR GLOBAL PROPERTY SB_${_sb_proj}_USE_SYSTEM_VAR)
  600. get_property(_sb_USE_SYSTEM GLOBAL PROPERTY SB_${_sb_proj}_USE_SYSTEM)
  601. get_property(_sb_EP_ARGS_VAR GLOBAL PROPERTY SB_${_sb_proj}_EP_ARGS_VAR)
  602. get_property(_sb_DEPENDS_VAR GLOBAL PROPERTY SB_${_sb_proj}_DEPENDS_VAR)
  603. get_property(_sb_DEPENDS GLOBAL PROPERTY SB_${_sb_proj}_DEPENDS)
  604. get_property(_sb_REQUIRED_DEPENDS GLOBAL PROPERTY SB_${_sb_proj}_REQUIRED_DEPENDS)
  605. # Use system ?
  606. set(_include_type "")
  607. if(_sb_USE_SYSTEM)
  608. set(_include_type " (SYSTEM)")
  609. endif()
  610. get_property(_optional GLOBAL PROPERTY SB_${_sb_proj}_OPTIONAL)
  611. ExternalProject_Message(${_sb_proj} "${_sb_proj}[OK]${_include_type}" SB_SECOND_PASS AND ${_sb_SB_VAR} AND NOT _optional)
  612. if(${_sb_proj} STREQUAL ${SUPERBUILD_TOPLEVEL_PROJECT} AND SB_FIRST_PASS)
  613. set(SB_FIRST_PASS FALSE)
  614. ExternalProject_Message(${_sb_proj} "First pass - done")
  615. if(${_sb_SB_VAR})
  616. foreach(possible_proj ${SB_${SUPERBUILD_TOPLEVEL_PROJECT}_POSSIBLE_DEPENDS})
  617. get_property(_optional GLOBAL PROPERTY SB_${possible_proj}_OPTIONAL)
  618. if(_optional)
  619. ExternalProject_Message(${_sb_proj} "${possible_proj}[OPTIONAL]")
  620. endif()
  621. set_property(GLOBAL PROPERTY SB_${possible_proj}_FILE_INCLUDED 0)
  622. endforeach()
  623. set(${_sb_PROJECT_VAR} ${_sb_proj})
  624. set(SB_SECOND_PASS TRUE)
  625. set(_ep_include_deps_EXTRA_ARGS )
  626. foreach(varname IN ITEMS
  627. "CMAKE_GENERATOR"
  628. "CMAKE_GENERATOR_PLATFORM"
  629. "CMAKE_GENERATOR_TOOLSET"
  630. )
  631. list(APPEND _ep_include_deps_EXTRA_ARGS
  632. ${varname} ${_sb_${varname}}
  633. )
  634. endforeach()
  635. ExternalProject_Include_Dependencies(${_sb_proj}
  636. PROJECT_VAR ${_sb_PROJECT_VAR}
  637. DEPENDS_VAR ${_sb_DEPENDS_VAR}
  638. EP_ARGS_VAR ${_sb_EP_ARGS_VAR}
  639. USE_SYSTEM_VAR _sb_USE_SYSTEM
  640. SUPERBUILD_VAR ${_sb_SB_VAR}
  641. ${_ep_include_deps_EXTRA_ARGS}
  642. )
  643. set(SB_SECOND_PASS FALSE)
  644. endif()
  645. endif()
  646. if(SB_FIRST_PASS OR _optional)
  647. if(NOT ${_sb_proj} STREQUAL ${SUPERBUILD_TOPLEVEL_PROJECT})
  648. return()
  649. endif()
  650. endif()
  651. if(SB_SECOND_PASS)
  652. _sb_get_external_project_arguments(${_sb_proj} ${_sb_EP_ARGS_VAR})
  653. endif()
  654. if(NOT SB_FIRST_PASS AND NOT SB_SECOND_PASS
  655. AND ${_sb_proj} STREQUAL ${SUPERBUILD_TOPLEVEL_PROJECT})
  656. #ExternalProject_Message(${_sb_proj} "Clean up")
  657. unset(_sb_SB_VAR)
  658. unset(SB_FIRST_PASS)
  659. unset(SB_SECOND_PASS)
  660. endif()
  661. # Set public variables
  662. set(${_sb_PROJECT_VAR} ${_sb_proj})
  663. set(${_sb_DEPENDS_VAR} ${_sb_REQUIRED_DEPENDS})
  664. set(${_sb_USE_SYSTEM_VAR} ${_sb_USE_SYSTEM})
  665. #message("[${_sb_proj}] #################################")
  666. #message("[${_sb_proj}] Setting ${_sb_PROJECT_VAR}:${_sb_proj}")
  667. #message("[${_sb_proj}] Setting ${_sb_EP_ARGS_VAR}:${${_sb_EP_ARGS_VAR}}")
  668. #message("[${_sb_proj}] Setting ${_sb_DEPENDS_VAR}:${${_sb_DEPENDS_VAR}}")
  669. #message("[${_sb_proj}] Setting ${_sb_USE_SYSTEM_VAR}:${_sb_USE_SYSTEM}")
  670. endmacro()
  671. #.rst:
  672. # .. cmake:function:: ExternalProject_Add_Empty
  673. #
  674. # .. code-block:: cmake
  675. #
  676. # ExternalProject_Add_Empty(<project_name>
  677. # DEPENDS <depends>
  678. # )
  679. #
  680. macro(ExternalProject_Add_Empty project_name)
  681. set(options)
  682. set(oneValueArgs)
  683. set(multiValueArgs DEPENDS)
  684. cmake_parse_arguments(_sb "${options}" "${oneValueArgs}" "${multiValueArgs}" ${ARGN})
  685. # Sanity checks
  686. if(x${project_name} STREQUAL xDEPENDS)
  687. message(FATAL_ERROR "Argument <project_name> is missing !")
  688. endif()
  689. if(_sb_UNPARSED_ARGUMENTS)
  690. message(FATAL_ERROR "Invalid arguments: ${_sb_UNPARSED_ARGUMENTS}")
  691. endif()
  692. ExternalProject_Add(${project_name}
  693. SOURCE_DIR ${CMAKE_BINARY_DIR}/${project_name}
  694. BINARY_DIR ${project_name}-build
  695. DOWNLOAD_COMMAND ""
  696. CONFIGURE_COMMAND ""
  697. BUILD_COMMAND ""
  698. INSTALL_COMMAND ""
  699. DEPENDS ${_sb_DEPENDS}
  700. )
  701. endmacro()
  702. #.rst:
  703. # .. cmake:function:: ExternalProject_Install_CMake
  704. #
  705. # Install an external CMake-based project as part of the ``install`` target.
  706. #
  707. # .. code-block:: cmake
  708. #
  709. # ExternalProject_Install_CMake(<project_name>)
  710. #
  711. # This causes building the main project's ``install`` target to also execute
  712. # the CMake install script for the specified external project. The project must
  713. # be previously declared with :command:`ExternalProject_Add`.
  714. #
  715. function(ExternalProject_Install_CMake project_name)
  716. ExternalProject_Get_Property(${project_name} binary_dir)
  717. install(SCRIPT ${binary_dir}/cmake_install.cmake)
  718. endfunction()
  719. #.rst:
  720. # .. cmake:function:: ExternalProject_SetIfNotDefined
  721. #
  722. # Set a variable to its default value if not already defined.
  723. #
  724. # .. code-block:: cmake
  725. #
  726. # ExternalProject_SetIfNotDefined(<var> <defaultvalue> [OBFUSCATE] [QUIET])
  727. #
  728. # The default value is set with:
  729. # (1) if set, the value environment variable <var>.
  730. # (2) if set, the value of local variable variable <var>.
  731. # (3) if none of the above, the value passed as a parameter.
  732. #
  733. # Setting the optional parameter 'OBFUSCATE' will display 'OBFUSCATED' instead of the real value.
  734. # Setting the optional parameter 'QUIET' will not display any message.
  735. macro(ExternalProject_SetIfNotDefined var defaultvalue)
  736. set(_obfuscate FALSE)
  737. set(_quiet FALSE)
  738. foreach(arg ${ARGN})
  739. if(arg STREQUAL "OBFUSCATE")
  740. set(_obfuscate TRUE)
  741. endif()
  742. if(arg STREQUAL "QUIET")
  743. set(_quiet TRUE)
  744. endif()
  745. endforeach()
  746. if(DEFINED ENV{${var}} AND NOT DEFINED ${var})
  747. set(_value "$ENV{${var}}")
  748. if(_obfuscate)
  749. set(_value "OBFUSCATED")
  750. endif()
  751. if(NOT _quiet)
  752. message(STATUS "Setting '${var}' variable with environment variable value '${_value}'")
  753. endif()
  754. set(${var} $ENV{${var}})
  755. endif()
  756. if(NOT DEFINED ${var})
  757. set(_value "${defaultvalue}")
  758. if(_obfuscate)
  759. set(_value "OBFUSCATED")
  760. endif()
  761. if(NOT _quiet)
  762. message(STATUS "Setting '${var}' variable with default value '${_value}'")
  763. endif()
  764. set(${var} "${defaultvalue}")
  765. endif()
  766. endmacro()