blob: e274fd4fd87df7921faa3fd0f2916331816e8316 [file] [log] [blame]
#! /bin/bash
#
# Copyright (C) 2018 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.
# Push ART artifacts and its dependencies to a chroot directory for on-device testing.
if [ -t 1 ]; then
# Color sequences if terminal is a tty.
red='\033[0;31m'
green='\033[0;32m'
yellow='\033[0;33m'
magenta='\033[0;35m'
nc='\033[0m'
fi
adb wait-for-device
if [[ -z "$ANDROID_BUILD_TOP" ]]; then
echo 'ANDROID_BUILD_TOP environment variable is empty; did you forget to run `lunch`?'
exit 1
fi
if [[ -z "$ANDROID_PRODUCT_OUT" ]]; then
echo 'ANDROID_PRODUCT_OUT environment variable is empty; did you forget to run `lunch`?'
exit 1
fi
if [[ -z "$ART_TEST_CHROOT" ]]; then
echo 'ART_TEST_CHROOT environment variable is empty; please set it before running this script.'
exit 1
fi
if [[ "$(build/soong/soong_ui.bash --dumpvar-mode TARGET_FLATTEN_APEX)" != "true" ]]; then
echo -e "${red}This script only works when APEX packages are flattened, but the build" \
"configuration is set up to use non-flattened APEX packages.${nc}"
echo -e "${magenta}You can force APEX flattening by setting the environment variable" \
"\`OVERRIDE_TARGET_FLATTEN_APEX\` to \"true\" before starting the build and running this" \
"script.${nc}"
exit 1
fi
# Linker configuration.
# ---------------------
# Adjust the chroot environment to have it use the system linker configuration
# of the built target ("guest system"), located in `/system/etc` under the
# chroot directory, even if the linker configuration flavor of the "guest
# system" (e.g. legacy configuration) does not match the one of the "host
# system" (e.g. full-VNDK configuration). This is done by renaming the
# configuration file provided by the "guest system" (created according to the
# build target configuration) within the chroot environment, using the name of
# the configuration file expected by the linker (governed by system properties
# of the "host system").
# Default linker configuration file name/stem.
ld_config_file_path="/system/etc/ld.config.txt";
# VNDK-lite linker configuration file name.
ld_config_vndk_lite_file_path="/system/etc/ld.config.vndk_lite.txt";
# Find linker configuration path name on the "host system".
#
# The logic here partly replicates (and simplifies) Bionic's linker logic around
# configuration file search (see `get_ld_config_file_path` in
# bionic/linker/linker.cpp).
get_ld_host_system_config_file_path() {
# Check whether the "host device" uses a VNDK-lite linker configuration.
local vndk_lite=$(adb shell getprop "ro.vndk.lite" false)
if [[ "$vndk_lite" = true ]]; then
if adb shell test -f "$ld_config_vndk_lite_file_path"; then
echo "$ld_config_vndk_lite_file_path"
return
fi
fi
# Check the "host device"'s VNDK version, if any.
local vndk_version=$(adb shell getprop "ro.vndk.version")
if [[ -n "$vndk_version" ]] && [[ "$vndk_version" != current ]]; then
# Insert the VNDK version after the last period (and add another period).
local ld_config_file_vdnk_path=$(echo "$ld_config_file_path" \
| sed -e "s/^\\(.*\\)\\.\\([^.]\\)/\\1.${vndk_version}.\\2/")
if adb shell test -f "$ld_config_file_vdnk_path"; then
echo "$ld_config_file_vdnk_path"
return
fi
else
if adb shell test -f "$ld_config_file_path"; then
echo "$ld_config_file_path"
return
fi
fi
# If all else fails, return the default linker configuration name.
echo -e "${yellow}Cannot find linker configuration; using default path name:" \
"\`$ld_config_file_path\`${nc}" >&2
echo "$ld_config_file_path"
return
}
# Find linker configuration path name on the "guest system".
#
# The logic here tries to "guess" the name of the linker configuration file,
# based on the contents of the build directory.
get_ld_guest_system_config_file_path() {
if [[ -z "$ANDROID_PRODUCT_OUT" ]]; then
echo -e "${red}ANDROID_PRODUCT_OUT environment variable is empty;" \
"did you forget to run \`lunch\`${nc}?" >&2
exit 1
fi
local ld_config_file_location="$ANDROID_PRODUCT_OUT/system/etc"
local ld_config_file_paths=$(find "$ld_config_file_location" -name "ld.*.txt")
local ld_config_file_path_number=$(wc -l <<< "$ld_config_file_paths")
if [[ "$ld_config_file_path_number" -eq 0 ]]; then
echo -e "${red}No linker configuration file found in \`$ld_config_file_location\`${nc}" >&2
exit 1
fi
if [[ "$ld_config_file_path_number" -gt 1 ]]; then
echo -e \
"${red}More than one linker configuration file found in \`$ld_config_file_location\`:" \
"\n${ld_config_file_paths}${nc}" >&2
exit 1
fi
# Strip the build prefix to make the path name relative to the "guest root directory".
sed -e "s|^$ANDROID_PRODUCT_OUT||" <<< "$ld_config_file_paths"
}
# Synchronization recipe.
# -----------------------
# Sync the system directory to the chroot.
echo -e "${green}Syncing system directory...${nc}"
adb shell mkdir -p "$ART_TEST_CHROOT/system"
adb push "$ANDROID_PRODUCT_OUT/system" "$ART_TEST_CHROOT/"
# Overwrite the default public.libraries.txt file with a smaller one that
# contains only the public libraries pushed to the chroot directory.
adb push "$ANDROID_BUILD_TOP/art/tools/public.libraries.buildbot.txt" \
"$ART_TEST_CHROOT/system/etc/public.libraries.txt"
# Manually "activate" the flattened APEX $1 by syncing it to /apex/$2 in the
# chroot. $2 defaults to $1.
#
# TODO: Handle the case of build targets using non-flatted APEX packages.
# As a workaround, one can run `export OVERRIDE_TARGET_FLATTEN_APEX=true` before building
# a target to have its APEX packages flattened.
activate_apex() {
local src_apex=${1}
local dst_apex=${2:-${src_apex}}
echo -e "${green}Activating APEX ${src_apex} as ${dst_apex}...${nc}"
# We copy the files from `/system/apex/${src_apex}` to `/apex/${dst_apex}` in
# the chroot directory, instead of simply using a symlink, as Bionic's linker
# relies on the real path name of a binary (e.g.
# `/apex/com.android.art/bin/dex2oat`) to select the linker configuration.
adb shell mkdir -p "$ART_TEST_CHROOT/apex"
adb shell rm -rf "$ART_TEST_CHROOT/apex/${dst_apex}"
adb shell cp -a "$ART_TEST_CHROOT/system/apex/${src_apex}" "$ART_TEST_CHROOT/apex/${dst_apex}" \
|| exit 1
}
# "Activate" the required APEX modules.
activate_apex com.android.art.testing com.android.art
activate_apex com.android.i18n
activate_apex com.android.runtime
activate_apex com.android.tzdata
# Adjust the linker configuration file (if needed).
#
# Check the linker configurations files on the "host system" and the "guest
# system". If these file names are different, rename the "guest system" linker
# configuration file within the chroot environment using the "host system"
# linker configuration file name.
ld_host_system_config_file_path=$(get_ld_host_system_config_file_path) || exit 1
echo -e "${green}Determining host system linker configuration:" \
"\`$ld_host_system_config_file_path\`${nc}"
ld_guest_system_config_file_path=$(get_ld_guest_system_config_file_path) || exit 1
echo -e "${green}Determining guest system linker configuration:" \
"\`$ld_guest_system_config_file_path\`${nc}"
if [[ "$ld_host_system_config_file_path" != "$ld_guest_system_config_file_path" ]]; then
echo -e "${green}Renaming linker configuration file in chroot environment:" \
"\`$ART_TEST_CHROOT$ld_guest_system_config_file_path\`" \
"-> \`$ART_TEST_CHROOT$ld_host_system_config_file_path\`${nc}"
adb shell mv -f "$ART_TEST_CHROOT$ld_guest_system_config_file_path" \
"$ART_TEST_CHROOT$ld_host_system_config_file_path"
fi
# Sync the data directory to the chroot.
echo -e "${green}Syncing data directory...${nc}"
adb shell mkdir -p "$ART_TEST_CHROOT/data"
adb push "$ANDROID_PRODUCT_OUT/data" "$ART_TEST_CHROOT/"