#!/bin/bash # # Copyright (C) 2020 The Android Open Source Project # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. # # This script creates the final boot image profile (suitable to include in the platform build). # The input to the script are: # 1) the boot.zip file which contains the boot classpath and system server jars. # This file can be obtained from running `m dist` or by configuring the device with # the `art/tools/boot-image-profile-configure-device.sh` script. # 2) the preloaded classes denylist which specify what clases should not be preloaded # in Zygote. Usually located in usually in frameworks/base/config/preloaded-classes-denylist # 3) a list of raw boot image profiles extracted from devices. An example how to do that is # by running `art/tools/boot-image-profile-extract-profile.sh` script. # # It is strongly recommended that you make use of extensive critical user journeys flows in order # to capture the raw boot image profiles described in #3. # # NOTE: by default, the script uses default arguments for producing the boot image profiles. # You might want to adjust the default generation arguments based on the shape of profile # and based on the metrics that matter for each product. # if [[ -z "$ANDROID_BUILD_TOP" ]]; then echo "You must run on this after running envsetup.sh and launch target" exit 1 fi if [[ "$#" -lt 4 ]]; then echo "Usage $0 ... " echo "Without any profman args the script will use defaults." echo "Example: $0 output-dir boot.zip frameworks/base/config/preloaded-classes-denylist android1.prof android2.prof" echo " $0 output-dir boot.zip frameworks/base/config/preloaded-classes-denylist android.prof --profman-arg --upgrade-startup-to-hot=true" echo "preloaded-deny-list-location is usually frameworks/base/config/preloaded-classes-denylist" exit 1 fi echo "Creating work dir" WORK_DIR=/tmp/android-bcp mkdir -p "$WORK_DIR" OUT_DIR="$1" BOOT_ZIP="$2" PRELOADED_DENYLIST="$3" shift 3 # Read the profile input args. profman_profile_input_args=() while [[ "$#" -ge 1 ]] && [[ ! "$1" = '--profman-arg' ]]; do profman_profile_input_args+=("--profile-file=$1") shift done # Read the profman args. profman_args=() while [[ "$#" -ge 2 ]] && [[ "$1" = '--profman-arg' ]]; do profman_args+=("$2") shift 2 done OUT_BOOT_PROFILE="$OUT_DIR"/boot-image-profile.txt OUT_PRELOADED_CLASSES="$OUT_DIR"/preloaded-classes OUT_SYSTEM_SERVER="$OUT_DIR"/art-profile echo "Changing dirs to the build top" cd "$ANDROID_BUILD_TOP" echo "Unziping boot.zip" BOOT_UNZIP_DIR="$WORK_DIR"/boot-dex ART_JARS="$BOOT_UNZIP_DIR"/dex_artjars_input BOOT_JARS="$BOOT_UNZIP_DIR"/dex_bootjars_input SYSTEM_SERVER_JAR="$BOOT_UNZIP_DIR"/system/framework/services.jar unzip -o "$BOOT_ZIP" -d "$BOOT_UNZIP_DIR" echo "Processing boot image jar files" jar_args=() for entry in "$ART_JARS"/* do jar_args+=("--apk=$entry") done for entry in "$BOOT_JARS"/* do jar_args+=("--apk=$entry") done profman_args+=("${jar_args[@]}") echo "Running profman for boot image profiles" # NOTE: # You might want to adjust the default generation arguments based on the data # For example, to update the selection thresholds you could specify: # --method-threshold=10 \ # --class-threshold=10 \ # --preloaded-class-threshold=10 \ # --special-package=android:1 \ # --special-package=com.android.systemui:1 \ # The threshold is percentage of total aggregation, that is, a method/class is # included in the profile only if it's used by at least x% of the packages. # (from 0% - include everything to 100% - include only the items that # are used by all packages on device). # The --special-package allows you to give a prioriority to certain packages, # meaning, if the methods is used by that package then the algorithm will use a # different selection thresholds. # (system server is identified as the "android" package) profman \ --generate-boot-image-profile \ "${profman_profile_input_args[@]}" \ --out-profile-path="$OUT_BOOT_PROFILE" \ --out-preloaded-classes-path="$OUT_PRELOADED_CLASSES" \ --preloaded-classes-denylist="$PRELOADED_DENYLIST" \ --special-package=android:1 \ --special-package=com.android.systemui:1 \ "${profman_args[@]}" echo "Done boot image profile" echo "Running profman for system server" # For system server profile we want to include everything usually # We also don't have a preloaded-classes file for it, so we ignore the argument. profman \ --generate-boot-image-profile \ "${profman_profile_input_args[@]}" \ --out-profile-path="$OUT_SYSTEM_SERVER" \ --apk="$SYSTEM_SERVER_JAR" \ --method-threshold=0 \ --class-threshold=0 echo "Done system server" echo "" echo "Boot profile methods+classes count: $(wc -l $OUT_BOOT_PROFILE)" echo "Preloaded classes count: $(wc -l $OUT_PRELOADED_CLASSES)" echo "System server profile methods+classes count: $(wc -l $OUT_SYSTEM_SERVER)" CLEAN_UP="${CLEAN_UP:-true}" if [[ "$CLEAN_UP" = "true" ]]; then rm -rf "$WORK_DIR" fi