CheckSubmodules.cmake 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226
  1. # Utility for validating and, if needed, cloning all submodules
  2. #
  3. # Looks for a .gitmodules in the root project folder
  4. # Loops over all modules looking well-known configure/build scripts
  5. #
  6. # Usage:
  7. # INCLUDE(CheckSubmodules)
  8. #
  9. # Options:
  10. # SET(PLUGIN_LIST "zynaddsubfx;...") # skips submodules for plugins not explicitely listed
  11. #
  12. # Or via command line:
  13. # cmake -PLUGIN_LIST=foo;bar
  14. #
  15. # Copyright (c) 2019, Tres Finocchiaro, <tres.finocchiaro@gmail.com>
  16. #
  17. # Redistribution and use is allowed according to the terms of the BSD license.
  18. # For details see the accompanying COPYING-CMAKE-SCRIPTS file.
  19. # Files which confirm a successful clone
  20. SET(VALID_CRUMBS "CMakeLists.txt;Makefile;Makefile.in;Makefile.am;configure.ac;configure.py;autogen.sh;.gitignore;LICENSE;Home.md")
  21. OPTION(NO_SHALLOW_CLONE "Disable shallow cloning of submodules" OFF)
  22. # Try and use the specified shallow clone on submodules, if supported
  23. SET(DEPTH_VALUE 100)
  24. # Number of times git commands will retry before failing
  25. SET(MAX_ATTEMPTS 2)
  26. MESSAGE("\nChecking submodules...")
  27. IF(NOT EXISTS "${CMAKE_SOURCE_DIR}/.gitmodules")
  28. MESSAGE("Skipping the check because .gitmodules not detected."
  29. "Please make sure you have all submodules in the source tree!"
  30. )
  31. RETURN()
  32. ENDIF()
  33. FILE(READ "${CMAKE_SOURCE_DIR}/.gitmodules" SUBMODULE_DATA)
  34. # Force English locale
  35. SET(LC_ALL_BACKUP "$ENV{LC_ALL}")
  36. SET(LANG_BACKUP "$ENV{LANG}")
  37. SET(ENV{LC_ALL} "C")
  38. SET(ENV{LANG} "en_US")
  39. # Submodule list pairs, unparsed (WARNING: Assumes alpha-numeric paths)
  40. STRING(REGEX MATCHALL "path = [-0-9A-Za-z/]+" SUBMODULE_LIST_RAW ${SUBMODULE_DATA})
  41. STRING(REGEX MATCHALL "url = [.:%-0-9A-Za-z/]+" SUBMODULE_URL_RAW ${SUBMODULE_DATA})
  42. # Submodule list pairs, parsed
  43. SET(SUBMODULE_LIST "")
  44. SET(SUBMODULE_URL "")
  45. FOREACH(_path ${SUBMODULE_LIST_RAW})
  46. # Parse SUBMODULE_PATH
  47. STRING(REPLACE "path = " "" SUBMODULE_PATH "${_path}")
  48. # Grab index for matching SUBMODULE_URL
  49. LIST(FIND SUBMODULE_LIST_RAW "${_path}" SUBMODULE_INDEX)
  50. LIST(GET SUBMODULE_URL_RAW ${SUBMODULE_INDEX} _url)
  51. # Parse SUBMODULE_URL
  52. STRING(REPLACE "url = " "" SUBMODULE_URL "${_url}")
  53. SET(SKIP false)
  54. # Loop over skipped plugins, add to SKIP_SUBMODULES (e.g. -DPLUGIN_LIST=foo;bar)
  55. IF(${SUBMODULE_PATH} MATCHES "^plugins/")
  56. SET(REMOVE_PLUGIN true)
  57. FOREACH(_plugin ${PLUGIN_LIST})
  58. IF(_plugin STREQUAL "")
  59. CONTINUE()
  60. ENDIF()
  61. IF(${SUBMODULE_PATH} MATCHES "${_plugin}")
  62. SET(REMOVE_PLUGIN false)
  63. ENDIF()
  64. ENDFOREACH()
  65. IF(REMOVE_PLUGIN)
  66. LIST(APPEND SKIP_SUBMODULES "${SUBMODULE_PATH}")
  67. ENDIF()
  68. ENDIF()
  69. # Finally, loop and mark "SKIP" on match
  70. IF(SKIP_SUBMODULES)
  71. FOREACH(_skip ${SKIP_SUBMODULES})
  72. IF("${SUBMODULE_PATH}" MATCHES "${_skip}")
  73. MESSAGE("-- Skipping ${SUBMODULE_PATH} matches \"${_skip}\" (absent in PLUGIN_LIST)")
  74. SET(SKIP true)
  75. BREAK()
  76. ENDIF()
  77. ENDFOREACH()
  78. ENDIF()
  79. IF(NOT SKIP)
  80. LIST(APPEND SUBMODULE_LIST "${SUBMODULE_PATH}")
  81. LIST(APPEND SUBMODULE_URL "${SUBMODULE_URL}")
  82. ENDIF()
  83. ENDFOREACH()
  84. # Once called, status is stored in GIT_RESULT respectively.
  85. # Note: Git likes to write to stderr. Don't assume stderr is error; Check GIT_RESULT instead.
  86. MACRO(GIT_SUBMODULE SUBMODULE_PATH FORCE_DEINIT FORCE_REMOTE FULL_CLONE)
  87. FIND_PACKAGE(Git REQUIRED)
  88. # Handle missing commits
  89. SET(FORCE_REMOTE_FLAG "${FORCE_REMOTE}")
  90. SET(FULL_CLONE_FLAG "${FULL_CLONE}")
  91. IF(FORCE_REMOTE_FLAG)
  92. MESSAGE("-- Adding remote submodulefix to ${SUBMODULE_PATH}")
  93. EXECUTE_PROCESS(
  94. COMMAND "${GIT_EXECUTABLE}" remote rm submodulefix
  95. COMMAND "${GIT_EXECUTABLE}" remote add submodulefix ${FORCE_REMOTE}
  96. COMMAND "${GIT_EXECUTABLE}" fetch submodulefix
  97. WORKING_DIRECTORY "${CMAKE_SOURCE_DIR}/${SUBMODULE_PATH}"
  98. OUTPUT_QUIET ERROR_QUIET
  99. )
  100. # Recurse
  101. GIT_SUBMODULE(${SUBMODULE_PATH} false false ${FULL_CLONE_FLAG})
  102. ELSEIF(${FORCE_DEINIT})
  103. MESSAGE("-- Resetting ${SUBMODULE_PATH}")
  104. EXECUTE_PROCESS(
  105. COMMAND "${GIT_EXECUTABLE}" submodule deinit -f "${CMAKE_SOURCE_DIR}/${SUBMODULE_PATH}"
  106. WORKING_DIRECTORY "${CMAKE_SOURCE_DIR}"
  107. OUTPUT_QUIET
  108. )
  109. MESSAGE("-- Deleting ${CMAKE_SOURCE_DIR}/.git/${SUBMODULE_PATH}")
  110. FILE(REMOVE_RECURSE "${CMAKE_SOURCE_DIR}/.git/modules/${SUBMODULE_PATH}")
  111. # Recurse
  112. GIT_SUBMODULE(${SUBMODULE_PATH} false false true)
  113. ELSE()
  114. # Try to use the depth switch
  115. IF(NO_SHALLOW_CLONE OR GIT_VERSION_STRING VERSION_LESS "1.8.4")
  116. # Shallow submodules were introduced in 1.8.4
  117. MESSAGE("-- Fetching ${SUBMODULE_PATH}")
  118. SET(DEPTH_CMD "")
  119. SET(DEPTH_VAL "")
  120. ELSEIF(FULL_CLONE_FLAG)
  121. # Depth doesn't revert easily... It should be "--no-recommend-shallow"
  122. # but it's ignored by nested submodules, use the highest value instead.
  123. MESSAGE("-- Fetching ${SUBMODULE_PATH}")
  124. SET(DEPTH_CMD "--depth")
  125. SET(DEPTH_VAL "2147483647")
  126. ELSE()
  127. MESSAGE("-- Fetching ${SUBMODULE_PATH} @ --depth ${DEPTH_VALUE}")
  128. SET(DEPTH_CMD "--depth")
  129. SET(DEPTH_VAL "${DEPTH_VALUE}")
  130. ENDIF()
  131. EXECUTE_PROCESS(
  132. COMMAND "${GIT_EXECUTABLE}" submodule update --init --recursive ${DEPTH_CMD} ${DEPTH_VAL} "${CMAKE_SOURCE_DIR}/${SUBMODULE_PATH}"
  133. WORKING_DIRECTORY "${CMAKE_SOURCE_DIR}"
  134. RESULT_VARIABLE GIT_RESULT
  135. OUTPUT_VARIABLE GIT_STDOUT
  136. ERROR_VARIABLE GIT_STDERR
  137. )
  138. SET(GIT_MESSAGE "${GIT_STDOUT}${GIT_STDERR}")
  139. MESSAGE("${GIT_MESSAGE}")
  140. ENDIF()
  141. ENDMACRO()
  142. SET(MISSING_COMMIT_PHRASES "no such remote ref;reference is not a tree;unadvertised object")
  143. SET(RETRY_PHRASES "Failed to recurse;cannot create directory;already exists;${MISSING_COMMIT_PHRASES}")
  144. # Attempt to do lazy clone
  145. FOREACH(_submodule ${SUBMODULE_LIST})
  146. STRING(REPLACE "/" ";" PATH_PARTS "${_submodule}")
  147. LIST(REVERSE PATH_PARTS)
  148. LIST(GET PATH_PARTS 0 SUBMODULE_NAME)
  149. MESSAGE("-- Checking ${SUBMODULE_NAME}...")
  150. SET(CRUMB_FOUND false)
  151. FOREACH(_crumb ${VALID_CRUMBS})
  152. IF(EXISTS "${CMAKE_SOURCE_DIR}/${_submodule}/${_crumb}")
  153. SET(CRUMB_FOUND true)
  154. MESSAGE("-- Found ${_submodule}/${_crumb}")
  155. BREAK()
  156. ENDIF()
  157. ENDFOREACH()
  158. IF(NOT CRUMB_FOUND)
  159. GIT_SUBMODULE("${_submodule}" false false false)
  160. SET(COUNTED 0)
  161. # Handle edge-cases where submodule didn't clone properly or re-uses a non-empty directory
  162. WHILE(NOT GIT_RESULT EQUAL 0 AND COUNTED LESS MAX_ATTEMPTS)
  163. MATH(EXPR COUNTED "${COUNTED}+1")
  164. SET(MISSING_COMMIT false)
  165. FOREACH(_phrase ${MISSING_COMMIT_PHRASES})
  166. IF("${GIT_MESSAGE}" MATCHES "${_phrase}")
  167. SET(MISSING_COMMIT true)
  168. BREAK()
  169. ENDIF()
  170. ENDFOREACH()
  171. FOREACH(_phrase ${RETRY_PHRASES})
  172. IF(${MISSING_COMMIT} AND COUNTED LESS 2)
  173. LIST(FIND SUBMODULE_LIST ${_submodule} SUBMODULE_INDEX)
  174. LIST(GET SUBMODULE_URL_LIST ${SUBMODULE_INDEX} SUBMODULE_URL)
  175. MESSAGE("-- Retrying ${_submodule} using 'remote add submodulefix' (attempt ${COUNTED} of ${MAX_ATTEMPTS})...")
  176. GIT_SUBMODULE("${_submodule}" false "${SUBMODULE_URL}" false)
  177. BREAK()
  178. ELSEIF("${GIT_MESSAGE}" MATCHES "${_phrase}")
  179. MESSAGE("-- Retrying ${_submodule} using 'deinit' (attempt ${COUNTED} of ${MAX_ATTEMPTS})...")
  180. IF(COUNTED LESS 2)
  181. SET(FULL_CLONE false)
  182. ELSE()
  183. SET(FULL_CLONE true)
  184. ENDIF()
  185. GIT_SUBMODULE("${_submodule}" true false ${FULL_CLONE})
  186. BREAK()
  187. ENDIF()
  188. ENDFOREACH()
  189. ENDWHILE()
  190. IF(NOT GIT_RESULT EQUAL 0)
  191. MESSAGE(FATAL_ERROR "${GIT_EXECUTABLE} exited with status of ${GIT_RESULT}")
  192. ENDIF()
  193. ENDIF()
  194. ENDFOREACH()
  195. MESSAGE("-- Done validating submodules.\n")
  196. # Reset locale
  197. SET(ENV{LC_ALL} "${LC_ALL_BACKUP}")
  198. SET(ENV{LANG} "${LANG_BACKUP}")