flutter/packages/flutter_tools/bin/xcode_backend.sh
Chris Bracken 1926d11188 Thin iOS app frameworks to the target architecture (#7913)
* Support thinning iOS frameworks to supported architectures

When building against frameworks that are distributed as
multi-architecture fat binaries, we want to strip the frameworks we
distribute down to only the architectures specified in $ARCHS.

This patch adds:
* The ability to specify commands to xcode_backend.sh (if none is
  specified, run BuildApp for backward compatibility).
* A 'thin' command that invokes lipo to thin down the distributed as
  described above.

* Add framework thinning step to iOS build

Invokes xcode_backend.sh thin on the build application.

* Limit architectures to arm64 in Xcode template

Flutter does not yet support armv7 iOS devices. Limit the $ARCHS build
variable to arm64 until then.
2017-02-07 12:04:36 -08:00

198 lines
5.6 KiB
Bash
Executable File

#!/bin/sh
# Copyright 2016 The Chromium Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
RunCommand() {
echo "♦ " $@
$@
return $?
}
EchoError() {
echo "$@" 1>&2
}
AssertExists() {
RunCommand ls $1
if [ $? -ne 0 ]; then
EchoError "The path $1 does not exist"
exit -1
fi
return 0
}
BuildApp() {
local project_path="${SOURCE_ROOT}/.."
if [[ -n "$FLUTTER_APPLICATION_PATH" ]]; then
project_path=${FLUTTER_APPLICATION_PATH}
fi
local target_path="lib/main.dart"
if [[ -n "$FLUTTER_TARGET" ]]; then
target_path=${FLUTTER_TARGET}
fi
local build_mode="release"
if [[ -n "$FLUTTER_BUILD_MODE" ]]; then
build_mode=${FLUTTER_BUILD_MODE}
fi
local artifact_variant="unknown"
case "$build_mode" in
release) artifact_variant="ios-release";;
profile) artifact_variant="ios-profile";;
debug) artifact_variant="ios";;
*) echo "Unknown FLUTTER_BUILD_MODE: $FLUTTER_BUILD_MODE";;
esac
local framework_path="${FLUTTER_ROOT}/bin/cache/artifacts/engine/${artifact_variant}"
if [[ -n "$FLUTTER_FRAMEWORK_DIR" ]]; then
framework_path="${FLUTTER_FRAMEWORK_DIR}"
fi
AssertExists ${project_path}
local derived_dir=${SOURCE_ROOT}/Flutter
RunCommand mkdir -p $derived_dir
AssertExists $derived_dir
RunCommand rm -f ${derived_dir}/Flutter.framework
RunCommand rm -f ${derived_dir}/app.dylib
RunCommand rm -f ${derived_dir}/app.flx
RunCommand cp -r ${framework_path}/Flutter.framework ${derived_dir}
RunCommand pushd ${project_path}
AssertExists ${target_path}
local build_dir=${FLUTTER_BUILD_DIR:-build}
local local_engine_flag=""
if [[ -n "$LOCAL_ENGINE" ]]; then
local_engine_flag="--local-engine=$LOCAL_ENGINE"
fi
if [[ $CURRENT_ARCH != "x86_64" ]]; then
local aot_flags=""
if [[ "$build_mode" == "debug" ]]; then
aot_flags="--interpreter --debug"
else
aot_flags="--${build_mode}"
fi
RunCommand ${FLUTTER_ROOT}/bin/flutter --suppress-analytics build aot \
--output-dir=${build_dir}/aot \
--target-platform=ios \
--target=${target_path} \
${aot_flags} \
${local_engine_flag}
if [[ $? -ne 0 ]]; then
EchoError "Failed to build ${project_path}."
exit -1
fi
RunCommand cp ${build_dir}/aot/app.dylib ${derived_dir}/app.dylib
else
RunCommand eval "$(echo \"static const int Moo = 88;\" | xcrun clang -x c --shared -o ${derived_dir}/app.dylib -)"
fi
local precompilation_flag=""
if [[ $CURRENT_ARCH != "x86_64" ]] && [[ "$build_mode" != "debug" ]]; then
precompilation_flag="--precompiled"
fi
RunCommand ${FLUTTER_ROOT}/bin/flutter --suppress-analytics build flx \
--target=${target_path} \
--output-file=${derived_dir}/app.flx \
--snapshot=${build_dir}/snapshot_blob.bin \
--depfile=${build_dir}/snapshot_blob.bin.d \
--working-dir=${build_dir}/flx \
${precompilation_flag} \
${local_engine_flag} \
if [[ $? -ne 0 ]]; then
EchoError "Failed to package ${project_path}."
exit -1
fi
RunCommand popd
echo "Project ${project_path} built and packaged successfully."
return 0
}
# Returns the CFBundleExecutable for the specified framework directory.
GetFrameworkExecutablePath() {
local framework_dir="$1"
local plist_path="${framework_dir}/Info.plist"
local executable="$(defaults read "${plist_path}" CFBundleExecutable)"
echo "${framework_dir}/${executable}"
}
# Destructively thins the specified executable file to include only the
# specified architectures.
LipoExecutable() {
local executable="$1"
shift
local archs="$@"
# Extract architecture-specific framework executables.
local all_executables=()
for arch in $archs; do
local output="${executable}_${arch}"
lipo -output "${output}" -extract "${arch}" "${executable}"
if [[ $? == 0 ]]; then
all_executables+=("${output}")
else
echo "Failed to extract ${arch} for ${executable}. Running lipo -info:"
lipo -info "${executable}"
exit 1
fi
done
# Merge desired architectures.
local merged="${executable}_merged"
lipo -output "${merged}" -create "${all_executables[@]}"
# Replace the original executable with the thinned one and clean up.
cp -f "${merged}" "${executable}" > /dev/null
rm -f "${merged}" "${all_executables[@]}"
}
# Destructively thins the specified framework to include only the specified
# architectures.
ThinFramework() {
local framework_dir="$1"
shift
local archs="$@"
local plist_path="${framework_dir}/Info.plist"
local executable="$(GetFrameworkExecutablePath "${framework_dir}")"
LipoExecutable "${executable}" "$archs"
}
ThinAppFrameworks() {
local app_path="${TARGET_BUILD_DIR}/${WRAPPER_NAME}"
local frameworks_dir="${app_path}/Frameworks"
[[ -d "$frameworks_dir" ]] || return 0
for framework_dir in "$(find "${app_path}" -type d -name "*.framework")"; do
ThinFramework "$framework_dir" "$ARCHS"
done
}
# Main entry point.
if [[ $# == 0 ]]; then
# Backwards-comptibility: if no args are provided, build.
BuildApp
else
case $1 in
"build")
BuildApp ;;
"thin")
ThinAppFrameworks ;;
esac
fi