CMakeIOSInstallCombined.cmake 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307
  1. #=============================================================================
  2. # Copyright 2014-2015 Ruslan Baratov, Gregor Jasny
  3. #
  4. # Distributed under the OSI-approved BSD License (the "License");
  5. # see accompanying file Copyright.txt for details.
  6. #
  7. # This software is distributed WITHOUT ANY WARRANTY; without even the
  8. # implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  9. # See the License for more information.
  10. #=============================================================================
  11. # (To distribute this file outside of CMake, substitute the full
  12. # License text for the above reference.)
  13. # Function to print messages of this module
  14. function(_ios_install_combined_message)
  15. message("[iOS combined] " ${ARGN})
  16. endfunction()
  17. # Get build settings for the current target/config/SDK by running
  18. # `xcodebuild -sdk ... -showBuildSettings` and parsing it's output
  19. function(_ios_install_combined_get_build_setting sdk variable resultvar)
  20. if("${sdk}" STREQUAL "")
  21. message(FATAL_ERROR "`sdk` is empty")
  22. endif()
  23. if("${variable}" STREQUAL "")
  24. message(FATAL_ERROR "`variable` is empty")
  25. endif()
  26. if("${resultvar}" STREQUAL "")
  27. message(FATAL_ERROR "`resultvar` is empty")
  28. endif()
  29. set(
  30. cmd
  31. xcodebuild -showBuildSettings
  32. -sdk "${sdk}"
  33. -target "${CURRENT_TARGET}"
  34. -config "${CURRENT_CONFIG}"
  35. )
  36. execute_process(
  37. COMMAND ${cmd}
  38. WORKING_DIRECTORY "${CMAKE_BINARY_DIR}"
  39. RESULT_VARIABLE result
  40. OUTPUT_VARIABLE output
  41. )
  42. if(NOT result EQUAL 0)
  43. message(FATAL_ERROR "Command failed (${result}): ${cmd}")
  44. endif()
  45. if(NOT output MATCHES " ${variable} = ([^\n]*)")
  46. if("${variable}" STREQUAL "VALID_ARCHS")
  47. # VALID_ARCHS may be unset by user for given SDK
  48. # (e.g. for build without simulator).
  49. set("${resultvar}" "" PARENT_SCOPE)
  50. return()
  51. else()
  52. message(FATAL_ERROR "${variable} not found.")
  53. endif()
  54. endif()
  55. set("${resultvar}" "${CMAKE_MATCH_1}" PARENT_SCOPE)
  56. endfunction()
  57. # Get architectures of given SDK (iphonesimulator/iphoneos)
  58. function(_ios_install_combined_get_valid_archs sdk resultvar)
  59. cmake_policy(SET CMP0007 NEW)
  60. if("${resultvar}" STREQUAL "")
  61. message(FATAL_ERROR "`resultvar` is empty")
  62. endif()
  63. _ios_install_combined_get_build_setting("${sdk}" "VALID_ARCHS" valid_archs)
  64. separate_arguments(valid_archs)
  65. list(REMOVE_ITEM valid_archs "") # remove empty elements
  66. list(REMOVE_DUPLICATES valid_archs)
  67. string(REPLACE ";" " " printable "${valid_archs}")
  68. _ios_install_combined_message("Architectures (${sdk}): ${printable}")
  69. set("${resultvar}" "${valid_archs}" PARENT_SCOPE)
  70. endfunction()
  71. # Final target can contain more architectures that specified by SDK. This
  72. # function will run 'lipo -info' and parse output. Result will be returned
  73. # as a CMake list.
  74. function(_ios_install_combined_get_real_archs filename resultvar)
  75. set(cmd "${_lipo_path}" -info "${filename}")
  76. execute_process(
  77. COMMAND ${cmd}
  78. RESULT_VARIABLE result
  79. OUTPUT_VARIABLE output
  80. ERROR_VARIABLE output
  81. OUTPUT_STRIP_TRAILING_WHITESPACE
  82. ERROR_STRIP_TRAILING_WHITESPACE
  83. )
  84. if(NOT result EQUAL 0)
  85. message(
  86. FATAL_ERROR "Command failed (${result}): ${cmd}\n\nOutput:\n${output}"
  87. )
  88. endif()
  89. if(NOT output MATCHES "(Architectures in the fat file: [^\n]+ are|Non-fat file: [^\n]+ is architecture): ([^\n]*)")
  90. message(FATAL_ERROR "Could not detect architecture from: ${output}")
  91. endif()
  92. separate_arguments(CMAKE_MATCH_2)
  93. set(${resultvar} ${CMAKE_MATCH_2} PARENT_SCOPE)
  94. endfunction()
  95. # Run build command for the given SDK
  96. function(_ios_install_combined_build sdk)
  97. if("${sdk}" STREQUAL "")
  98. message(FATAL_ERROR "`sdk` is empty")
  99. endif()
  100. _ios_install_combined_message("Build `${CURRENT_TARGET}` for `${sdk}`")
  101. execute_process(
  102. COMMAND
  103. "${CMAKE_COMMAND}"
  104. --build
  105. .
  106. --target "${CURRENT_TARGET}"
  107. --config ${CURRENT_CONFIG}
  108. --
  109. -sdk "${sdk}"
  110. WORKING_DIRECTORY "${CMAKE_BINARY_DIR}"
  111. RESULT_VARIABLE result
  112. )
  113. if(NOT result EQUAL 0)
  114. message(FATAL_ERROR "Build failed")
  115. endif()
  116. endfunction()
  117. # Remove given architecture from file. This step needed only in rare cases
  118. # when target was built in "unusual" way. Emit warning message.
  119. function(_ios_install_combined_remove_arch lib arch)
  120. _ios_install_combined_message(
  121. "Warning! Unexpected architecture `${arch}` detected and will be removed "
  122. "from file `${lib}`")
  123. set(cmd "${_lipo_path}" -remove ${arch} -output ${lib} ${lib})
  124. execute_process(
  125. COMMAND ${cmd}
  126. RESULT_VARIABLE result
  127. OUTPUT_VARIABLE output
  128. ERROR_VARIABLE output
  129. OUTPUT_STRIP_TRAILING_WHITESPACE
  130. ERROR_STRIP_TRAILING_WHITESPACE
  131. )
  132. if(NOT result EQUAL 0)
  133. message(
  134. FATAL_ERROR "Command failed (${result}): ${cmd}\n\nOutput:\n${output}"
  135. )
  136. endif()
  137. endfunction()
  138. # Check that 'lib' contains only 'archs' architectures (remove others).
  139. function(_ios_install_combined_keep_archs lib archs)
  140. _ios_install_combined_get_real_archs("${lib}" real_archs)
  141. set(archs_to_remove ${real_archs})
  142. list(REMOVE_ITEM archs_to_remove ${archs})
  143. foreach(x ${archs_to_remove})
  144. _ios_install_combined_remove_arch("${lib}" "${x}")
  145. endforeach()
  146. endfunction()
  147. function(_ios_install_combined_detect_sdks this_sdk_var corr_sdk_var)
  148. cmake_policy(SET CMP0057 NEW)
  149. set(this_sdk "$ENV{PLATFORM_NAME}")
  150. if("${this_sdk}" STREQUAL "")
  151. message(FATAL_ERROR "Environment variable PLATFORM_NAME is empty")
  152. endif()
  153. set(all_platforms "$ENV{SUPPORTED_PLATFORMS}")
  154. if("${all_platforms}" STREQUAL "")
  155. message(FATAL_ERROR "Environment variable SUPPORTED_PLATFORMS is empty")
  156. endif()
  157. separate_arguments(all_platforms)
  158. if(NOT this_sdk IN_LIST all_platforms)
  159. message(FATAL_ERROR "`${this_sdk}` not found in `${all_platforms}`")
  160. endif()
  161. list(REMOVE_ITEM all_platforms "" "${this_sdk}")
  162. list(LENGTH all_platforms all_platforms_length)
  163. if(NOT all_platforms_length EQUAL 1)
  164. message(FATAL_ERROR "Expected one element: ${all_platforms}")
  165. endif()
  166. set(${this_sdk_var} "${this_sdk}" PARENT_SCOPE)
  167. set(${corr_sdk_var} "${all_platforms}" PARENT_SCOPE)
  168. endfunction()
  169. # Create combined binary for the given target.
  170. #
  171. # Preconditions:
  172. # * Target already installed at ${destination}
  173. # for the ${PLATFORM_NAME} platform
  174. #
  175. # This function will:
  176. # * Run build for the lacking platform, i.e. opposite to the ${PLATFORM_NAME}
  177. # * Fuse both libraries by running lipo
  178. function(ios_install_combined target destination)
  179. if("${target}" STREQUAL "")
  180. message(FATAL_ERROR "`target` is empty")
  181. endif()
  182. if("${destination}" STREQUAL "")
  183. message(FATAL_ERROR "`destination` is empty")
  184. endif()
  185. if(NOT IS_ABSOLUTE "${destination}")
  186. message(FATAL_ERROR "`destination` is not absolute: ${destination}")
  187. endif()
  188. if(IS_DIRECTORY "${destination}" OR IS_SYMLINK "${destination}")
  189. message(FATAL_ERROR "`destination` is no regular file: ${destination}")
  190. endif()
  191. if("${CMAKE_BINARY_DIR}" STREQUAL "")
  192. message(FATAL_ERROR "`CMAKE_BINARY_DIR` is empty")
  193. endif()
  194. if(NOT IS_DIRECTORY "${CMAKE_BINARY_DIR}")
  195. message(FATAL_ERROR "Is not a directory: ${CMAKE_BINARY_DIR}")
  196. endif()
  197. if("${CMAKE_INSTALL_CONFIG_NAME}" STREQUAL "")
  198. message(FATAL_ERROR "CMAKE_INSTALL_CONFIG_NAME is empty")
  199. endif()
  200. set(cmd xcrun -f lipo)
  201. execute_process(
  202. COMMAND ${cmd}
  203. RESULT_VARIABLE result
  204. OUTPUT_VARIABLE output
  205. ERROR_VARIABLE output
  206. OUTPUT_STRIP_TRAILING_WHITESPACE
  207. ERROR_STRIP_TRAILING_WHITESPACE
  208. )
  209. if(NOT result EQUAL 0)
  210. message(
  211. FATAL_ERROR "Command failed (${result}): ${cmd}\n\nOutput:\n${output}"
  212. )
  213. endif()
  214. set(_lipo_path ${output})
  215. set(CURRENT_CONFIG "${CMAKE_INSTALL_CONFIG_NAME}")
  216. set(CURRENT_TARGET "${target}")
  217. _ios_install_combined_message("Target: ${CURRENT_TARGET}")
  218. _ios_install_combined_message("Config: ${CURRENT_CONFIG}")
  219. _ios_install_combined_message("Destination: ${destination}")
  220. # Get SDKs
  221. _ios_install_combined_detect_sdks(this_sdk corr_sdk)
  222. # Get architectures of the target
  223. _ios_install_combined_get_valid_archs("${corr_sdk}" corr_valid_archs)
  224. _ios_install_combined_get_valid_archs("${this_sdk}" this_valid_archs)
  225. # Return if there are no valid architectures for the SDK.
  226. # (note that library already installed)
  227. if("${corr_valid_archs}" STREQUAL "")
  228. _ios_install_combined_message(
  229. "No architectures detected for `${corr_sdk}` (skip)"
  230. )
  231. return()
  232. endif()
  233. # Trigger build of corresponding target
  234. _ios_install_combined_build("${corr_sdk}")
  235. # Get location of the library in build directory
  236. _ios_install_combined_get_build_setting(
  237. "${corr_sdk}" "CONFIGURATION_BUILD_DIR" corr_build_dir)
  238. _ios_install_combined_get_build_setting(
  239. "${corr_sdk}" "EXECUTABLE_PATH" corr_executable_path)
  240. set(corr "${corr_build_dir}/${corr_executable_path}")
  241. _ios_install_combined_keep_archs("${corr}" "${corr_valid_archs}")
  242. _ios_install_combined_keep_archs("${destination}" "${this_valid_archs}")
  243. _ios_install_combined_message("Current: ${destination}")
  244. _ios_install_combined_message("Corresponding: ${corr}")
  245. set(cmd "${_lipo_path}" -create ${corr} ${destination} -output ${destination})
  246. execute_process(
  247. COMMAND ${cmd}
  248. WORKING_DIRECTORY ${CMAKE_BINARY_DIR}
  249. RESULT_VARIABLE result
  250. )
  251. if(NOT result EQUAL 0)
  252. message(FATAL_ERROR "Command failed: ${cmd}")
  253. endif()
  254. _ios_install_combined_message("Install done: ${destination}")
  255. endfunction()