bc7e0a9f25
Introduce a new option `--allow-partial-ab` in merge_target_files, which allows merging a non-AB framework with an AB vendor. The reason for adding this option is to support merging a real device framework with a cuttlefish vendor. Cuttlefish enables AB partition by default; however, some real devices do not. Bug: 318326532 Test: merge_target_files Test: atest --host releasetools_test Change-Id: Iaebd06796153fe82fbf56e86fcc8c500b6d60771
686 lines
24 KiB
Python
Executable file
686 lines
24 KiB
Python
Executable file
#!/usr/bin/env python
|
|
#
|
|
# Copyright (C) 2022 The Android Open Source Project
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may not
|
|
# use this file except in compliance with the License. You may obtain a copy of
|
|
# the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
# License for the specific language governing permissions and limitations under
|
|
# the License.
|
|
#
|
|
"""This script merges two partial target files packages.
|
|
|
|
One input package contains framework files, and the other contains vendor files.
|
|
|
|
This script produces a complete, merged target files package:
|
|
- This package can be used to generate a flashable IMG package.
|
|
See --output-img.
|
|
- This package can be used to generate an OTA package. See --output-ota.
|
|
- The merged package is checked for compatibility between the two inputs.
|
|
|
|
Usage: merge_target_files [args]
|
|
|
|
--framework-target-files framework-target-files-package
|
|
The input target files package containing framework bits. This is a zip
|
|
archive or a directory.
|
|
|
|
--framework-item-list framework-item-list-file
|
|
The optional path to a newline-separated config file of items that
|
|
are extracted as-is from the framework target files package.
|
|
|
|
--framework-misc-info-keys framework-misc-info-keys-file
|
|
The optional path to a newline-separated config file of keys to
|
|
extract from the framework META/misc_info.txt file.
|
|
|
|
--vendor-target-files vendor-target-files-package
|
|
The input target files package containing vendor bits. This is a zip
|
|
archive or a directory.
|
|
|
|
--vendor-item-list vendor-item-list-file
|
|
The optional path to a newline-separated config file of items that
|
|
are extracted as-is from the vendor target files package.
|
|
|
|
--boot-image-dir-path
|
|
The input boot image directory path. This path contains IMAGES/boot.img
|
|
file.
|
|
|
|
--output-target-files output-target-files-package
|
|
If provided, the output merged target files package. Also a zip archive.
|
|
|
|
--output-dir output-directory
|
|
If provided, the destination directory for saving merged files. Requires
|
|
the --output-item-list flag.
|
|
Can be provided alongside --output-target-files, or by itself.
|
|
|
|
--output-item-list output-item-list-file.
|
|
The optional path to a newline-separated config file that specifies the
|
|
file patterns to copy into the --output-dir. Required if providing
|
|
the --output-dir flag.
|
|
|
|
--output-ota output-ota-package
|
|
The output ota package. This is a zip archive. Use of this flag may
|
|
require passing the --path common flag; see common.py.
|
|
|
|
--output-img output-img-package
|
|
The output img package, suitable for use with 'fastboot update'. Use of
|
|
this flag may require passing the --path common flag; see common.py.
|
|
|
|
--output-super-empty output-super-empty-image
|
|
If provided, creates a super_empty.img file from the merged target
|
|
files package and saves it at this path.
|
|
|
|
--rebuild_recovery
|
|
Copy the recovery image used by non-A/B devices, used when
|
|
regenerating vendor images with --rebuild-sepolicy.
|
|
|
|
--allow-duplicate-apkapex-keys
|
|
If provided, duplicate APK/APEX keys are ignored and the value from the
|
|
framework is used.
|
|
|
|
--rebuild-sepolicy
|
|
If provided, rebuilds odm.img or vendor.img to include merged sepolicy
|
|
files. If odm is present then odm is preferred.
|
|
|
|
--vendor-otatools otatools.zip
|
|
If provided, use this otatools.zip when recompiling the odm or vendor
|
|
image to include sepolicy.
|
|
|
|
--keep-tmp
|
|
Keep tempoary files for debugging purposes.
|
|
|
|
--avb-resolve-rollback-index-location-conflict
|
|
If provided, resolve the conflict AVB rollback index location when
|
|
necessary.
|
|
|
|
--allow-partial-ab
|
|
If provided, allow merging non-AB framework target files with AB vendor
|
|
target files, which means that only the vendor has AB partitions.
|
|
|
|
The following only apply when using the VSDK to perform dexopt on vendor apps:
|
|
|
|
--framework-dexpreopt-config
|
|
If provided, the location of framwework's dexpreopt_config.zip.
|
|
|
|
--framework-dexpreopt-tools
|
|
if provided, the location of framework's dexpreopt_tools.zip.
|
|
|
|
--vendor-dexpreopt-config
|
|
If provided, the location of vendor's dexpreopt_config.zip.
|
|
"""
|
|
|
|
import logging
|
|
import os
|
|
import shutil
|
|
import subprocess
|
|
import sys
|
|
import zipfile
|
|
|
|
import add_img_to_target_files
|
|
import build_image
|
|
import build_super_image
|
|
import common
|
|
import img_from_target_files
|
|
import merge_compatibility_checks
|
|
import merge_dexopt
|
|
import merge_meta
|
|
import merge_utils
|
|
import ota_from_target_files
|
|
|
|
from common import ExternalError
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
OPTIONS = common.OPTIONS
|
|
# Always turn on verbose logging.
|
|
OPTIONS.verbose = True
|
|
OPTIONS.framework_target_files = None
|
|
OPTIONS.framework_item_list = []
|
|
OPTIONS.framework_misc_info_keys = []
|
|
OPTIONS.vendor_target_files = None
|
|
OPTIONS.vendor_item_list = []
|
|
OPTIONS.boot_image_dir_path = None
|
|
OPTIONS.output_target_files = None
|
|
OPTIONS.output_dir = None
|
|
OPTIONS.output_item_list = []
|
|
OPTIONS.output_ota = None
|
|
OPTIONS.output_img = None
|
|
OPTIONS.output_super_empty = None
|
|
OPTIONS.rebuild_recovery = False
|
|
# TODO(b/150582573): Remove this option.
|
|
OPTIONS.allow_duplicate_apkapex_keys = False
|
|
OPTIONS.vendor_otatools = None
|
|
OPTIONS.rebuild_sepolicy = False
|
|
OPTIONS.keep_tmp = False
|
|
OPTIONS.avb_resolve_rollback_index_location_conflict = False
|
|
OPTIONS.allow_partial_ab = False
|
|
OPTIONS.framework_dexpreopt_config = None
|
|
OPTIONS.framework_dexpreopt_tools = None
|
|
OPTIONS.vendor_dexpreopt_config = None
|
|
|
|
|
|
def move_only_exists(source, destination):
|
|
"""Judge whether the file exists and then move the file."""
|
|
|
|
if os.path.exists(source):
|
|
shutil.move(source, destination)
|
|
|
|
|
|
def remove_file_if_exists(file_name):
|
|
"""Remove the file if it exists and skip otherwise."""
|
|
|
|
try:
|
|
os.remove(file_name)
|
|
except FileNotFoundError:
|
|
pass
|
|
|
|
|
|
def include_extra_in_list(item_list):
|
|
"""
|
|
1. Include all `META/*` files in the item list.
|
|
|
|
To ensure that `AddImagesToTargetFiles` can still be used with vendor item
|
|
list that do not specify all of the required META/ files, those files should
|
|
be included by default. This preserves the backward compatibility of
|
|
`rebuild_image_with_sepolicy`.
|
|
|
|
2. Include `SYSTEM/build.prop` file in the item list.
|
|
|
|
To ensure that `AddImagesToTargetFiles` for GRF vendor images, can still
|
|
access SYSTEM/build.prop to pass GetPartitionFingerprint check in BuildInfo
|
|
constructor.
|
|
"""
|
|
if not item_list:
|
|
return None
|
|
return list(item_list) + ['META/*'] + ['SYSTEM/build.prop']
|
|
|
|
|
|
def create_merged_package(temp_dir):
|
|
"""Merges two target files packages into one target files structure.
|
|
|
|
Returns:
|
|
Path to merged package under temp directory.
|
|
"""
|
|
# Extract "as is" items from the input framework and vendor partial target
|
|
# files packages directly into the output temporary directory, since these
|
|
# items do not need special case processing.
|
|
|
|
output_target_files_temp_dir = os.path.join(temp_dir, 'output')
|
|
merge_utils.CollectTargetFiles(
|
|
input_zipfile_or_dir=OPTIONS.framework_target_files,
|
|
output_dir=output_target_files_temp_dir,
|
|
item_list=OPTIONS.framework_item_list)
|
|
merge_utils.CollectTargetFiles(
|
|
input_zipfile_or_dir=OPTIONS.vendor_target_files,
|
|
output_dir=output_target_files_temp_dir,
|
|
item_list=OPTIONS.vendor_item_list)
|
|
|
|
if OPTIONS.boot_image_dir_path:
|
|
merge_utils.CollectTargetFiles(
|
|
input_zipfile_or_dir=OPTIONS.boot_image_dir_path,
|
|
output_dir=output_target_files_temp_dir,
|
|
item_list=['IMAGES/boot.img'])
|
|
|
|
# Perform special case processing on META/* items.
|
|
# After this function completes successfully, all the files we need to create
|
|
# the output target files package are in place.
|
|
merge_meta.MergeMetaFiles(
|
|
temp_dir=temp_dir,
|
|
merged_dir=output_target_files_temp_dir,
|
|
framework_partitions=OPTIONS.framework_partition_set)
|
|
|
|
merge_dexopt.MergeDexopt(
|
|
temp_dir=temp_dir, output_target_files_dir=output_target_files_temp_dir)
|
|
|
|
return output_target_files_temp_dir
|
|
|
|
|
|
def generate_missing_images(target_files_dir):
|
|
"""Generate any missing images from target files."""
|
|
|
|
# Regenerate IMAGES in the target directory.
|
|
|
|
add_img_args = [
|
|
'--verbose',
|
|
'--add_missing',
|
|
]
|
|
if OPTIONS.rebuild_recovery:
|
|
add_img_args.append('--rebuild_recovery')
|
|
if OPTIONS.avb_resolve_rollback_index_location_conflict:
|
|
add_img_args.append('--avb_resolve_rollback_index_location_conflict')
|
|
add_img_args.append(target_files_dir)
|
|
|
|
add_img_to_target_files.main(add_img_args)
|
|
|
|
|
|
def rebuild_image_with_sepolicy(target_files_dir):
|
|
"""Rebuilds odm.img or vendor.img to include merged sepolicy files.
|
|
|
|
If odm is present then odm is preferred -- otherwise vendor is used.
|
|
"""
|
|
partition = 'vendor'
|
|
if os.path.exists(os.path.join(target_files_dir, 'ODM')):
|
|
partition = 'odm'
|
|
partition_img = '{}.img'.format(partition)
|
|
partition_map = '{}.map'.format(partition)
|
|
|
|
logger.info('Recompiling %s using the merged sepolicy files.', partition_img)
|
|
|
|
# Copy the combined SEPolicy file and framework hashes to the image that is
|
|
# being rebuilt.
|
|
def copy_selinux_file(input_path, output_filename):
|
|
input_filename = os.path.join(target_files_dir, input_path)
|
|
if not os.path.exists(input_filename):
|
|
input_filename = input_filename.replace('SYSTEM_EXT/',
|
|
'SYSTEM/system_ext/') \
|
|
.replace('PRODUCT/', 'SYSTEM/product/')
|
|
if not os.path.exists(input_filename):
|
|
logger.info('Skipping copy_selinux_file for %s', input_filename)
|
|
return
|
|
shutil.copy(
|
|
input_filename,
|
|
os.path.join(target_files_dir, partition.upper(), 'etc/selinux',
|
|
output_filename))
|
|
|
|
copy_selinux_file('META/combined_sepolicy', 'precompiled_sepolicy')
|
|
copy_selinux_file('SYSTEM/etc/selinux/plat_sepolicy_and_mapping.sha256',
|
|
'precompiled_sepolicy.plat_sepolicy_and_mapping.sha256')
|
|
copy_selinux_file(
|
|
'SYSTEM_EXT/etc/selinux/system_ext_sepolicy_and_mapping.sha256',
|
|
'precompiled_sepolicy.system_ext_sepolicy_and_mapping.sha256')
|
|
copy_selinux_file('PRODUCT/etc/selinux/product_sepolicy_and_mapping.sha256',
|
|
'precompiled_sepolicy.product_sepolicy_and_mapping.sha256')
|
|
|
|
if not OPTIONS.vendor_otatools:
|
|
# Remove the partition from the merged target-files archive. It will be
|
|
# rebuilt later automatically by generate_missing_images().
|
|
remove_file_if_exists(
|
|
os.path.join(target_files_dir, 'IMAGES', partition_img))
|
|
return
|
|
|
|
# TODO(b/192253131): Remove the need for vendor_otatools by fixing
|
|
# backwards-compatibility issues when compiling images across releases.
|
|
if not OPTIONS.vendor_target_files:
|
|
raise ValueError(
|
|
'Expected vendor_target_files if vendor_otatools is not None.')
|
|
logger.info(
|
|
'%s recompilation will be performed using the vendor otatools.zip',
|
|
partition_img)
|
|
|
|
# Unzip the vendor build's otatools.zip and target-files archive.
|
|
vendor_otatools_dir = common.MakeTempDir(
|
|
prefix='merge_target_files_vendor_otatools_')
|
|
vendor_target_files_dir = common.MakeTempDir(
|
|
prefix='merge_target_files_vendor_target_files_')
|
|
common.UnzipToDir(OPTIONS.vendor_otatools, vendor_otatools_dir)
|
|
merge_utils.CollectTargetFiles(
|
|
input_zipfile_or_dir=OPTIONS.vendor_target_files,
|
|
output_dir=vendor_target_files_dir,
|
|
item_list=include_extra_in_list(OPTIONS.vendor_item_list))
|
|
|
|
# Copy the partition contents from the merged target-files archive to the
|
|
# vendor target-files archive.
|
|
shutil.rmtree(os.path.join(vendor_target_files_dir, partition.upper()))
|
|
shutil.copytree(
|
|
os.path.join(target_files_dir, partition.upper()),
|
|
os.path.join(vendor_target_files_dir, partition.upper()),
|
|
symlinks=True)
|
|
|
|
# Delete then rebuild the partition.
|
|
remove_file_if_exists(
|
|
os.path.join(vendor_target_files_dir, 'IMAGES', partition_img))
|
|
rebuild_partition_command = [
|
|
os.path.join(vendor_otatools_dir, 'bin', 'add_img_to_target_files'),
|
|
'--verbose',
|
|
'--add_missing',
|
|
]
|
|
if OPTIONS.rebuild_recovery:
|
|
rebuild_partition_command.append('--rebuild_recovery')
|
|
rebuild_partition_command.append(vendor_target_files_dir)
|
|
logger.info('Recompiling %s: %s', partition_img,
|
|
' '.join(rebuild_partition_command))
|
|
common.RunAndCheckOutput(rebuild_partition_command, verbose=True)
|
|
|
|
# Move the newly-created image to the merged target files dir.
|
|
if not os.path.exists(os.path.join(target_files_dir, 'IMAGES')):
|
|
os.makedirs(os.path.join(target_files_dir, 'IMAGES'))
|
|
shutil.move(
|
|
os.path.join(vendor_target_files_dir, 'IMAGES', partition_img),
|
|
os.path.join(target_files_dir, 'IMAGES', partition_img))
|
|
move_only_exists(
|
|
os.path.join(vendor_target_files_dir, 'IMAGES', partition_map),
|
|
os.path.join(target_files_dir, 'IMAGES', partition_map))
|
|
|
|
def copy_recovery_file(filename):
|
|
for subdir in ('VENDOR', 'SYSTEM/vendor'):
|
|
source = os.path.join(vendor_target_files_dir, subdir, filename)
|
|
if os.path.exists(source):
|
|
dest = os.path.join(target_files_dir, subdir, filename)
|
|
shutil.copy(source, dest)
|
|
return
|
|
logger.info('Skipping copy_recovery_file for %s, file not found', filename)
|
|
|
|
if OPTIONS.rebuild_recovery:
|
|
copy_recovery_file('etc/recovery.img')
|
|
copy_recovery_file('bin/install-recovery.sh')
|
|
copy_recovery_file('recovery-from-boot.p')
|
|
|
|
|
|
def generate_super_empty_image(target_dir, output_super_empty):
|
|
"""Generates super_empty image from target package.
|
|
|
|
Args:
|
|
target_dir: Path to the target file package which contains misc_info.txt for
|
|
detailed information for super image.
|
|
output_super_empty: If provided, copies a super_empty.img file from the
|
|
target files package to this path.
|
|
"""
|
|
# Create super_empty.img using the merged misc_info.txt.
|
|
|
|
misc_info_txt = os.path.join(target_dir, 'META', 'misc_info.txt')
|
|
|
|
use_dynamic_partitions = common.LoadDictionaryFromFile(misc_info_txt).get(
|
|
'use_dynamic_partitions')
|
|
|
|
if use_dynamic_partitions != 'true' and output_super_empty:
|
|
raise ValueError(
|
|
'Building super_empty.img requires use_dynamic_partitions=true.')
|
|
elif use_dynamic_partitions == 'true':
|
|
super_empty_img = os.path.join(target_dir, 'IMAGES', 'super_empty.img')
|
|
build_super_image_args = [
|
|
misc_info_txt,
|
|
super_empty_img,
|
|
]
|
|
build_super_image.main(build_super_image_args)
|
|
|
|
# Copy super_empty.img to the user-provided output_super_empty location.
|
|
if output_super_empty:
|
|
shutil.copyfile(super_empty_img, output_super_empty)
|
|
|
|
|
|
def create_target_files_archive(output_zip, source_dir, temp_dir):
|
|
"""Creates a target_files zip archive from the input source dir.
|
|
|
|
Args:
|
|
output_zip: The name of the zip archive target files package.
|
|
source_dir: The target directory contains package to be archived.
|
|
temp_dir: Path to temporary directory for any intermediate files.
|
|
"""
|
|
output_target_files_list = os.path.join(temp_dir, 'output.list')
|
|
output_target_files_meta_dir = os.path.join(source_dir, 'META')
|
|
|
|
def files_from_path(target_path, extra_args=None):
|
|
"""Gets files under the given path and return a sorted list."""
|
|
find_command = ['find', target_path] + (extra_args or [])
|
|
find_process = common.Run(
|
|
find_command, stdout=subprocess.PIPE, verbose=False)
|
|
return common.RunAndCheckOutput(['sort'],
|
|
stdin=find_process.stdout,
|
|
verbose=False)
|
|
|
|
# META content appears first in the zip. This is done by the
|
|
# standard build system for optimized extraction of those files,
|
|
# so we do the same step for merged target_files.zips here too.
|
|
meta_content = files_from_path(output_target_files_meta_dir)
|
|
other_content = files_from_path(
|
|
source_dir,
|
|
['-path', output_target_files_meta_dir, '-prune', '-o', '-print'])
|
|
|
|
with open(output_target_files_list, 'w') as f:
|
|
f.write(meta_content)
|
|
f.write(other_content)
|
|
|
|
command = [
|
|
'soong_zip',
|
|
'-d',
|
|
'-o',
|
|
os.path.abspath(output_zip),
|
|
'-C',
|
|
source_dir,
|
|
'-r',
|
|
output_target_files_list,
|
|
]
|
|
|
|
logger.info('creating %s', output_zip)
|
|
common.RunAndCheckOutput(command, verbose=True)
|
|
logger.info('finished creating %s', output_zip)
|
|
|
|
|
|
def merge_target_files(temp_dir):
|
|
"""Merges two target files packages together.
|
|
|
|
This function uses framework and vendor target files packages as input,
|
|
performs various file extractions, special case processing, and finally
|
|
creates a merged zip archive as output.
|
|
|
|
Args:
|
|
temp_dir: The name of a directory we use when we extract items from the
|
|
input target files packages, and also a scratch directory that we use for
|
|
temporary files.
|
|
"""
|
|
|
|
logger.info('starting: merge framework %s and vendor %s into output %s',
|
|
OPTIONS.framework_target_files, OPTIONS.vendor_target_files,
|
|
OPTIONS.output_target_files)
|
|
|
|
output_target_files_temp_dir = create_merged_package(temp_dir)
|
|
|
|
partition_map = common.PartitionMapFromTargetFiles(
|
|
output_target_files_temp_dir)
|
|
|
|
compatibility_errors = merge_compatibility_checks.CheckCompatibility(
|
|
target_files_dir=output_target_files_temp_dir,
|
|
partition_map=partition_map)
|
|
if compatibility_errors:
|
|
for error in compatibility_errors:
|
|
logger.error(error)
|
|
raise ExternalError(
|
|
'Found incompatibilities in the merged target files package.')
|
|
|
|
# Include the compiled policy in an image if requested.
|
|
if OPTIONS.rebuild_sepolicy:
|
|
rebuild_image_with_sepolicy(output_target_files_temp_dir)
|
|
|
|
generate_missing_images(output_target_files_temp_dir)
|
|
|
|
generate_super_empty_image(output_target_files_temp_dir,
|
|
OPTIONS.output_super_empty)
|
|
|
|
# Finally, create the output target files zip archive and/or copy the
|
|
# output items to the output target files directory.
|
|
|
|
if OPTIONS.output_dir:
|
|
merge_utils.CopyItems(output_target_files_temp_dir, OPTIONS.output_dir,
|
|
OPTIONS.output_item_list)
|
|
|
|
if not OPTIONS.output_target_files:
|
|
return
|
|
|
|
create_target_files_archive(OPTIONS.output_target_files,
|
|
output_target_files_temp_dir, temp_dir)
|
|
|
|
# Create the IMG package from the merged target files package.
|
|
if OPTIONS.output_img:
|
|
img_from_target_files.main(
|
|
[OPTIONS.output_target_files, OPTIONS.output_img])
|
|
|
|
# Create the OTA package from the merged target files package.
|
|
|
|
if OPTIONS.output_ota:
|
|
ota_from_target_files.main(
|
|
[OPTIONS.output_target_files, OPTIONS.output_ota])
|
|
|
|
|
|
def main():
|
|
"""The main function.
|
|
|
|
Process command line arguments, then call merge_target_files to
|
|
perform the heavy lifting.
|
|
"""
|
|
|
|
common.InitLogging()
|
|
|
|
def option_handler(o, a):
|
|
if o == '--system-target-files':
|
|
logger.warning(
|
|
'--system-target-files has been renamed to --framework-target-files')
|
|
OPTIONS.framework_target_files = a
|
|
elif o == '--framework-target-files':
|
|
OPTIONS.framework_target_files = a
|
|
elif o == '--system-item-list':
|
|
logger.warning(
|
|
'--system-item-list has been renamed to --framework-item-list')
|
|
OPTIONS.framework_item_list = a
|
|
elif o == '--framework-item-list':
|
|
OPTIONS.framework_item_list = a
|
|
elif o == '--system-misc-info-keys':
|
|
logger.warning('--system-misc-info-keys has been renamed to '
|
|
'--framework-misc-info-keys')
|
|
OPTIONS.framework_misc_info_keys = a
|
|
elif o == '--framework-misc-info-keys':
|
|
OPTIONS.framework_misc_info_keys = a
|
|
elif o == '--other-target-files':
|
|
logger.warning(
|
|
'--other-target-files has been renamed to --vendor-target-files')
|
|
OPTIONS.vendor_target_files = a
|
|
elif o == '--vendor-target-files':
|
|
OPTIONS.vendor_target_files = a
|
|
elif o == '--other-item-list':
|
|
logger.warning('--other-item-list has been renamed to --vendor-item-list')
|
|
OPTIONS.vendor_item_list = a
|
|
elif o == '--vendor-item-list':
|
|
OPTIONS.vendor_item_list = a
|
|
elif o == '--boot-image-dir-path':
|
|
OPTIONS.boot_image_dir_path = a
|
|
elif o == '--output-target-files':
|
|
OPTIONS.output_target_files = a
|
|
elif o == '--output-dir':
|
|
OPTIONS.output_dir = a
|
|
elif o == '--output-item-list':
|
|
OPTIONS.output_item_list = a
|
|
elif o == '--output-ota':
|
|
OPTIONS.output_ota = a
|
|
elif o == '--output-img':
|
|
OPTIONS.output_img = a
|
|
elif o == '--output-super-empty':
|
|
OPTIONS.output_super_empty = a
|
|
elif o == '--rebuild_recovery' or o == '--rebuild-recovery':
|
|
OPTIONS.rebuild_recovery = True
|
|
elif o == '--allow-duplicate-apkapex-keys':
|
|
OPTIONS.allow_duplicate_apkapex_keys = True
|
|
elif o == '--vendor-otatools':
|
|
OPTIONS.vendor_otatools = a
|
|
elif o == '--rebuild-sepolicy':
|
|
OPTIONS.rebuild_sepolicy = True
|
|
elif o == '--keep-tmp':
|
|
OPTIONS.keep_tmp = True
|
|
elif o == '--avb-resolve-rollback-index-location-conflict':
|
|
OPTIONS.avb_resolve_rollback_index_location_conflict = True
|
|
elif o == '--allow-partial-ab':
|
|
OPTIONS.allow_partial_ab = True
|
|
elif o == '--framework-dexpreopt-config':
|
|
OPTIONS.framework_dexpreopt_config = a
|
|
elif o == '--framework-dexpreopt-tools':
|
|
OPTIONS.framework_dexpreopt_tools = a
|
|
elif o == '--vendor-dexpreopt-config':
|
|
OPTIONS.vendor_dexpreopt_config = a
|
|
else:
|
|
return False
|
|
return True
|
|
|
|
args = common.ParseOptions(
|
|
sys.argv[1:],
|
|
__doc__,
|
|
extra_long_opts=[
|
|
'system-target-files=',
|
|
'framework-target-files=',
|
|
'system-item-list=',
|
|
'framework-item-list=',
|
|
'system-misc-info-keys=',
|
|
'framework-misc-info-keys=',
|
|
'other-target-files=',
|
|
'vendor-target-files=',
|
|
'other-item-list=',
|
|
'vendor-item-list=',
|
|
'boot-image-dir-path=',
|
|
'output-target-files=',
|
|
'output-dir=',
|
|
'output-item-list=',
|
|
'output-ota=',
|
|
'output-img=',
|
|
'output-super-empty=',
|
|
'framework-dexpreopt-config=',
|
|
'framework-dexpreopt-tools=',
|
|
'vendor-dexpreopt-config=',
|
|
'rebuild_recovery',
|
|
'rebuild-recovery',
|
|
'allow-duplicate-apkapex-keys',
|
|
'vendor-otatools=',
|
|
'rebuild-sepolicy',
|
|
'keep-tmp',
|
|
'avb-resolve-rollback-index-location-conflict',
|
|
'allow-partial-ab',
|
|
],
|
|
extra_option_handler=option_handler)
|
|
|
|
# pylint: disable=too-many-boolean-expressions
|
|
if (args or OPTIONS.framework_target_files is None or
|
|
OPTIONS.vendor_target_files is None or
|
|
(OPTIONS.output_target_files is None and OPTIONS.output_dir is None) or
|
|
(OPTIONS.output_dir is not None and not OPTIONS.output_item_list) or
|
|
(OPTIONS.rebuild_recovery and not OPTIONS.rebuild_sepolicy)):
|
|
common.Usage(__doc__)
|
|
sys.exit(1)
|
|
|
|
framework_namelist = merge_utils.GetTargetFilesItems(
|
|
OPTIONS.framework_target_files)
|
|
vendor_namelist = merge_utils.GetTargetFilesItems(
|
|
OPTIONS.vendor_target_files)
|
|
|
|
if OPTIONS.framework_item_list:
|
|
OPTIONS.framework_item_list = common.LoadListFromFile(
|
|
OPTIONS.framework_item_list)
|
|
else:
|
|
OPTIONS.framework_item_list = merge_utils.InferItemList(
|
|
input_namelist=framework_namelist, framework=True)
|
|
OPTIONS.framework_partition_set = merge_utils.ItemListToPartitionSet(
|
|
OPTIONS.framework_item_list)
|
|
|
|
if OPTIONS.framework_misc_info_keys:
|
|
OPTIONS.framework_misc_info_keys = common.LoadListFromFile(
|
|
OPTIONS.framework_misc_info_keys)
|
|
else:
|
|
OPTIONS.framework_misc_info_keys = merge_utils.InferFrameworkMiscInfoKeys(
|
|
input_namelist=framework_namelist)
|
|
|
|
if OPTIONS.vendor_item_list:
|
|
OPTIONS.vendor_item_list = common.LoadListFromFile(OPTIONS.vendor_item_list)
|
|
else:
|
|
OPTIONS.vendor_item_list = merge_utils.InferItemList(
|
|
input_namelist=vendor_namelist, framework=False)
|
|
OPTIONS.vendor_partition_set = merge_utils.ItemListToPartitionSet(
|
|
OPTIONS.vendor_item_list)
|
|
|
|
if OPTIONS.output_item_list:
|
|
OPTIONS.output_item_list = common.LoadListFromFile(OPTIONS.output_item_list)
|
|
|
|
if not merge_utils.ValidateConfigLists():
|
|
sys.exit(1)
|
|
|
|
temp_dir = common.MakeTempDir(prefix='merge_target_files_')
|
|
try:
|
|
merge_target_files(temp_dir)
|
|
finally:
|
|
if OPTIONS.keep_tmp:
|
|
logger.info('Keeping temp_dir %s', temp_dir)
|
|
else:
|
|
common.Cleanup()
|
|
|
|
|
|
if __name__ == '__main__':
|
|
main()
|