Merge "Add compatibility metadata for Treble-enabled non-A/B OTA packages."

am: 02b158d7a4

Change-Id: I54e64fcd3fbff266f8082a4347432941a8c6e8fe
This commit is contained in:
Tao Bao
2017-08-29 19:20:08 +00:00
committed by android-build-merger
2 changed files with 103 additions and 64 deletions

View File

@@ -251,15 +251,16 @@ def LoadInfoDict(input_file, input_dir=None):
else: else:
d["fstab"] = None d["fstab"] = None
d["build.prop"] = LoadBuildProp(read_helper) d["build.prop"] = LoadBuildProp(read_helper, 'SYSTEM/build.prop')
d["vendor.build.prop"] = LoadBuildProp(read_helper, 'VENDOR/build.prop')
return d return d
def LoadBuildProp(read_helper): def LoadBuildProp(read_helper, prop_file):
try: try:
data = read_helper("SYSTEM/build.prop") data = read_helper(prop_file)
except KeyError: except KeyError:
print("Warning: could not find SYSTEM/build.prop in %s" % (zip,)) print("Warning: could not read %s" % (prop_file,))
data = "" data = ""
return LoadDictionaryFromLines(data.split("\n")) return LoadDictionaryFromLines(data.split("\n"))

View File

@@ -275,6 +275,11 @@ def HasVendorPartition(target_files_zip):
return False return False
def HasTrebleEnabled(target_files_zip, info_dict):
return (HasVendorPartition(target_files_zip) and
GetBuildProp("ro.treble.enabled", info_dict) == "true")
def GetOemProperty(name, oem_props, oem_dict, info_dict): def GetOemProperty(name, oem_props, oem_dict, info_dict):
if oem_props is not None and name in oem_props: if oem_props is not None and name in oem_props:
return oem_dict[name] return oem_dict[name]
@@ -316,57 +321,93 @@ def GetImage(which, tmpdir):
return sparse_img.SparseImage(path, mappath, clobbered_blocks) return sparse_img.SparseImage(path, mappath, clobbered_blocks)
def AddCompatibilityArchive(target_zip, output_zip, system_included=True, def AddCompatibilityArchiveIfTrebleEnabled(target_zip, output_zip,
vendor_included=True): target_info_dict,
"""Adds compatibility info from target files into the output zip. source_info_dict=None):
"""Adds compatibility info into the output zip if it's Treble-enabled target.
Metadata used for on-device compatibility verification is retrieved from Metadata used for on-device compatibility verification is retrieved from
target_zip then added to compatibility.zip which is added to the output_zip target_zip then added to compatibility.zip which is added to the output_zip
archive. archive.
Compatibility archive should only be included for devices with a vendor Compatibility archive should only be included for devices that have enabled
partition as checking provides value when system and vendor are independently Treble support.
versioned.
Args: Args:
target_zip: Zip file containing the source files to be included for OTA. target_zip: Zip file containing the source files to be included for OTA.
output_zip: Zip file that will be sent for OTA. output_zip: Zip file that will be sent for OTA.
system_included: If True, the system image will be updated and therefore target_info_dict: The dict that holds the target build info.
its metadata should be included. source_info_dict: The dict that holds the source build info, if generating
vendor_included: If True, the vendor image will be updated and therefore an incremental OTA; None otherwise.
its metadata should be included.
""" """
# Determine what metadata we need. Files are names relative to META/. def AddCompatibilityArchive(system_updated, vendor_updated):
compatibility_files = [] """Adds compatibility info based on system/vendor update status.
vendor_metadata = ("vendor_manifest.xml", "vendor_matrix.xml")
system_metadata = ("system_manifest.xml", "system_matrix.xml")
if vendor_included:
compatibility_files += vendor_metadata
if system_included:
compatibility_files += system_metadata
# Create new archive. Args:
compatibility_archive = tempfile.NamedTemporaryFile() system_updated: If True, the system image will be updated and therefore
compatibility_archive_zip = zipfile.ZipFile(compatibility_archive, "w", its metadata should be included.
compression=zipfile.ZIP_DEFLATED) vendor_updated: If True, the vendor image will be updated and therefore
its metadata should be included.
"""
# Determine what metadata we need. Files are names relative to META/.
compatibility_files = []
vendor_metadata = ("vendor_manifest.xml", "vendor_matrix.xml")
system_metadata = ("system_manifest.xml", "system_matrix.xml")
if vendor_updated:
compatibility_files += vendor_metadata
if system_updated:
compatibility_files += system_metadata
# Add metadata. # Create new archive.
for file_name in compatibility_files: compatibility_archive = tempfile.NamedTemporaryFile()
target_file_name = "META/" + file_name compatibility_archive_zip = zipfile.ZipFile(compatibility_archive, "w",
compression=zipfile.ZIP_DEFLATED)
if target_file_name in target_zip.namelist(): # Add metadata.
data = target_zip.read(target_file_name) for file_name in compatibility_files:
common.ZipWriteStr(compatibility_archive_zip, file_name, data) target_file_name = "META/" + file_name
# Ensure files are written before we copy into output_zip. if target_file_name in target_zip.namelist():
compatibility_archive_zip.close() data = target_zip.read(target_file_name)
common.ZipWriteStr(compatibility_archive_zip, file_name, data)
# Only add the archive if we have any compatibility info. # Ensure files are written before we copy into output_zip.
if compatibility_archive_zip.namelist(): compatibility_archive_zip.close()
common.ZipWrite(output_zip, compatibility_archive.name,
arcname="compatibility.zip", # Only add the archive if we have any compatibility info.
compress_type=zipfile.ZIP_STORED) if compatibility_archive_zip.namelist():
common.ZipWrite(output_zip, compatibility_archive.name,
arcname="compatibility.zip",
compress_type=zipfile.ZIP_STORED)
# Will only proceed if the target has enabled the Treble support (as well as
# having a /vendor partition).
if not HasTrebleEnabled(target_zip, target_info_dict):
return
# We don't support OEM thumbprint in Treble world (which calculates
# fingerprints in a different way as shown in CalculateFingerprint()).
assert not target_info_dict.get("oem_fingerprint_properties")
# Full OTA carries the info for system/vendor both.
if source_info_dict is None:
AddCompatibilityArchive(True, True)
return
assert not source_info_dict.get("oem_fingerprint_properties")
source_fp = GetBuildProp("ro.build.fingerprint", source_info_dict)
target_fp = GetBuildProp("ro.build.fingerprint", target_info_dict)
system_updated = source_fp != target_fp
source_fp_vendor = GetVendorBuildProp("ro.vendor.build.fingerprint",
source_info_dict)
target_fp_vendor = GetVendorBuildProp("ro.vendor.build.fingerprint",
target_info_dict)
vendor_updated = source_fp_vendor != target_fp_vendor
AddCompatibilityArchive(system_updated, vendor_updated)
def WriteFullOTAPackage(input_zip, output_zip): def WriteFullOTAPackage(input_zip, output_zip):
@@ -494,6 +535,9 @@ else if get_stage("%(bcb_dev)s") == "3/3" then
vendor_diff = common.BlockDifference("vendor", vendor_tgt) vendor_diff = common.BlockDifference("vendor", vendor_tgt)
vendor_diff.WriteScript(script, output_zip) vendor_diff.WriteScript(script, output_zip)
AddCompatibilityArchiveIfTrebleEnabled(input_zip, output_zip,
OPTIONS.info_dict)
common.CheckSize(boot_img.data, "boot.img", OPTIONS.info_dict) common.CheckSize(boot_img.data, "boot.img", OPTIONS.info_dict)
common.ZipWriteStr(output_zip, "boot.img", boot_img.data) common.ZipWriteStr(output_zip, "boot.img", boot_img.data)
@@ -542,13 +586,22 @@ def WriteMetadata(metadata, output_zip):
def GetBuildProp(prop, info_dict): def GetBuildProp(prop, info_dict):
"""Return the fingerprint of the build of a given target-files info_dict.""" """Returns the inquired build property from a given info_dict."""
try: try:
return info_dict.get("build.prop", {})[prop] return info_dict.get("build.prop", {})[prop]
except KeyError: except KeyError:
raise common.ExternalError("couldn't find %s in build.prop" % (prop,)) raise common.ExternalError("couldn't find %s in build.prop" % (prop,))
def GetVendorBuildProp(prop, info_dict):
"""Returns the inquired vendor build property from a given info_dict."""
try:
return info_dict.get("vendor.build.prop", {})[prop]
except KeyError:
raise common.ExternalError(
"couldn't find %s in vendor.build.prop" % (prop,))
def HandleDowngradeMetadata(metadata): def HandleDowngradeMetadata(metadata):
# Only incremental OTAs are allowed to reach here. # Only incremental OTAs are allowed to reach here.
assert OPTIONS.incremental_source is not None assert OPTIONS.incremental_source is not None
@@ -679,6 +732,10 @@ def WriteBlockIncrementalOTAPackage(target_zip, source_zip, output_zip):
else: else:
vendor_diff = None vendor_diff = None
AddCompatibilityArchiveIfTrebleEnabled(
target_zip, output_zip, OPTIONS.target_info_dict,
OPTIONS.source_info_dict)
AppendAssertions(script, OPTIONS.target_info_dict, oem_dicts) AppendAssertions(script, OPTIONS.target_info_dict, oem_dicts)
device_specific.IncrementalOTA_Assertions() device_specific.IncrementalOTA_Assertions()
@@ -1169,32 +1226,13 @@ def WriteABOTAPackageWithBrilloScript(target_file, output_file,
else: else:
print("Warning: cannot find care map file in target_file package") print("Warning: cannot find care map file in target_file package")
if HasVendorPartition(target_zip): # OPTIONS.source_info_dict must be None for incrementals.
update_vendor = True if source_file is None:
update_system = True assert OPTIONS.source_info_dict is None
# If incremental then figure out what is being updated so metadata only for AddCompatibilityArchiveIfTrebleEnabled(
# the updated image is included. target_zip, output_zip, OPTIONS.info_dict, OPTIONS.source_info_dict)
if source_file is not None:
input_tmp, input_zip = common.UnzipTemp(
target_file, UNZIP_PATTERN)
source_tmp, source_zip = common.UnzipTemp(
source_file, UNZIP_PATTERN)
vendor_src = GetImage("vendor", source_tmp)
vendor_tgt = GetImage("vendor", input_tmp)
system_src = GetImage("system", source_tmp)
system_tgt = GetImage("system", input_tmp)
update_system = system_src.TotalSha1() != system_tgt.TotalSha1()
update_vendor = vendor_src.TotalSha1() != vendor_tgt.TotalSha1()
input_zip.close()
source_zip.close()
target_zip = zipfile.ZipFile(target_file, "r")
AddCompatibilityArchive(target_zip, output_zip, update_system,
update_vendor)
common.ZipClose(target_zip) common.ZipClose(target_zip)
# Write the current metadata entry with placeholders. # Write the current metadata entry with placeholders.