| # Copyright (C) 2011 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 is used on host and device. It uses a common subset |
| # shell dialect that should work on the host (e.g. bash), and |
| # Android (e.g. mksh). |
| |
| # Globals |
| ART_BINARY=dalvikvm |
| DELETE_ANDROID_DATA="no" |
| LAUNCH_WRAPPER= |
| LIBART=libart.so |
| JIT_PROFILE="no" |
| VERBOSE="no" |
| EXTRA_OPTIONS=() |
| |
| # Follow all sym links to get the program name. |
| if [ z"$BASH_SOURCE" != z ]; then |
| PROG_NAME="$BASH_SOURCE" |
| else |
| PROG_NAME="$0" |
| fi |
| while [ -h "$PROG_NAME" ]; do |
| # On Mac OS, readlink -f doesn't work. |
| PROG_NAME="$(readlink "$PROG_NAME")" |
| done |
| |
| function find_libdir() { |
| # Get the actual file, $1 is the ART_BINARY_PATH and may be a symbolic link. |
| # Use realpath instead of readlink because Android does not have a readlink. |
| if [[ "$(realpath "$1")" == *dalvikvm64 ]]; then |
| echo "lib64" |
| else |
| echo "lib" |
| fi |
| } |
| |
| function replace_compiler_filter_with_quicken() { |
| ARGS_WITH_QUICKEN=("$@") |
| |
| found="false" |
| ((index=0)) |
| while ((index <= $#)); do |
| what="${ARGS_WITH_QUICKEN[$index]}" |
| |
| case "$what" in |
| --compiler-filter=*) |
| ARGS_WITH_QUICKEN[$index]="--compiler-filter=quicken" |
| found="true" |
| ;; |
| esac |
| |
| ((index++)) |
| shift |
| done |
| if [ "$found" != "true" ]; then |
| ARGS_WITH_QUICKEN=(-Xcompiler-option --compiler-filter=quicken "${ARGS_WITH_QUICKEN[@]}") |
| fi |
| } |
| |
| function usage() { |
| cat 1>&2 <<EOF |
| Usage: art [OPTIONS] [--] [ART_OPTIONS] CLASS |
| |
| Supported OPTIONS include: |
| --32 Use the 32-bit Android Runtime. |
| --64 Use the 64-bit Android Runtime. |
| --callgrind Launch the Android Runtime in callgrind. |
| -d Use the debug ART library (libartd.so). |
| --debug Equivalent to -d. |
| --gdb Launch the Android Runtime in gdb. |
| --help Display usage message. |
| --invoke-with <program> Launch the Android Runtime in <program>. |
| --perf Launch the Android Runtime with perf recording. |
| --perf-report Launch the Android Runtime with perf recording with |
| report upon completion. |
| --profile Run with profiling, then run using profile data. |
| --verbose Run script verbosely. |
| |
| The ART_OPTIONS are passed directly to the Android Runtime. |
| |
| Example: |
| art --32 -cp my_classes.dex MainClass |
| |
| Common errors: |
| 1) Not having core.art available (see $ANDROID_BUILD_TOP/art/Android.mk). |
| eg m -j32 build-art-host |
| 2) Not having boot.art available (see $ANDROID_BUILD_TOP/build/make/core/dex_preopt_libart_boot.mk) |
| eg m -j32 out/target/product/generic_x86_64/dex_bootjars/system/framework/x86_64/boot.art |
| EOF |
| } |
| |
| function clean_android_data() { |
| if [ "$DELETE_ANDROID_DATA" = "yes" ]; then |
| rm -rf $ANDROID_DATA |
| fi |
| } |
| |
| # Given 'VAR1=VAL VAR2=VAL2 ... cmd arg1 arg2 ... argN' run the 'cmd' with the args |
| # with the modified environment {VAR1=VAL,VAL2=,...}. |
| # |
| # Also prints the command to be run if verbose mode is enabled. |
| function verbose_run() { |
| if [ "$VERBOSE" = "yes" ]; then |
| echo "$@" |
| fi |
| |
| env "$@" |
| } |
| |
| # Parse a colon-separated list into an array (e.g. "foo.dex:bar.dex" -> (foo.dex bar.dex)) |
| PARSE_CLASSPATH_RESULT=() # Return value will be here due to shell limitations. |
| parse_classpath() { |
| local cp="$1" |
| local oldifs=$IFS |
| |
| local cp_array |
| cp_array=() |
| |
| IFS=":" |
| for part in $cp; do |
| cp_array+=("$part") |
| done |
| IFS=$oldifs |
| |
| PARSE_CLASSPATH_RESULT=("${cp_array[@]}") |
| } |
| |
| # Sets 'PARSE_CLASSPATH_RESULT' to an array of class path dex files. |
| # e.g. (-cp foo/classes.dex:bar/classes.dex) -> (foo/classes.dex bar/classes.dex) |
| find_cp_in_args() { |
| local found="false" |
| local index=0 |
| local what |
| |
| while [[ $# -gt 0 ]]; do |
| case "$1" in |
| -cp|-classpath) |
| parse_classpath "$2" |
| # Sets 'PARSE_CLASSPATH_RESULT' to an array of class path dex files. |
| # Subsequent parses will overwrite the preceding. |
| shift |
| ;; |
| esac |
| shift |
| done |
| } |
| |
| # Delete the 'oat' directories relative to the classpath's dex files. |
| # e.g. (foo/classes.dex bar/classes.dex) would delete (foo/oat bar/oat) directories. |
| cleanup_oat_directory() { |
| local classpath |
| classpath=("$@") |
| |
| local dirpath |
| |
| for path in "${classpath[@]}"; do |
| dirpath="$(dirname "$path")" |
| [[ -d "$dirpath" ]] && verbose_run rm -rf "$dirpath/oat" |
| done |
| } |
| |
| # Parse -cp <CP>, -classpath <CP>, and $CLASSPATH to find the dex files. |
| # Each dex file's directory will have an 'oat' file directory, delete it. |
| # Input: Command line arguments to the art script. |
| # e.g. -cp foo/classes.dex:bar/classes.dex would delete (foo/oat bar/oat) directories. |
| cleanup_oat_directory_for_classpath() { |
| # First try: Use $CLASSPATH environment variable. |
| parse_classpath "$CLASSPATH" |
| # Second try: Look for latest -cp or -classpath arg which will take precedence. |
| find_cp_in_args "$@" |
| |
| cleanup_oat_directory "${PARSE_CLASSPATH_RESULT[@]}" |
| } |
| |
| # Attempt to find $ANDROID_ROOT/framework/<isa>/core.art' without knowing what <isa> is. |
| function check_if_boot_image_file_exists() { |
| local image_location_dir="$1" |
| local image_location_name="$2" |
| |
| # Expand image_files to a list of existing image files on the disk. |
| # If no such files exist, it expands to single element 'dir/*/file' with a literal '*'. |
| local image_files |
| image_files=("$image_location_dir"/*/"$image_location_name") # avoid treating "*" as literal. |
| |
| # Array always has at least 1 element. Test explicitly whether the file exists. |
| [[ -e "${image_files[0]}" ]] |
| } |
| |
| # Automatically find the boot image location. It uses core.art by default. |
| # On a real device, it might only have a boot.art, so use that instead when core.art does not exist. |
| function detect_boot_image_location() { |
| local image_location_dir="$ANDROID_ROOT/framework" |
| local image_location_name="core.art" |
| |
| # If there are no existing core.art, try to find boot.art. |
| # If there is no boot.art then leave it as-is, assumes -Ximage is explicitly used. |
| # Otherwise let dalvikvm give the error message about an invalid image file. |
| if ! check_if_boot_image_file_exists "$image_location_dir" "core.art" && \ |
| check_if_boot_image_file_exists "$image_location_dir" "boot.art"; then |
| image_location_name="boot.art" |
| fi |
| |
| local image_location="$image_location_dir/$image_location_name" |
| echo "$image_location" |
| } |
| |
| # Runs dalvikvm, returns its exit code. |
| # (Oat directories are cleaned up in between runs) |
| function run_art() { |
| local image_location="$(detect_boot_image_location)" |
| local ret |
| |
| # First cleanup any left-over 'oat' files from the last time dalvikvm was run. |
| cleanup_oat_directory_for_classpath "$@" |
| # Run dalvikvm. |
| verbose_run ANDROID_DATA=$ANDROID_DATA \ |
| ANDROID_ROOT=$ANDROID_ROOT \ |
| LD_LIBRARY_PATH=$LD_LIBRARY_PATH \ |
| PATH=$ANDROID_ROOT/bin:$PATH \ |
| LD_USE_LOAD_BIAS=1 \ |
| $LAUNCH_WRAPPER $ART_BINARY_PATH $lib \ |
| -XXlib:$LIBART \ |
| -Xnorelocate \ |
| -Ximage:"$image_location" \ |
| "$@" |
| ret=$? |
| |
| # Avoid polluting disk with 'oat' files after dalvikvm has finished. |
| cleanup_oat_directory_for_classpath "$@" |
| |
| # Forward exit code of dalvikvm. |
| return $ret |
| } |
| |
| while [[ "$1" = "-"* ]]; do |
| case "$1" in |
| --) |
| # No more arguments for this script. |
| shift |
| break |
| ;; |
| --32) |
| ART_BINARY=dalvikvm32 |
| ;; |
| --64) |
| ART_BINARY=dalvikvm64 |
| ;; |
| --callgrind) |
| LAUNCH_WRAPPER="valgrind --tool=callgrind" |
| ;; |
| -d) |
| ;& # Fallthrough |
| --debug) |
| LIBART="libartd.so" |
| # Expect that debug mode wants all checks. |
| EXTRA_OPTIONS+=(-XX:SlowDebug=true) |
| ;; |
| --gdb) |
| LIBART="libartd.so" |
| LAUNCH_WRAPPER="gdb --args" |
| ;; |
| --help) |
| usage |
| exit 0 |
| ;; |
| --invoke-with) |
| LAUNCH_WRAPPER=$2 |
| shift |
| ;; |
| --perf) |
| PERF="record" |
| ;; |
| --perf-report) |
| PERF="report" |
| ;; |
| --profile) |
| JIT_PROFILE="yes" |
| ;; |
| --verbose) |
| VERBOSE="yes" |
| ;; |
| --*) |
| echo "unknown option: $1" 1>&2 |
| usage |
| exit 1 |
| ;; |
| *) |
| break |
| ;; |
| esac |
| shift |
| done |
| |
| if [ $# -eq 0 ]; then |
| usage |
| exit 1 |
| fi |
| |
| PROG_DIR="$(cd "${PROG_NAME%/*}" ; pwd -P)" |
| ANDROID_ROOT=$PROG_DIR/.. |
| ART_BINARY_PATH=$ANDROID_ROOT/bin/$ART_BINARY |
| |
| if [ ! -x "$ART_BINARY_PATH" ]; then |
| cat 1>&2 <<EOF |
| Android Runtime not found: $ART_BINARY_PATH |
| This script should be in the same directory as the Android Runtime ($ART_BINARY). |
| EOF |
| exit 1 |
| fi |
| |
| LIBDIR="$(find_libdir $ART_BINARY_PATH)" |
| LD_LIBRARY_PATH=$ANDROID_ROOT/$LIBDIR |
| |
| # If ANDROID_DATA is the system ANDROID_DATA or is not set, use our own, |
| # and ensure we delete it at the end. |
| if [ "$ANDROID_DATA" = "/data" ] || [ "$ANDROID_DATA" = "" ]; then |
| if [[ $PWD != / ]]; then |
| ANDROID_DATA="$PWD/android-data$$" |
| else |
| # Use /data/local/tmp when running this from adb shell, since it starts out in / |
| # by default. |
| ANDROID_DATA="$ANDROID_DATA/local/tmp/android-data$$" |
| fi |
| mkdir -p "$ANDROID_DATA" |
| DELETE_ANDROID_DATA="yes" |
| fi |
| |
| if [ "$PERF" != "" ]; then |
| LAUNCH_WRAPPER="perf record -g -o $ANDROID_DATA/perf.data -e cycles:u $LAUNCH_WRAPPER" |
| EXTRA_OPTIONS+=(-Xcompiler-option --generate-debug-info) |
| fi |
| |
| if [ "$JIT_PROFILE" = "yes" ]; then |
| # Create the profile. The runtime expects profiles to be created before |
| # execution. |
| PROFILE_PATH="$ANDROID_DATA/primary.prof" |
| touch "$PROFILE_PATH" |
| |
| # Replace the compiler filter with quicken so that we |
| # can capture the profile. |
| ARGS_WITH_QUICKEN= |
| replace_compiler_filter_with_quicken "$@" |
| |
| run_art -Xjitsaveprofilinginfo \ |
| -Xps-min-methods-to-save:1 \ |
| -Xps-min-classes-to-save:1 \ |
| -Xps-min-notification-before-wake:10 \ |
| -Xps-profile-path:$PROFILE_PATH \ |
| -Xusejit:true \ |
| "${ARGS_WITH_QUICKEN[@]}" \ |
| &> "$ANDROID_DATA/profile_gen.log" |
| EXIT_STATUS=$? |
| |
| if [ $EXIT_STATUS != 0 ]; then |
| echo "Profile run failed: " >&2 |
| cat "$ANDROID_DATA/profile_gen.log" >&2 |
| clean_android_data |
| exit $EXIT_STATUS |
| fi |
| |
| # Wipe dalvik-cache so that a subsequent run_art must regenerate it. |
| # Leave $ANDROID_DATA intact since it contains our profile file. |
| rm -rf "$ANDROID_DATA/dalvik-cache" |
| |
| # Append arguments so next invocation of run_art uses the profile. |
| EXTRA_OPTIONS+=(-Xcompiler-option --profile-file="$PROFILE_PATH") |
| fi |
| |
| # Protect additional arguments in quotes to preserve whitespaces (used by |
| # run-jdwp-test.sh when running on device), '$' (may be used as part of |
| # classpath) and other special characters when evaluated. |
| EXTRA_OPTIONS+=("$@") |
| |
| run_art "${EXTRA_OPTIONS[@]}" |
| EXIT_STATUS=$? |
| |
| if [ "$PERF" != "" ]; then |
| if [ "$PERF" = report ]; then |
| perf report -i $ANDROID_DATA/perf.data |
| fi |
| echo "Perf data saved in: $ANDROID_DATA/perf.data" |
| else |
| # Perf output is placed under $ANDROID_DATA so not cleaned when perf options used. |
| clean_android_data |
| fi |
| |
| exit $EXIT_STATUS |