1#!/bin/bash -ex 2usage () { 3 echo "Create a Mixed Build archive with the given system and device archives." 4 echo 5 echo "Usage: $0 [-v <vendor_version>] [-m <modify_system_image_path>]" 6 echo " [-t <prebuilt_otatools_path>] [-p <override_vbmeta_image_path>]" 7 echo " [-b <override_boot_image_path>]" 8 echo " [-s] [-d] system_build_dir device_build_dir out_dir" 9 echo 10 echo "Options -v, -m, -t, -p, -b, -s, -d must precede positional arguments." 11 echo 12 echo "vendor_version is the version of the vendor image when Keymaster v3" 13 echo " related modifications to the system image is necessary. Optional." 14 echo " eg. 8.1.0 for a mixed build of GSI and O-MR1 vendor image." 15 echo "modify_system_image_path is the path to the script that modifies the" 16 echo " system image, needed for Keymaster v3. Optional." 17 echo "prebuilt_otatools_path is the path to otatools.zip file that has all" 18 echo " required host binaries to modify system image. It also must include" 19 echo " VINTF check tool to verify the compatibility of the given images." 20 echo " Optional." 21 echo "override_vbmeta_image_path is the path to a vbmeta.img to use" 22 echo " to override the existing vbmeta.img of device. Optional." 23 echo "override_boot_image_path is the path to a boot imgage to use to" 24 echo " override the existing boot.img of device. Optional." 25 echo "-s is used to fetch and flash both product.img and system.img from" 26 echo " the system_build_dir for devices with a product partition." 27 echo " product.img will be removed if system_build_dir does not have" 28 echo " product.img when -s option is declared." 29 echo " By default, only system.img is flashed to the target device for" 30 echo " independent system update. No parameter required. Optional" 31 echo "-d is used to skip vbmeta.img replacement in its entirety and the" 32 echo " one from the device build (if exists) is left untouched" 33 echo "system_build_dir is the path to the system build" 34 echo " eg. aosp_arm64-userdebug." 35 echo "device_build_dir is the path to the device build" 36 echo " eg. sailfish-user." 37 echo "out_dir is the path to where the new build will be placed." 38} 39 40# Print error message and exit. 41# Usage: exit_badparam message 42# 43# message is a string to be displayed before exit. 44exit_badparam () { 45 echo "ERROR: $1" >&2 46 usage 47 exit 1 48} 49 50cleanup_and_exit () { 51 readonly result="$?" 52 rm -rf "$TEMP_DIR" 53 exit "$result" 54} 55 56trap cleanup_and_exit EXIT 57 58while getopts :v:m:p:b:t:sd opt; do 59 case "$opt" in 60 v) 61 readonly VENDOR_VERSION="$OPTARG" 62 ;; 63 m) 64 readonly MODIFY_SYSTEM_SCRIPT="$OPTARG" 65 ;; 66 p) 67 readonly OVERRIDE_VBMETA_IMAGE_PATH="$OPTARG" 68 ;; 69 b) 70 readonly OVERRIDE_BOOT_IMAGE_PATH="$OPTARG" 71 ;; 72 t) 73 readonly OTATOOLS_ZIP="$OPTARG" 74 ;; 75 s) 76 readonly INCLUDE_PRODUCT=true 77 ;; 78 d) 79 readonly SKIP_VBMETA_REPLACE=true 80 ;; 81 \?) 82 exit_badparam "Invalid options: -"$OPTARG"" 83 ;; 84 :) 85 exit_badparam "Option -"$OPTARG" requires an argument." 86 ;; 87 esac 88done 89 90if [[ -z "${VENDOR_VERSION+x}" && ! -z "${MODIFY_SYSTEM_SCRIPT+x}" ]] || \ 91 [[ ! -z "${VENDOR_VERSION+x}" && -z "${MODIFY_SYSTEM_SCRIPT+x}" ]]; then 92 exit_badparam "Options -v and -m must be set together." 93fi 94 95shift "$((OPTIND-1))" 96 97if [[ $# -lt 3 ]]; then 98 exit_badparam "Unexpected number of arguments" 99fi 100 101readonly SYSTEM_DIR="$1" 102readonly DEVICE_DIR="$2" 103readonly DIST_DIR="$3" 104readonly TEMP_DIR="$(mktemp -d /tmp/"$(basename $0)"_XXXXXXXX)" 105 106readonly SYSTEM_TARGET_FILES_ARCHIVE="$(find "$SYSTEM_DIR" -name "*-target_files-*.zip" -print)" 107if [[ ! -f "$SYSTEM_TARGET_FILES_ARCHIVE" ]]; then 108 exit_badparam "Could not find system target files archive in $SYSTEM_DIR." 109fi 110 111readonly DEVICE_ARCHIVE="$(find "$DEVICE_DIR" -name "*-img-*.zip" -print)" 112if [[ ! -f "$DEVICE_ARCHIVE" ]]; then 113 exit_badparam "Could not find device img archive in $DEVICE_DIR." 114fi 115 116readonly DEVICE_TARGET_FILES_ARCHIVE="$(find "$DEVICE_DIR" -name "*-target_files-*.zip" -print)" 117if [[ ! -f "$DEVICE_TARGET_FILES_ARCHIVE" ]]; then 118 exit_badparam "Could not find device target_files archive in $DEVICE_DIR." 119fi 120 121if [[ ! -z "${MODIFY_SYSTEM_SCRIPT+x}" && ! -f "$MODIFY_SYSTEM_SCRIPT" ]]; then 122 exit_badparam "Script not found: "$MODIFY_SYSTEM_SCRIPT"" 123fi 124 125if [[ ! -z "${OVERRIDE_VBMETA_IMAGE_PATH+x}" && ! -f "$OVERRIDE_VBMETA_IMAGE_PATH" ]]; then 126 exit_badparam "Specified vbmeta.img not found: "$OVERRIDE_VBMETA_IMAGE_PATH"" 127fi 128 129if [[ ! -z "${OVERRIDE_BOOT_IMAGE_PATH+x}" && ! -f "$OVERRIDE_BOOT_IMAGE_PATH" ]]; then 130 exit_badparam "Specified boot image not found: "$OVERRIDE_BOOT_IMAGE_PATH"" 131fi 132 133readonly DEVICE_ARTIFACTS_DIR="$TEMP_DIR"/device_archive_artifacts 134readonly DEVICE_IMAGES_DIR="$DEVICE_ARTIFACTS_DIR"/IMAGES 135readonly SYSTEM_ARTIFACTS_DIR="$TEMP_DIR"/system_artifacts 136readonly SYSTEM_IMAGES_DIR="$SYSTEM_ARTIFACTS_DIR"/IMAGES 137readonly OTATOOLS_DIR="$TEMP_DIR"/otatools 138 139readonly SPL_PROPERTY_NAME="ro.build.version.security_patch" 140readonly SYSTEM_BUILD_PROP="SYSTEM/build.prop" 141 142declare -a EXTRACT_SYSTEM_FILE_LIST 143EXTRACT_SYSTEM_FILE_LIST=( 144 IMAGES/system.img \ 145 IMAGES/vbmeta.img \ 146 "$SYSTEM_BUILD_PROP" \ 147) 148 149declare -a EXTRACT_VINTF_SYSTEM_FILE_LIST 150EXTRACT_VINTF_SYSTEM_FILE_LIST=( 151 "$SYSTEM_BUILD_PROP" \ 152) 153 154declare -a EXTRACT_DEVICE_FILE_LIST 155EXTRACT_DEVICE_FILE_LIST=( 156 */build.prop \ 157 META/* \ 158) 159 160declare -A SYSTEM_SEARCH_PATH 161SYSTEM_SEARCH_PATH=( \ 162 [/system]="SYSTEM" \ 163 [/product]="PRODUCT SYSTEM/product" \ 164 [/system_ext]="SYSTEM_EXT SYSTEM/system_ext" \ 165) 166 167declare -A DEVICE_SEARCH_PATH 168# Mixed build will not have /vendor to SYSTEM/vendor case 169DEVICE_SEARCH_PATH=( \ 170 [/vendor]="VENDOR" \ 171 [/odm]="ODM VENDOR/odm" \ 172) 173 174### 175# Uncompress otatools.zip and get vintf file list. 176if [[ ! -f "$OTATOOLS_ZIP" ]]; then 177 echo "WARNING: otatools.zip is missing. Add \"-t otatools.zip\" to enable checkvintf" 178else 179 readonly OTATOOLS_AVAILABLE=true 180 # Uncompress otatools 181 mkdir -p "$OTATOOLS_DIR" 182 unzip "$OTATOOLS_ZIP" bin/* lib64/* -d "$OTATOOLS_DIR" 183 # Set paths for using prebuilt host binaries. 184 export PATH="$OTATOOLS_DIR"/bin:"$PATH" 185 export LD_LIBRARY_PATH="$OTATOOLS_DIR"/lib64:"$LD_LIBRARY_PATH" 186 187 # Add vintf file to extract file list 188 declare -a VINTF_DUMP_FILE_LIST 189 VINTF_DUMP_FILE_LIST=( "$(checkvintf --dump-file-list)" ) 190 191 for vintf_file_list in ${VINTF_DUMP_FILE_LIST[*]}; do 192 if [[ "$vintf_file_list" == */ ]]; then 193 vintf_file_list="$vintf_file_list"\* 194 # Create system vintf file list for system target files archive 195 for system_dir in "${!SYSTEM_SEARCH_PATH[@]}"; do 196 if [[ "$vintf_file_list" == "$system_dir"/* ]]; then 197 for search_dir in ${SYSTEM_SEARCH_PATH["$system_dir"]}; do 198 search_file=${vintf_file_list/$system_dir/$search_dir} 199 unzip -l "$SYSTEM_TARGET_FILES_ARCHIVE" "$search_file" > /dev/null && \ 200 EXTRACT_VINTF_SYSTEM_FILE_LIST+=( "$search_file" ) 201 done 202 break 203 fi 204 done 205 # Create device vintf file list for device target files archive 206 for device_dir in "${!DEVICE_SEARCH_PATH[@]}"; do 207 if [[ "$vintf_file_list" == "$device_dir"/* ]]; then 208 for search_dir in ${DEVICE_SEARCH_PATH["$device_dir"]}; do 209 search_file=${vintf_file_list/$device_dir/$search_dir} 210 unzip -l "$DEVICE_TARGET_FILES_ARCHIVE" "$search_file" > /dev/null && \ 211 EXTRACT_DEVICE_FILE_LIST+=( "$search_file" ) 212 done 213 break 214 fi 215 done 216 fi 217 done 218fi 219 220### 221# Uncompress the system archives. 222if [[ "$INCLUDE_PRODUCT" == true ]]; then 223 unzip -l "$SYSTEM_TARGET_FILES_ARCHIVE" | grep -q IMAGES/product.img && 224 EXTRACT_SYSTEM_FILE_LIST+=(IMAGES/product.img) 225fi 226 227mkdir -p "$SYSTEM_ARTIFACTS_DIR" 228# Get system images. 229unzip "$SYSTEM_TARGET_FILES_ARCHIVE" "${EXTRACT_SYSTEM_FILE_LIST[@]}" \ 230 -d "$SYSTEM_ARTIFACTS_DIR" 231 232### 233# Uncompress the device archives. 234mkdir -p "$DEVICE_IMAGES_DIR" 235# Get device images. 236unzip "$DEVICE_ARCHIVE" -d "$DEVICE_IMAGES_DIR" 237# Get the device meta data. 238unzip "$DEVICE_TARGET_FILES_ARCHIVE" "${EXTRACT_DEVICE_FILE_LIST[@]}" \ 239 -d "$DEVICE_ARTIFACTS_DIR" 240 241### 242# Modify system.img if vendor version is provided. 243if [[ ! -z "${VENDOR_VERSION+x}" ]]; then 244 # Create copy of system target files package that can be modified 245 # since the original $SYSTEM_TARGET_FILES_ARCHIVE is a symlink to 246 # prebuilt files in cache 247 cp "$SYSTEM_TARGET_FILES_ARCHIVE" "$TEMP_DIR" 248 readonly COPY_SYSTEM_TARGET_FILES_ARCHIVE="$TEMP_DIR"/"$(basename "$SYSTEM_TARGET_FILES_ARCHIVE")" 249 250 # Check compatibility of security patch level 251 readonly SYSTEM_SPL=$(sed -n -r "s/^"$SPL_PROPERTY_NAME"=(.*)$/\1/p" "$SYSTEM_ARTIFACTS_DIR"/"$SYSTEM_BUILD_PROP") 252 readonly VENDOR_SPL=$(sed -n -r "s/^"$SPL_PROPERTY_NAME"=(.*)$/\1/p" "$DEVICE_ARTIFACTS_DIR"/"$SYSTEM_BUILD_PROP") 253 declare -a args 254 args=(-v "$VENDOR_VERSION" "$COPY_SYSTEM_TARGET_FILES_ARCHIVE") 255 if [[ "$SYSTEM_SPL" != "$VENDOR_SPL" ]]; then 256 echo "Security patch level mismatch detected..." 257 echo " SPL of system: "$SYSTEM_SPL"" 258 echo " SPL of vendor: "$VENDOR_SPL"" 259 args+=("$VENDOR_SPL") 260 fi 261 "$MODIFY_SYSTEM_SCRIPT" "${args[@]}" 262 # Replace system.img with newly modified system.img 263 unzip -o "$COPY_SYSTEM_TARGET_FILES_ARCHIVE" IMAGES/system.img -d "$SYSTEM_ARTIFACTS_DIR" 264fi 265 266# Check vintf 267if [[ "$OTATOOLS_AVAILABLE" == true ]]; then 268 # Overwrite VINTF system matrix to device artifacts dir 269 unzip -o "$SYSTEM_TARGET_FILES_ARCHIVE" "${EXTRACT_VINTF_SYSTEM_FILE_LIST[@]}" \ 270 -d "$DEVICE_ARTIFACTS_DIR" 271 check_target_files_vintf "$DEVICE_ARTIFACTS_DIR" 272fi 273 274### 275# Overwrite artifacts in the device archive to create the Mixed Build artifacts. 276cp "$SYSTEM_IMAGES_DIR"/system.img "$DEVICE_IMAGES_DIR"/ 277if [[ "$INCLUDE_PRODUCT" == true ]]; then 278 if [[ -f "$SYSTEM_IMAGES_DIR"/product.img ]]; then 279 cp "$SYSTEM_IMAGES_DIR"/product.img "$DEVICE_IMAGES_DIR"/ 280 else 281 rm -f "$DEVICE_IMAGES_DIR"/product.img 282 # Removed product partition from required partition list 283 sed -i "/partition-exists=product$/d" "$DEVICE_IMAGES_DIR"/android-info.txt 284 fi 285fi 286 287if [[ "$SKIP_VBMETA_REPLACE" == true ]]; then 288 # Totally skip the vbmeta.img replacement 289 echo "Skip vbmeta.img replacement." 290else 291 # Only override vbmeta if it is already present since fastboot update will try 292 # to flash whatever is in the archive. 293 if [[ -f "$DEVICE_IMAGES_DIR"/vbmeta.img ]]; then 294 readonly VBMETA_IMAGE_PATH="${OVERRIDE_VBMETA_IMAGE_PATH:-"$SYSTEM_IMAGES_DIR"/vbmeta.img}" 295 cp "$VBMETA_IMAGE_PATH" "$DEVICE_IMAGES_DIR"/ 296 fi 297fi 298 299# Override boot.img with the provided boot image file since fastboot update cmd 300# will try to flash boot.img in the archive. 301if [[ ! -z "${OVERRIDE_BOOT_IMAGE_PATH+x}" && -f "$DEVICE_IMAGES_DIR"/boot.img ]]; then 302 cp "$OVERRIDE_BOOT_IMAGE_PATH" "$DEVICE_IMAGES_DIR"/boot.img 303fi 304 305### 306# Create the Mixed Build archive. 307( 308 cd "$DEVICE_IMAGES_DIR" 309 zip -r mixed.zip ./* 310) 311 312### 313# Archive the artifacts. 314if [ -n "$DIST_DIR" ]; then 315 mkdir -p "$DIST_DIR" || true 316fi 317# Archive all the device artifacts. 318rsync --archive --verbose --copy-links --exclude='logs' \ 319 "$DEVICE_DIR"/* "$DIST_DIR" 320# Overwrite the image archive with the Mixed Build archive. 321OUT_ARCHIVE="$DIST_DIR"/"$(basename $DEVICE_ARCHIVE)" 322cp "$DEVICE_IMAGES_DIR"/mixed.zip "$OUT_ARCHIVE" 323# Overwrite android-info.txt with the updated one. 324cp "$DEVICE_IMAGES_DIR"/android-info.txt "$DIST_DIR"/ 325