| # Distributed under the OSI-approved BSD 3-Clause License. See accompanying | |
| # file Copyright.txt or https://cmake.org/licensing for details. | |
| cmake_policy(PUSH) | |
| cmake_policy(SET CMP0057 NEW) # if IN_LIST | |
| cmake_policy(SET CMP0054 NEW) | |
| # Function to print messages of this module | |
| function(_ios_install_combined_message) | |
| message(STATUS "[iOS combined] " ${ARGN}) | |
| endfunction() | |
| # Get build settings for the current target/config/SDK by running | |
| # `xcodebuild -sdk ... -showBuildSettings` and parsing it's output | |
| function(_ios_install_combined_get_build_setting sdk variable resultvar) | |
| if("${sdk}" STREQUAL "") | |
| message(FATAL_ERROR "`sdk` is empty") | |
| endif() | |
| if("${variable}" STREQUAL "") | |
| message(FATAL_ERROR "`variable` is empty") | |
| endif() | |
| if("${resultvar}" STREQUAL "") | |
| message(FATAL_ERROR "`resultvar` is empty") | |
| endif() | |
| set( | |
| cmd | |
| xcodebuild -showBuildSettings | |
| -sdk "${sdk}" | |
| -target "${CURRENT_TARGET}" | |
| -config "${CURRENT_CONFIG}" | |
| ) | |
| execute_process( | |
| COMMAND ${cmd} | |
| WORKING_DIRECTORY "${CMAKE_BINARY_DIR}" | |
| RESULT_VARIABLE result | |
| OUTPUT_VARIABLE output | |
| ) | |
| if(NOT result EQUAL 0) | |
| message(FATAL_ERROR "Command failed (${result}): ${cmd}") | |
| endif() | |
| if(NOT output MATCHES " ${variable} = ([^\n]*)") | |
| if("${variable}" STREQUAL "VALID_ARCHS") | |
| # VALID_ARCHS may be unset by user for given SDK | |
| # (e.g. for build without simulator). | |
| set("${resultvar}" "" PARENT_SCOPE) | |
| return() | |
| else() | |
| message(FATAL_ERROR "${variable} not found.") | |
| endif() | |
| endif() | |
| set("${resultvar}" "${CMAKE_MATCH_1}" PARENT_SCOPE) | |
| endfunction() | |
| # Get architectures of given SDK (iphonesimulator/iphoneos) | |
| function(_ios_install_combined_get_valid_archs sdk resultvar) | |
| cmake_policy(PUSH) | |
| cmake_policy(SET CMP0007 NEW) | |
| if("${resultvar}" STREQUAL "") | |
| message(FATAL_ERROR "`resultvar` is empty") | |
| endif() | |
| _ios_install_combined_get_build_setting("${sdk}" "VALID_ARCHS" valid_archs) | |
| separate_arguments(valid_archs) | |
| list(REMOVE_ITEM valid_archs "") # remove empty elements | |
| list(REMOVE_DUPLICATES valid_archs) | |
| string(REPLACE ";" " " printable "${valid_archs}") | |
| _ios_install_combined_message("Architectures (${sdk}): ${printable}") | |
| set("${resultvar}" "${valid_archs}" PARENT_SCOPE) | |
| cmake_policy(POP) | |
| endfunction() | |
| # Make both arch lists a disjoint set by preferring the current SDK | |
| # (starting with Xcode 12 arm64 is available as device and simulator arch on iOS) | |
| function(_ios_install_combined_prune_common_archs corr_sdk corr_archs_var this_archs_var) | |
| list(REMOVE_ITEM ${corr_archs_var} ${${this_archs_var}}) | |
| string(REPLACE ";" " " printable "${${corr_archs_var}}") | |
| _ios_install_combined_message("Architectures (${corr_sdk}) after pruning: ${printable}") | |
| set("${corr_archs_var}" "${${corr_archs_var}}" PARENT_SCOPE) | |
| endfunction() | |
| # Final target can contain more architectures that specified by SDK. This | |
| # function will run 'lipo -info' and parse output. Result will be returned | |
| # as a CMake list. | |
| function(_ios_install_combined_get_real_archs filename resultvar) | |
| set(cmd "${_lipo_path}" -info "${filename}") | |
| execute_process( | |
| COMMAND ${cmd} | |
| RESULT_VARIABLE result | |
| OUTPUT_VARIABLE output | |
| ERROR_VARIABLE output | |
| OUTPUT_STRIP_TRAILING_WHITESPACE | |
| ERROR_STRIP_TRAILING_WHITESPACE | |
| ) | |
| if(NOT result EQUAL 0) | |
| message( | |
| FATAL_ERROR "Command failed (${result}): ${cmd}\n\nOutput:\n${output}" | |
| ) | |
| endif() | |
| if(NOT output MATCHES "(Architectures in the fat file: [^\n]+ are|Non-fat file: [^\n]+ is architecture): ([^\n]*)") | |
| message(FATAL_ERROR "Could not detect architecture from: ${output}") | |
| endif() | |
| separate_arguments(CMAKE_MATCH_2) | |
| set(${resultvar} ${CMAKE_MATCH_2} PARENT_SCOPE) | |
| endfunction() | |
| # Run build command for the given SDK | |
| function(_ios_install_combined_build sdk) | |
| if("${sdk}" STREQUAL "") | |
| message(FATAL_ERROR "`sdk` is empty") | |
| endif() | |
| _ios_install_combined_message("Build `${CURRENT_TARGET}` for `${sdk}`") | |
| execute_process( | |
| COMMAND | |
| "${CMAKE_COMMAND}" | |
| --build | |
| . | |
| --target "${CURRENT_TARGET}" | |
| --config ${CURRENT_CONFIG} | |
| -- | |
| -sdk "${sdk}" | |
| WORKING_DIRECTORY "${CMAKE_BINARY_DIR}" | |
| RESULT_VARIABLE result | |
| ) | |
| if(NOT result EQUAL 0) | |
| message(FATAL_ERROR "Build failed") | |
| endif() | |
| endfunction() | |
| # Remove given architecture from file. This step needed only in rare cases | |
| # when target was built in "unusual" way. Emit warning message. | |
| function(_ios_install_combined_remove_arch lib arch) | |
| _ios_install_combined_message( | |
| "Warning! Unexpected architecture `${arch}` detected and will be removed " | |
| "from file `${lib}`") | |
| set(cmd "${_lipo_path}" -remove ${arch} -output ${lib} ${lib}) | |
| execute_process( | |
| COMMAND ${cmd} | |
| RESULT_VARIABLE result | |
| OUTPUT_VARIABLE output | |
| ERROR_VARIABLE output | |
| OUTPUT_STRIP_TRAILING_WHITESPACE | |
| ERROR_STRIP_TRAILING_WHITESPACE | |
| ) | |
| if(NOT result EQUAL 0) | |
| message( | |
| FATAL_ERROR "Command failed (${result}): ${cmd}\n\nOutput:\n${output}" | |
| ) | |
| endif() | |
| endfunction() | |
| # Check that 'lib' contains only 'archs' architectures (remove others). | |
| function(_ios_install_combined_keep_archs lib archs) | |
| _ios_install_combined_get_real_archs("${lib}" real_archs) | |
| set(archs_to_remove ${real_archs}) | |
| list(REMOVE_ITEM archs_to_remove ${archs}) | |
| foreach(x ${archs_to_remove}) | |
| _ios_install_combined_remove_arch("${lib}" "${x}") | |
| endforeach() | |
| endfunction() | |
| function(_ios_install_combined_detect_associated_sdk corr_sdk_var) | |
| if("${PLATFORM_NAME}" STREQUAL "") | |
| message(FATAL_ERROR "PLATFORM_NAME should not be empty") | |
| endif() | |
| set(all_platforms "$ENV{SUPPORTED_PLATFORMS}") | |
| if("${SUPPORTED_PLATFORMS}" STREQUAL "") | |
| _ios_install_combined_get_build_setting( | |
| ${PLATFORM_NAME} SUPPORTED_PLATFORMS all_platforms) | |
| if("${all_platforms}" STREQUAL "") | |
| message(FATAL_ERROR | |
| "SUPPORTED_PLATFORMS not set as an environment variable nor " | |
| "able to be determined from project") | |
| endif() | |
| endif() | |
| separate_arguments(all_platforms) | |
| if(NOT PLATFORM_NAME IN_LIST all_platforms) | |
| message(FATAL_ERROR "`${PLATFORM_NAME}` not found in `${all_platforms}`") | |
| endif() | |
| list(REMOVE_ITEM all_platforms "" "${PLATFORM_NAME}") | |
| list(LENGTH all_platforms all_platforms_length) | |
| if(NOT all_platforms_length EQUAL 1) | |
| message(FATAL_ERROR "Expected one element: ${all_platforms}") | |
| endif() | |
| set(${corr_sdk_var} "${all_platforms}" PARENT_SCOPE) | |
| endfunction() | |
| # Create combined binary for the given target. | |
| # | |
| # Preconditions: | |
| # * Target already installed at ${destination} | |
| # for the ${PLATFORM_NAME} platform | |
| # | |
| # This function will: | |
| # * Run build for the lacking platform, i.e. opposite to the ${PLATFORM_NAME} | |
| # * Fuse both libraries by running lipo | |
| function(ios_install_combined target destination) | |
| if("${target}" STREQUAL "") | |
| message(FATAL_ERROR "`target` is empty") | |
| endif() | |
| if("${destination}" STREQUAL "") | |
| message(FATAL_ERROR "`destination` is empty") | |
| endif() | |
| if(NOT IS_ABSOLUTE "${destination}") | |
| message(FATAL_ERROR "`destination` is not absolute: ${destination}") | |
| endif() | |
| if(IS_DIRECTORY "${destination}" OR IS_SYMLINK "${destination}") | |
| message(FATAL_ERROR "`destination` is no regular file: ${destination}") | |
| endif() | |
| if("${CMAKE_BINARY_DIR}" STREQUAL "") | |
| message(FATAL_ERROR "`CMAKE_BINARY_DIR` is empty") | |
| endif() | |
| if(NOT IS_DIRECTORY "${CMAKE_BINARY_DIR}") | |
| message(FATAL_ERROR "Is not a directory: ${CMAKE_BINARY_DIR}") | |
| endif() | |
| if("${CMAKE_INSTALL_CONFIG_NAME}" STREQUAL "") | |
| message(FATAL_ERROR "CMAKE_INSTALL_CONFIG_NAME is empty") | |
| endif() | |
| set(cmd xcrun -f lipo) | |
| # Do not merge OUTPUT_VARIABLE and ERROR_VARIABLE since latter may contain | |
| # some diagnostic information even for the successful run. | |
| execute_process( | |
| COMMAND ${cmd} | |
| RESULT_VARIABLE result | |
| OUTPUT_VARIABLE output | |
| ERROR_VARIABLE error_output | |
| OUTPUT_STRIP_TRAILING_WHITESPACE | |
| ERROR_STRIP_TRAILING_WHITESPACE | |
| ) | |
| if(NOT result EQUAL 0) | |
| message( | |
| FATAL_ERROR "Command failed (${result}): ${cmd}\n\nOutput:\n${output}\nOutput(error):\n${error_output}" | |
| ) | |
| endif() | |
| set(_lipo_path ${output}) | |
| list(LENGTH _lipo_path len) | |
| if(NOT len EQUAL 1) | |
| message(FATAL_ERROR "Unexpected xcrun output: ${_lipo_path}") | |
| endif() | |
| if(NOT EXISTS "${_lipo_path}") | |
| message(FATAL_ERROR "File not found: ${_lipo_path}") | |
| endif() | |
| set(CURRENT_CONFIG "${CMAKE_INSTALL_CONFIG_NAME}") | |
| set(CURRENT_TARGET "${target}") | |
| _ios_install_combined_message("Target: ${CURRENT_TARGET}") | |
| _ios_install_combined_message("Config: ${CURRENT_CONFIG}") | |
| _ios_install_combined_message("Destination: ${destination}") | |
| # Get SDKs | |
| _ios_install_combined_detect_associated_sdk(corr_sdk) | |
| # Get architectures of the target | |
| _ios_install_combined_get_valid_archs("${PLATFORM_NAME}" this_valid_archs) | |
| _ios_install_combined_get_valid_archs("${corr_sdk}" corr_valid_archs) | |
| _ios_install_combined_prune_common_archs("${corr_sdk}" corr_valid_archs this_valid_archs) | |
| # Return if there are no valid architectures for the SDK. | |
| # (note that library already installed) | |
| if("${corr_valid_archs}" STREQUAL "") | |
| _ios_install_combined_message( | |
| "No architectures detected for `${corr_sdk}` (skip)" | |
| ) | |
| return() | |
| endif() | |
| # Trigger build of corresponding target | |
| _ios_install_combined_build("${corr_sdk}") | |
| # Get location of the library in build directory | |
| _ios_install_combined_get_build_setting( | |
| "${corr_sdk}" "CONFIGURATION_BUILD_DIR" corr_build_dir) | |
| _ios_install_combined_get_build_setting( | |
| "${corr_sdk}" "EXECUTABLE_PATH" corr_executable_path) | |
| set(corr "${corr_build_dir}/${corr_executable_path}") | |
| _ios_install_combined_keep_archs("${corr}" "${corr_valid_archs}") | |
| _ios_install_combined_keep_archs("${destination}" "${this_valid_archs}") | |
| _ios_install_combined_message("Current: ${destination}") | |
| _ios_install_combined_message("Corresponding: ${corr}") | |
| set(cmd "${_lipo_path}" -create ${corr} ${destination} -output ${destination}) | |
| execute_process( | |
| COMMAND ${cmd} | |
| WORKING_DIRECTORY ${CMAKE_BINARY_DIR} | |
| RESULT_VARIABLE result | |
| ) | |
| if(NOT result EQUAL 0) | |
| message(FATAL_ERROR "Command failed: ${cmd}") | |
| endif() | |
| _ios_install_combined_message("Install done: ${destination}") | |
| endfunction() | |
| cmake_policy(POP) |