Merge "Trigger data wipe on spl-downgrade as well"

This commit is contained in:
Treehugger Robot 2021-03-12 18:50:38 +00:00 committed by Gerrit Code Review
commit b7ade64e3f
1 changed files with 38 additions and 32 deletions

View File

@ -297,8 +297,6 @@ SECONDARY_PAYLOAD_SKIPPED_IMAGES = [
'vendor_boot'] 'vendor_boot']
class PayloadSigner(object): class PayloadSigner(object):
"""A class that wraps the payload signing works. """A class that wraps the payload signing works.
@ -765,10 +763,12 @@ def GetTargetFilesZipWithoutPostinstallConfig(input_file):
common.ZipDelete(target_file, POSTINSTALL_CONFIG) common.ZipDelete(target_file, POSTINSTALL_CONFIG)
return target_file return target_file
def ParseInfoDict(target_file_path): def ParseInfoDict(target_file_path):
with zipfile.ZipFile(target_file_path, 'r', allowZip64=True) as zfp: with zipfile.ZipFile(target_file_path, 'r', allowZip64=True) as zfp:
return common.LoadInfoDict(zfp) return common.LoadInfoDict(zfp)
def GetTargetFilesZipForPartialUpdates(input_file, ab_partitions): def GetTargetFilesZipForPartialUpdates(input_file, ab_partitions):
"""Returns a target-files.zip for partial ota update package generation. """Returns a target-files.zip for partial ota update package generation.
@ -889,7 +889,7 @@ def GetTargetFilesZipForRetrofitDynamicPartitions(input_file,
with open(new_ab_partitions, 'w') as f: with open(new_ab_partitions, 'w') as f:
for partition in ab_partitions: for partition in ab_partitions:
if (partition in dynamic_partition_list and if (partition in dynamic_partition_list and
partition not in super_block_devices): partition not in super_block_devices):
logger.info("Dropping %s from ab_partitions.txt", partition) logger.info("Dropping %s from ab_partitions.txt", partition)
continue continue
f.write(partition + "\n") f.write(partition + "\n")
@ -963,32 +963,37 @@ def GetTargetFilesZipForCustomImagesUpdates(input_file, custom_images):
return target_file return target_file
def GeneratePartitionTimestampFlags(partition_state): def GeneratePartitionTimestampFlags(partition_state):
partition_timestamps = [ partition_timestamps = [
part.partition_name + ":" + part.version part.partition_name + ":" + part.version
for part in partition_state] for part in partition_state]
return ["--partition_timestamps", ",".join(partition_timestamps)] return ["--partition_timestamps", ",".join(partition_timestamps)]
def GeneratePartitionTimestampFlagsDowngrade( def GeneratePartitionTimestampFlagsDowngrade(
pre_partition_state, post_partition_state): pre_partition_state, post_partition_state):
assert pre_partition_state is not None assert pre_partition_state is not None
partition_timestamps = {} partition_timestamps = {}
for part in pre_partition_state: for part in pre_partition_state:
partition_timestamps[part.partition_name] = part.version partition_timestamps[part.partition_name] = part.version
for part in post_partition_state: for part in post_partition_state:
partition_timestamps[part.partition_name] = \ partition_timestamps[part.partition_name] = \
max(part.version, partition_timestamps[part.partition_name]) max(part.version, partition_timestamps[part.partition_name])
return [ return [
"--partition_timestamps", "--partition_timestamps",
",".join([key + ":" + val for (key, val) in partition_timestamps.items()]) ",".join([key + ":" + val for (key, val)
in partition_timestamps.items()])
] ]
def IsSparseImage(filepath): def IsSparseImage(filepath):
with open(filepath, 'rb') as fp: with open(filepath, 'rb') as fp:
# Magic for android sparse image format # Magic for android sparse image format
# https://source.android.com/devices/bootloader/images # https://source.android.com/devices/bootloader/images
return fp.read(4) == b'\x3A\xFF\x26\xED' return fp.read(4) == b'\x3A\xFF\x26\xED'
def SupportsMainlineGkiUpdates(target_file): def SupportsMainlineGkiUpdates(target_file):
"""Return True if the build supports MainlineGKIUpdates. """Return True if the build supports MainlineGKIUpdates.
@ -1027,6 +1032,7 @@ def SupportsMainlineGkiUpdates(target_file):
pattern = re.compile(r"com\.android\.gki\..*\.apex") pattern = re.compile(r"com\.android\.gki\..*\.apex")
return pattern.search(output) is not None return pattern.search(output) is not None
def GenerateAbOtaPackage(target_file, output_file, source_file=None): def GenerateAbOtaPackage(target_file, output_file, source_file=None):
"""Generates an Android OTA package that has A/B update payload.""" """Generates an Android OTA package that has A/B update payload."""
# Stage the output zip package for package signing. # Stage the output zip package for package signing.
@ -1047,7 +1053,7 @@ def GenerateAbOtaPackage(target_file, output_file, source_file=None):
source_info = common.BuildInfo(OPTIONS.source_info_dict, OPTIONS.oem_dicts) source_info = common.BuildInfo(OPTIONS.source_info_dict, OPTIONS.oem_dicts)
vendor_prop = source_info.info_dict.get("vendor.build.prop") vendor_prop = source_info.info_dict.get("vendor.build.prop")
if vendor_prop and \ if vendor_prop and \
vendor_prop.GetProp("ro.virtual_ab.compression.enabled") == "true": vendor_prop.GetProp("ro.virtual_ab.compression.enabled") == "true":
# TODO(zhangkelvin) Remove this once FEC on VABC is supported # TODO(zhangkelvin) Remove this once FEC on VABC is supported
logger.info("Virtual AB Compression enabled, disabling FEC") logger.info("Virtual AB Compression enabled, disabling FEC")
OPTIONS.disable_fec_computation = True OPTIONS.disable_fec_computation = True
@ -1103,7 +1109,8 @@ def GenerateAbOtaPackage(target_file, output_file, source_file=None):
additional_args += ["--max_timestamp", max_timestamp] additional_args += ["--max_timestamp", max_timestamp]
if SupportsMainlineGkiUpdates(source_file): if SupportsMainlineGkiUpdates(source_file):
logger.warning("Detected build with mainline GKI, include full boot image.") logger.warning(
"Detected build with mainline GKI, include full boot image.")
additional_args.extend(["--full_boot", "true"]) additional_args.extend(["--full_boot", "true"])
payload.Generate( payload.Generate(
@ -1137,7 +1144,7 @@ def GenerateAbOtaPackage(target_file, output_file, source_file=None):
# into A/B OTA package. # into A/B OTA package.
target_zip = zipfile.ZipFile(target_file, "r", allowZip64=True) target_zip = zipfile.ZipFile(target_file, "r", allowZip64=True)
if (target_info.get("verity") == "true" or if (target_info.get("verity") == "true" or
target_info.get("avb_enable") == "true"): target_info.get("avb_enable") == "true"):
care_map_list = [x for x in ["care_map.pb", "care_map.txt"] if care_map_list = [x for x in ["care_map.pb", "care_map.txt"] if
"META/" + x in target_zip.namelist()] "META/" + x in target_zip.namelist()]
@ -1157,7 +1164,7 @@ def GenerateAbOtaPackage(target_file, output_file, source_file=None):
apex_info_entry = target_zip.getinfo("META/apex_info.pb") apex_info_entry = target_zip.getinfo("META/apex_info.pb")
with target_zip.open(apex_info_entry, "r") as zfp: with target_zip.open(apex_info_entry, "r") as zfp:
common.ZipWriteStr(output_zip, "apex_info.pb", zfp.read(), common.ZipWriteStr(output_zip, "apex_info.pb", zfp.read(),
compress_type=zipfile.ZIP_STORED) compress_type=zipfile.ZIP_STORED)
except KeyError: except KeyError:
logger.warning("target_file doesn't contain apex_info.pb %s", target_file) logger.warning("target_file doesn't contain apex_info.pb %s", target_file)
@ -1271,6 +1278,7 @@ def main(argv):
OPTIONS.disable_vabc = True OPTIONS.disable_vabc = True
elif o == "--spl_downgrade": elif o == "--spl_downgrade":
OPTIONS.spl_downgrade = True OPTIONS.spl_downgrade = True
OPTIONS.wipe_user_data = True
else: else:
return False return False
return True return True
@ -1341,7 +1349,6 @@ def main(argv):
if OPTIONS.incremental_source is None: if OPTIONS.incremental_source is None:
raise ValueError("Cannot generate downgradable full OTAs") raise ValueError("Cannot generate downgradable full OTAs")
# TODO(xunchang) for retrofit and partial updates, maybe we should rebuild the # TODO(xunchang) for retrofit and partial updates, maybe we should rebuild the
# target-file and reload the info_dict. So the info will be consistent with # target-file and reload the info_dict. So the info will be consistent with
# the modified target-file. # the modified target-file.
@ -1349,7 +1356,6 @@ def main(argv):
logger.info("--- target info ---") logger.info("--- target info ---")
common.DumpInfoDict(OPTIONS.info_dict) common.DumpInfoDict(OPTIONS.info_dict)
# Load the source build dict if applicable. # Load the source build dict if applicable.
if OPTIONS.incremental_source is not None: if OPTIONS.incremental_source is not None:
OPTIONS.target_info_dict = OPTIONS.info_dict OPTIONS.target_info_dict = OPTIONS.info_dict
@ -1360,15 +1366,15 @@ def main(argv):
if OPTIONS.partial: if OPTIONS.partial:
OPTIONS.info_dict['ab_partitions'] = \ OPTIONS.info_dict['ab_partitions'] = \
list( list(
set(OPTIONS.info_dict['ab_partitions']) & set(OPTIONS.partial) set(OPTIONS.info_dict['ab_partitions']) & set(OPTIONS.partial)
) )
if OPTIONS.source_info_dict: if OPTIONS.source_info_dict:
OPTIONS.source_info_dict['ab_partitions'] = \ OPTIONS.source_info_dict['ab_partitions'] = \
list( list(
set(OPTIONS.source_info_dict['ab_partitions']) & set(OPTIONS.source_info_dict['ab_partitions']) &
set(OPTIONS.partial) set(OPTIONS.partial)
) )
# Load OEM dicts if provided. # Load OEM dicts if provided.
OPTIONS.oem_dicts = _LoadOemDicts(OPTIONS.oem_source) OPTIONS.oem_dicts = _LoadOemDicts(OPTIONS.oem_source)
@ -1377,7 +1383,7 @@ def main(argv):
# use_dynamic_partitions but target build does. # use_dynamic_partitions but target build does.
if (OPTIONS.source_info_dict and if (OPTIONS.source_info_dict and
OPTIONS.source_info_dict.get("use_dynamic_partitions") != "true" and OPTIONS.source_info_dict.get("use_dynamic_partitions") != "true" and
OPTIONS.target_info_dict.get("use_dynamic_partitions") == "true"): OPTIONS.target_info_dict.get("use_dynamic_partitions") == "true"):
if OPTIONS.target_info_dict.get("dynamic_partition_retrofit") != "true": if OPTIONS.target_info_dict.get("dynamic_partition_retrofit") != "true":
raise common.ExternalError( raise common.ExternalError(
"Expect to generate incremental OTA for retrofitting dynamic " "Expect to generate incremental OTA for retrofitting dynamic "
@ -1394,7 +1400,7 @@ def main(argv):
allow_non_ab = OPTIONS.info_dict.get("allow_non_ab") == "true" allow_non_ab = OPTIONS.info_dict.get("allow_non_ab") == "true"
if OPTIONS.force_non_ab: if OPTIONS.force_non_ab:
assert allow_non_ab,\ assert allow_non_ab,\
"--force_non_ab only allowed on devices that supports non-A/B" "--force_non_ab only allowed on devices that supports non-A/B"
assert ab_update, "--force_non_ab only allowed on A/B devices" assert ab_update, "--force_non_ab only allowed on A/B devices"
generate_ab = not OPTIONS.force_non_ab and ab_update generate_ab = not OPTIONS.force_non_ab and ab_update
@ -1412,10 +1418,10 @@ def main(argv):
private_key_path = OPTIONS.package_key + OPTIONS.private_key_suffix private_key_path = OPTIONS.package_key + OPTIONS.private_key_suffix
if not os.path.exists(private_key_path): if not os.path.exists(private_key_path):
raise common.ExternalError( raise common.ExternalError(
"Private key {} doesn't exist. Make sure you passed the" "Private key {} doesn't exist. Make sure you passed the"
" correct key path through -k option".format( " correct key path through -k option".format(
private_key_path) private_key_path)
) )
if OPTIONS.source_info_dict: if OPTIONS.source_info_dict:
source_build_prop = OPTIONS.source_info_dict["build.prop"] source_build_prop = OPTIONS.source_info_dict["build.prop"]
@ -1423,14 +1429,14 @@ def main(argv):
source_spl = source_build_prop.GetProp(SECURITY_PATCH_LEVEL_PROP_NAME) source_spl = source_build_prop.GetProp(SECURITY_PATCH_LEVEL_PROP_NAME)
target_spl = target_build_prop.GetProp(SECURITY_PATCH_LEVEL_PROP_NAME) target_spl = target_build_prop.GetProp(SECURITY_PATCH_LEVEL_PROP_NAME)
is_spl_downgrade = target_spl < source_spl is_spl_downgrade = target_spl < source_spl
if is_spl_downgrade and not OPTIONS.spl_downgrade: if is_spl_downgrade and not OPTIONS.spl_downgrade and not OPTIONS.downgrade:
raise common.ExternalError( raise common.ExternalError(
"Target security patch level {} is older than source SPL {} applying " "Target security patch level {} is older than source SPL {} applying "
"such OTA will likely cause device fail to boot. Pass --spl_downgrade " "such OTA will likely cause device fail to boot. Pass --spl_downgrade "
"to override this check. This script expects security patch level to " "to override this check. This script expects security patch level to "
"be in format yyyy-mm-dd (e.x. 2021-02-05). It's possible to use " "be in format yyyy-mm-dd (e.x. 2021-02-05). It's possible to use "
"separators other than -, so as long as it's used consistenly across " "separators other than -, so as long as it's used consistenly across "
"all SPL dates".format(target_spl, source_spl)) "all SPL dates".format(target_spl, source_spl))
elif not is_spl_downgrade and OPTIONS.spl_downgrade: elif not is_spl_downgrade and OPTIONS.spl_downgrade:
raise ValueError("--spl_downgrade specified but no actual SPL downgrade" raise ValueError("--spl_downgrade specified but no actual SPL downgrade"
" detected. Please only pass in this flag if you want a" " detected. Please only pass in this flag if you want a"