| #!/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() { |
| if [[ ! -e "$1" ]]; then |
| if [[ -h "$1" ]]; then |
| EchoError "The path $1 is a symlink to a path that does not exist" |
| else |
| EchoError "The path $1 does not exist" |
| fi |
| 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}" |
| local lipo_info=$(lipo -info "${executable}") |
| if [[ "${lipo_info}" == "Non-fat file:"* ]]; then |
| if [[ "${lipo_info}" != *"${arch}" ]]; then |
| echo "Non-fat binary ${executable} is not ${arch}. Running lipo -info:" |
| echo "${lipo_info}" |
| exit 1 |
| fi |
| else |
| 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 |
| 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 |