1*5a923131SAndroid Build Coastguard Worker#!/bin/bash 2*5a923131SAndroid Build Coastguard Worker 3*5a923131SAndroid Build Coastguard Worker# 4*5a923131SAndroid Build Coastguard Worker# Copyright (C) 2015 The Android Open Source Project 5*5a923131SAndroid Build Coastguard Worker# 6*5a923131SAndroid Build Coastguard Worker# Licensed under the Apache License, Version 2.0 (the "License"); 7*5a923131SAndroid Build Coastguard Worker# you may not use this file except in compliance with the License. 8*5a923131SAndroid Build Coastguard Worker# You may obtain a copy of the License at 9*5a923131SAndroid Build Coastguard Worker# 10*5a923131SAndroid Build Coastguard Worker# http://www.apache.org/licenses/LICENSE-2.0 11*5a923131SAndroid Build Coastguard Worker# 12*5a923131SAndroid Build Coastguard Worker# Unless required by applicable law or agreed to in writing, software 13*5a923131SAndroid Build Coastguard Worker# distributed under the License is distributed on an "AS IS" BASIS, 14*5a923131SAndroid Build Coastguard Worker# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 15*5a923131SAndroid Build Coastguard Worker# See the License for the specific language governing permissions and 16*5a923131SAndroid Build Coastguard Worker# limitations under the License. 17*5a923131SAndroid Build Coastguard Worker# 18*5a923131SAndroid Build Coastguard Worker 19*5a923131SAndroid Build Coastguard Worker# Script to generate a Brillo update for use by the update engine. 20*5a923131SAndroid Build Coastguard Worker# 21*5a923131SAndroid Build Coastguard Worker# usage: brillo_update_payload COMMAND [ARGS] 22*5a923131SAndroid Build Coastguard Worker# The following commands are supported: 23*5a923131SAndroid Build Coastguard Worker# generate generate an unsigned payload 24*5a923131SAndroid Build Coastguard Worker# hash generate a payload or metadata hash 25*5a923131SAndroid Build Coastguard Worker# sign generate a signed payload 26*5a923131SAndroid Build Coastguard Worker# properties generate a properties file from a payload 27*5a923131SAndroid Build Coastguard Worker# verify verify a payload by recreating a target image. 28*5a923131SAndroid Build Coastguard Worker# check verify a payload using paycheck (static testing) 29*5a923131SAndroid Build Coastguard Worker# 30*5a923131SAndroid Build Coastguard Worker# Generate command arguments: 31*5a923131SAndroid Build Coastguard Worker# --payload generated unsigned payload output file 32*5a923131SAndroid Build Coastguard Worker# --source_image if defined, generate a delta payload from the 33*5a923131SAndroid Build Coastguard Worker# specified image to the target_image 34*5a923131SAndroid Build Coastguard Worker# --target_image the target image that should be sent to clients 35*5a923131SAndroid Build Coastguard Worker# --metadata_size_file if defined, generate a file containing the size 36*5a923131SAndroid Build Coastguard Worker# of the ayload metadata in bytes to the specified 37*5a923131SAndroid Build Coastguard Worker# file 38*5a923131SAndroid Build Coastguard Worker# --disable_fec_computation Disable the on device fec data computation for 39*5a923131SAndroid Build Coastguard Worker# incremental update. This feature is enabled by 40*5a923131SAndroid Build Coastguard Worker# default 41*5a923131SAndroid Build Coastguard Worker# --force_minor_version Override the minor version used for delta 42*5a923131SAndroid Build Coastguard Worker# generation. 43*5a923131SAndroid Build Coastguard Worker# 44*5a923131SAndroid Build Coastguard Worker# Hash command arguments: 45*5a923131SAndroid Build Coastguard Worker# --unsigned_payload the input unsigned payload to generate the hash from 46*5a923131SAndroid Build Coastguard Worker# --signature_size signature sizes in bytes in the following format: 47*5a923131SAndroid Build Coastguard Worker# "size1:size2[:...]" 48*5a923131SAndroid Build Coastguard Worker# --payload_hash_file if defined, generate a payload hash and output to the 49*5a923131SAndroid Build Coastguard Worker# specified file 50*5a923131SAndroid Build Coastguard Worker# --metadata_hash_file if defined, generate a metadata hash and output to the 51*5a923131SAndroid Build Coastguard Worker# specified file 52*5a923131SAndroid Build Coastguard Worker# 53*5a923131SAndroid Build Coastguard Worker# Sign command arguments: 54*5a923131SAndroid Build Coastguard Worker# --unsigned_payload the input unsigned payload to insert the signatures 55*5a923131SAndroid Build Coastguard Worker# --payload the output signed payload 56*5a923131SAndroid Build Coastguard Worker# --signature_size signature sizes in bytes in the following format: 57*5a923131SAndroid Build Coastguard Worker# "size1:size2[:...]" 58*5a923131SAndroid Build Coastguard Worker# --payload_signature_file the payload signature files in the following 59*5a923131SAndroid Build Coastguard Worker# format: 60*5a923131SAndroid Build Coastguard Worker# "payload_signature1:payload_signature2[:...]" 61*5a923131SAndroid Build Coastguard Worker# --metadata_signature_file the metadata signature files in the following 62*5a923131SAndroid Build Coastguard Worker# format: 63*5a923131SAndroid Build Coastguard Worker# "metadata_signature1:metadata_signature2[:...]" 64*5a923131SAndroid Build Coastguard Worker# --metadata_size_file if defined, generate a file containing the size of 65*5a923131SAndroid Build Coastguard Worker# the signed payload metadata in bytes to the 66*5a923131SAndroid Build Coastguard Worker# specified file 67*5a923131SAndroid Build Coastguard Worker# Note that the number of signature sizes and payload signatures have to match. 68*5a923131SAndroid Build Coastguard Worker# 69*5a923131SAndroid Build Coastguard Worker# Properties command arguments: 70*5a923131SAndroid Build Coastguard Worker# --payload the input signed or unsigned payload 71*5a923131SAndroid Build Coastguard Worker# --properties_file the output path where to write the properties, or 72*5a923131SAndroid Build Coastguard Worker# '-' for stdout. 73*5a923131SAndroid Build Coastguard Worker# Verify command arguments: 74*5a923131SAndroid Build Coastguard Worker# --payload payload input file 75*5a923131SAndroid Build Coastguard Worker# --source_image verify payload to the specified source image. 76*5a923131SAndroid Build Coastguard Worker# --target_image the target image to verify upon. 77*5a923131SAndroid Build Coastguard Worker# 78*5a923131SAndroid Build Coastguard Worker# Check command arguments: 79*5a923131SAndroid Build Coastguard Worker# Symmetrical with the verify command. 80*5a923131SAndroid Build Coastguard Worker 81*5a923131SAndroid Build Coastguard Worker 82*5a923131SAndroid Build Coastguard Worker# Exit codes: 83*5a923131SAndroid Build Coastguard WorkerEX_UNSUPPORTED_DELTA=100 84*5a923131SAndroid Build Coastguard Worker 85*5a923131SAndroid Build Coastguard Workerwarn() { 86*5a923131SAndroid Build Coastguard Worker echo "brillo_update_payload: warning: $*" >&2 87*5a923131SAndroid Build Coastguard Worker} 88*5a923131SAndroid Build Coastguard Worker 89*5a923131SAndroid Build Coastguard Workerdie() { 90*5a923131SAndroid Build Coastguard Worker echo "brillo_update_payload: error: $*" >&2 91*5a923131SAndroid Build Coastguard Worker exit 1 92*5a923131SAndroid Build Coastguard Worker} 93*5a923131SAndroid Build Coastguard Worker 94*5a923131SAndroid Build Coastguard Worker# Loads shflags. We first look at the default install location; then our own 95*5a923131SAndroid Build Coastguard Worker# directory; finally the parent directory. 96*5a923131SAndroid Build Coastguard Workerload_shflags() { 97*5a923131SAndroid Build Coastguard Worker local my_dir="$(dirname "$(readlink -f "$0")")" 98*5a923131SAndroid Build Coastguard Worker local path 99*5a923131SAndroid Build Coastguard Worker for path in /usr/share/misc \ 100*5a923131SAndroid Build Coastguard Worker "${my_dir}"/lib/shflags \ 101*5a923131SAndroid Build Coastguard Worker "${my_dir}"/../lib/shflags; do 102*5a923131SAndroid Build Coastguard Worker if [[ -r "${path}/shflags" ]]; then 103*5a923131SAndroid Build Coastguard Worker . "${path}/shflags" || die "Could not load ${path}/shflags." 104*5a923131SAndroid Build Coastguard Worker return 105*5a923131SAndroid Build Coastguard Worker fi 106*5a923131SAndroid Build Coastguard Worker done 107*5a923131SAndroid Build Coastguard Worker die "Could not find shflags." 108*5a923131SAndroid Build Coastguard Worker} 109*5a923131SAndroid Build Coastguard Worker 110*5a923131SAndroid Build Coastguard Workerload_shflags 111*5a923131SAndroid Build Coastguard Worker 112*5a923131SAndroid Build Coastguard WorkerHELP_GENERATE="generate: Generate an unsigned update payload." 113*5a923131SAndroid Build Coastguard WorkerHELP_HASH="hash: Generate the hashes of the unsigned payload and metadata used \ 114*5a923131SAndroid Build Coastguard Workerfor signing." 115*5a923131SAndroid Build Coastguard WorkerHELP_SIGN="sign: Insert the signatures into the unsigned payload." 116*5a923131SAndroid Build Coastguard WorkerHELP_PROPERTIES="properties: Extract payload properties to a file." 117*5a923131SAndroid Build Coastguard WorkerHELP_VERIFY="verify: Verify a (signed) update payload using delta_generator." 118*5a923131SAndroid Build Coastguard WorkerHELP_CHECK="check: Check a (signed) update payload using paycheck (static \ 119*5a923131SAndroid Build Coastguard Workertesting)." 120*5a923131SAndroid Build Coastguard Worker 121*5a923131SAndroid Build Coastguard Workerusage() { 122*5a923131SAndroid Build Coastguard Worker echo "Supported commands:" 123*5a923131SAndroid Build Coastguard Worker echo 124*5a923131SAndroid Build Coastguard Worker echo "${HELP_GENERATE}" 125*5a923131SAndroid Build Coastguard Worker echo "${HELP_HASH}" 126*5a923131SAndroid Build Coastguard Worker echo "${HELP_SIGN}" 127*5a923131SAndroid Build Coastguard Worker echo "${HELP_PROPERTIES}" 128*5a923131SAndroid Build Coastguard Worker echo "${HELP_VERIFY}" 129*5a923131SAndroid Build Coastguard Worker echo "${HELP_CHECK}" 130*5a923131SAndroid Build Coastguard Worker echo 131*5a923131SAndroid Build Coastguard Worker echo "Use: \"$0 <command> --help\" for more options." 132*5a923131SAndroid Build Coastguard Worker} 133*5a923131SAndroid Build Coastguard Worker 134*5a923131SAndroid Build Coastguard Worker# Check that a command is specified. 135*5a923131SAndroid Build Coastguard Workerif [[ $# -lt 1 ]]; then 136*5a923131SAndroid Build Coastguard Worker echo "Please specify a command [generate|hash|sign|properties|verify|check]" 137*5a923131SAndroid Build Coastguard Worker exit 1 138*5a923131SAndroid Build Coastguard Workerfi 139*5a923131SAndroid Build Coastguard Worker 140*5a923131SAndroid Build Coastguard Worker# Parse command. 141*5a923131SAndroid Build Coastguard WorkerCOMMAND="${1:-}" 142*5a923131SAndroid Build Coastguard Workershift 143*5a923131SAndroid Build Coastguard Worker 144*5a923131SAndroid Build Coastguard Workercase "${COMMAND}" in 145*5a923131SAndroid Build Coastguard Worker generate) 146*5a923131SAndroid Build Coastguard Worker FLAGS_HELP="${HELP_GENERATE}" 147*5a923131SAndroid Build Coastguard Worker ;; 148*5a923131SAndroid Build Coastguard Worker 149*5a923131SAndroid Build Coastguard Worker hash) 150*5a923131SAndroid Build Coastguard Worker FLAGS_HELP="${HELP_HASH}" 151*5a923131SAndroid Build Coastguard Worker ;; 152*5a923131SAndroid Build Coastguard Worker 153*5a923131SAndroid Build Coastguard Worker sign) 154*5a923131SAndroid Build Coastguard Worker FLAGS_HELP="${HELP_SIGN}" 155*5a923131SAndroid Build Coastguard Worker ;; 156*5a923131SAndroid Build Coastguard Worker 157*5a923131SAndroid Build Coastguard Worker properties) 158*5a923131SAndroid Build Coastguard Worker FLAGS_HELP="${HELP_PROPERTIES}" 159*5a923131SAndroid Build Coastguard Worker ;; 160*5a923131SAndroid Build Coastguard Worker 161*5a923131SAndroid Build Coastguard Worker verify) 162*5a923131SAndroid Build Coastguard Worker FLAGS_HELP="${HELP_VERIFY}" 163*5a923131SAndroid Build Coastguard Worker ;; 164*5a923131SAndroid Build Coastguard Worker 165*5a923131SAndroid Build Coastguard Worker check) 166*5a923131SAndroid Build Coastguard Worker FLAGS_HELP="${HELP_CHECK}" 167*5a923131SAndroid Build Coastguard Worker ;; 168*5a923131SAndroid Build Coastguard Worker 169*5a923131SAndroid Build Coastguard Worker *) 170*5a923131SAndroid Build Coastguard Worker echo "Unrecognized command: \"${COMMAND}\"" >&2 171*5a923131SAndroid Build Coastguard Worker usage >&2 172*5a923131SAndroid Build Coastguard Worker exit 1 173*5a923131SAndroid Build Coastguard Worker ;; 174*5a923131SAndroid Build Coastguard Workeresac 175*5a923131SAndroid Build Coastguard Worker 176*5a923131SAndroid Build Coastguard Worker# Flags 177*5a923131SAndroid Build Coastguard WorkerFLAGS_HELP="Usage: $0 ${COMMAND} [flags] 178*5a923131SAndroid Build Coastguard Worker${FLAGS_HELP}" 179*5a923131SAndroid Build Coastguard Worker 180*5a923131SAndroid Build Coastguard Workerif [[ "${COMMAND}" == "generate" ]]; then 181*5a923131SAndroid Build Coastguard Worker DEFINE_string payload "" \ 182*5a923131SAndroid Build Coastguard Worker "Path to output the generated unsigned payload file." 183*5a923131SAndroid Build Coastguard Worker DEFINE_string target_image "" \ 184*5a923131SAndroid Build Coastguard Worker "Path to the target image that should be sent to clients." 185*5a923131SAndroid Build Coastguard Worker DEFINE_string source_image "" \ 186*5a923131SAndroid Build Coastguard Worker "Optional: Path to a source image. If specified, this makes a delta update." 187*5a923131SAndroid Build Coastguard Worker DEFINE_string metadata_size_file "" \ 188*5a923131SAndroid Build Coastguard Worker "Optional: Path to output metadata size." 189*5a923131SAndroid Build Coastguard Worker DEFINE_string max_timestamp "" \ 190*5a923131SAndroid Build Coastguard Worker "Optional: The maximum unix timestamp of the OS allowed to apply this \ 191*5a923131SAndroid Build Coastguard Workerpayload, should be set to a number higher than the build timestamp of the \ 192*5a923131SAndroid Build Coastguard Workersystem running on the device, 0 if not specified." 193*5a923131SAndroid Build Coastguard Worker DEFINE_string partition_timestamps "" \ 194*5a923131SAndroid Build Coastguard Worker "Optional: Per-partition maximum unix timestamp of the OS allowed to \ 195*5a923131SAndroid Build Coastguard Workerapply this payload. Should be a comma separated key value pairs. e.x.\ 196*5a923131SAndroid Build Coastguard Workersystem:1234,vendor:456" 197*5a923131SAndroid Build Coastguard Worker DEFINE_string disable_fec_computation "" \ 198*5a923131SAndroid Build Coastguard Worker "Optional: Disables the on device fec data computation for incremental \ 199*5a923131SAndroid Build Coastguard Workerupdate. This feature is enabled by default." 200*5a923131SAndroid Build Coastguard Worker DEFINE_string disable_verity_computation "" \ 201*5a923131SAndroid Build Coastguard Worker "Optional: Disables the on device verity computation for incremental \ 202*5a923131SAndroid Build Coastguard Workerupdate. This feature is enabled by default." 203*5a923131SAndroid Build Coastguard Worker DEFINE_string is_partial_update "" \ 204*5a923131SAndroid Build Coastguard Worker "Optional: True if the payload is for partial update. i.e. it only updates \ 205*5a923131SAndroid Build Coastguard Workera subset of partitions on device." 206*5a923131SAndroid Build Coastguard Worker DEFINE_string full_boot "" "Will include full boot image" 207*5a923131SAndroid Build Coastguard Worker DEFINE_string disable_vabc "" \ 208*5a923131SAndroid Build Coastguard Worker "Optional: Disables Virtual AB Compression when installing the OTA" 209*5a923131SAndroid Build Coastguard Worker DEFINE_string enable_vabc_xor "" \ 210*5a923131SAndroid Build Coastguard Worker "Optional: Enable the use of Virtual AB Compression XOR feature" 211*5a923131SAndroid Build Coastguard Worker DEFINE_string force_minor_version "" \ 212*5a923131SAndroid Build Coastguard Worker "Optional: Override the minor version for the delta generation." 213*5a923131SAndroid Build Coastguard Worker DEFINE_string compressor_types "" \ 214*5a923131SAndroid Build Coastguard Worker "Optional: allowed compressor types. Colon separated, allowe values are bz2 and brotli" 215*5a923131SAndroid Build Coastguard Worker DEFINE_string enable_zucchini "" \ 216*5a923131SAndroid Build Coastguard Worker "Optional: Whether to enable zucchini diffing" 217*5a923131SAndroid Build Coastguard Worker DEFINE_string enable_lz4diff "" \ 218*5a923131SAndroid Build Coastguard Worker "Optional: Whether to enable lz4 diffing for EROFS" 219*5a923131SAndroid Build Coastguard Worker DEFINE_string liblz4_path "" \ 220*5a923131SAndroid Build Coastguard Worker "Required if --enabled_lz4diff true is passed. Path to liblz4.so. delta_generator will use this copy of liblz4.so for compression. It is important that this copy of liblz4.so is the same as the one on source build." 221*5a923131SAndroid Build Coastguard Worker DEFINE_string erofs_compression_param "" \ 222*5a923131SAndroid Build Coastguard Worker "Compression parameter passed to mkfs.erofs's -z option." 223*5a923131SAndroid Build Coastguard Worker DEFINE_string security_patch_level "" \ 224*5a923131SAndroid Build Coastguard Worker "Optional: security patch level of this OTA" 225*5a923131SAndroid Build Coastguard Worker DEFINE_string max_threads "" \ 226*5a923131SAndroid Build Coastguard Worker "Optional: specifies max_threads used to generate OTA" 227*5a923131SAndroid Build Coastguard Workerfi 228*5a923131SAndroid Build Coastguard Workerif [[ "${COMMAND}" == "hash" || "${COMMAND}" == "sign" ]]; then 229*5a923131SAndroid Build Coastguard Worker DEFINE_string unsigned_payload "" "Path to the input unsigned payload." 230*5a923131SAndroid Build Coastguard Worker DEFINE_string signature_size "" \ 231*5a923131SAndroid Build Coastguard Worker "Signature sizes in bytes in the following format: size1:size2[:...]" 232*5a923131SAndroid Build Coastguard Workerfi 233*5a923131SAndroid Build Coastguard Workerif [[ "${COMMAND}" == "hash" ]]; then 234*5a923131SAndroid Build Coastguard Worker DEFINE_string metadata_hash_file "" \ 235*5a923131SAndroid Build Coastguard Worker "Optional: Path to output metadata hash file." 236*5a923131SAndroid Build Coastguard Worker DEFINE_string payload_hash_file "" \ 237*5a923131SAndroid Build Coastguard Worker "Optional: Path to output payload hash file." 238*5a923131SAndroid Build Coastguard Workerfi 239*5a923131SAndroid Build Coastguard Workerif [[ "${COMMAND}" == "sign" ]]; then 240*5a923131SAndroid Build Coastguard Worker DEFINE_string payload "" \ 241*5a923131SAndroid Build Coastguard Worker "Path to output the generated unsigned payload file." 242*5a923131SAndroid Build Coastguard Worker DEFINE_string metadata_signature_file "" \ 243*5a923131SAndroid Build Coastguard Worker "The metatada signatures in the following format: \ 244*5a923131SAndroid Build Coastguard Workermetadata_signature1:metadata_signature2[:...]" 245*5a923131SAndroid Build Coastguard Worker DEFINE_string payload_signature_file "" \ 246*5a923131SAndroid Build Coastguard Worker "The payload signatures in the following format: \ 247*5a923131SAndroid Build Coastguard Workerpayload_signature1:payload_signature2[:...]" 248*5a923131SAndroid Build Coastguard Worker DEFINE_string metadata_size_file "" \ 249*5a923131SAndroid Build Coastguard Worker "Optional: Path to output metadata size." 250*5a923131SAndroid Build Coastguard Workerfi 251*5a923131SAndroid Build Coastguard Workerif [[ "${COMMAND}" == "properties" ]]; then 252*5a923131SAndroid Build Coastguard Worker DEFINE_string payload "" \ 253*5a923131SAndroid Build Coastguard Worker "Path to the input signed or unsigned payload file." 254*5a923131SAndroid Build Coastguard Worker DEFINE_string properties_file "-" \ 255*5a923131SAndroid Build Coastguard Worker "Path to output the extracted property files. If '-' is passed stdout will \ 256*5a923131SAndroid Build Coastguard Workerbe used." 257*5a923131SAndroid Build Coastguard Workerfi 258*5a923131SAndroid Build Coastguard Workerif [[ "${COMMAND}" == "verify" || "${COMMAND}" == "check" ]]; then 259*5a923131SAndroid Build Coastguard Worker DEFINE_string payload "" \ 260*5a923131SAndroid Build Coastguard Worker "Path to the input payload file." 261*5a923131SAndroid Build Coastguard Worker DEFINE_string target_image "" \ 262*5a923131SAndroid Build Coastguard Worker "Path to the target image to verify upon." 263*5a923131SAndroid Build Coastguard Worker DEFINE_string source_image "" \ 264*5a923131SAndroid Build Coastguard Worker "Optional: Path to a source image. If specified, the delta update is \ 265*5a923131SAndroid Build Coastguard Workerapplied to this." 266*5a923131SAndroid Build Coastguard Workerfi 267*5a923131SAndroid Build Coastguard Worker 268*5a923131SAndroid Build Coastguard WorkerDEFINE_string work_dir "${TMPDIR:-/tmp}" "Where to dump temporary files." 269*5a923131SAndroid Build Coastguard Worker 270*5a923131SAndroid Build Coastguard Worker# Parse command line flag arguments 271*5a923131SAndroid Build Coastguard WorkerFLAGS "$@" || exit 1 272*5a923131SAndroid Build Coastguard Workereval set -- "${FLAGS_ARGV}" 273*5a923131SAndroid Build Coastguard Workerset -e 274*5a923131SAndroid Build Coastguard Worker 275*5a923131SAndroid Build Coastguard Worker# Override the TMPDIR with the passed work_dir flags, which anyway defaults to 276*5a923131SAndroid Build Coastguard Worker# ${TMPDIR}. 277*5a923131SAndroid Build Coastguard WorkerTMPDIR="${FLAGS_work_dir}" 278*5a923131SAndroid Build Coastguard Workerexport TMPDIR 279*5a923131SAndroid Build Coastguard Worker 280*5a923131SAndroid Build Coastguard Worker# Associative arrays from partition name to file in the source and target 281*5a923131SAndroid Build Coastguard Worker# images. The size of the updated area must be the size of the file. 282*5a923131SAndroid Build Coastguard Workerdeclare -A SRC_PARTITIONS 283*5a923131SAndroid Build Coastguard Workerdeclare -A DST_PARTITIONS 284*5a923131SAndroid Build Coastguard Worker 285*5a923131SAndroid Build Coastguard Worker# Associative arrays for the .map files associated with each src/dst partition 286*5a923131SAndroid Build Coastguard Worker# file in SRC_PARTITIONS and DST_PARTITIONS. 287*5a923131SAndroid Build Coastguard Workerdeclare -A SRC_PARTITIONS_MAP 288*5a923131SAndroid Build Coastguard Workerdeclare -A DST_PARTITIONS_MAP 289*5a923131SAndroid Build Coastguard Worker 290*5a923131SAndroid Build Coastguard Worker# List of partition names in order. 291*5a923131SAndroid Build Coastguard Workerdeclare -a PARTITIONS_ORDER 292*5a923131SAndroid Build Coastguard Worker 293*5a923131SAndroid Build Coastguard Worker# A list of PIDs of the extract_image workers. 294*5a923131SAndroid Build Coastguard WorkerEXTRACT_IMAGE_PIDS=() 295*5a923131SAndroid Build Coastguard Worker 296*5a923131SAndroid Build Coastguard Worker# A list of temporary files to remove during cleanup. 297*5a923131SAndroid Build Coastguard WorkerCLEANUP_FILES=() 298*5a923131SAndroid Build Coastguard Worker 299*5a923131SAndroid Build Coastguard Worker# Global options to force the version of the payload. 300*5a923131SAndroid Build Coastguard WorkerFORCE_MAJOR_VERSION="" 301*5a923131SAndroid Build Coastguard WorkerFORCE_MINOR_VERSION="" 302*5a923131SAndroid Build Coastguard Worker 303*5a923131SAndroid Build Coastguard Worker# Path to the postinstall config file in target image if exists. 304*5a923131SAndroid Build Coastguard WorkerPOSTINSTALL_CONFIG_FILE="" 305*5a923131SAndroid Build Coastguard Worker 306*5a923131SAndroid Build Coastguard Worker# Path to the dynamic partition info file in target image if exists. 307*5a923131SAndroid Build Coastguard WorkerDYNAMIC_PARTITION_INFO_FILE="" 308*5a923131SAndroid Build Coastguard Worker 309*5a923131SAndroid Build Coastguard Worker# Path to the META/apex_info.pb found in target build 310*5a923131SAndroid Build Coastguard WorkerAPEX_INFO_FILE="" 311*5a923131SAndroid Build Coastguard Worker 312*5a923131SAndroid Build Coastguard Worker# read_option_int <file.txt> <option_key> [default_value] 313*5a923131SAndroid Build Coastguard Worker# 314*5a923131SAndroid Build Coastguard Worker# Reads the unsigned integer value associated with |option_key| in a key=value 315*5a923131SAndroid Build Coastguard Worker# file |file.txt|. Prints the read value if found and valid, otherwise prints 316*5a923131SAndroid Build Coastguard Worker# the |default_value|. 317*5a923131SAndroid Build Coastguard Workerread_option_uint() { 318*5a923131SAndroid Build Coastguard Worker local file_txt="$1" 319*5a923131SAndroid Build Coastguard Worker local option_key="$2" 320*5a923131SAndroid Build Coastguard Worker local default_value="${3:-}" 321*5a923131SAndroid Build Coastguard Worker local value 322*5a923131SAndroid Build Coastguard Worker if value=$(grep "^${option_key}=" "${file_txt}" | tail -n 1); then 323*5a923131SAndroid Build Coastguard Worker if value=$(echo "${value}" | cut -f 2- -d "=" | grep -E "^[0-9]+$"); then 324*5a923131SAndroid Build Coastguard Worker echo "${value}" 325*5a923131SAndroid Build Coastguard Worker return 326*5a923131SAndroid Build Coastguard Worker fi 327*5a923131SAndroid Build Coastguard Worker fi 328*5a923131SAndroid Build Coastguard Worker echo "${default_value}" 329*5a923131SAndroid Build Coastguard Worker} 330*5a923131SAndroid Build Coastguard Worker 331*5a923131SAndroid Build Coastguard Worker# truncate_file <file_path> <file_size> 332*5a923131SAndroid Build Coastguard Worker# 333*5a923131SAndroid Build Coastguard Worker# Truncate the given |file_path| to |file_size| using python. 334*5a923131SAndroid Build Coastguard Worker# The truncate binary might not be available. 335*5a923131SAndroid Build Coastguard Workertruncate_file() { 336*5a923131SAndroid Build Coastguard Worker local file_path="$1" 337*5a923131SAndroid Build Coastguard Worker local file_size="$2" 338*5a923131SAndroid Build Coastguard Worker python -c "open(\"${file_path}\", 'a').truncate(${file_size})" 339*5a923131SAndroid Build Coastguard Worker} 340*5a923131SAndroid Build Coastguard Worker 341*5a923131SAndroid Build Coastguard Worker# Create a temporary file in the work_dir with an optional pattern name. 342*5a923131SAndroid Build Coastguard Worker# Prints the name of the newly created file. 343*5a923131SAndroid Build Coastguard Workercreate_tempfile() { 344*5a923131SAndroid Build Coastguard Worker local pattern="${1:-tempfile.XXXXXX}" 345*5a923131SAndroid Build Coastguard Worker mktemp --tmpdir="${FLAGS_work_dir}" "${pattern}" 346*5a923131SAndroid Build Coastguard Worker} 347*5a923131SAndroid Build Coastguard Worker 348*5a923131SAndroid Build Coastguard Workercleanup() { 349*5a923131SAndroid Build Coastguard Worker local err="" 350*5a923131SAndroid Build Coastguard Worker rm -f "${CLEANUP_FILES[@]}" || err=1 351*5a923131SAndroid Build Coastguard Worker 352*5a923131SAndroid Build Coastguard Worker # If we are cleaning up after an error, or if we got an error during 353*5a923131SAndroid Build Coastguard Worker # cleanup (even if we eventually succeeded) return a non-zero exit 354*5a923131SAndroid Build Coastguard Worker # code. This triggers additional logging in most environments that call 355*5a923131SAndroid Build Coastguard Worker # this script. 356*5a923131SAndroid Build Coastguard Worker if [[ -n "${err}" ]]; then 357*5a923131SAndroid Build Coastguard Worker die "Cleanup encountered an error." 358*5a923131SAndroid Build Coastguard Worker fi 359*5a923131SAndroid Build Coastguard Worker} 360*5a923131SAndroid Build Coastguard Worker 361*5a923131SAndroid Build Coastguard Workercleanup_on_error() { 362*5a923131SAndroid Build Coastguard Worker trap - INT TERM ERR EXIT 363*5a923131SAndroid Build Coastguard Worker cleanup 364*5a923131SAndroid Build Coastguard Worker die "Cleanup success after an error." 365*5a923131SAndroid Build Coastguard Worker} 366*5a923131SAndroid Build Coastguard Worker 367*5a923131SAndroid Build Coastguard Workercleanup_on_exit() { 368*5a923131SAndroid Build Coastguard Worker trap - INT TERM ERR EXIT 369*5a923131SAndroid Build Coastguard Worker cleanup 370*5a923131SAndroid Build Coastguard Worker} 371*5a923131SAndroid Build Coastguard Worker 372*5a923131SAndroid Build Coastguard Workertrap cleanup_on_error INT TERM ERR 373*5a923131SAndroid Build Coastguard Workertrap cleanup_on_exit EXIT 374*5a923131SAndroid Build Coastguard Worker 375*5a923131SAndroid Build Coastguard Worker# extract_file <zip_file> <entry_name> <destination> 376*5a923131SAndroid Build Coastguard Worker# 377*5a923131SAndroid Build Coastguard Worker# Extracts |entry_name| from |zip_file| to |destination|. 378*5a923131SAndroid Build Coastguard Workerextract_file() { 379*5a923131SAndroid Build Coastguard Worker local zip_file="$1" 380*5a923131SAndroid Build Coastguard Worker local entry_name="$2" 381*5a923131SAndroid Build Coastguard Worker local destination="$3" 382*5a923131SAndroid Build Coastguard Worker 383*5a923131SAndroid Build Coastguard Worker # unzip -p won't report error upon ENOSPC. Therefore, create a temp directory 384*5a923131SAndroid Build Coastguard Worker # as the destination of the unzip, and move the file to the intended 385*5a923131SAndroid Build Coastguard Worker # destination. 386*5a923131SAndroid Build Coastguard Worker local output_directory=$( 387*5a923131SAndroid Build Coastguard Worker mktemp --directory --tmpdir="${FLAGS_work_dir}" "TEMP.XXXXXX") 388*5a923131SAndroid Build Coastguard Worker unzip "${zip_file}" "${entry_name}" -d "${output_directory}" || 389*5a923131SAndroid Build Coastguard Worker { rm -rf "${output_directory}"; die "Failed to extract ${entry_name}"; } 390*5a923131SAndroid Build Coastguard Worker 391*5a923131SAndroid Build Coastguard Worker mv "${output_directory}/${entry_name}" "${destination}" 392*5a923131SAndroid Build Coastguard Worker rm -rf "${output_directory}" 393*5a923131SAndroid Build Coastguard Worker} 394*5a923131SAndroid Build Coastguard Worker 395*5a923131SAndroid Build Coastguard Worker# extract_image <image> <partitions_array> [partitions_order] 396*5a923131SAndroid Build Coastguard Worker# 397*5a923131SAndroid Build Coastguard Worker# Detect the format of the |image| file and extract its updatable partitions 398*5a923131SAndroid Build Coastguard Worker# into new temporary files. Add the list of partition names and its files to the 399*5a923131SAndroid Build Coastguard Worker# associative array passed in |partitions_array|. If |partitions_order| is 400*5a923131SAndroid Build Coastguard Worker# passed, set it to list of partition names in order. 401*5a923131SAndroid Build Coastguard Workerextract_image() { 402*5a923131SAndroid Build Coastguard Worker local image="$1" 403*5a923131SAndroid Build Coastguard Worker 404*5a923131SAndroid Build Coastguard Worker # Brillo images are zip files. We detect the 4-byte magic header of the zip 405*5a923131SAndroid Build Coastguard Worker # file. 406*5a923131SAndroid Build Coastguard Worker local magic=$(xxd -p -l4 "${image}") 407*5a923131SAndroid Build Coastguard Worker if [[ "${magic}" == "504b0304" ]]; then 408*5a923131SAndroid Build Coastguard Worker echo "Detected .zip file, extracting Brillo image." 409*5a923131SAndroid Build Coastguard Worker extract_image_brillo "$@" 410*5a923131SAndroid Build Coastguard Worker return 411*5a923131SAndroid Build Coastguard Worker fi 412*5a923131SAndroid Build Coastguard Worker 413*5a923131SAndroid Build Coastguard Worker # Chrome OS images are GPT partitioned disks. We should have the cgpt binary 414*5a923131SAndroid Build Coastguard Worker # bundled here and we will use it to extract the partitions, so the GPT 415*5a923131SAndroid Build Coastguard Worker # headers must be valid. 416*5a923131SAndroid Build Coastguard Worker if cgpt show -q -n "${image}" >/dev/null; then 417*5a923131SAndroid Build Coastguard Worker echo "Detected GPT image, extracting Chrome OS image." 418*5a923131SAndroid Build Coastguard Worker extract_image_cros "$@" 419*5a923131SAndroid Build Coastguard Worker return 420*5a923131SAndroid Build Coastguard Worker fi 421*5a923131SAndroid Build Coastguard Worker 422*5a923131SAndroid Build Coastguard Worker die "Couldn't detect the image format of ${image}" 423*5a923131SAndroid Build Coastguard Worker} 424*5a923131SAndroid Build Coastguard Worker 425*5a923131SAndroid Build Coastguard Worker# extract_image_cros <image.bin> <partitions_array> [partitions_order] 426*5a923131SAndroid Build Coastguard Worker# 427*5a923131SAndroid Build Coastguard Worker# Extract Chromium OS recovery images into new temporary files. 428*5a923131SAndroid Build Coastguard Workerextract_image_cros() { 429*5a923131SAndroid Build Coastguard Worker local image="$1" 430*5a923131SAndroid Build Coastguard Worker local partitions_array="$2" 431*5a923131SAndroid Build Coastguard Worker local partitions_order="${3:-}" 432*5a923131SAndroid Build Coastguard Worker 433*5a923131SAndroid Build Coastguard Worker local kernel root 434*5a923131SAndroid Build Coastguard Worker kernel=$(create_tempfile "kernel.bin.XXXXXX") 435*5a923131SAndroid Build Coastguard Worker CLEANUP_FILES+=("${kernel}") 436*5a923131SAndroid Build Coastguard Worker root=$(create_tempfile "root.bin.XXXXXX") 437*5a923131SAndroid Build Coastguard Worker CLEANUP_FILES+=("${root}") 438*5a923131SAndroid Build Coastguard Worker 439*5a923131SAndroid Build Coastguard Worker cros_generate_update_payload --extract \ 440*5a923131SAndroid Build Coastguard Worker --image "${image}" \ 441*5a923131SAndroid Build Coastguard Worker --kern_path "${kernel}" --root_path "${root}" 442*5a923131SAndroid Build Coastguard Worker 443*5a923131SAndroid Build Coastguard Worker # Chrome OS now uses major_version 2 payloads for all boards. 444*5a923131SAndroid Build Coastguard Worker # See crbug.com/794404 for more information. 445*5a923131SAndroid Build Coastguard Worker FORCE_MAJOR_VERSION="2" 446*5a923131SAndroid Build Coastguard Worker 447*5a923131SAndroid Build Coastguard Worker eval ${partitions_array}[kernel]=\""${kernel}"\" 448*5a923131SAndroid Build Coastguard Worker eval ${partitions_array}[root]=\""${root}"\" 449*5a923131SAndroid Build Coastguard Worker 450*5a923131SAndroid Build Coastguard Worker if [[ -n "${partitions_order}" ]]; then 451*5a923131SAndroid Build Coastguard Worker eval "${partitions_order}=( \"root\" \"kernel\" )" 452*5a923131SAndroid Build Coastguard Worker fi 453*5a923131SAndroid Build Coastguard Worker 454*5a923131SAndroid Build Coastguard Worker local part varname 455*5a923131SAndroid Build Coastguard Worker for part in kernel root; do 456*5a923131SAndroid Build Coastguard Worker varname="${partitions_array}[${part}]" 457*5a923131SAndroid Build Coastguard Worker printf "md5sum of %s: " "${varname}" 458*5a923131SAndroid Build Coastguard Worker md5sum "${!varname}" 459*5a923131SAndroid Build Coastguard Worker done 460*5a923131SAndroid Build Coastguard Worker} 461*5a923131SAndroid Build Coastguard Worker 462*5a923131SAndroid Build Coastguard Worker# extract_partition_brillo <target_files.zip> <partitions_array> <partition> 463*5a923131SAndroid Build Coastguard Worker# <part_file> <part_map_file> 464*5a923131SAndroid Build Coastguard Worker# 465*5a923131SAndroid Build Coastguard Worker# Extract the <partition> from target_files zip file into <part_file> and its 466*5a923131SAndroid Build Coastguard Worker# map file into <part_map_file>. 467*5a923131SAndroid Build Coastguard Workerextract_partition_brillo() { 468*5a923131SAndroid Build Coastguard Worker local image="$1" 469*5a923131SAndroid Build Coastguard Worker local partitions_array="$2" 470*5a923131SAndroid Build Coastguard Worker local part="$3" 471*5a923131SAndroid Build Coastguard Worker local part_file="$4" 472*5a923131SAndroid Build Coastguard Worker local part_map_file="$5" 473*5a923131SAndroid Build Coastguard Worker 474*5a923131SAndroid Build Coastguard Worker # For each partition, we in turn look for its image file under IMAGES/ and 475*5a923131SAndroid Build Coastguard Worker # RADIO/ in the given target_files zip file. 476*5a923131SAndroid Build Coastguard Worker local path path_in_zip 477*5a923131SAndroid Build Coastguard Worker for path in IMAGES RADIO; do 478*5a923131SAndroid Build Coastguard Worker if unzip -l "${image}" "${path}/${part}.img" >/dev/null; then 479*5a923131SAndroid Build Coastguard Worker path_in_zip="${path}" 480*5a923131SAndroid Build Coastguard Worker break 481*5a923131SAndroid Build Coastguard Worker fi 482*5a923131SAndroid Build Coastguard Worker done 483*5a923131SAndroid Build Coastguard Worker [[ -n "${path_in_zip}" ]] || die "Failed to find ${part}.img" 484*5a923131SAndroid Build Coastguard Worker extract_file "${image}" "${path_in_zip}/${part}.img" "${part_file}" 485*5a923131SAndroid Build Coastguard Worker 486*5a923131SAndroid Build Coastguard Worker # If the partition is stored as an Android sparse image file, we need to 487*5a923131SAndroid Build Coastguard Worker # convert them to a raw image for the update. 488*5a923131SAndroid Build Coastguard Worker local magic=$(xxd -p -l4 "${part_file}") 489*5a923131SAndroid Build Coastguard Worker if [[ "${magic}" == "3aff26ed" ]]; then 490*5a923131SAndroid Build Coastguard Worker local temp_sparse=$(create_tempfile "${part}.sparse.XXXXXX") 491*5a923131SAndroid Build Coastguard Worker echo "Converting Android sparse image ${part}.img to RAW." 492*5a923131SAndroid Build Coastguard Worker mv "${part_file}" "${temp_sparse}" 493*5a923131SAndroid Build Coastguard Worker simg2img "${temp_sparse}" "${part_file}" 494*5a923131SAndroid Build Coastguard Worker rm -f "${temp_sparse}" 495*5a923131SAndroid Build Coastguard Worker fi 496*5a923131SAndroid Build Coastguard Worker 497*5a923131SAndroid Build Coastguard Worker # Extract the .map file (if one is available). 498*5a923131SAndroid Build Coastguard Worker if unzip -l "${image}" "${path_in_zip}/${part}.map" > /dev/null; then 499*5a923131SAndroid Build Coastguard Worker extract_file "${image}" "${path_in_zip}/${part}.map" "${part_map_file}" 500*5a923131SAndroid Build Coastguard Worker fi 501*5a923131SAndroid Build Coastguard Worker 502*5a923131SAndroid Build Coastguard Worker # delta_generator only supports images multiple of 4 KiB. For target images 503*5a923131SAndroid Build Coastguard Worker # we pad the data with zeros if needed, but for source images we truncate 504*5a923131SAndroid Build Coastguard Worker # down the data since the last block of the old image could be padded on 505*5a923131SAndroid Build Coastguard Worker # disk with unknown data. 506*5a923131SAndroid Build Coastguard Worker local filesize=$(stat -c%s "${part_file}") 507*5a923131SAndroid Build Coastguard Worker if [[ $(( filesize % 4096 )) -ne 0 ]]; then 508*5a923131SAndroid Build Coastguard Worker if [[ "${partitions_array}" == "SRC_PARTITIONS" ]]; then 509*5a923131SAndroid Build Coastguard Worker echo "Rounding DOWN partition ${part}.img to a multiple of 4 KiB." 510*5a923131SAndroid Build Coastguard Worker : $(( filesize = filesize & -4096 )) 511*5a923131SAndroid Build Coastguard Worker else 512*5a923131SAndroid Build Coastguard Worker echo "Rounding UP partition ${part}.img to a multiple of 4 KiB." 513*5a923131SAndroid Build Coastguard Worker : $(( filesize = (filesize + 4095) & -4096 )) 514*5a923131SAndroid Build Coastguard Worker fi 515*5a923131SAndroid Build Coastguard Worker truncate_file "${part_file}" "${filesize}" 516*5a923131SAndroid Build Coastguard Worker fi 517*5a923131SAndroid Build Coastguard Worker 518*5a923131SAndroid Build Coastguard Worker echo "Extracted ${partitions_array}[${part}]: ${filesize} bytes" 519*5a923131SAndroid Build Coastguard Worker} 520*5a923131SAndroid Build Coastguard Worker 521*5a923131SAndroid Build Coastguard Worker# extract_image_brillo <target_files.zip> <partitions_array> [partitions_order] 522*5a923131SAndroid Build Coastguard Worker# 523*5a923131SAndroid Build Coastguard Worker# Extract the A/B updated partitions from a Brillo target_files zip file into 524*5a923131SAndroid Build Coastguard Worker# new temporary files. 525*5a923131SAndroid Build Coastguard Workerextract_image_brillo() { 526*5a923131SAndroid Build Coastguard Worker local image="$1" 527*5a923131SAndroid Build Coastguard Worker local partitions_array="$2" 528*5a923131SAndroid Build Coastguard Worker local partitions_order="${3:-}" 529*5a923131SAndroid Build Coastguard Worker 530*5a923131SAndroid Build Coastguard Worker local partitions=( "boot" "system" ) 531*5a923131SAndroid Build Coastguard Worker local ab_partitions_list 532*5a923131SAndroid Build Coastguard Worker ab_partitions_list=$(create_tempfile "ab_partitions_list.XXXXXX") 533*5a923131SAndroid Build Coastguard Worker CLEANUP_FILES+=("${ab_partitions_list}") 534*5a923131SAndroid Build Coastguard Worker if unzip -l "${image}" "META/ab_partitions.txt" > /dev/null; then 535*5a923131SAndroid Build Coastguard Worker extract_file "${image}" "META/ab_partitions.txt" "${ab_partitions_list}" 536*5a923131SAndroid Build Coastguard Worker if grep -v -E '^[a-zA-Z0-9_-]*$' "${ab_partitions_list}" >&2; then 537*5a923131SAndroid Build Coastguard Worker die "Invalid partition names found in the partition list." 538*5a923131SAndroid Build Coastguard Worker fi 539*5a923131SAndroid Build Coastguard Worker # Get partition list without duplicates. 540*5a923131SAndroid Build Coastguard Worker partitions=($(awk '!seen[$0]++' "${ab_partitions_list}")) 541*5a923131SAndroid Build Coastguard Worker if [[ ${#partitions[@]} -eq 0 ]]; then 542*5a923131SAndroid Build Coastguard Worker die "The list of partitions is empty. Can't generate a payload." 543*5a923131SAndroid Build Coastguard Worker fi 544*5a923131SAndroid Build Coastguard Worker else 545*5a923131SAndroid Build Coastguard Worker warn "No ab_partitions.txt found. Using default." 546*5a923131SAndroid Build Coastguard Worker fi 547*5a923131SAndroid Build Coastguard Worker echo "List of A/B partitions for ${partitions_array}: ${partitions[@]}" 548*5a923131SAndroid Build Coastguard Worker 549*5a923131SAndroid Build Coastguard Worker if [[ -n "${partitions_order}" ]]; then 550*5a923131SAndroid Build Coastguard Worker eval "${partitions_order}=(${partitions[@]})" 551*5a923131SAndroid Build Coastguard Worker fi 552*5a923131SAndroid Build Coastguard Worker 553*5a923131SAndroid Build Coastguard Worker # All Brillo updaters support major version 2. 554*5a923131SAndroid Build Coastguard Worker FORCE_MAJOR_VERSION="2" 555*5a923131SAndroid Build Coastguard Worker 556*5a923131SAndroid Build Coastguard Worker if [[ "${partitions_array}" == "SRC_PARTITIONS" ]]; then 557*5a923131SAndroid Build Coastguard Worker # Source image 558*5a923131SAndroid Build Coastguard Worker local ue_config=$(create_tempfile "ue_config.XXXXXX") 559*5a923131SAndroid Build Coastguard Worker CLEANUP_FILES+=("${ue_config}") 560*5a923131SAndroid Build Coastguard Worker if unzip -l "${image}" "META/update_engine_config.txt" > /dev/null; then 561*5a923131SAndroid Build Coastguard Worker extract_file "${image}" "META/update_engine_config.txt" "${ue_config}" 562*5a923131SAndroid Build Coastguard Worker else 563*5a923131SAndroid Build Coastguard Worker warn "No update_engine_config.txt found. Assuming pre-release image, \ 564*5a923131SAndroid Build Coastguard Workerusing payload minor version 2" 565*5a923131SAndroid Build Coastguard Worker fi 566*5a923131SAndroid Build Coastguard Worker # For delta payloads, we use the major and minor version supported by the 567*5a923131SAndroid Build Coastguard Worker # old updater. 568*5a923131SAndroid Build Coastguard Worker FORCE_MINOR_VERSION=$(read_option_uint "${ue_config}" \ 569*5a923131SAndroid Build Coastguard Worker "PAYLOAD_MINOR_VERSION" 2) 570*5a923131SAndroid Build Coastguard Worker if [[ -n "${FLAGS_force_minor_version}" ]]; then 571*5a923131SAndroid Build Coastguard Worker FORCE_MINOR_VERSION="${FLAGS_force_minor_version}" 572*5a923131SAndroid Build Coastguard Worker fi 573*5a923131SAndroid Build Coastguard Worker FORCE_MAJOR_VERSION=$(read_option_uint "${ue_config}" \ 574*5a923131SAndroid Build Coastguard Worker "PAYLOAD_MAJOR_VERSION" 2) 575*5a923131SAndroid Build Coastguard Worker 576*5a923131SAndroid Build Coastguard Worker # Brillo support for deltas started with minor version 3. 577*5a923131SAndroid Build Coastguard Worker if [[ "${FORCE_MINOR_VERSION}" -le 2 ]]; then 578*5a923131SAndroid Build Coastguard Worker warn "No delta support from minor version ${FORCE_MINOR_VERSION}. \ 579*5a923131SAndroid Build Coastguard WorkerDisabling deltas for this source version." 580*5a923131SAndroid Build Coastguard Worker exit ${EX_UNSUPPORTED_DELTA} 581*5a923131SAndroid Build Coastguard Worker fi 582*5a923131SAndroid Build Coastguard Worker else 583*5a923131SAndroid Build Coastguard Worker # Target image 584*5a923131SAndroid Build Coastguard Worker local postinstall_config=$(create_tempfile "postinstall_config.XXXXXX") 585*5a923131SAndroid Build Coastguard Worker CLEANUP_FILES+=("${postinstall_config}") 586*5a923131SAndroid Build Coastguard Worker if unzip -l "${image}" "META/postinstall_config.txt" > /dev/null; then 587*5a923131SAndroid Build Coastguard Worker extract_file "${image}" "META/postinstall_config.txt" \ 588*5a923131SAndroid Build Coastguard Worker "${postinstall_config}" 589*5a923131SAndroid Build Coastguard Worker POSTINSTALL_CONFIG_FILE="${postinstall_config}" 590*5a923131SAndroid Build Coastguard Worker fi 591*5a923131SAndroid Build Coastguard Worker local dynamic_partitions_info=$(create_tempfile "dynamic_partitions_info.XXXXXX") 592*5a923131SAndroid Build Coastguard Worker CLEANUP_FILES+=("${dynamic_partitions_info}") 593*5a923131SAndroid Build Coastguard Worker if unzip -l "${image}" "META/dynamic_partitions_info.txt" > /dev/null; then 594*5a923131SAndroid Build Coastguard Worker extract_file "${image}" "META/dynamic_partitions_info.txt" \ 595*5a923131SAndroid Build Coastguard Worker "${dynamic_partitions_info}" 596*5a923131SAndroid Build Coastguard Worker DYNAMIC_PARTITION_INFO_FILE="${dynamic_partitions_info}" 597*5a923131SAndroid Build Coastguard Worker fi 598*5a923131SAndroid Build Coastguard Worker local apex_info=$(create_tempfile "apex_info.XXXXXX") 599*5a923131SAndroid Build Coastguard Worker CLEANUP_FILES+=("${apex_info}") 600*5a923131SAndroid Build Coastguard Worker if unzip -l "${image}" "META/apex_info.pb" > /dev/null; then 601*5a923131SAndroid Build Coastguard Worker extract_file "${image}" "META/apex_info.pb" \ 602*5a923131SAndroid Build Coastguard Worker "${apex_info}" 603*5a923131SAndroid Build Coastguard Worker APEX_INFO_FILE="${apex_info}" 604*5a923131SAndroid Build Coastguard Worker fi 605*5a923131SAndroid Build Coastguard Worker fi 606*5a923131SAndroid Build Coastguard Worker 607*5a923131SAndroid Build Coastguard Worker local part 608*5a923131SAndroid Build Coastguard Worker for part in "${partitions[@]}"; do 609*5a923131SAndroid Build Coastguard Worker local part_file=$(create_tempfile "${part}.img.XXXXXX") 610*5a923131SAndroid Build Coastguard Worker local part_map_file=$(create_tempfile "${part}.map.XXXXXX") 611*5a923131SAndroid Build Coastguard Worker CLEANUP_FILES+=("${part_file}" "${part_map_file}") 612*5a923131SAndroid Build Coastguard Worker # Extract partitions in background. 613*5a923131SAndroid Build Coastguard Worker extract_partition_brillo "${image}" "${partitions_array}" "${part}" \ 614*5a923131SAndroid Build Coastguard Worker "${part_file}" "${part_map_file}" & 615*5a923131SAndroid Build Coastguard Worker EXTRACT_IMAGE_PIDS+=("$!") 616*5a923131SAndroid Build Coastguard Worker eval "${partitions_array}[\"${part}\"]=\"${part_file}\"" 617*5a923131SAndroid Build Coastguard Worker eval "${partitions_array}_MAP[\"${part}\"]=\"${part_map_file}\"" 618*5a923131SAndroid Build Coastguard Worker done 619*5a923131SAndroid Build Coastguard Worker} 620*5a923131SAndroid Build Coastguard Worker 621*5a923131SAndroid Build Coastguard Worker# cleanup_partition_array <partitions_array> 622*5a923131SAndroid Build Coastguard Worker# 623*5a923131SAndroid Build Coastguard Worker# Remove all empty files in <partitions_array>. 624*5a923131SAndroid Build Coastguard Workercleanup_partition_array() { 625*5a923131SAndroid Build Coastguard Worker local partitions_array="$1" 626*5a923131SAndroid Build Coastguard Worker # Have to use eval to iterate over associative array keys with variable array 627*5a923131SAndroid Build Coastguard Worker # names, we should change it to use nameref once bash 4.3 is available 628*5a923131SAndroid Build Coastguard Worker # everywhere. 629*5a923131SAndroid Build Coastguard Worker for part in $(eval "echo \${!${partitions_array}[@]}"); do 630*5a923131SAndroid Build Coastguard Worker local path="${partitions_array}[$part]" 631*5a923131SAndroid Build Coastguard Worker if [[ ! -s "${!path}" ]]; then 632*5a923131SAndroid Build Coastguard Worker eval "unset ${partitions_array}[${part}]" 633*5a923131SAndroid Build Coastguard Worker fi 634*5a923131SAndroid Build Coastguard Worker done 635*5a923131SAndroid Build Coastguard Worker} 636*5a923131SAndroid Build Coastguard Worker 637*5a923131SAndroid Build Coastguard Workerextract_payload_images() { 638*5a923131SAndroid Build Coastguard Worker local payload_type=$1 639*5a923131SAndroid Build Coastguard Worker echo "Extracting images for ${payload_type} update." 640*5a923131SAndroid Build Coastguard Worker 641*5a923131SAndroid Build Coastguard Worker if [[ "${payload_type}" == "delta" ]]; then 642*5a923131SAndroid Build Coastguard Worker extract_image "${FLAGS_source_image}" SRC_PARTITIONS 643*5a923131SAndroid Build Coastguard Worker fi 644*5a923131SAndroid Build Coastguard Worker extract_image "${FLAGS_target_image}" DST_PARTITIONS PARTITIONS_ORDER 645*5a923131SAndroid Build Coastguard Worker # Wait for all subprocesses to finish. Not using `wait` since it doesn't die 646*5a923131SAndroid Build Coastguard Worker # on non-zero subprocess exit code. Not using `wait ${EXTRACT_IMAGE_PIDS[@]}` 647*5a923131SAndroid Build Coastguard Worker # as it gives the status of the last process it has waited for. 648*5a923131SAndroid Build Coastguard Worker for pid in ${EXTRACT_IMAGE_PIDS[@]}; do 649*5a923131SAndroid Build Coastguard Worker wait ${pid} 650*5a923131SAndroid Build Coastguard Worker done 651*5a923131SAndroid Build Coastguard Worker cleanup_partition_array SRC_PARTITIONS 652*5a923131SAndroid Build Coastguard Worker cleanup_partition_array SRC_PARTITIONS_MAP 653*5a923131SAndroid Build Coastguard Worker cleanup_partition_array DST_PARTITIONS 654*5a923131SAndroid Build Coastguard Worker cleanup_partition_array DST_PARTITIONS_MAP 655*5a923131SAndroid Build Coastguard Worker} 656*5a923131SAndroid Build Coastguard Worker 657*5a923131SAndroid Build Coastguard Workerget_payload_type() { 658*5a923131SAndroid Build Coastguard Worker if [[ -z "${FLAGS_source_image}" ]]; then 659*5a923131SAndroid Build Coastguard Worker echo "full" 660*5a923131SAndroid Build Coastguard Worker else 661*5a923131SAndroid Build Coastguard Worker echo "delta" 662*5a923131SAndroid Build Coastguard Worker fi 663*5a923131SAndroid Build Coastguard Worker} 664*5a923131SAndroid Build Coastguard Worker 665*5a923131SAndroid Build Coastguard Workervalidate_generate() { 666*5a923131SAndroid Build Coastguard Worker [[ -n "${FLAGS_payload}" ]] || 667*5a923131SAndroid Build Coastguard Worker die "You must specify an output filename with --payload FILENAME" 668*5a923131SAndroid Build Coastguard Worker 669*5a923131SAndroid Build Coastguard Worker [[ -n "${FLAGS_target_image}" ]] || 670*5a923131SAndroid Build Coastguard Worker die "You must specify a target image with --target_image FILENAME" 671*5a923131SAndroid Build Coastguard Worker} 672*5a923131SAndroid Build Coastguard Worker 673*5a923131SAndroid Build Coastguard Workercmd_generate() { 674*5a923131SAndroid Build Coastguard Worker local payload_type=$(get_payload_type) 675*5a923131SAndroid Build Coastguard Worker extract_payload_images ${payload_type} 676*5a923131SAndroid Build Coastguard Worker 677*5a923131SAndroid Build Coastguard Worker echo "Generating ${payload_type} update." 678*5a923131SAndroid Build Coastguard Worker # Common payload args: 679*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS=( --out_file="${FLAGS_payload}" ) 680*5a923131SAndroid Build Coastguard Worker 681*5a923131SAndroid Build Coastguard Worker local part old_partitions="" new_partitions="" partition_names="" 682*5a923131SAndroid Build Coastguard Worker local old_mapfiles="" new_mapfiles="" 683*5a923131SAndroid Build Coastguard Worker for part in "${PARTITIONS_ORDER[@]}"; do 684*5a923131SAndroid Build Coastguard Worker if [[ -n "${partition_names}" ]]; then 685*5a923131SAndroid Build Coastguard Worker partition_names+=":" 686*5a923131SAndroid Build Coastguard Worker new_partitions+=":" 687*5a923131SAndroid Build Coastguard Worker old_partitions+=":" 688*5a923131SAndroid Build Coastguard Worker new_mapfiles+=":" 689*5a923131SAndroid Build Coastguard Worker old_mapfiles+=":" 690*5a923131SAndroid Build Coastguard Worker fi 691*5a923131SAndroid Build Coastguard Worker partition_names+="${part}" 692*5a923131SAndroid Build Coastguard Worker new_partitions+="${DST_PARTITIONS[${part}]}" 693*5a923131SAndroid Build Coastguard Worker if [ "${FLAGS_full_boot}" == "true" ] && [ "${part}" == "boot" ]; then 694*5a923131SAndroid Build Coastguard Worker # Skip boot partition. 695*5a923131SAndroid Build Coastguard Worker old_partitions+="" 696*5a923131SAndroid Build Coastguard Worker else 697*5a923131SAndroid Build Coastguard Worker old_partitions+="${SRC_PARTITIONS[${part}]:-}" 698*5a923131SAndroid Build Coastguard Worker fi 699*5a923131SAndroid Build Coastguard Worker new_mapfiles+="${DST_PARTITIONS_MAP[${part}]:-}" 700*5a923131SAndroid Build Coastguard Worker old_mapfiles+="${SRC_PARTITIONS_MAP[${part}]:-}" 701*5a923131SAndroid Build Coastguard Worker done 702*5a923131SAndroid Build Coastguard Worker 703*5a923131SAndroid Build Coastguard Worker # Target image args: 704*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( 705*5a923131SAndroid Build Coastguard Worker --partition_names="${partition_names}" 706*5a923131SAndroid Build Coastguard Worker --new_partitions="${new_partitions}" 707*5a923131SAndroid Build Coastguard Worker --new_mapfiles="${new_mapfiles}" 708*5a923131SAndroid Build Coastguard Worker ) 709*5a923131SAndroid Build Coastguard Worker 710*5a923131SAndroid Build Coastguard Worker if [[ "${FLAGS_is_partial_update}" == "true" ]]; then 711*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( --is_partial_update="true" ) 712*5a923131SAndroid Build Coastguard Worker # Need at least minor version 7 for partial update, so generate with minor 713*5a923131SAndroid Build Coastguard Worker # version 7 if we don't have a source image. Let the delta_generator to fail 714*5a923131SAndroid Build Coastguard Worker # the other incompatiable minor versions. 715*5a923131SAndroid Build Coastguard Worker if [[ -z "${FORCE_MINOR_VERSION}" ]]; then 716*5a923131SAndroid Build Coastguard Worker FORCE_MINOR_VERSION="7" 717*5a923131SAndroid Build Coastguard Worker fi 718*5a923131SAndroid Build Coastguard Worker fi 719*5a923131SAndroid Build Coastguard Worker 720*5a923131SAndroid Build Coastguard Worker if [[ "${payload_type}" == "delta" ]]; then 721*5a923131SAndroid Build Coastguard Worker # Source image args: 722*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( 723*5a923131SAndroid Build Coastguard Worker --old_partitions="${old_partitions}" 724*5a923131SAndroid Build Coastguard Worker --old_mapfiles="${old_mapfiles}" 725*5a923131SAndroid Build Coastguard Worker ) 726*5a923131SAndroid Build Coastguard Worker if [[ -n "${FLAGS_disable_fec_computation}" ]]; then 727*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( 728*5a923131SAndroid Build Coastguard Worker --disable_fec_computation="${FLAGS_disable_fec_computation}" ) 729*5a923131SAndroid Build Coastguard Worker fi 730*5a923131SAndroid Build Coastguard Worker if [[ -n "${FLAGS_disable_verity_computation}" ]]; then 731*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( 732*5a923131SAndroid Build Coastguard Worker --disable_verity_computation="${FLAGS_disable_verity_computation}" ) 733*5a923131SAndroid Build Coastguard Worker fi 734*5a923131SAndroid Build Coastguard Worker if [[ -n "${FLAGS_compressor_types}" ]]; then 735*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( 736*5a923131SAndroid Build Coastguard Worker --compressor_types="${FLAGS_compressor_types}" ) 737*5a923131SAndroid Build Coastguard Worker fi 738*5a923131SAndroid Build Coastguard Worker if [[ -n "${FLAGS_enable_zucchini}" ]]; then 739*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( 740*5a923131SAndroid Build Coastguard Worker --enable_zucchini="${FLAGS_enable_zucchini}" ) 741*5a923131SAndroid Build Coastguard Worker fi 742*5a923131SAndroid Build Coastguard Worker if [[ -n "${FLAGS_enable_lz4diff}" ]]; then 743*5a923131SAndroid Build Coastguard Worker if [[ "${FLAGS_enable_lz4diff}" == "true" && -z "${FLAGS_liblz4_path}" ]]; then 744*5a923131SAndroid Build Coastguard Worker echo "--liblz4_path is required when enable_lz4diff is set to true." 745*5a923131SAndroid Build Coastguard Worker exit 1 746*5a923131SAndroid Build Coastguard Worker fi 747*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( 748*5a923131SAndroid Build Coastguard Worker --enable_lz4diff="${FLAGS_enable_lz4diff}" ) 749*5a923131SAndroid Build Coastguard Worker fi 750*5a923131SAndroid Build Coastguard Worker if [[ -n "${FLAGS_erofs_compression_param}" ]]; then 751*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( 752*5a923131SAndroid Build Coastguard Worker --erofs_compression_param="${FLAGS_erofs_compression_param}" ) 753*5a923131SAndroid Build Coastguard Worker fi 754*5a923131SAndroid Build Coastguard Worker fi 755*5a923131SAndroid Build Coastguard Worker 756*5a923131SAndroid Build Coastguard Worker if [[ -n "${FLAGS_enable_vabc_xor}" ]]; then 757*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( 758*5a923131SAndroid Build Coastguard Worker --enable_vabc_xor="${FLAGS_enable_vabc_xor}" ) 759*5a923131SAndroid Build Coastguard Worker fi 760*5a923131SAndroid Build Coastguard Worker 761*5a923131SAndroid Build Coastguard Worker if [[ -n "${FLAGS_disable_vabc}" ]]; then 762*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( 763*5a923131SAndroid Build Coastguard Worker --disable_vabc="${FLAGS_disable_vabc}" ) 764*5a923131SAndroid Build Coastguard Worker fi 765*5a923131SAndroid Build Coastguard Worker 766*5a923131SAndroid Build Coastguard Worker if [[ -n "${FLAGS_max_threads}" ]]; then 767*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( 768*5a923131SAndroid Build Coastguard Worker --max_threads="${FLAGS_max_threads}" ) 769*5a923131SAndroid Build Coastguard Worker fi 770*5a923131SAndroid Build Coastguard Worker 771*5a923131SAndroid Build Coastguard Worker # minor version is set only for delta or partial payload. 772*5a923131SAndroid Build Coastguard Worker if [[ -n "${FORCE_MINOR_VERSION}" ]]; then 773*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( --minor_version="${FORCE_MINOR_VERSION}" ) 774*5a923131SAndroid Build Coastguard Worker fi 775*5a923131SAndroid Build Coastguard Worker 776*5a923131SAndroid Build Coastguard Worker if [[ -n "${FORCE_MAJOR_VERSION}" ]]; then 777*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( --major_version="${FORCE_MAJOR_VERSION}" ) 778*5a923131SAndroid Build Coastguard Worker fi 779*5a923131SAndroid Build Coastguard Worker 780*5a923131SAndroid Build Coastguard Worker if [[ -n "${FLAGS_metadata_size_file}" ]]; then 781*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( --out_metadata_size_file="${FLAGS_metadata_size_file}" ) 782*5a923131SAndroid Build Coastguard Worker fi 783*5a923131SAndroid Build Coastguard Worker 784*5a923131SAndroid Build Coastguard Worker if [[ -n "${FLAGS_max_timestamp}" ]]; then 785*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( --max_timestamp="${FLAGS_max_timestamp}" ) 786*5a923131SAndroid Build Coastguard Worker fi 787*5a923131SAndroid Build Coastguard Worker 788*5a923131SAndroid Build Coastguard Worker if [[ -n "${FLAGS_security_patch_level}" ]]; then 789*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( --security_patch_level="${FLAGS_security_patch_level}" ) 790*5a923131SAndroid Build Coastguard Worker fi 791*5a923131SAndroid Build Coastguard Worker 792*5a923131SAndroid Build Coastguard Worker if [[ -n "${FLAGS_partition_timestamps}" ]]; then 793*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( --partition_timestamps="${FLAGS_partition_timestamps}" ) 794*5a923131SAndroid Build Coastguard Worker fi 795*5a923131SAndroid Build Coastguard Worker 796*5a923131SAndroid Build Coastguard Worker if [[ -n "${POSTINSTALL_CONFIG_FILE}" ]]; then 797*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( 798*5a923131SAndroid Build Coastguard Worker --new_postinstall_config_file="${POSTINSTALL_CONFIG_FILE}" 799*5a923131SAndroid Build Coastguard Worker ) 800*5a923131SAndroid Build Coastguard Worker fi 801*5a923131SAndroid Build Coastguard Worker 802*5a923131SAndroid Build Coastguard Worker if [[ -n "{DYNAMIC_PARTITION_INFO_FILE}" ]]; then 803*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( 804*5a923131SAndroid Build Coastguard Worker --dynamic_partition_info_file="${DYNAMIC_PARTITION_INFO_FILE}" 805*5a923131SAndroid Build Coastguard Worker ) 806*5a923131SAndroid Build Coastguard Worker fi 807*5a923131SAndroid Build Coastguard Worker if [[ -n "{APEX_INFO_FILE}" ]]; then 808*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( 809*5a923131SAndroid Build Coastguard Worker --apex_info_file="${APEX_INFO_FILE}" 810*5a923131SAndroid Build Coastguard Worker ) 811*5a923131SAndroid Build Coastguard Worker fi 812*5a923131SAndroid Build Coastguard Worker 813*5a923131SAndroid Build Coastguard Worker echo "Running delta_generator with args: ${GENERATOR_ARGS[@]}" 814*5a923131SAndroid Build Coastguard Worker if [[ -n "${FLAGS_enable_lz4diff}" ]]; then 815*5a923131SAndroid Build Coastguard Worker LD_PRELOAD=${LD_PRELOAD}:${FLAGS_liblz4_path} "${GENERATOR}" "${GENERATOR_ARGS[@]}" 816*5a923131SAndroid Build Coastguard Worker else 817*5a923131SAndroid Build Coastguard Worker "${GENERATOR}" "${GENERATOR_ARGS[@]}" 818*5a923131SAndroid Build Coastguard Worker fi 819*5a923131SAndroid Build Coastguard Worker 820*5a923131SAndroid Build Coastguard Worker echo "Done generating ${payload_type} update." 821*5a923131SAndroid Build Coastguard Worker} 822*5a923131SAndroid Build Coastguard Worker 823*5a923131SAndroid Build Coastguard Workervalidate_hash() { 824*5a923131SAndroid Build Coastguard Worker [[ -n "${FLAGS_signature_size}" ]] || 825*5a923131SAndroid Build Coastguard Worker die "You must specify signature size with --signature_size SIZES" 826*5a923131SAndroid Build Coastguard Worker 827*5a923131SAndroid Build Coastguard Worker [[ -n "${FLAGS_unsigned_payload}" ]] || 828*5a923131SAndroid Build Coastguard Worker die "You must specify the input unsigned payload with \ 829*5a923131SAndroid Build Coastguard Worker--unsigned_payload FILENAME" 830*5a923131SAndroid Build Coastguard Worker 831*5a923131SAndroid Build Coastguard Worker [[ -n "${FLAGS_payload_hash_file}" ]] || 832*5a923131SAndroid Build Coastguard Worker die "You must specify --payload_hash_file FILENAME" 833*5a923131SAndroid Build Coastguard Worker 834*5a923131SAndroid Build Coastguard Worker [[ -n "${FLAGS_metadata_hash_file}" ]] || 835*5a923131SAndroid Build Coastguard Worker die "You must specify --metadata_hash_file FILENAME" 836*5a923131SAndroid Build Coastguard Worker} 837*5a923131SAndroid Build Coastguard Worker 838*5a923131SAndroid Build Coastguard Workercmd_hash() { 839*5a923131SAndroid Build Coastguard Worker "${GENERATOR}" \ 840*5a923131SAndroid Build Coastguard Worker --in_file="${FLAGS_unsigned_payload}" \ 841*5a923131SAndroid Build Coastguard Worker --signature_size="${FLAGS_signature_size}" \ 842*5a923131SAndroid Build Coastguard Worker --out_hash_file="${FLAGS_payload_hash_file}" \ 843*5a923131SAndroid Build Coastguard Worker --out_metadata_hash_file="${FLAGS_metadata_hash_file}" 844*5a923131SAndroid Build Coastguard Worker 845*5a923131SAndroid Build Coastguard Worker echo "Done generating hash." 846*5a923131SAndroid Build Coastguard Worker} 847*5a923131SAndroid Build Coastguard Worker 848*5a923131SAndroid Build Coastguard Workervalidate_sign() { 849*5a923131SAndroid Build Coastguard Worker [[ -n "${FLAGS_signature_size}" ]] || 850*5a923131SAndroid Build Coastguard Worker die "You must specify signature size with --signature_size SIZES" 851*5a923131SAndroid Build Coastguard Worker 852*5a923131SAndroid Build Coastguard Worker [[ -n "${FLAGS_unsigned_payload}" ]] || 853*5a923131SAndroid Build Coastguard Worker die "You must specify the input unsigned payload with \ 854*5a923131SAndroid Build Coastguard Worker--unsigned_payload FILENAME" 855*5a923131SAndroid Build Coastguard Worker 856*5a923131SAndroid Build Coastguard Worker [[ -n "${FLAGS_payload}" ]] || 857*5a923131SAndroid Build Coastguard Worker die "You must specify the output signed payload with --payload FILENAME" 858*5a923131SAndroid Build Coastguard Worker 859*5a923131SAndroid Build Coastguard Worker [[ -n "${FLAGS_payload_signature_file}" ]] || 860*5a923131SAndroid Build Coastguard Worker die "You must specify the payload signature file with \ 861*5a923131SAndroid Build Coastguard Worker--payload_signature_file SIGNATURES" 862*5a923131SAndroid Build Coastguard Worker 863*5a923131SAndroid Build Coastguard Worker [[ -n "${FLAGS_metadata_signature_file}" ]] || 864*5a923131SAndroid Build Coastguard Worker die "You must specify the metadata signature file with \ 865*5a923131SAndroid Build Coastguard Worker--metadata_signature_file SIGNATURES" 866*5a923131SAndroid Build Coastguard Worker} 867*5a923131SAndroid Build Coastguard Worker 868*5a923131SAndroid Build Coastguard Workercmd_sign() { 869*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS=( 870*5a923131SAndroid Build Coastguard Worker --in_file="${FLAGS_unsigned_payload}" 871*5a923131SAndroid Build Coastguard Worker --signature_size="${FLAGS_signature_size}" 872*5a923131SAndroid Build Coastguard Worker --payload_signature_file="${FLAGS_payload_signature_file}" 873*5a923131SAndroid Build Coastguard Worker --metadata_signature_file="${FLAGS_metadata_signature_file}" 874*5a923131SAndroid Build Coastguard Worker --out_file="${FLAGS_payload}" 875*5a923131SAndroid Build Coastguard Worker ) 876*5a923131SAndroid Build Coastguard Worker 877*5a923131SAndroid Build Coastguard Worker if [[ -n "${FLAGS_metadata_size_file}" ]]; then 878*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( --out_metadata_size_file="${FLAGS_metadata_size_file}" ) 879*5a923131SAndroid Build Coastguard Worker fi 880*5a923131SAndroid Build Coastguard Worker 881*5a923131SAndroid Build Coastguard Worker "${GENERATOR}" "${GENERATOR_ARGS[@]}" 882*5a923131SAndroid Build Coastguard Worker echo "Done signing payload." 883*5a923131SAndroid Build Coastguard Worker} 884*5a923131SAndroid Build Coastguard Worker 885*5a923131SAndroid Build Coastguard Workervalidate_properties() { 886*5a923131SAndroid Build Coastguard Worker [[ -n "${FLAGS_payload}" ]] || 887*5a923131SAndroid Build Coastguard Worker die "You must specify the payload file with --payload FILENAME" 888*5a923131SAndroid Build Coastguard Worker 889*5a923131SAndroid Build Coastguard Worker [[ -n "${FLAGS_properties_file}" ]] || 890*5a923131SAndroid Build Coastguard Worker die "You must specify a non empty --properties_file FILENAME" 891*5a923131SAndroid Build Coastguard Worker} 892*5a923131SAndroid Build Coastguard Worker 893*5a923131SAndroid Build Coastguard Workercmd_properties() { 894*5a923131SAndroid Build Coastguard Worker "${GENERATOR}" \ 895*5a923131SAndroid Build Coastguard Worker --in_file="${FLAGS_payload}" \ 896*5a923131SAndroid Build Coastguard Worker --properties_file="${FLAGS_properties_file}" 897*5a923131SAndroid Build Coastguard Worker} 898*5a923131SAndroid Build Coastguard Worker 899*5a923131SAndroid Build Coastguard Workervalidate_verify_and_check() { 900*5a923131SAndroid Build Coastguard Worker [[ -n "${FLAGS_payload}" ]] || 901*5a923131SAndroid Build Coastguard Worker die "Error: you must specify an input filename with --payload FILENAME" 902*5a923131SAndroid Build Coastguard Worker 903*5a923131SAndroid Build Coastguard Worker [[ -n "${FLAGS_target_image}" ]] || 904*5a923131SAndroid Build Coastguard Worker die "Error: you must specify a target image with --target_image FILENAME" 905*5a923131SAndroid Build Coastguard Worker} 906*5a923131SAndroid Build Coastguard Worker 907*5a923131SAndroid Build Coastguard Workercmd_verify() { 908*5a923131SAndroid Build Coastguard Worker local payload_type=$(get_payload_type) 909*5a923131SAndroid Build Coastguard Worker extract_payload_images ${payload_type} 910*5a923131SAndroid Build Coastguard Worker 911*5a923131SAndroid Build Coastguard Worker declare -A TMP_PARTITIONS 912*5a923131SAndroid Build Coastguard Worker for part in "${PARTITIONS_ORDER[@]}"; do 913*5a923131SAndroid Build Coastguard Worker local tmp_part=$(create_tempfile "tmp_part.bin.XXXXXX") 914*5a923131SAndroid Build Coastguard Worker echo "Creating temporary target partition ${tmp_part} for ${part}" 915*5a923131SAndroid Build Coastguard Worker CLEANUP_FILES+=("${tmp_part}") 916*5a923131SAndroid Build Coastguard Worker TMP_PARTITIONS[${part}]=${tmp_part} 917*5a923131SAndroid Build Coastguard Worker local FILESIZE=$(stat -c%s "${DST_PARTITIONS[${part}]}") 918*5a923131SAndroid Build Coastguard Worker echo "Truncating ${TMP_PARTITIONS[${part}]} to ${FILESIZE}" 919*5a923131SAndroid Build Coastguard Worker truncate_file "${TMP_PARTITIONS[${part}]}" "${FILESIZE}" 920*5a923131SAndroid Build Coastguard Worker done 921*5a923131SAndroid Build Coastguard Worker 922*5a923131SAndroid Build Coastguard Worker echo "Verifying ${payload_type} update." 923*5a923131SAndroid Build Coastguard Worker # Common payload args: 924*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS=( --in_file="${FLAGS_payload}" ) 925*5a923131SAndroid Build Coastguard Worker 926*5a923131SAndroid Build Coastguard Worker local part old_partitions="" new_partitions="" partition_names="" 927*5a923131SAndroid Build Coastguard Worker for part in "${PARTITIONS_ORDER[@]}"; do 928*5a923131SAndroid Build Coastguard Worker if [[ -n "${partition_names}" ]]; then 929*5a923131SAndroid Build Coastguard Worker partition_names+=":" 930*5a923131SAndroid Build Coastguard Worker new_partitions+=":" 931*5a923131SAndroid Build Coastguard Worker old_partitions+=":" 932*5a923131SAndroid Build Coastguard Worker fi 933*5a923131SAndroid Build Coastguard Worker partition_names+="${part}" 934*5a923131SAndroid Build Coastguard Worker new_partitions+="${TMP_PARTITIONS[${part}]}" 935*5a923131SAndroid Build Coastguard Worker old_partitions+="${SRC_PARTITIONS[${part}]:-}" 936*5a923131SAndroid Build Coastguard Worker done 937*5a923131SAndroid Build Coastguard Worker 938*5a923131SAndroid Build Coastguard Worker # Target image args: 939*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( 940*5a923131SAndroid Build Coastguard Worker --partition_names="${partition_names}" 941*5a923131SAndroid Build Coastguard Worker --new_partitions="${new_partitions}" 942*5a923131SAndroid Build Coastguard Worker ) 943*5a923131SAndroid Build Coastguard Worker 944*5a923131SAndroid Build Coastguard Worker if [[ "${payload_type}" == "delta" ]]; then 945*5a923131SAndroid Build Coastguard Worker # Source image args: 946*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( 947*5a923131SAndroid Build Coastguard Worker --old_partitions="${old_partitions}" 948*5a923131SAndroid Build Coastguard Worker ) 949*5a923131SAndroid Build Coastguard Worker fi 950*5a923131SAndroid Build Coastguard Worker 951*5a923131SAndroid Build Coastguard Worker if [[ -n "${FORCE_MAJOR_VERSION}" ]]; then 952*5a923131SAndroid Build Coastguard Worker GENERATOR_ARGS+=( --major_version="${FORCE_MAJOR_VERSION}" ) 953*5a923131SAndroid Build Coastguard Worker fi 954*5a923131SAndroid Build Coastguard Worker 955*5a923131SAndroid Build Coastguard Worker echo "Running delta_generator to verify ${payload_type} payload with args: \ 956*5a923131SAndroid Build Coastguard Worker${GENERATOR_ARGS[@]}" 957*5a923131SAndroid Build Coastguard Worker "${GENERATOR}" "${GENERATOR_ARGS[@]}" || true 958*5a923131SAndroid Build Coastguard Worker 959*5a923131SAndroid Build Coastguard Worker echo "Done applying ${payload_type} update." 960*5a923131SAndroid Build Coastguard Worker echo "Checking the newly generated partitions against the target partitions" 961*5a923131SAndroid Build Coastguard Worker local need_pause=false 962*5a923131SAndroid Build Coastguard Worker for part in "${PARTITIONS_ORDER[@]}"; do 963*5a923131SAndroid Build Coastguard Worker local not_str="" 964*5a923131SAndroid Build Coastguard Worker if ! cmp "${TMP_PARTITIONS[${part}]}" "${DST_PARTITIONS[${part}]}"; then 965*5a923131SAndroid Build Coastguard Worker not_str="in" 966*5a923131SAndroid Build Coastguard Worker need_pause=true 967*5a923131SAndroid Build Coastguard Worker fi 968*5a923131SAndroid Build Coastguard Worker echo "The new partition (${part}) is ${not_str}valid." 969*5a923131SAndroid Build Coastguard Worker done 970*5a923131SAndroid Build Coastguard Worker # All images will be cleaned up when script exits, pause here to give a chance 971*5a923131SAndroid Build Coastguard Worker # to inspect the images. 972*5a923131SAndroid Build Coastguard Worker if [[ "$need_pause" == true ]]; then 973*5a923131SAndroid Build Coastguard Worker read -n1 -r -s -p "Paused to investigate invalid partitions, \ 974*5a923131SAndroid Build Coastguard Workerpress any key to exit." 975*5a923131SAndroid Build Coastguard Worker fi 976*5a923131SAndroid Build Coastguard Worker} 977*5a923131SAndroid Build Coastguard Worker 978*5a923131SAndroid Build Coastguard Workercmd_check() { 979*5a923131SAndroid Build Coastguard Worker local payload_type=$(get_payload_type) 980*5a923131SAndroid Build Coastguard Worker extract_payload_images ${payload_type} 981*5a923131SAndroid Build Coastguard Worker 982*5a923131SAndroid Build Coastguard Worker local part dst_partitions="" src_partitions="" 983*5a923131SAndroid Build Coastguard Worker for part in "${PARTITIONS_ORDER[@]}"; do 984*5a923131SAndroid Build Coastguard Worker if [[ -n "${dst_partitions}" ]]; then 985*5a923131SAndroid Build Coastguard Worker dst_partitions+=" " 986*5a923131SAndroid Build Coastguard Worker src_partitions+=" " 987*5a923131SAndroid Build Coastguard Worker fi 988*5a923131SAndroid Build Coastguard Worker dst_partitions+="${DST_PARTITIONS[${part}]}" 989*5a923131SAndroid Build Coastguard Worker src_partitions+="${SRC_PARTITIONS[${part}]:-}" 990*5a923131SAndroid Build Coastguard Worker done 991*5a923131SAndroid Build Coastguard Worker 992*5a923131SAndroid Build Coastguard Worker # Common payload args: 993*5a923131SAndroid Build Coastguard Worker PAYCHECK_ARGS=( "${FLAGS_payload}" --type ${payload_type} \ 994*5a923131SAndroid Build Coastguard Worker --part_names ${PARTITIONS_ORDER[@]} \ 995*5a923131SAndroid Build Coastguard Worker --dst_part_paths ${dst_partitions} ) 996*5a923131SAndroid Build Coastguard Worker 997*5a923131SAndroid Build Coastguard Worker if [[ ! -z "${SRC_PARTITIONS[@]}" ]]; then 998*5a923131SAndroid Build Coastguard Worker PAYCHECK_ARGS+=( --src_part_paths ${src_partitions} ) 999*5a923131SAndroid Build Coastguard Worker fi 1000*5a923131SAndroid Build Coastguard Worker 1001*5a923131SAndroid Build Coastguard Worker echo "Checking ${payload_type} update." 1002*5a923131SAndroid Build Coastguard Worker check_update_payload ${PAYCHECK_ARGS[@]} --check 1003*5a923131SAndroid Build Coastguard Worker} 1004*5a923131SAndroid Build Coastguard Worker 1005*5a923131SAndroid Build Coastguard Worker# Check that the real generator exists: 1006*5a923131SAndroid Build Coastguard Worker[[ -x "${GENERATOR}" ]] || GENERATOR="$(which delta_generator || true)" 1007*5a923131SAndroid Build Coastguard Worker[[ -x "${GENERATOR}" ]] || die "can't find delta_generator" 1008*5a923131SAndroid Build Coastguard Worker 1009*5a923131SAndroid Build Coastguard Workercase "$COMMAND" in 1010*5a923131SAndroid Build Coastguard Worker generate) validate_generate 1011*5a923131SAndroid Build Coastguard Worker cmd_generate 1012*5a923131SAndroid Build Coastguard Worker ;; 1013*5a923131SAndroid Build Coastguard Worker hash) validate_hash 1014*5a923131SAndroid Build Coastguard Worker cmd_hash 1015*5a923131SAndroid Build Coastguard Worker ;; 1016*5a923131SAndroid Build Coastguard Worker sign) validate_sign 1017*5a923131SAndroid Build Coastguard Worker cmd_sign 1018*5a923131SAndroid Build Coastguard Worker ;; 1019*5a923131SAndroid Build Coastguard Worker properties) validate_properties 1020*5a923131SAndroid Build Coastguard Worker cmd_properties 1021*5a923131SAndroid Build Coastguard Worker ;; 1022*5a923131SAndroid Build Coastguard Worker verify) validate_verify_and_check 1023*5a923131SAndroid Build Coastguard Worker cmd_verify 1024*5a923131SAndroid Build Coastguard Worker ;; 1025*5a923131SAndroid Build Coastguard Worker check) validate_verify_and_check 1026*5a923131SAndroid Build Coastguard Worker cmd_check 1027*5a923131SAndroid Build Coastguard Worker ;; 1028*5a923131SAndroid Build Coastguard Workeresac 1029