blob: 46192464ab2fddec7653e883c3bee9a49520a0be [file] [log] [blame]
Bill Peckhame9eb5f92019-02-01 15:52:10 -08001#!/usr/bin/env python
2#
Daniel Norman03747412022-02-25 10:38:37 -08003# Copyright (C) 2022 The Android Open Source Project
Bill Peckhame9eb5f92019-02-01 15:52:10 -08004#
5# Licensed under the Apache License, Version 2.0 (the "License"); you may not
6# use this file except in compliance with the License. You may obtain a copy of
7# the License at
8#
9# http://www.apache.org/licenses/LICENSE-2.0
10#
11# Unless required by applicable law or agreed to in writing, software
12# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
13# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
14# License for the specific language governing permissions and limitations under
15# the License.
Daniel Norman4cc9df62019-07-18 10:11:07 -070016#
17"""This script merges two partial target files packages.
Bill Peckhame9eb5f92019-02-01 15:52:10 -080018
Daniel Normandbbf5a32020-10-22 16:03:32 -070019One input package contains framework files, and the other contains vendor files.
Bill Peckhame9eb5f92019-02-01 15:52:10 -080020
Daniel Normandbbf5a32020-10-22 16:03:32 -070021This script produces a complete, merged target files package:
22 - This package can be used to generate a flashable IMG package.
23 See --output-img.
24 - This package can be used to generate an OTA package. See --output-ota.
25 - The merged package is checked for compatibility between the two inputs.
26
27Usage: merge_target_files [args]
Bill Peckhame9eb5f92019-02-01 15:52:10 -080028
Dennis Song5bfa43e2023-03-30 18:28:00 +080029 --framework-target-files framework-target-files-package
Daniel Normand5d70ea2019-06-05 15:13:43 -070030 The input target files package containing framework bits. This is a zip
Dennis Song5bfa43e2023-03-30 18:28:00 +080031 archive or a directory.
Bill Peckhame9eb5f92019-02-01 15:52:10 -080032
Daniel Normand5d70ea2019-06-05 15:13:43 -070033 --framework-item-list framework-item-list-file
Daniel Norman5f476772022-03-02 15:46:34 -080034 The optional path to a newline-separated config file of items that
35 are extracted as-is from the framework target files package.
Daniel Norman2c99c5b2019-03-07 13:01:48 -080036
Daniel Normand5d70ea2019-06-05 15:13:43 -070037 --framework-misc-info-keys framework-misc-info-keys-file
Daniel Norman5f476772022-03-02 15:46:34 -080038 The optional path to a newline-separated config file of keys to
39 extract from the framework META/misc_info.txt file.
Daniel Norman2c99c5b2019-03-07 13:01:48 -080040
Dennis Song5bfa43e2023-03-30 18:28:00 +080041 --vendor-target-files vendor-target-files-package
Daniel Normand5d70ea2019-06-05 15:13:43 -070042 The input target files package containing vendor bits. This is a zip
Dennis Song5bfa43e2023-03-30 18:28:00 +080043 archive or a directory.
Bill Peckhame9eb5f92019-02-01 15:52:10 -080044
Daniel Normand5d70ea2019-06-05 15:13:43 -070045 --vendor-item-list vendor-item-list-file
Daniel Norman5f476772022-03-02 15:46:34 -080046 The optional path to a newline-separated config file of items that
47 are extracted as-is from the vendor target files package.
Daniel Norman2c99c5b2019-03-07 13:01:48 -080048
Sundong Ahn1930fd12023-11-02 19:13:35 +090049 --boot-image-dir-path
50 The input boot image directory path. This path contains IMAGES/boot.img
51 file.
52
Bill Peckhame9eb5f92019-02-01 15:52:10 -080053 --output-target-files output-target-files-package
Daniel Normanfdb38812019-04-15 09:47:24 -070054 If provided, the output merged target files package. Also a zip archive.
55
56 --output-dir output-directory
57 If provided, the destination directory for saving merged files. Requires
58 the --output-item-list flag.
59 Can be provided alongside --output-target-files, or by itself.
60
61 --output-item-list output-item-list-file.
62 The optional path to a newline-separated config file that specifies the
63 file patterns to copy into the --output-dir. Required if providing
64 the --output-dir flag.
Daniel Normana4911da2019-03-15 14:36:21 -070065
Daniel Norman3b64ce12019-04-16 16:11:35 -070066 --output-ota output-ota-package
67 The output ota package. This is a zip archive. Use of this flag may
68 require passing the --path common flag; see common.py.
69
Daniel Norman1bd2a1d2019-04-18 12:32:18 -070070 --output-img output-img-package
71 The output img package, suitable for use with 'fastboot update'. Use of
72 this flag may require passing the --path common flag; see common.py.
73
Daniel Normanf0318252019-04-15 11:34:56 -070074 --output-super-empty output-super-empty-image
75 If provided, creates a super_empty.img file from the merged target
76 files package and saves it at this path.
77
Daniel Normana4911da2019-03-15 14:36:21 -070078 --rebuild_recovery
Po Hua6c59122022-02-16 08:41:29 +000079 Copy the recovery image used by non-A/B devices, used when
80 regenerating vendor images with --rebuild-sepolicy.
Bill Peckham364c1cc2019-03-29 18:27:23 -070081
Daniel Normanb0c75912020-09-24 14:30:21 -070082 --allow-duplicate-apkapex-keys
83 If provided, duplicate APK/APEX keys are ignored and the value from the
84 framework is used.
85
Daniel Norman571e1822021-06-25 17:18:25 -070086 --rebuild-sepolicy
87 If provided, rebuilds odm.img or vendor.img to include merged sepolicy
88 files. If odm is present then odm is preferred.
89
90 --vendor-otatools otatools.zip
91 If provided, use this otatools.zip when recompiling the odm or vendor
92 image to include sepolicy.
93
Bill Peckham364c1cc2019-03-29 18:27:23 -070094 --keep-tmp
95 Keep tempoary files for debugging purposes.
Jose Galmes9c8f6eb2021-07-21 09:34:08 -070096
Dennis Song4aae62e2023-10-02 04:31:34 +000097 --avb-resolve-rollback-index-location-conflict
98 If provided, resolve the conflict AVB rollback index location when
99 necessary.
100
Jose Galmes9c8f6eb2021-07-21 09:34:08 -0700101 The following only apply when using the VSDK to perform dexopt on vendor apps:
102
103 --framework-dexpreopt-config
104 If provided, the location of framwework's dexpreopt_config.zip.
105
106 --framework-dexpreopt-tools
107 if provided, the location of framework's dexpreopt_tools.zip.
108
109 --vendor-dexpreopt-config
110 If provided, the location of vendor's dexpreopt_config.zip.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800111"""
112
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800113import logging
114import os
Daniel Normanfdb38812019-04-15 09:47:24 -0700115import shutil
Bill Peckham540d91a2019-04-25 14:18:16 -0700116import subprocess
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800117import sys
Daniel Norman5f476772022-03-02 15:46:34 -0800118import zipfile
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800119
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800120import add_img_to_target_files
Daniel Normandb8cacc2021-04-09 15:34:43 -0700121import build_image
Daniel Normanf0318252019-04-15 11:34:56 -0700122import build_super_image
123import common
Daniel Norman1bd2a1d2019-04-18 12:32:18 -0700124import img_from_target_files
Daniel Norman2465fc82022-03-02 12:01:20 -0800125import merge_compatibility_checks
126import merge_dexopt
127import merge_meta
128import merge_utils
Daniel Norman3b64ce12019-04-16 16:11:35 -0700129import ota_from_target_files
Daniel Normandb8cacc2021-04-09 15:34:43 -0700130
Daniel Normana84d13b2022-02-17 14:16:40 -0800131from common import ExternalError
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800132
133logger = logging.getLogger(__name__)
Tao Bao2ad4b822019-06-27 16:52:12 -0700134
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800135OPTIONS = common.OPTIONS
Bill Peckhamcb848172020-04-03 12:50:47 -0700136# Always turn on verbose logging.
137OPTIONS.verbose = True
Daniel Normand5d70ea2019-06-05 15:13:43 -0700138OPTIONS.framework_target_files = None
Daniel Norman5f476772022-03-02 15:46:34 -0800139OPTIONS.framework_item_list = []
140OPTIONS.framework_misc_info_keys = []
Daniel Normand5d70ea2019-06-05 15:13:43 -0700141OPTIONS.vendor_target_files = None
Daniel Norman5f476772022-03-02 15:46:34 -0800142OPTIONS.vendor_item_list = []
Sundong Ahn1930fd12023-11-02 19:13:35 +0900143OPTIONS.boot_image_dir_path = None
Bill Peckhamf753e152019-02-19 18:02:46 -0800144OPTIONS.output_target_files = None
Daniel Normanfdb38812019-04-15 09:47:24 -0700145OPTIONS.output_dir = None
Daniel Norman5f476772022-03-02 15:46:34 -0800146OPTIONS.output_item_list = []
Daniel Norman3b64ce12019-04-16 16:11:35 -0700147OPTIONS.output_ota = None
Daniel Norman1bd2a1d2019-04-18 12:32:18 -0700148OPTIONS.output_img = None
Daniel Normanf0318252019-04-15 11:34:56 -0700149OPTIONS.output_super_empty = None
Daniel Normana4911da2019-03-15 14:36:21 -0700150OPTIONS.rebuild_recovery = False
Daniel Normanb0c75912020-09-24 14:30:21 -0700151# TODO(b/150582573): Remove this option.
152OPTIONS.allow_duplicate_apkapex_keys = False
Daniel Norman571e1822021-06-25 17:18:25 -0700153OPTIONS.vendor_otatools = None
154OPTIONS.rebuild_sepolicy = False
Bill Peckhamf753e152019-02-19 18:02:46 -0800155OPTIONS.keep_tmp = False
Dennis Song4aae62e2023-10-02 04:31:34 +0000156OPTIONS.avb_resolve_rollback_index_location_conflict = False
Jose Galmes9c8f6eb2021-07-21 09:34:08 -0700157OPTIONS.framework_dexpreopt_config = None
158OPTIONS.framework_dexpreopt_tools = None
159OPTIONS.vendor_dexpreopt_config = None
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800160
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800161
jiangxu52bd42732022-04-11 16:02:48 +0800162def move_only_exists(source, destination):
163 """Judge whether the file exists and then move the file."""
164
165 if os.path.exists(source):
166 shutil.move(source, destination)
167
168
Dennis Song16d6fe02023-02-23 23:37:42 +0800169def remove_file_if_exists(file_name):
170 """Remove the file if it exists and skip otherwise."""
171
172 try:
173 os.remove(file_name)
174 except FileNotFoundError:
175 pass
176
177
Po Hu09de3142023-04-26 08:49:42 +0800178def include_extra_in_list(item_list):
179 """
180 1. Include all `META/*` files in the item list.
Dennis Song09f5c5d2023-04-14 11:43:04 +0800181
182 To ensure that `AddImagesToTargetFiles` can still be used with vendor item
183 list that do not specify all of the required META/ files, those files should
184 be included by default. This preserves the backward compatibility of
185 `rebuild_image_with_sepolicy`.
Po Hu09de3142023-04-26 08:49:42 +0800186
187 2. Include `SYSTEM/build.prop` file in the item list.
188
189 To ensure that `AddImagesToTargetFiles` for GRF vendor images, can still
190 access SYSTEM/build.prop to pass GetPartitionFingerprint check in BuildInfo
191 constructor.
Dennis Song09f5c5d2023-04-14 11:43:04 +0800192 """
193 if not item_list:
194 return None
Po Hu09de3142023-04-26 08:49:42 +0800195 return list(item_list) + ['META/*'] + ['SYSTEM/build.prop']
Dennis Song09f5c5d2023-04-14 11:43:04 +0800196
197
Daniel Norman03747412022-02-25 10:38:37 -0800198def create_merged_package(temp_dir):
Tao Bao2ad4b822019-06-27 16:52:12 -0700199 """Merges two target files packages into one target files structure.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800200
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900201 Returns:
202 Path to merged package under temp directory.
203 """
Daniel Normandbbf5a32020-10-22 16:03:32 -0700204 # Extract "as is" items from the input framework and vendor partial target
Dennis Song5bfa43e2023-03-30 18:28:00 +0800205 # files packages directly into the output temporary directory, since these
206 # items do not need special case processing.
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800207
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800208 output_target_files_temp_dir = os.path.join(temp_dir, 'output')
Dennis Song5bfa43e2023-03-30 18:28:00 +0800209 merge_utils.CollectTargetFiles(
210 input_zipfile_or_dir=OPTIONS.framework_target_files,
Daniel Norman03747412022-02-25 10:38:37 -0800211 output_dir=output_target_files_temp_dir,
Dennis Song5bfa43e2023-03-30 18:28:00 +0800212 item_list=OPTIONS.framework_item_list)
213 merge_utils.CollectTargetFiles(
214 input_zipfile_or_dir=OPTIONS.vendor_target_files,
Daniel Norman03747412022-02-25 10:38:37 -0800215 output_dir=output_target_files_temp_dir,
Dennis Song5bfa43e2023-03-30 18:28:00 +0800216 item_list=OPTIONS.vendor_item_list)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800217
Sundong Ahn1930fd12023-11-02 19:13:35 +0900218 if OPTIONS.boot_image_dir_path:
219 merge_utils.CollectTargetFiles(
220 input_zipfile_or_dir=OPTIONS.boot_image_dir_path,
221 output_dir=output_target_files_temp_dir,
222 item_list=['IMAGES/boot.img'])
223
Daniel Normandbbf5a32020-10-22 16:03:32 -0700224 # Perform special case processing on META/* items.
225 # After this function completes successfully, all the files we need to create
226 # the output target files package are in place.
Daniel Norman2465fc82022-03-02 12:01:20 -0800227 merge_meta.MergeMetaFiles(
Dennis Song36ce3262023-09-13 06:53:00 +0000228 temp_dir=temp_dir,
229 merged_dir=output_target_files_temp_dir,
230 framework_partitions=OPTIONS.framework_partition_set)
Daniel Norman03747412022-02-25 10:38:37 -0800231
Daniel Norman2465fc82022-03-02 12:01:20 -0800232 merge_dexopt.MergeDexopt(
Daniel Norman03747412022-02-25 10:38:37 -0800233 temp_dir=temp_dir, output_target_files_dir=output_target_files_temp_dir)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800234
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900235 return output_target_files_temp_dir
236
237
Daniel Norman03747412022-02-25 10:38:37 -0800238def generate_missing_images(target_files_dir):
239 """Generate any missing images from target files."""
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900240
241 # Regenerate IMAGES in the target directory.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800242
Daniel Normandbbf5a32020-10-22 16:03:32 -0700243 add_img_args = [
244 '--verbose',
245 '--add_missing',
246 ]
Daniel Norman03747412022-02-25 10:38:37 -0800247 if OPTIONS.rebuild_recovery:
Daniel Normana4911da2019-03-15 14:36:21 -0700248 add_img_args.append('--rebuild_recovery')
Dennis Song4aae62e2023-10-02 04:31:34 +0000249 if OPTIONS.avb_resolve_rollback_index_location_conflict:
250 add_img_args.append('--avb_resolve_rollback_index_location_conflict')
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900251 add_img_args.append(target_files_dir)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800252
253 add_img_to_target_files.main(add_img_args)
254
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900255
Daniel Norman03747412022-02-25 10:38:37 -0800256def rebuild_image_with_sepolicy(target_files_dir):
Daniel Norman571e1822021-06-25 17:18:25 -0700257 """Rebuilds odm.img or vendor.img to include merged sepolicy files.
258
259 If odm is present then odm is preferred -- otherwise vendor is used.
Daniel Norman571e1822021-06-25 17:18:25 -0700260 """
261 partition = 'vendor'
Dennis Song16d6fe02023-02-23 23:37:42 +0800262 if os.path.exists(os.path.join(target_files_dir, 'ODM')):
Daniel Norman571e1822021-06-25 17:18:25 -0700263 partition = 'odm'
264 partition_img = '{}.img'.format(partition)
Po Hua6c59122022-02-16 08:41:29 +0000265 partition_map = '{}.map'.format(partition)
Daniel Norman571e1822021-06-25 17:18:25 -0700266
267 logger.info('Recompiling %s using the merged sepolicy files.', partition_img)
268
269 # Copy the combined SEPolicy file and framework hashes to the image that is
270 # being rebuilt.
271 def copy_selinux_file(input_path, output_filename):
Po Hu0e4403e2021-07-06 17:05:56 +0800272 input_filename = os.path.join(target_files_dir, input_path)
273 if not os.path.exists(input_filename):
Dennis Song5bfa43e2023-03-30 18:28:00 +0800274 input_filename = input_filename.replace('SYSTEM_EXT/',
275 'SYSTEM/system_ext/') \
Po Hu0e4403e2021-07-06 17:05:56 +0800276 .replace('PRODUCT/', 'SYSTEM/product/')
277 if not os.path.exists(input_filename):
278 logger.info('Skipping copy_selinux_file for %s', input_filename)
279 return
Daniel Norman571e1822021-06-25 17:18:25 -0700280 shutil.copy(
Po Hu0e4403e2021-07-06 17:05:56 +0800281 input_filename,
Daniel Norman571e1822021-06-25 17:18:25 -0700282 os.path.join(target_files_dir, partition.upper(), 'etc/selinux',
283 output_filename))
284
285 copy_selinux_file('META/combined_sepolicy', 'precompiled_sepolicy')
286 copy_selinux_file('SYSTEM/etc/selinux/plat_sepolicy_and_mapping.sha256',
287 'precompiled_sepolicy.plat_sepolicy_and_mapping.sha256')
288 copy_selinux_file(
289 'SYSTEM_EXT/etc/selinux/system_ext_sepolicy_and_mapping.sha256',
290 'precompiled_sepolicy.system_ext_sepolicy_and_mapping.sha256')
291 copy_selinux_file('PRODUCT/etc/selinux/product_sepolicy_and_mapping.sha256',
292 'precompiled_sepolicy.product_sepolicy_and_mapping.sha256')
293
Daniel Norman03747412022-02-25 10:38:37 -0800294 if not OPTIONS.vendor_otatools:
Daniel Norman571e1822021-06-25 17:18:25 -0700295 # Remove the partition from the merged target-files archive. It will be
Daniel Norman03747412022-02-25 10:38:37 -0800296 # rebuilt later automatically by generate_missing_images().
Dennis Song16d6fe02023-02-23 23:37:42 +0800297 remove_file_if_exists(
298 os.path.join(target_files_dir, 'IMAGES', partition_img))
Daniel Norman03747412022-02-25 10:38:37 -0800299 return
Daniel Norman571e1822021-06-25 17:18:25 -0700300
Daniel Norman03747412022-02-25 10:38:37 -0800301 # TODO(b/192253131): Remove the need for vendor_otatools by fixing
302 # backwards-compatibility issues when compiling images across releases.
303 if not OPTIONS.vendor_target_files:
304 raise ValueError(
305 'Expected vendor_target_files if vendor_otatools is not None.')
306 logger.info(
307 '%s recompilation will be performed using the vendor otatools.zip',
308 partition_img)
Daniel Norman571e1822021-06-25 17:18:25 -0700309
Daniel Norman03747412022-02-25 10:38:37 -0800310 # Unzip the vendor build's otatools.zip and target-files archive.
311 vendor_otatools_dir = common.MakeTempDir(
312 prefix='merge_target_files_vendor_otatools_')
313 vendor_target_files_dir = common.MakeTempDir(
314 prefix='merge_target_files_vendor_target_files_')
315 common.UnzipToDir(OPTIONS.vendor_otatools, vendor_otatools_dir)
Dennis Song5bfa43e2023-03-30 18:28:00 +0800316 merge_utils.CollectTargetFiles(
317 input_zipfile_or_dir=OPTIONS.vendor_target_files,
318 output_dir=vendor_target_files_dir,
Po Hu09de3142023-04-26 08:49:42 +0800319 item_list=include_extra_in_list(OPTIONS.vendor_item_list))
Daniel Norman571e1822021-06-25 17:18:25 -0700320
Daniel Norman03747412022-02-25 10:38:37 -0800321 # Copy the partition contents from the merged target-files archive to the
322 # vendor target-files archive.
323 shutil.rmtree(os.path.join(vendor_target_files_dir, partition.upper()))
324 shutil.copytree(
325 os.path.join(target_files_dir, partition.upper()),
326 os.path.join(vendor_target_files_dir, partition.upper()),
327 symlinks=True)
Daniel Norman571e1822021-06-25 17:18:25 -0700328
Daniel Norman03747412022-02-25 10:38:37 -0800329 # Delete then rebuild the partition.
Dennis Song16d6fe02023-02-23 23:37:42 +0800330 remove_file_if_exists(
331 os.path.join(vendor_target_files_dir, 'IMAGES', partition_img))
Daniel Norman03747412022-02-25 10:38:37 -0800332 rebuild_partition_command = [
333 os.path.join(vendor_otatools_dir, 'bin', 'add_img_to_target_files'),
334 '--verbose',
335 '--add_missing',
336 ]
337 if OPTIONS.rebuild_recovery:
338 rebuild_partition_command.append('--rebuild_recovery')
339 rebuild_partition_command.append(vendor_target_files_dir)
340 logger.info('Recompiling %s: %s', partition_img,
341 ' '.join(rebuild_partition_command))
342 common.RunAndCheckOutput(rebuild_partition_command, verbose=True)
Po Hua6c59122022-02-16 08:41:29 +0000343
Daniel Norman03747412022-02-25 10:38:37 -0800344 # Move the newly-created image to the merged target files dir.
345 if not os.path.exists(os.path.join(target_files_dir, 'IMAGES')):
346 os.makedirs(os.path.join(target_files_dir, 'IMAGES'))
347 shutil.move(
348 os.path.join(vendor_target_files_dir, 'IMAGES', partition_img),
349 os.path.join(target_files_dir, 'IMAGES', partition_img))
Dennis Song5bfa43e2023-03-30 18:28:00 +0800350 move_only_exists(
351 os.path.join(vendor_target_files_dir, 'IMAGES', partition_map),
352 os.path.join(target_files_dir, 'IMAGES', partition_map))
Po Hua6c59122022-02-16 08:41:29 +0000353
Daniel Norman03747412022-02-25 10:38:37 -0800354 def copy_recovery_file(filename):
355 for subdir in ('VENDOR', 'SYSTEM/vendor'):
356 source = os.path.join(vendor_target_files_dir, subdir, filename)
357 if os.path.exists(source):
358 dest = os.path.join(target_files_dir, subdir, filename)
359 shutil.copy(source, dest)
360 return
361 logger.info('Skipping copy_recovery_file for %s, file not found', filename)
362
363 if OPTIONS.rebuild_recovery:
364 copy_recovery_file('etc/recovery.img')
365 copy_recovery_file('bin/install-recovery.sh')
366 copy_recovery_file('recovery-from-boot.p')
Daniel Norman571e1822021-06-25 17:18:25 -0700367
368
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900369def generate_super_empty_image(target_dir, output_super_empty):
Tao Bao2ad4b822019-06-27 16:52:12 -0700370 """Generates super_empty image from target package.
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900371
372 Args:
373 target_dir: Path to the target file package which contains misc_info.txt for
374 detailed information for super image.
375 output_super_empty: If provided, copies a super_empty.img file from the
376 target files package to this path.
377 """
Daniel Norman1bd2a1d2019-04-18 12:32:18 -0700378 # Create super_empty.img using the merged misc_info.txt.
379
Daniel Norman4cc9df62019-07-18 10:11:07 -0700380 misc_info_txt = os.path.join(target_dir, 'META', 'misc_info.txt')
Daniel Norman1bd2a1d2019-04-18 12:32:18 -0700381
Kiyoung Kimebe7c9c2019-06-25 17:09:55 +0900382 use_dynamic_partitions = common.LoadDictionaryFromFile(misc_info_txt).get(
383 'use_dynamic_partitions')
Daniel Norman1bd2a1d2019-04-18 12:32:18 -0700384
385 if use_dynamic_partitions != 'true' and output_super_empty:
386 raise ValueError(
387 'Building super_empty.img requires use_dynamic_partitions=true.')
388 elif use_dynamic_partitions == 'true':
Daniel Norman4cc9df62019-07-18 10:11:07 -0700389 super_empty_img = os.path.join(target_dir, 'IMAGES', 'super_empty.img')
Daniel Norman1bd2a1d2019-04-18 12:32:18 -0700390 build_super_image_args = [
391 misc_info_txt,
392 super_empty_img,
393 ]
394 build_super_image.main(build_super_image_args)
395
396 # Copy super_empty.img to the user-provided output_super_empty location.
397 if output_super_empty:
398 shutil.copyfile(super_empty_img, output_super_empty)
399
Daniel Normanb8a2f9d2019-04-24 12:55:51 -0700400
Daniel Norman03747412022-02-25 10:38:37 -0800401def create_target_files_archive(output_zip, source_dir, temp_dir):
402 """Creates a target_files zip archive from the input source dir.
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900403
404 Args:
Daniel Norman03747412022-02-25 10:38:37 -0800405 output_zip: The name of the zip archive target files package.
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900406 source_dir: The target directory contains package to be archived.
407 temp_dir: Path to temporary directory for any intermediate files.
408 """
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800409 output_target_files_list = os.path.join(temp_dir, 'output.list')
Daniel Norman4cc9df62019-07-18 10:11:07 -0700410 output_target_files_meta_dir = os.path.join(source_dir, 'META')
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800411
Daniel Normandbbf5a32020-10-22 16:03:32 -0700412 def files_from_path(target_path, extra_args=None):
413 """Gets files under the given path and return a sorted list."""
414 find_command = ['find', target_path] + (extra_args or [])
415 find_process = common.Run(
416 find_command, stdout=subprocess.PIPE, verbose=False)
417 return common.RunAndCheckOutput(['sort'],
418 stdin=find_process.stdout,
419 verbose=False)
420
Daniel Norman03747412022-02-25 10:38:37 -0800421 # META content appears first in the zip. This is done by the
422 # standard build system for optimized extraction of those files,
423 # so we do the same step for merged target_files.zips here too.
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900424 meta_content = files_from_path(output_target_files_meta_dir)
Daniel Norman4cc9df62019-07-18 10:11:07 -0700425 other_content = files_from_path(
426 source_dir,
427 ['-path', output_target_files_meta_dir, '-prune', '-o', '-print'])
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800428
Tao Bao2ad4b822019-06-27 16:52:12 -0700429 with open(output_target_files_list, 'w') as f:
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800430 f.write(meta_content)
431 f.write(other_content)
432
433 command = [
Bill Peckhamf753e152019-02-19 18:02:46 -0800434 'soong_zip',
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800435 '-d',
Daniel Normane5b134a2019-04-17 14:54:06 -0700436 '-o',
Daniel Norman03747412022-02-25 10:38:37 -0800437 os.path.abspath(output_zip),
Daniel Normane5b134a2019-04-17 14:54:06 -0700438 '-C',
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900439 source_dir,
Daniel Normaneaf5c1d2021-02-09 11:01:42 -0800440 '-r',
Daniel Normane5b134a2019-04-17 14:54:06 -0700441 output_target_files_list,
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800442 ]
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900443
Daniel Norman03747412022-02-25 10:38:37 -0800444 logger.info('creating %s', output_zip)
Daniel Normaneaf5c1d2021-02-09 11:01:42 -0800445 common.RunAndCheckOutput(command, verbose=True)
Daniel Norman03747412022-02-25 10:38:37 -0800446 logger.info('finished creating %s', output_zip)
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900447
448
Daniel Norman03747412022-02-25 10:38:37 -0800449def merge_target_files(temp_dir):
Tao Bao2ad4b822019-06-27 16:52:12 -0700450 """Merges two target files packages together.
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900451
Daniel Norman03747412022-02-25 10:38:37 -0800452 This function uses framework and vendor target files packages as input,
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900453 performs various file extractions, special case processing, and finally
454 creates a merged zip archive as output.
455
456 Args:
457 temp_dir: The name of a directory we use when we extract items from the
458 input target files packages, and also a scratch directory that we use for
459 temporary files.
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900460 """
461
462 logger.info('starting: merge framework %s and vendor %s into output %s',
Daniel Norman03747412022-02-25 10:38:37 -0800463 OPTIONS.framework_target_files, OPTIONS.vendor_target_files,
464 OPTIONS.output_target_files)
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900465
Daniel Norman03747412022-02-25 10:38:37 -0800466 output_target_files_temp_dir = create_merged_package(temp_dir)
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900467
Daniel Norman21c34f72020-11-11 17:25:50 -0800468 partition_map = common.PartitionMapFromTargetFiles(
469 output_target_files_temp_dir)
470
Daniel Norman2465fc82022-03-02 12:01:20 -0800471 compatibility_errors = merge_compatibility_checks.CheckCompatibility(
472 target_files_dir=output_target_files_temp_dir,
473 partition_map=partition_map)
474 if compatibility_errors:
475 for error in compatibility_errors:
476 logger.error(error)
477 raise ExternalError(
478 'Found incompatibilities in the merged target files package.')
Daniel Normand3351562020-10-29 12:33:11 -0700479
Daniel Norman571e1822021-06-25 17:18:25 -0700480 # Include the compiled policy in an image if requested.
Daniel Norman03747412022-02-25 10:38:37 -0800481 if OPTIONS.rebuild_sepolicy:
482 rebuild_image_with_sepolicy(output_target_files_temp_dir)
Daniel Norman48603ff2021-02-22 15:15:24 -0800483
Daniel Norman03747412022-02-25 10:38:37 -0800484 generate_missing_images(output_target_files_temp_dir)
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900485
Daniel Norman03747412022-02-25 10:38:37 -0800486 generate_super_empty_image(output_target_files_temp_dir,
487 OPTIONS.output_super_empty)
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900488
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900489 # Finally, create the output target files zip archive and/or copy the
490 # output items to the output target files directory.
491
Daniel Norman03747412022-02-25 10:38:37 -0800492 if OPTIONS.output_dir:
Daniel Norman2465fc82022-03-02 12:01:20 -0800493 merge_utils.CopyItems(output_target_files_temp_dir, OPTIONS.output_dir,
494 OPTIONS.output_item_list)
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900495
Daniel Norman03747412022-02-25 10:38:37 -0800496 if not OPTIONS.output_target_files:
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900497 return
498
Daniel Norman03747412022-02-25 10:38:37 -0800499 create_target_files_archive(OPTIONS.output_target_files,
500 output_target_files_temp_dir, temp_dir)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800501
Daniel Norman74eb74b2019-09-18 14:01:48 -0700502 # Create the IMG package from the merged target files package.
Daniel Norman03747412022-02-25 10:38:37 -0800503 if OPTIONS.output_img:
504 img_from_target_files.main(
505 [OPTIONS.output_target_files, OPTIONS.output_img])
Daniel Norman74eb74b2019-09-18 14:01:48 -0700506
Daniel Norman3b64ce12019-04-16 16:11:35 -0700507 # Create the OTA package from the merged target files package.
508
Daniel Norman03747412022-02-25 10:38:37 -0800509 if OPTIONS.output_ota:
510 ota_from_target_files.main(
511 [OPTIONS.output_target_files, OPTIONS.output_ota])
Daniel Norman3b64ce12019-04-16 16:11:35 -0700512
Daniel Norman1bd2a1d2019-04-18 12:32:18 -0700513
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800514def main():
515 """The main function.
516
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800517 Process command line arguments, then call merge_target_files to
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800518 perform the heavy lifting.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800519 """
520
521 common.InitLogging()
522
Bill Peckhamf753e152019-02-19 18:02:46 -0800523 def option_handler(o, a):
524 if o == '--system-target-files':
Daniel Normand5d70ea2019-06-05 15:13:43 -0700525 logger.warning(
526 '--system-target-files has been renamed to --framework-target-files')
527 OPTIONS.framework_target_files = a
528 elif o == '--framework-target-files':
529 OPTIONS.framework_target_files = a
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800530 elif o == '--system-item-list':
Daniel Normand5d70ea2019-06-05 15:13:43 -0700531 logger.warning(
532 '--system-item-list has been renamed to --framework-item-list')
533 OPTIONS.framework_item_list = a
534 elif o == '--framework-item-list':
535 OPTIONS.framework_item_list = a
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800536 elif o == '--system-misc-info-keys':
Daniel Norman4cc9df62019-07-18 10:11:07 -0700537 logger.warning('--system-misc-info-keys has been renamed to '
538 '--framework-misc-info-keys')
Daniel Normand5d70ea2019-06-05 15:13:43 -0700539 OPTIONS.framework_misc_info_keys = a
540 elif o == '--framework-misc-info-keys':
541 OPTIONS.framework_misc_info_keys = a
Bill Peckhamf753e152019-02-19 18:02:46 -0800542 elif o == '--other-target-files':
Daniel Normand5d70ea2019-06-05 15:13:43 -0700543 logger.warning(
544 '--other-target-files has been renamed to --vendor-target-files')
545 OPTIONS.vendor_target_files = a
546 elif o == '--vendor-target-files':
547 OPTIONS.vendor_target_files = a
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800548 elif o == '--other-item-list':
Daniel Norman2d7989a2021-04-05 17:40:47 +0000549 logger.warning('--other-item-list has been renamed to --vendor-item-list')
Daniel Normand5d70ea2019-06-05 15:13:43 -0700550 OPTIONS.vendor_item_list = a
551 elif o == '--vendor-item-list':
552 OPTIONS.vendor_item_list = a
Sundong Ahn1930fd12023-11-02 19:13:35 +0900553 elif o == '--boot-image-dir-path':
554 OPTIONS.boot_image_dir_path = a
Bill Peckhamf753e152019-02-19 18:02:46 -0800555 elif o == '--output-target-files':
556 OPTIONS.output_target_files = a
Daniel Normanfdb38812019-04-15 09:47:24 -0700557 elif o == '--output-dir':
558 OPTIONS.output_dir = a
559 elif o == '--output-item-list':
560 OPTIONS.output_item_list = a
Daniel Norman3b64ce12019-04-16 16:11:35 -0700561 elif o == '--output-ota':
562 OPTIONS.output_ota = a
Daniel Norman1bd2a1d2019-04-18 12:32:18 -0700563 elif o == '--output-img':
564 OPTIONS.output_img = a
Daniel Normanf0318252019-04-15 11:34:56 -0700565 elif o == '--output-super-empty':
566 OPTIONS.output_super_empty = a
Daniel Norman2465fc82022-03-02 12:01:20 -0800567 elif o == '--rebuild_recovery' or o == '--rebuild-recovery':
Daniel Normana4911da2019-03-15 14:36:21 -0700568 OPTIONS.rebuild_recovery = True
Daniel Normanb0c75912020-09-24 14:30:21 -0700569 elif o == '--allow-duplicate-apkapex-keys':
570 OPTIONS.allow_duplicate_apkapex_keys = True
Daniel Norman571e1822021-06-25 17:18:25 -0700571 elif o == '--vendor-otatools':
572 OPTIONS.vendor_otatools = a
573 elif o == '--rebuild-sepolicy':
574 OPTIONS.rebuild_sepolicy = True
Bill Peckham364c1cc2019-03-29 18:27:23 -0700575 elif o == '--keep-tmp':
Bill Peckhamf753e152019-02-19 18:02:46 -0800576 OPTIONS.keep_tmp = True
Dennis Song4aae62e2023-10-02 04:31:34 +0000577 elif o == '--avb-resolve-rollback-index-location-conflict':
578 OPTIONS.avb_resolve_rollback_index_location_conflict = True
Jose Galmes9c8f6eb2021-07-21 09:34:08 -0700579 elif o == '--framework-dexpreopt-config':
580 OPTIONS.framework_dexpreopt_config = a
581 elif o == '--framework-dexpreopt-tools':
582 OPTIONS.framework_dexpreopt_tools = a
583 elif o == '--vendor-dexpreopt-config':
584 OPTIONS.vendor_dexpreopt_config = a
Bill Peckhamf753e152019-02-19 18:02:46 -0800585 else:
586 return False
587 return True
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800588
Bill Peckhamf753e152019-02-19 18:02:46 -0800589 args = common.ParseOptions(
Daniel Normane5b134a2019-04-17 14:54:06 -0700590 sys.argv[1:],
591 __doc__,
Bill Peckhamf753e152019-02-19 18:02:46 -0800592 extra_long_opts=[
593 'system-target-files=',
Daniel Normand5d70ea2019-06-05 15:13:43 -0700594 'framework-target-files=',
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800595 'system-item-list=',
Daniel Normand5d70ea2019-06-05 15:13:43 -0700596 'framework-item-list=',
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800597 'system-misc-info-keys=',
Daniel Normand5d70ea2019-06-05 15:13:43 -0700598 'framework-misc-info-keys=',
Bill Peckhamf753e152019-02-19 18:02:46 -0800599 'other-target-files=',
Daniel Normand5d70ea2019-06-05 15:13:43 -0700600 'vendor-target-files=',
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800601 'other-item-list=',
Daniel Normand5d70ea2019-06-05 15:13:43 -0700602 'vendor-item-list=',
Sundong Ahn1930fd12023-11-02 19:13:35 +0900603 'boot-image-dir-path=',
Bill Peckhamf753e152019-02-19 18:02:46 -0800604 'output-target-files=',
Daniel Normanfdb38812019-04-15 09:47:24 -0700605 'output-dir=',
606 'output-item-list=',
Daniel Norman3b64ce12019-04-16 16:11:35 -0700607 'output-ota=',
Daniel Norman1bd2a1d2019-04-18 12:32:18 -0700608 'output-img=',
Daniel Normanf0318252019-04-15 11:34:56 -0700609 'output-super-empty=',
Jose Galmes9c8f6eb2021-07-21 09:34:08 -0700610 'framework-dexpreopt-config=',
611 'framework-dexpreopt-tools=',
612 'vendor-dexpreopt-config=',
Daniel Normana4911da2019-03-15 14:36:21 -0700613 'rebuild_recovery',
Daniel Norman2465fc82022-03-02 12:01:20 -0800614 'rebuild-recovery',
Daniel Normanb0c75912020-09-24 14:30:21 -0700615 'allow-duplicate-apkapex-keys',
Daniel Norman571e1822021-06-25 17:18:25 -0700616 'vendor-otatools=',
617 'rebuild-sepolicy',
Bill Peckham364c1cc2019-03-29 18:27:23 -0700618 'keep-tmp',
Dennis Song4aae62e2023-10-02 04:31:34 +0000619 'avb-resolve-rollback-index-location-conflict',
Bill Peckhamf753e152019-02-19 18:02:46 -0800620 ],
621 extra_option_handler=option_handler)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800622
Tao Bao2ad4b822019-06-27 16:52:12 -0700623 # pylint: disable=too-many-boolean-expressions
Daniel Normand5d70ea2019-06-05 15:13:43 -0700624 if (args or OPTIONS.framework_target_files is None or
625 OPTIONS.vendor_target_files is None or
Daniel Normane5b134a2019-04-17 14:54:06 -0700626 (OPTIONS.output_target_files is None and OPTIONS.output_dir is None) or
Daniel Norman5f476772022-03-02 15:46:34 -0800627 (OPTIONS.output_dir is not None and not OPTIONS.output_item_list) or
Po Hua6c59122022-02-16 08:41:29 +0000628 (OPTIONS.rebuild_recovery and not OPTIONS.rebuild_sepolicy)):
Bill Peckhamf753e152019-02-19 18:02:46 -0800629 common.Usage(__doc__)
Bill Peckham889b0c62019-02-21 18:53:37 -0800630 sys.exit(1)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800631
Dennis Song5bfa43e2023-03-30 18:28:00 +0800632 framework_namelist = merge_utils.GetTargetFilesItems(
633 OPTIONS.framework_target_files)
634 vendor_namelist = merge_utils.GetTargetFilesItems(
635 OPTIONS.vendor_target_files)
Daniel Norman5f476772022-03-02 15:46:34 -0800636
Daniel Normand5d70ea2019-06-05 15:13:43 -0700637 if OPTIONS.framework_item_list:
Daniel Norman03747412022-02-25 10:38:37 -0800638 OPTIONS.framework_item_list = common.LoadListFromFile(
639 OPTIONS.framework_item_list)
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800640 else:
Daniel Norman5f476772022-03-02 15:46:34 -0800641 OPTIONS.framework_item_list = merge_utils.InferItemList(
642 input_namelist=framework_namelist, framework=True)
Daniel Norman2465fc82022-03-02 12:01:20 -0800643 OPTIONS.framework_partition_set = merge_utils.ItemListToPartitionSet(
Daniel Norman03747412022-02-25 10:38:37 -0800644 OPTIONS.framework_item_list)
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800645
Daniel Normand5d70ea2019-06-05 15:13:43 -0700646 if OPTIONS.framework_misc_info_keys:
Daniel Norman03747412022-02-25 10:38:37 -0800647 OPTIONS.framework_misc_info_keys = common.LoadListFromFile(
Daniel Normand5d70ea2019-06-05 15:13:43 -0700648 OPTIONS.framework_misc_info_keys)
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800649 else:
Daniel Norman5f476772022-03-02 15:46:34 -0800650 OPTIONS.framework_misc_info_keys = merge_utils.InferFrameworkMiscInfoKeys(
651 input_namelist=framework_namelist)
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800652
Daniel Normand5d70ea2019-06-05 15:13:43 -0700653 if OPTIONS.vendor_item_list:
Daniel Norman03747412022-02-25 10:38:37 -0800654 OPTIONS.vendor_item_list = common.LoadListFromFile(OPTIONS.vendor_item_list)
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800655 else:
Daniel Norman5f476772022-03-02 15:46:34 -0800656 OPTIONS.vendor_item_list = merge_utils.InferItemList(
657 input_namelist=vendor_namelist, framework=False)
Daniel Norman2465fc82022-03-02 12:01:20 -0800658 OPTIONS.vendor_partition_set = merge_utils.ItemListToPartitionSet(
Daniel Norman03747412022-02-25 10:38:37 -0800659 OPTIONS.vendor_item_list)
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800660
Daniel Normanfdb38812019-04-15 09:47:24 -0700661 if OPTIONS.output_item_list:
Daniel Norman03747412022-02-25 10:38:37 -0800662 OPTIONS.output_item_list = common.LoadListFromFile(OPTIONS.output_item_list)
Daniel Normanfdb38812019-04-15 09:47:24 -0700663
Daniel Norman2465fc82022-03-02 12:01:20 -0800664 if not merge_utils.ValidateConfigLists():
Daniel Normane5964522019-03-19 10:32:03 -0700665 sys.exit(1)
666
Daniel Norman2465fc82022-03-02 12:01:20 -0800667 temp_dir = common.MakeTempDir(prefix='merge_target_files_')
668 try:
669 merge_target_files(temp_dir)
670 finally:
671 if OPTIONS.keep_tmp:
672 logger.info('Keeping temp_dir %s', temp_dir)
673 else:
674 common.Cleanup()
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800675
676
677if __name__ == '__main__':
Bill Peckham889b0c62019-02-21 18:53:37 -0800678 main()