blob: 81b95c807b1fb3eee48807c7a5d8a07ff379b57d [file] [log] [blame]
Bill Peckhame9eb5f92019-02-01 15:52:10 -08001#!/usr/bin/env python
2#
3# Copyright (C) 2019 The Android Open Source Project
4#
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 Normand5d70ea2019-06-05 15:13:43 -070019One package contains framework files, and the other contains vendor files.
Daniel Normane5b134a2019-04-17 14:54:06 -070020It produces a complete target files package that can be used to generate an
21OTA package.
Bill Peckhame9eb5f92019-02-01 15:52:10 -080022
23Usage: merge_target_files.py [args]
24
Daniel Normand5d70ea2019-06-05 15:13:43 -070025 --framework-target-files framework-target-files-zip-archive
26 The input target files package containing framework bits. This is a zip
Bill Peckhame9eb5f92019-02-01 15:52:10 -080027 archive.
28
Daniel Normand5d70ea2019-06-05 15:13:43 -070029 --framework-item-list framework-item-list-file
Daniel Norman2c99c5b2019-03-07 13:01:48 -080030 The optional path to a newline-separated config file that replaces the
Daniel Normand5d70ea2019-06-05 15:13:43 -070031 contents of DEFAULT_FRAMEWORK_ITEM_LIST if provided.
Daniel Norman2c99c5b2019-03-07 13:01:48 -080032
Daniel Normand5d70ea2019-06-05 15:13:43 -070033 --framework-misc-info-keys framework-misc-info-keys-file
Daniel Norman2c99c5b2019-03-07 13:01:48 -080034 The optional path to a newline-separated config file that replaces the
Daniel Normand5d70ea2019-06-05 15:13:43 -070035 contents of DEFAULT_FRAMEWORK_MISC_INFO_KEYS if provided.
Daniel Norman2c99c5b2019-03-07 13:01:48 -080036
Daniel Normand5d70ea2019-06-05 15:13:43 -070037 --vendor-target-files vendor-target-files-zip-archive
38 The input target files package containing vendor bits. This is a zip
Bill Peckhame9eb5f92019-02-01 15:52:10 -080039 archive.
40
Daniel Normand5d70ea2019-06-05 15:13:43 -070041 --vendor-item-list vendor-item-list-file
Daniel Norman2c99c5b2019-03-07 13:01:48 -080042 The optional path to a newline-separated config file that replaces the
Daniel Normand5d70ea2019-06-05 15:13:43 -070043 contents of DEFAULT_VENDOR_ITEM_LIST if provided.
Daniel Norman2c99c5b2019-03-07 13:01:48 -080044
Bill Peckhame9eb5f92019-02-01 15:52:10 -080045 --output-target-files output-target-files-package
Daniel Normanfdb38812019-04-15 09:47:24 -070046 If provided, the output merged target files package. Also a zip archive.
47
48 --output-dir output-directory
49 If provided, the destination directory for saving merged files. Requires
50 the --output-item-list flag.
51 Can be provided alongside --output-target-files, or by itself.
52
53 --output-item-list output-item-list-file.
54 The optional path to a newline-separated config file that specifies the
55 file patterns to copy into the --output-dir. Required if providing
56 the --output-dir flag.
Daniel Normana4911da2019-03-15 14:36:21 -070057
Daniel Norman3b64ce12019-04-16 16:11:35 -070058 --output-ota output-ota-package
59 The output ota package. This is a zip archive. Use of this flag may
60 require passing the --path common flag; see common.py.
61
Daniel Norman1bd2a1d2019-04-18 12:32:18 -070062 --output-img output-img-package
63 The output img package, suitable for use with 'fastboot update'. Use of
64 this flag may require passing the --path common flag; see common.py.
65
Daniel Normanf0318252019-04-15 11:34:56 -070066 --output-super-empty output-super-empty-image
67 If provided, creates a super_empty.img file from the merged target
68 files package and saves it at this path.
69
Daniel Normana4911da2019-03-15 14:36:21 -070070 --rebuild_recovery
71 Rebuild the recovery patch used by non-A/B devices and write it to the
72 system image.
Bill Peckham364c1cc2019-03-29 18:27:23 -070073
74 --keep-tmp
75 Keep tempoary files for debugging purposes.
Bill Peckhame9eb5f92019-02-01 15:52:10 -080076"""
77
78from __future__ import print_function
79
Bill Peckhame9eb5f92019-02-01 15:52:10 -080080import fnmatch
81import logging
82import os
Daniel Normanfdb38812019-04-15 09:47:24 -070083import shutil
Bill Peckham540d91a2019-04-25 14:18:16 -070084import subprocess
Bill Peckhame9eb5f92019-02-01 15:52:10 -080085import sys
86import zipfile
87
Bill Peckhame9eb5f92019-02-01 15:52:10 -080088import add_img_to_target_files
Daniel Normanf0318252019-04-15 11:34:56 -070089import build_super_image
90import common
Daniel Norman1bd2a1d2019-04-18 12:32:18 -070091import img_from_target_files
Daniel Norman3b64ce12019-04-16 16:11:35 -070092import ota_from_target_files
Bill Peckhame9eb5f92019-02-01 15:52:10 -080093
94logger = logging.getLogger(__name__)
Tao Bao2ad4b822019-06-27 16:52:12 -070095
Bill Peckhame9eb5f92019-02-01 15:52:10 -080096OPTIONS = common.OPTIONS
97OPTIONS.verbose = True
Daniel Normand5d70ea2019-06-05 15:13:43 -070098OPTIONS.framework_target_files = None
99OPTIONS.framework_item_list = None
100OPTIONS.framework_misc_info_keys = None
101OPTIONS.vendor_target_files = None
102OPTIONS.vendor_item_list = None
Bill Peckhamf753e152019-02-19 18:02:46 -0800103OPTIONS.output_target_files = None
Daniel Normanfdb38812019-04-15 09:47:24 -0700104OPTIONS.output_dir = None
105OPTIONS.output_item_list = None
Daniel Norman3b64ce12019-04-16 16:11:35 -0700106OPTIONS.output_ota = None
Daniel Norman1bd2a1d2019-04-18 12:32:18 -0700107OPTIONS.output_img = None
Daniel Normanf0318252019-04-15 11:34:56 -0700108OPTIONS.output_super_empty = None
Daniel Normana4911da2019-03-15 14:36:21 -0700109OPTIONS.rebuild_recovery = False
Bill Peckhamf753e152019-02-19 18:02:46 -0800110OPTIONS.keep_tmp = False
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800111
Daniel Normand5d70ea2019-06-05 15:13:43 -0700112# DEFAULT_FRAMEWORK_ITEM_LIST is a list of items to extract from the partial
113# framework target files package as is, meaning these items will land in the
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800114# output target files package exactly as they appear in the input partial
Daniel Normand5d70ea2019-06-05 15:13:43 -0700115# framework target files package.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800116
Daniel Normand5d70ea2019-06-05 15:13:43 -0700117DEFAULT_FRAMEWORK_ITEM_LIST = (
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800118 'META/apkcerts.txt',
119 'META/filesystem_config.txt',
120 'META/root_filesystem_config.txt',
121 'META/system_manifest.xml',
122 'META/system_matrix.xml',
123 'META/update_engine_config.txt',
124 'PRODUCT/*',
125 'ROOT/*',
126 'SYSTEM/*',
Daniel Normanedf12472019-05-22 10:47:08 -0700127)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800128
Daniel Normand5d70ea2019-06-05 15:13:43 -0700129# FRAMEWORK_EXTRACT_SPECIAL_ITEM_LIST is a list of items to extract from the
130# partial framework target files package that need some special processing, such
131# as some sort of combination with items from the partial vendor target files
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800132# package.
133
Daniel Normand5d70ea2019-06-05 15:13:43 -0700134FRAMEWORK_EXTRACT_SPECIAL_ITEM_LIST = ('META/*',)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800135
Daniel Normand5d70ea2019-06-05 15:13:43 -0700136# DEFAULT_FRAMEWORK_MISC_INFO_KEYS is a list of keys to obtain from the
137# framework instance of META/misc_info.txt. The remaining keys from the
138# vendor instance.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800139
Daniel Normand5d70ea2019-06-05 15:13:43 -0700140DEFAULT_FRAMEWORK_MISC_INFO_KEYS = (
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800141 'avb_system_hashtree_enable',
142 'avb_system_add_hashtree_footer_args',
143 'avb_system_key_path',
144 'avb_system_algorithm',
145 'avb_system_rollback_index_location',
146 'avb_product_hashtree_enable',
147 'avb_product_add_hashtree_footer_args',
Justin Yun6151e3f2019-06-25 15:58:13 +0900148 'avb_system_ext_hashtree_enable',
149 'avb_system_ext_add_hashtree_footer_args',
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800150 'system_root_image',
151 'root_dir',
152 'ab_update',
153 'default_system_dev_certificate',
154 'system_size',
Daniel Normanedf12472019-05-22 10:47:08 -0700155)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800156
Daniel Normand5d70ea2019-06-05 15:13:43 -0700157# DEFAULT_VENDOR_ITEM_LIST is a list of items to extract from the partial
158# vendor target files package as is, meaning these items will land in the output
159# target files package exactly as they appear in the input partial vendor target
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800160# files package.
161
Daniel Normand5d70ea2019-06-05 15:13:43 -0700162DEFAULT_VENDOR_ITEM_LIST = (
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800163 'META/boot_filesystem_config.txt',
164 'META/otakeys.txt',
165 'META/releasetools.py',
166 'META/vendor_filesystem_config.txt',
167 'META/vendor_manifest.xml',
168 'META/vendor_matrix.xml',
169 'BOOT/*',
170 'DATA/*',
171 'ODM/*',
172 'OTA/android-info.txt',
173 'PREBUILT_IMAGES/*',
174 'RADIO/*',
175 'VENDOR/*',
Daniel Normanedf12472019-05-22 10:47:08 -0700176)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800177
Daniel Normand5d70ea2019-06-05 15:13:43 -0700178# VENDOR_EXTRACT_SPECIAL_ITEM_LIST is a list of items to extract from the
179# partial vendor target files package that need some special processing, such as
180# some sort of combination with items from the partial framework target files
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800181# package.
182
Daniel Normand5d70ea2019-06-05 15:13:43 -0700183VENDOR_EXTRACT_SPECIAL_ITEM_LIST = ('META/*',)
Daniel Normanedf12472019-05-22 10:47:08 -0700184
185# The merge config lists should not attempt to extract items from both
186# builds for any of the following partitions. The partitions in
187# SINGLE_BUILD_PARTITIONS should come entirely from a single build (either
Daniel Normand5d70ea2019-06-05 15:13:43 -0700188# framework or vendor, but not both).
Daniel Normanedf12472019-05-22 10:47:08 -0700189
190SINGLE_BUILD_PARTITIONS = (
191 'BOOT/',
192 'DATA/',
193 'ODM/',
194 'PRODUCT/',
Justin Yun6151e3f2019-06-25 15:58:13 +0900195 'SYSTEM_EXT/',
Daniel Normanedf12472019-05-22 10:47:08 -0700196 'RADIO/',
197 'RECOVERY/',
198 'ROOT/',
199 'SYSTEM/',
200 'SYSTEM_OTHER/',
201 'VENDOR/',
202)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800203
204
Chris Grossfabf50a2019-05-02 12:42:09 -0700205def write_sorted_data(data, path):
Tao Bao2ad4b822019-06-27 16:52:12 -0700206 """Writes the sorted contents of either a list or dict to file.
Chris Grossfabf50a2019-05-02 12:42:09 -0700207
Tao Bao2ad4b822019-06-27 16:52:12 -0700208 This function sorts the contents of the list or dict and then writes the
209 resulting sorted contents to a file specified by path.
Chris Grossfabf50a2019-05-02 12:42:09 -0700210
211 Args:
212 data: The list or dict to sort and write.
213 path: Path to the file to write the sorted values to. The file at path will
214 be overridden if it exists.
215 """
216 with open(path, 'w') as output:
Daniel Normand5d70ea2019-06-05 15:13:43 -0700217 for entry in sorted(data):
Chris Grossfabf50a2019-05-02 12:42:09 -0700218 out_str = '{}={}\n'.format(entry, data[entry]) if isinstance(
219 data, dict) else '{}\n'.format(entry)
220 output.write(out_str)
221
222
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800223def extract_items(target_files, target_files_temp_dir, extract_item_list):
Tao Bao2ad4b822019-06-27 16:52:12 -0700224 """Extracts items from target files to temporary directory.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800225
226 This function extracts from the specified target files zip archive into the
227 specified temporary directory, the items specified in the extract item list.
228
229 Args:
230 target_files: The target files zip archive from which to extract items.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800231 target_files_temp_dir: The temporary directory where the extracted items
Daniel Normane5b134a2019-04-17 14:54:06 -0700232 will land.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800233 extract_item_list: A list of items to extract.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800234 """
235
236 logger.info('extracting from %s', target_files)
237
238 # Filter the extract_item_list to remove any items that do not exist in the
239 # zip file. Otherwise, the extraction step will fail.
240
Daniel Norman4cc9df62019-07-18 10:11:07 -0700241 with zipfile.ZipFile(target_files, allowZip64=True) as target_files_zipfile:
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800242 target_files_namelist = target_files_zipfile.namelist()
243
244 filtered_extract_item_list = []
245 for pattern in extract_item_list:
246 matching_namelist = fnmatch.filter(target_files_namelist, pattern)
247 if not matching_namelist:
248 logger.warning('no match for %s', pattern)
249 else:
250 filtered_extract_item_list.append(pattern)
251
Bill Peckham8ff3fbd2019-02-22 10:57:43 -0800252 # Extract from target_files into target_files_temp_dir the
253 # filtered_extract_item_list.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800254
Daniel Normane5b134a2019-04-17 14:54:06 -0700255 common.UnzipToDir(target_files, target_files_temp_dir,
256 filtered_extract_item_list)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800257
258
Daniel Normanfdb38812019-04-15 09:47:24 -0700259def copy_items(from_dir, to_dir, patterns):
260 """Similar to extract_items() except uses an input dir instead of zip."""
261 file_paths = []
262 for dirpath, _, filenames in os.walk(from_dir):
Daniel Normane5b134a2019-04-17 14:54:06 -0700263 file_paths.extend(
264 os.path.relpath(path=os.path.join(dirpath, filename), start=from_dir)
265 for filename in filenames)
Daniel Normanfdb38812019-04-15 09:47:24 -0700266
267 filtered_file_paths = set()
268 for pattern in patterns:
269 filtered_file_paths.update(fnmatch.filter(file_paths, pattern))
270
271 for file_path in filtered_file_paths:
272 original_file_path = os.path.join(from_dir, file_path)
273 copied_file_path = os.path.join(to_dir, file_path)
274 copied_file_dir = os.path.dirname(copied_file_path)
275 if not os.path.exists(copied_file_dir):
276 os.makedirs(copied_file_dir)
277 if os.path.islink(original_file_path):
278 os.symlink(os.readlink(original_file_path), copied_file_path)
279 else:
280 shutil.copyfile(original_file_path, copied_file_path)
281
282
Daniel Normand5d70ea2019-06-05 15:13:43 -0700283def validate_config_lists(framework_item_list, framework_misc_info_keys,
284 vendor_item_list):
Daniel Normane5964522019-03-19 10:32:03 -0700285 """Performs validations on the merge config lists.
286
287 Args:
Daniel Normand5d70ea2019-06-05 15:13:43 -0700288 framework_item_list: The list of items to extract from the partial framework
Daniel Normane5b134a2019-04-17 14:54:06 -0700289 target files package as is.
Daniel Normand5d70ea2019-06-05 15:13:43 -0700290 framework_misc_info_keys: A list of keys to obtain from the framework
291 instance of META/misc_info.txt. The remaining keys from the vendor
292 instance.
293 vendor_item_list: The list of items to extract from the partial vendor
294 target files package as is.
Daniel Normane5964522019-03-19 10:32:03 -0700295
296 Returns:
297 False if a validation fails, otherwise true.
298 """
Daniel Normanedf12472019-05-22 10:47:08 -0700299 has_error = False
300
Daniel Normand5d70ea2019-06-05 15:13:43 -0700301 default_combined_item_set = set(DEFAULT_FRAMEWORK_ITEM_LIST)
302 default_combined_item_set.update(DEFAULT_VENDOR_ITEM_LIST)
Daniel Normane5964522019-03-19 10:32:03 -0700303
Daniel Normand5d70ea2019-06-05 15:13:43 -0700304 combined_item_set = set(framework_item_list)
305 combined_item_set.update(vendor_item_list)
Daniel Normane5964522019-03-19 10:32:03 -0700306
307 # Check that the merge config lists are not missing any item specified
308 # by the default config lists.
309 difference = default_combined_item_set.difference(combined_item_set)
310 if difference:
Daniel Normane5b134a2019-04-17 14:54:06 -0700311 logger.error('Missing merge config items: %s', list(difference))
Daniel Normane5964522019-03-19 10:32:03 -0700312 logger.error('Please ensure missing items are in either the '
Daniel Normand5d70ea2019-06-05 15:13:43 -0700313 'framework-item-list or vendor-item-list files provided to '
Daniel Normane5964522019-03-19 10:32:03 -0700314 'this script.')
Daniel Normanedf12472019-05-22 10:47:08 -0700315 has_error = True
316
317 for partition in SINGLE_BUILD_PARTITIONS:
Daniel Normand5d70ea2019-06-05 15:13:43 -0700318 in_framework = any(
319 item.startswith(partition) for item in framework_item_list)
320 in_vendor = any(item.startswith(partition) for item in vendor_item_list)
321 if in_framework and in_vendor:
Daniel Normanedf12472019-05-22 10:47:08 -0700322 logger.error(
Tao Bao2ad4b822019-06-27 16:52:12 -0700323 'Cannot extract items from %s for both the framework and vendor'
Kiyoung Kimebe7c9c2019-06-25 17:09:55 +0900324 ' builds. Please ensure only one merge config item list'
Tao Bao2ad4b822019-06-27 16:52:12 -0700325 ' includes %s.', partition, partition)
Daniel Normanedf12472019-05-22 10:47:08 -0700326 has_error = True
Daniel Normane5964522019-03-19 10:32:03 -0700327
Daniel Normand5d70ea2019-06-05 15:13:43 -0700328 if ('dynamic_partition_list' in framework_misc_info_keys) or (
329 'super_partition_groups' in framework_misc_info_keys):
Daniel Norman19b9fe92019-03-19 14:48:02 -0700330 logger.error('Dynamic partition misc info keys should come from '
Daniel Normand5d70ea2019-06-05 15:13:43 -0700331 'the vendor instance of META/misc_info.txt.')
Daniel Normanedf12472019-05-22 10:47:08 -0700332 has_error = True
Daniel Norman19b9fe92019-03-19 14:48:02 -0700333
Daniel Normanedf12472019-05-22 10:47:08 -0700334 return not has_error
Daniel Normane5964522019-03-19 10:32:03 -0700335
336
Daniel Normand5d70ea2019-06-05 15:13:43 -0700337def process_ab_partitions_txt(framework_target_files_temp_dir,
338 vendor_target_files_temp_dir,
Daniel Normane5b134a2019-04-17 14:54:06 -0700339 output_target_files_temp_dir):
Tao Bao2ad4b822019-06-27 16:52:12 -0700340 """Performs special processing for META/ab_partitions.txt.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800341
Tao Bao2ad4b822019-06-27 16:52:12 -0700342 This function merges the contents of the META/ab_partitions.txt files from the
343 framework directory and the vendor directory, placing the merged result in the
344 output directory. The precondition in that the files are already extracted.
345 The post condition is that the output META/ab_partitions.txt contains the
346 merged content. The format for each ab_partitions.txt a one partition name per
347 line. The output file contains the union of the parition names.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800348
349 Args:
Daniel Normand5d70ea2019-06-05 15:13:43 -0700350 framework_target_files_temp_dir: The name of a directory containing the
351 special items extracted from the framework target files package.
352 vendor_target_files_temp_dir: The name of a directory containing the special
353 items extracted from the vendor target files package.
Daniel Normane5b134a2019-04-17 14:54:06 -0700354 output_target_files_temp_dir: The name of a directory that will be used to
355 create the output target files package after all the special cases are
356 processed.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800357 """
358
Daniel Normand5d70ea2019-06-05 15:13:43 -0700359 framework_ab_partitions_txt = os.path.join(framework_target_files_temp_dir,
360 'META', 'ab_partitions.txt')
361
362 vendor_ab_partitions_txt = os.path.join(vendor_target_files_temp_dir, 'META',
Daniel Normane5b134a2019-04-17 14:54:06 -0700363 'ab_partitions.txt')
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800364
Daniel Normand5d70ea2019-06-05 15:13:43 -0700365 with open(framework_ab_partitions_txt) as f:
366 framework_ab_partitions = f.read().splitlines()
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800367
Daniel Normand5d70ea2019-06-05 15:13:43 -0700368 with open(vendor_ab_partitions_txt) as f:
369 vendor_ab_partitions = f.read().splitlines()
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800370
Daniel Normand5d70ea2019-06-05 15:13:43 -0700371 output_ab_partitions = set(framework_ab_partitions + vendor_ab_partitions)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800372
Daniel Normane5b134a2019-04-17 14:54:06 -0700373 output_ab_partitions_txt = os.path.join(output_target_files_temp_dir, 'META',
374 'ab_partitions.txt')
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800375
Chris Grossfabf50a2019-05-02 12:42:09 -0700376 write_sorted_data(data=output_ab_partitions, path=output_ab_partitions_txt)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800377
378
Bill Peckham364c1cc2019-03-29 18:27:23 -0700379def append_recovery_to_filesystem_config(output_target_files_temp_dir):
Tao Bao2ad4b822019-06-27 16:52:12 -0700380 """Performs special processing for META/filesystem_config.txt.
Bill Peckham364c1cc2019-03-29 18:27:23 -0700381
Tao Bao2ad4b822019-06-27 16:52:12 -0700382 This function appends recovery information to META/filesystem_config.txt so
383 that recovery patch regeneration will succeed.
Bill Peckham364c1cc2019-03-29 18:27:23 -0700384
385 Args:
Daniel Normane5b134a2019-04-17 14:54:06 -0700386 output_target_files_temp_dir: The name of a directory that will be used to
387 create the output target files package after all the special cases are
388 processed. We find filesystem_config.txt here.
Bill Peckham364c1cc2019-03-29 18:27:23 -0700389 """
390
Daniel Normane5b134a2019-04-17 14:54:06 -0700391 filesystem_config_txt = os.path.join(output_target_files_temp_dir, 'META',
392 'filesystem_config.txt')
Bill Peckham364c1cc2019-03-29 18:27:23 -0700393
394 with open(filesystem_config_txt, 'a') as f:
395 # TODO(bpeckham) this data is hard coded. It should be generated
396 # programmatically.
Daniel Normane5b134a2019-04-17 14:54:06 -0700397 f.write('system/bin/install-recovery.sh 0 0 750 '
398 'selabel=u:object_r:install_recovery_exec:s0 capabilities=0x0\n')
399 f.write('system/recovery-from-boot.p 0 0 644 '
400 'selabel=u:object_r:system_file:s0 capabilities=0x0\n')
401 f.write('system/etc/recovery.img 0 0 440 '
402 'selabel=u:object_r:install_recovery_exec:s0 capabilities=0x0\n')
Bill Peckham364c1cc2019-03-29 18:27:23 -0700403
404
Daniel Normand5d70ea2019-06-05 15:13:43 -0700405def process_misc_info_txt(framework_target_files_temp_dir,
406 vendor_target_files_temp_dir,
407 output_target_files_temp_dir,
408 framework_misc_info_keys):
Tao Bao2ad4b822019-06-27 16:52:12 -0700409 """Performs special processing for META/misc_info.txt.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800410
411 This function merges the contents of the META/misc_info.txt files from the
Daniel Normand5d70ea2019-06-05 15:13:43 -0700412 framework directory and the vendor directory, placing the merged result in the
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800413 output directory. The precondition in that the files are already extracted.
414 The post condition is that the output META/misc_info.txt contains the merged
415 content.
416
417 Args:
Daniel Normand5d70ea2019-06-05 15:13:43 -0700418 framework_target_files_temp_dir: The name of a directory containing the
419 special items extracted from the framework target files package.
420 vendor_target_files_temp_dir: The name of a directory containing the special
421 items extracted from the vendor target files package.
Daniel Normane5b134a2019-04-17 14:54:06 -0700422 output_target_files_temp_dir: The name of a directory that will be used to
423 create the output target files package after all the special cases are
424 processed.
Daniel Normand5d70ea2019-06-05 15:13:43 -0700425 framework_misc_info_keys: A list of keys to obtain from the framework
426 instance of META/misc_info.txt. The remaining keys from the vendor
427 instance.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800428 """
429
Kiyoung Kimebe7c9c2019-06-25 17:09:55 +0900430 misc_info_path = ['META', 'misc_info.txt']
431 framework_dict = common.LoadDictionaryFromFile(
432 os.path.join(framework_target_files_temp_dir, *misc_info_path))
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800433
Daniel Normand5d70ea2019-06-05 15:13:43 -0700434 # We take most of the misc info from the vendor target files.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800435
Kiyoung Kimebe7c9c2019-06-25 17:09:55 +0900436 merged_dict = common.LoadDictionaryFromFile(
437 os.path.join(vendor_target_files_temp_dir, *misc_info_path))
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800438
Daniel Normand5d70ea2019-06-05 15:13:43 -0700439 # Replace certain values in merged_dict with values from
440 # framework_dict.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800441
Daniel Normand5d70ea2019-06-05 15:13:43 -0700442 for key in framework_misc_info_keys:
443 merged_dict[key] = framework_dict[key]
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800444
Daniel Norman19b9fe92019-03-19 14:48:02 -0700445 # Merge misc info keys used for Dynamic Partitions.
Daniel Normand5d70ea2019-06-05 15:13:43 -0700446 if (merged_dict.get('use_dynamic_partitions') == 'true') and (
447 framework_dict.get('use_dynamic_partitions') == 'true'):
Daniel Normanbfc51ef2019-07-24 14:34:54 -0700448 merged_dynamic_partitions_dict = common.MergeDynamicPartitionInfoDicts(
Daniel Normand5d70ea2019-06-05 15:13:43 -0700449 framework_dict=framework_dict,
450 vendor_dict=merged_dict,
Daniel Normana61cde02019-05-03 14:19:13 -0700451 size_prefix='super_',
452 size_suffix='_group_size',
453 list_prefix='super_',
454 list_suffix='_partition_list')
Daniel Normand5d70ea2019-06-05 15:13:43 -0700455 merged_dict.update(merged_dynamic_partitions_dict)
Tao Bao48a2feb2019-06-28 11:00:05 -0700456 # Ensure that add_img_to_target_files rebuilds super split images for
457 # devices that retrofit dynamic partitions. This flag may have been set to
458 # false in the partial builds to prevent duplicate building of super.img.
Daniel Norman0bf940c2019-06-10 12:50:19 -0700459 merged_dict['build_super_partition'] = 'true'
Daniel Norman19b9fe92019-03-19 14:48:02 -0700460
Daniel Normand5d70ea2019-06-05 15:13:43 -0700461 # Replace <image>_selinux_fc values with framework or vendor file_contexts.bin
Daniel Norman72c626f2019-05-13 15:58:14 -0700462 # depending on which dictionary the key came from.
463 # Only the file basename is required because all selinux_fc properties are
464 # replaced with the full path to the file under META/ when misc_info.txt is
465 # loaded from target files for repacking. See common.py LoadInfoDict().
Daniel Normand5d70ea2019-06-05 15:13:43 -0700466 for key in merged_dict:
Daniel Norman72c626f2019-05-13 15:58:14 -0700467 if key.endswith('_selinux_fc'):
Daniel Normand5d70ea2019-06-05 15:13:43 -0700468 merged_dict[key] = 'vendor_file_contexts.bin'
469 for key in framework_dict:
Daniel Norman72c626f2019-05-13 15:58:14 -0700470 if key.endswith('_selinux_fc'):
Daniel Normand5d70ea2019-06-05 15:13:43 -0700471 merged_dict[key] = 'framework_file_contexts.bin'
Daniel Norman72c626f2019-05-13 15:58:14 -0700472
Daniel Normane5b134a2019-04-17 14:54:06 -0700473 output_misc_info_txt = os.path.join(output_target_files_temp_dir, 'META',
474 'misc_info.txt')
Daniel Normand5d70ea2019-06-05 15:13:43 -0700475 write_sorted_data(data=merged_dict, path=output_misc_info_txt)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800476
477
Daniel Normand5d70ea2019-06-05 15:13:43 -0700478def process_dynamic_partitions_info_txt(framework_target_files_dir,
479 vendor_target_files_dir,
Daniel Normana61cde02019-05-03 14:19:13 -0700480 output_target_files_dir):
Tao Bao2ad4b822019-06-27 16:52:12 -0700481 """Performs special processing for META/dynamic_partitions_info.txt.
Daniel Normana61cde02019-05-03 14:19:13 -0700482
483 This function merges the contents of the META/dynamic_partitions_info.txt
Daniel Normand5d70ea2019-06-05 15:13:43 -0700484 files from the framework directory and the vendor directory, placing the
485 merged result in the output directory.
Daniel Normana61cde02019-05-03 14:19:13 -0700486
Daniel Normand5d70ea2019-06-05 15:13:43 -0700487 This function does nothing if META/dynamic_partitions_info.txt from the vendor
Daniel Normana61cde02019-05-03 14:19:13 -0700488 directory does not exist.
489
490 Args:
Daniel Normand5d70ea2019-06-05 15:13:43 -0700491 framework_target_files_dir: The name of a directory containing the special
492 items extracted from the framework target files package.
493 vendor_target_files_dir: The name of a directory containing the special
494 items extracted from the vendor target files package.
Daniel Normana61cde02019-05-03 14:19:13 -0700495 output_target_files_dir: The name of a directory that will be used to create
496 the output target files package after all the special cases are processed.
497 """
498
499 if not os.path.exists(
Daniel Normand5d70ea2019-06-05 15:13:43 -0700500 os.path.join(vendor_target_files_dir, 'META',
Daniel Normana61cde02019-05-03 14:19:13 -0700501 'dynamic_partitions_info.txt')):
502 return
503
Kiyoung Kimebe7c9c2019-06-25 17:09:55 +0900504 dynamic_partitions_info_path = ['META', 'dynamic_partitions_info.txt']
Daniel Normana61cde02019-05-03 14:19:13 -0700505
Kiyoung Kimebe7c9c2019-06-25 17:09:55 +0900506 framework_dynamic_partitions_dict = common.LoadDictionaryFromFile(
507 os.path.join(framework_target_files_dir, *dynamic_partitions_info_path))
508 vendor_dynamic_partitions_dict = common.LoadDictionaryFromFile(
509 os.path.join(vendor_target_files_dir, *dynamic_partitions_info_path))
Daniel Normana61cde02019-05-03 14:19:13 -0700510
Daniel Normanbfc51ef2019-07-24 14:34:54 -0700511 merged_dynamic_partitions_dict = common.MergeDynamicPartitionInfoDicts(
Daniel Normand5d70ea2019-06-05 15:13:43 -0700512 framework_dict=framework_dynamic_partitions_dict,
513 vendor_dict=vendor_dynamic_partitions_dict,
Daniel Normana61cde02019-05-03 14:19:13 -0700514 # META/dynamic_partitions_info.txt does not use dynamic_partition_list.
515 include_dynamic_partition_list=False,
516 size_suffix='_size',
517 list_suffix='_partition_list')
518
519 output_dynamic_partitions_info_txt = os.path.join(
520 output_target_files_dir, 'META', 'dynamic_partitions_info.txt')
Chris Grossfabf50a2019-05-02 12:42:09 -0700521 write_sorted_data(
522 data=merged_dynamic_partitions_dict,
523 path=output_dynamic_partitions_info_txt)
524
525
Daniel Normand5d70ea2019-06-05 15:13:43 -0700526def process_apex_keys_apk_certs_common(framework_target_files_dir,
527 vendor_target_files_dir,
Chris Grossfabf50a2019-05-02 12:42:09 -0700528 output_target_files_dir, file_name):
Tao Bao2ad4b822019-06-27 16:52:12 -0700529 """Performs special processing for META/apexkeys.txt or META/apkcerts.txt.
Chris Grossfabf50a2019-05-02 12:42:09 -0700530
531 This function merges the contents of the META/apexkeys.txt or
Tao Bao2ad4b822019-06-27 16:52:12 -0700532 META/apkcerts.txt files from the framework directory and the vendor directory,
533 placing the merged result in the output directory. The precondition in that
534 the files are already extracted. The post condition is that the output
535 META/apexkeys.txt or META/apkcerts.txt contains the merged content.
Chris Grossfabf50a2019-05-02 12:42:09 -0700536
537 Args:
Daniel Normand5d70ea2019-06-05 15:13:43 -0700538 framework_target_files_dir: The name of a directory containing the special
539 items extracted from the framework target files package.
540 vendor_target_files_dir: The name of a directory containing the special
541 items extracted from the vendor target files package.
Chris Grossfabf50a2019-05-02 12:42:09 -0700542 output_target_files_dir: The name of a directory that will be used to create
543 the output target files package after all the special cases are processed.
544 file_name: The name of the file to merge. One of apkcerts.txt or
545 apexkeys.txt.
546 """
547
548 def read_helper(d):
549 temp = {}
550 file_path = os.path.join(d, 'META', file_name)
551 with open(file_path) as f:
552 for line in f:
553 if line.strip():
554 temp[line.split()[0]] = line.strip()
555 return temp
556
Daniel Normand5d70ea2019-06-05 15:13:43 -0700557 framework_dict = read_helper(framework_target_files_dir)
558 vendor_dict = read_helper(vendor_target_files_dir)
Chris Grossfabf50a2019-05-02 12:42:09 -0700559
Daniel Normand5d70ea2019-06-05 15:13:43 -0700560 for key in framework_dict:
561 if key in vendor_dict and vendor_dict[key] != framework_dict[key]:
Chris Grossfabf50a2019-05-02 12:42:09 -0700562 raise ValueError('Conflicting entries found in %s:\n %s and\n %s' %
Daniel Normand5d70ea2019-06-05 15:13:43 -0700563 (file_name, framework_dict[key], vendor_dict[key]))
564 vendor_dict[key] = framework_dict[key]
Chris Grossfabf50a2019-05-02 12:42:09 -0700565
566 output_file = os.path.join(output_target_files_dir, 'META', file_name)
567
Daniel Normand5d70ea2019-06-05 15:13:43 -0700568 write_sorted_data(data=vendor_dict.values(), path=output_file)
Daniel Normana61cde02019-05-03 14:19:13 -0700569
570
Daniel Normand5d70ea2019-06-05 15:13:43 -0700571def copy_file_contexts(framework_target_files_dir, vendor_target_files_dir,
Daniel Norman72c626f2019-05-13 15:58:14 -0700572 output_target_files_dir):
573 """Creates named copies of each build's file_contexts.bin in output META/."""
Daniel Normand5d70ea2019-06-05 15:13:43 -0700574 framework_fc_path = os.path.join(framework_target_files_dir, 'META',
575 'framework_file_contexts.bin')
576 if not os.path.exists(framework_fc_path):
577 framework_fc_path = os.path.join(framework_target_files_dir, 'META',
578 'file_contexts.bin')
579 if not os.path.exists(framework_fc_path):
580 raise ValueError('Missing framework file_contexts.bin.')
581 shutil.copyfile(
582 framework_fc_path,
583 os.path.join(output_target_files_dir, 'META',
584 'framework_file_contexts.bin'))
585
586 vendor_fc_path = os.path.join(vendor_target_files_dir, 'META',
587 'vendor_file_contexts.bin')
588 if not os.path.exists(vendor_fc_path):
589 vendor_fc_path = os.path.join(vendor_target_files_dir, 'META',
Daniel Normanedf12472019-05-22 10:47:08 -0700590 'file_contexts.bin')
Daniel Normand5d70ea2019-06-05 15:13:43 -0700591 if not os.path.exists(vendor_fc_path):
592 raise ValueError('Missing vendor file_contexts.bin.')
Daniel Norman72c626f2019-05-13 15:58:14 -0700593 shutil.copyfile(
Daniel Normand5d70ea2019-06-05 15:13:43 -0700594 vendor_fc_path,
595 os.path.join(output_target_files_dir, 'META', 'vendor_file_contexts.bin'))
Daniel Norman72c626f2019-05-13 15:58:14 -0700596
597
Daniel Normand5d70ea2019-06-05 15:13:43 -0700598def process_special_cases(framework_target_files_temp_dir,
599 vendor_target_files_temp_dir,
600 output_target_files_temp_dir,
601 framework_misc_info_keys, rebuild_recovery):
Tao Bao2ad4b822019-06-27 16:52:12 -0700602 """Performs special-case processing for certain target files items.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800603
604 Certain files in the output target files package require special-case
605 processing. This function performs all that special-case processing.
606
607 Args:
Daniel Normand5d70ea2019-06-05 15:13:43 -0700608 framework_target_files_temp_dir: The name of a directory containing the
609 special items extracted from the framework target files package.
610 vendor_target_files_temp_dir: The name of a directory containing the special
611 items extracted from the vendor target files package.
Daniel Normane5b134a2019-04-17 14:54:06 -0700612 output_target_files_temp_dir: The name of a directory that will be used to
613 create the output target files package after all the special cases are
614 processed.
Daniel Normand5d70ea2019-06-05 15:13:43 -0700615 framework_misc_info_keys: A list of keys to obtain from the framework
616 instance of META/misc_info.txt. The remaining keys from the vendor
617 instance.
Bill Peckham364c1cc2019-03-29 18:27:23 -0700618 rebuild_recovery: If true, rebuild the recovery patch used by non-A/B
Daniel Normane5b134a2019-04-17 14:54:06 -0700619 devices and write it to the system image.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800620 """
621
Daniel Normand5d70ea2019-06-05 15:13:43 -0700622 if 'ab_update' in framework_misc_info_keys:
Bill Peckham364c1cc2019-03-29 18:27:23 -0700623 process_ab_partitions_txt(
Daniel Normand5d70ea2019-06-05 15:13:43 -0700624 framework_target_files_temp_dir=framework_target_files_temp_dir,
625 vendor_target_files_temp_dir=vendor_target_files_temp_dir,
Bill Peckham364c1cc2019-03-29 18:27:23 -0700626 output_target_files_temp_dir=output_target_files_temp_dir)
627
628 if rebuild_recovery:
629 append_recovery_to_filesystem_config(
630 output_target_files_temp_dir=output_target_files_temp_dir)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800631
Daniel Norman72c626f2019-05-13 15:58:14 -0700632 copy_file_contexts(
Daniel Normand5d70ea2019-06-05 15:13:43 -0700633 framework_target_files_dir=framework_target_files_temp_dir,
634 vendor_target_files_dir=vendor_target_files_temp_dir,
Daniel Norman72c626f2019-05-13 15:58:14 -0700635 output_target_files_dir=output_target_files_temp_dir)
636
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800637 process_misc_info_txt(
Daniel Normand5d70ea2019-06-05 15:13:43 -0700638 framework_target_files_temp_dir=framework_target_files_temp_dir,
639 vendor_target_files_temp_dir=vendor_target_files_temp_dir,
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800640 output_target_files_temp_dir=output_target_files_temp_dir,
Daniel Normand5d70ea2019-06-05 15:13:43 -0700641 framework_misc_info_keys=framework_misc_info_keys)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800642
Daniel Normana61cde02019-05-03 14:19:13 -0700643 process_dynamic_partitions_info_txt(
Daniel Normand5d70ea2019-06-05 15:13:43 -0700644 framework_target_files_dir=framework_target_files_temp_dir,
645 vendor_target_files_dir=vendor_target_files_temp_dir,
Daniel Norman714bd122019-05-08 16:20:02 -0700646 output_target_files_dir=output_target_files_temp_dir)
Daniel Normana61cde02019-05-03 14:19:13 -0700647
Chris Grossfabf50a2019-05-02 12:42:09 -0700648 process_apex_keys_apk_certs_common(
Daniel Normand5d70ea2019-06-05 15:13:43 -0700649 framework_target_files_dir=framework_target_files_temp_dir,
650 vendor_target_files_dir=vendor_target_files_temp_dir,
Chris Grossfabf50a2019-05-02 12:42:09 -0700651 output_target_files_dir=output_target_files_temp_dir,
652 file_name='apkcerts.txt')
653
654 process_apex_keys_apk_certs_common(
Daniel Normand5d70ea2019-06-05 15:13:43 -0700655 framework_target_files_dir=framework_target_files_temp_dir,
656 vendor_target_files_dir=vendor_target_files_temp_dir,
Chris Grossfabf50a2019-05-02 12:42:09 -0700657 output_target_files_dir=output_target_files_temp_dir,
658 file_name='apexkeys.txt')
659
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800660
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900661def files_from_path(target_path, extra_args=None):
Tao Bao2ad4b822019-06-27 16:52:12 -0700662 """Gets files under given path.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800663
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900664 Get (sub)files from given target path and return sorted list.
665
666 Args:
667 target_path: Target path to get subfiles.
668 extra_args: List of extra argument for find command. Optional.
669
670 Returns:
671 Sorted files and directories list.
672 """
673
674 find_command = ['find', target_path] + (extra_args or [])
675 find_process = common.Run(find_command, stdout=subprocess.PIPE, verbose=False)
Daniel Norman4cc9df62019-07-18 10:11:07 -0700676 return common.RunAndCheckOutput(['sort'],
677 stdin=find_process.stdout,
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900678 verbose=False)
679
680
681def create_merged_package(temp_dir, framework_target_files, framework_item_list,
682 vendor_target_files, vendor_item_list,
Daniel Norman4cc9df62019-07-18 10:11:07 -0700683 framework_misc_info_keys, rebuild_recovery):
Tao Bao2ad4b822019-06-27 16:52:12 -0700684 """Merges two target files packages into one target files structure.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800685
686 Args:
687 temp_dir: The name of a directory we use when we extract items from the
Daniel Normane5b134a2019-04-17 14:54:06 -0700688 input target files packages, and also a scratch directory that we use for
689 temporary files.
Daniel Normand5d70ea2019-06-05 15:13:43 -0700690 framework_target_files: The name of the zip archive containing the framework
Daniel Normane5b134a2019-04-17 14:54:06 -0700691 partial target files package.
Daniel Normand5d70ea2019-06-05 15:13:43 -0700692 framework_item_list: The list of items to extract from the partial framework
Daniel Normane5b134a2019-04-17 14:54:06 -0700693 target files package as is, meaning these items will land in the output
Daniel Normand5d70ea2019-06-05 15:13:43 -0700694 target files package exactly as they appear in the input partial framework
Daniel Normane5b134a2019-04-17 14:54:06 -0700695 target files package.
Daniel Normand5d70ea2019-06-05 15:13:43 -0700696 vendor_target_files: The name of the zip archive containing the vendor
697 partial target files package.
698 vendor_item_list: The list of items to extract from the partial vendor
699 target files package as is, meaning these items will land in the output
700 target files package exactly as they appear in the input partial vendor
Daniel Normane5b134a2019-04-17 14:54:06 -0700701 target files package.
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900702 framework_misc_info_keys: The list of keys to obtain from the framework
703 instance of META/misc_info.txt. The remaining keys from the vendor
704 instance.
Daniel Normana4911da2019-03-15 14:36:21 -0700705 rebuild_recovery: If true, rebuild the recovery patch used by non-A/B
Daniel Normane5b134a2019-04-17 14:54:06 -0700706 devices and write it to the system image.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800707
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900708 Returns:
709 Path to merged package under temp directory.
710 """
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800711
Daniel Normand5d70ea2019-06-05 15:13:43 -0700712 # Create directory names that we'll use when we extract files from framework,
713 # and vendor, and for zipping the final output.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800714
Daniel Normand5d70ea2019-06-05 15:13:43 -0700715 framework_target_files_temp_dir = os.path.join(temp_dir, 'framework')
716 vendor_target_files_temp_dir = os.path.join(temp_dir, 'vendor')
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800717 output_target_files_temp_dir = os.path.join(temp_dir, 'output')
718
Daniel Normand5d70ea2019-06-05 15:13:43 -0700719 # Extract "as is" items from the input framework partial target files package.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800720 # We extract them directly into the output temporary directory since the
721 # items do not need special case processing.
722
Bill Peckham889b0c62019-02-21 18:53:37 -0800723 extract_items(
Daniel Normand5d70ea2019-06-05 15:13:43 -0700724 target_files=framework_target_files,
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800725 target_files_temp_dir=output_target_files_temp_dir,
Daniel Normand5d70ea2019-06-05 15:13:43 -0700726 extract_item_list=framework_item_list)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800727
Daniel Normand5d70ea2019-06-05 15:13:43 -0700728 # Extract "as is" items from the input vendor partial target files package. We
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800729 # extract them directly into the output temporary directory since the items
730 # do not need special case processing.
731
Bill Peckham889b0c62019-02-21 18:53:37 -0800732 extract_items(
Daniel Normand5d70ea2019-06-05 15:13:43 -0700733 target_files=vendor_target_files,
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800734 target_files_temp_dir=output_target_files_temp_dir,
Daniel Normand5d70ea2019-06-05 15:13:43 -0700735 extract_item_list=vendor_item_list)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800736
Daniel Normand5d70ea2019-06-05 15:13:43 -0700737 # Extract "special" items from the input framework partial target files
738 # package. We extract these items to different directory since they require
739 # special processing before they will end up in the output directory.
740
741 extract_items(
742 target_files=framework_target_files,
743 target_files_temp_dir=framework_target_files_temp_dir,
744 extract_item_list=FRAMEWORK_EXTRACT_SPECIAL_ITEM_LIST)
745
746 # Extract "special" items from the input vendor partial target files package.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800747 # We extract these items to different directory since they require special
748 # processing before they will end up in the output directory.
749
Bill Peckham889b0c62019-02-21 18:53:37 -0800750 extract_items(
Daniel Normand5d70ea2019-06-05 15:13:43 -0700751 target_files=vendor_target_files,
752 target_files_temp_dir=vendor_target_files_temp_dir,
753 extract_item_list=VENDOR_EXTRACT_SPECIAL_ITEM_LIST)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800754
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800755 # Now that the temporary directories contain all the extracted files, perform
756 # special case processing on any items that need it. After this function
757 # completes successfully, all the files we need to create the output target
758 # files package are in place.
759
Bill Peckham889b0c62019-02-21 18:53:37 -0800760 process_special_cases(
Daniel Normand5d70ea2019-06-05 15:13:43 -0700761 framework_target_files_temp_dir=framework_target_files_temp_dir,
762 vendor_target_files_temp_dir=vendor_target_files_temp_dir,
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800763 output_target_files_temp_dir=output_target_files_temp_dir,
Daniel Normand5d70ea2019-06-05 15:13:43 -0700764 framework_misc_info_keys=framework_misc_info_keys,
Bill Peckham364c1cc2019-03-29 18:27:23 -0700765 rebuild_recovery=rebuild_recovery)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800766
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900767 return output_target_files_temp_dir
768
769
770def generate_images(target_files_dir, rebuild_recovery):
771 """Generate images from target files.
772
773 This function takes merged output temporary directory and create images
774 from it.
775
776 Args:
777 target_files_dir: Path to merged temp directory.
778 rebuild_recovery: If true, rebuild the recovery patch used by non-A/B
779 devices and write it to the system image.
780 """
781
782 # Regenerate IMAGES in the target directory.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800783
Daniel Normana4911da2019-03-15 14:36:21 -0700784 add_img_args = ['--verbose']
785 if rebuild_recovery:
786 add_img_args.append('--rebuild_recovery')
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900787 add_img_args.append(target_files_dir)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800788
789 add_img_to_target_files.main(add_img_args)
790
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900791
792def generate_super_empty_image(target_dir, output_super_empty):
Tao Bao2ad4b822019-06-27 16:52:12 -0700793 """Generates super_empty image from target package.
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900794
795 Args:
796 target_dir: Path to the target file package which contains misc_info.txt for
797 detailed information for super image.
798 output_super_empty: If provided, copies a super_empty.img file from the
799 target files package to this path.
800 """
Daniel Norman1bd2a1d2019-04-18 12:32:18 -0700801 # Create super_empty.img using the merged misc_info.txt.
802
Daniel Norman4cc9df62019-07-18 10:11:07 -0700803 misc_info_txt = os.path.join(target_dir, 'META', 'misc_info.txt')
Daniel Norman1bd2a1d2019-04-18 12:32:18 -0700804
Kiyoung Kimebe7c9c2019-06-25 17:09:55 +0900805 use_dynamic_partitions = common.LoadDictionaryFromFile(misc_info_txt).get(
806 'use_dynamic_partitions')
Daniel Norman1bd2a1d2019-04-18 12:32:18 -0700807
808 if use_dynamic_partitions != 'true' and output_super_empty:
809 raise ValueError(
810 'Building super_empty.img requires use_dynamic_partitions=true.')
811 elif use_dynamic_partitions == 'true':
Daniel Norman4cc9df62019-07-18 10:11:07 -0700812 super_empty_img = os.path.join(target_dir, 'IMAGES', 'super_empty.img')
Daniel Norman1bd2a1d2019-04-18 12:32:18 -0700813 build_super_image_args = [
814 misc_info_txt,
815 super_empty_img,
816 ]
817 build_super_image.main(build_super_image_args)
818
819 # Copy super_empty.img to the user-provided output_super_empty location.
820 if output_super_empty:
821 shutil.copyfile(super_empty_img, output_super_empty)
822
Daniel Normanb8a2f9d2019-04-24 12:55:51 -0700823
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900824def create_target_files_archive(output_file, source_dir, temp_dir):
Tao Bao2ad4b822019-06-27 16:52:12 -0700825 """Creates archive from target package.
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900826
827 Args:
828 output_file: The name of the zip archive target files package.
829 source_dir: The target directory contains package to be archived.
830 temp_dir: Path to temporary directory for any intermediate files.
831 """
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800832 output_target_files_list = os.path.join(temp_dir, 'output.list')
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900833 output_zip = os.path.abspath(output_file)
Daniel Norman4cc9df62019-07-18 10:11:07 -0700834 output_target_files_meta_dir = os.path.join(source_dir, 'META')
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800835
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900836 meta_content = files_from_path(output_target_files_meta_dir)
Daniel Norman4cc9df62019-07-18 10:11:07 -0700837 other_content = files_from_path(
838 source_dir,
839 ['-path', output_target_files_meta_dir, '-prune', '-o', '-print'])
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800840
Tao Bao2ad4b822019-06-27 16:52:12 -0700841 with open(output_target_files_list, 'w') as f:
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800842 f.write(meta_content)
843 f.write(other_content)
844
845 command = [
Bill Peckhamf753e152019-02-19 18:02:46 -0800846 'soong_zip',
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800847 '-d',
Daniel Normane5b134a2019-04-17 14:54:06 -0700848 '-o',
849 output_zip,
850 '-C',
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900851 source_dir,
Daniel Normane5b134a2019-04-17 14:54:06 -0700852 '-l',
853 output_target_files_list,
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800854 ]
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900855
856 logger.info('creating %s', output_file)
Bill Peckham889b0c62019-02-21 18:53:37 -0800857 common.RunAndWait(command, verbose=True)
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900858 logger.info('finished creating %s', output_file)
859
860 return output_zip
861
862
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900863def merge_target_files(temp_dir, framework_target_files, framework_item_list,
864 framework_misc_info_keys, vendor_target_files,
865 vendor_item_list, output_target_files, output_dir,
866 output_item_list, output_ota, output_img,
867 output_super_empty, rebuild_recovery):
Tao Bao2ad4b822019-06-27 16:52:12 -0700868 """Merges two target files packages together.
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900869
870 This function takes framework and vendor target files packages as input,
871 performs various file extractions, special case processing, and finally
872 creates a merged zip archive as output.
873
874 Args:
875 temp_dir: The name of a directory we use when we extract items from the
876 input target files packages, and also a scratch directory that we use for
877 temporary files.
878 framework_target_files: The name of the zip archive containing the framework
879 partial target files package.
880 framework_item_list: The list of items to extract from the partial framework
881 target files package as is, meaning these items will land in the output
882 target files package exactly as they appear in the input partial framework
883 target files package.
884 framework_misc_info_keys: The list of keys to obtain from the framework
885 instance of META/misc_info.txt. The remaining keys from the vendor
886 instance.
887 vendor_target_files: The name of the zip archive containing the vendor
888 partial target files package.
889 vendor_item_list: The list of items to extract from the partial vendor
890 target files package as is, meaning these items will land in the output
891 target files package exactly as they appear in the input partial vendor
892 target files package.
893 output_target_files: The name of the output zip archive target files package
894 created by merging framework and vendor.
895 output_dir: The destination directory for saving merged files.
896 output_item_list: The list of items to copy into the output_dir.
897 output_ota: The name of the output zip archive ota package.
898 output_img: The name of the output zip archive img package.
899 output_super_empty: If provided, creates a super_empty.img file from the
900 merged target files package and saves it at this path.
901 rebuild_recovery: If true, rebuild the recovery patch used by non-A/B
902 devices and write it to the system image.
903 """
904
905 logger.info('starting: merge framework %s and vendor %s into output %s',
906 framework_target_files, vendor_target_files, output_target_files)
907
908 output_target_files_temp_dir = create_merged_package(
909 temp_dir, framework_target_files, framework_item_list,
910 vendor_target_files, vendor_item_list, framework_misc_info_keys,
911 rebuild_recovery)
912
913 generate_images(output_target_files_temp_dir, rebuild_recovery)
914
915 generate_super_empty_image(output_target_files_temp_dir, output_super_empty)
916
917 if output_img:
918 # Create the IMG package from the merged target files (before zipping, in
919 # order to avoid an unnecessary unzip and copy).
Daniel Norman4cc9df62019-07-18 10:11:07 -0700920 img_from_target_files.main([output_target_files_temp_dir, output_img])
Kiyoung Kim7cbeda72019-06-28 13:26:04 +0900921
922 # Finally, create the output target files zip archive and/or copy the
923 # output items to the output target files directory.
924
925 if output_dir:
926 copy_items(output_target_files_temp_dir, output_dir, output_item_list)
927
928 if not output_target_files:
929 return
930
931 output_zip = create_target_files_archive(output_target_files,
932 output_target_files_temp_dir,
933 temp_dir)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800934
Daniel Norman3b64ce12019-04-16 16:11:35 -0700935 # Create the OTA package from the merged target files package.
936
937 if output_ota:
Daniel Norman4cc9df62019-07-18 10:11:07 -0700938 ota_from_target_files.main([output_zip, output_ota])
Daniel Norman3b64ce12019-04-16 16:11:35 -0700939
Daniel Norman1bd2a1d2019-04-18 12:32:18 -0700940
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800941def call_func_with_temp_dir(func, keep_tmp):
Tao Bao2ad4b822019-06-27 16:52:12 -0700942 """Manages the creation and cleanup of the temporary directory.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800943
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800944 This function calls the given function after first creating a temporary
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800945 directory. It also cleans up the temporary directory.
946
947 Args:
Daniel Normane5b134a2019-04-17 14:54:06 -0700948 func: The function to call. Should accept one parameter, the path to the
949 temporary directory.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800950 keep_tmp: Keep the temporary directory after processing is complete.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800951 """
952
953 # Create a temporary directory. This will serve as the parent of directories
954 # we use when we extract items from the input target files packages, and also
955 # a scratch directory that we use for temporary files.
956
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800957 temp_dir = common.MakeTempDir(prefix='merge_target_files_')
958
959 try:
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800960 func(temp_dir)
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800961 finally:
962 if keep_tmp:
963 logger.info('keeping %s', temp_dir)
964 else:
965 common.Cleanup()
966
967
968def main():
969 """The main function.
970
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800971 Process command line arguments, then call merge_target_files to
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800972 perform the heavy lifting.
Bill Peckhame9eb5f92019-02-01 15:52:10 -0800973 """
974
975 common.InitLogging()
976
Bill Peckhamf753e152019-02-19 18:02:46 -0800977 def option_handler(o, a):
978 if o == '--system-target-files':
Daniel Normand5d70ea2019-06-05 15:13:43 -0700979 logger.warning(
980 '--system-target-files has been renamed to --framework-target-files')
981 OPTIONS.framework_target_files = a
982 elif o == '--framework-target-files':
983 OPTIONS.framework_target_files = a
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800984 elif o == '--system-item-list':
Daniel Normand5d70ea2019-06-05 15:13:43 -0700985 logger.warning(
986 '--system-item-list has been renamed to --framework-item-list')
987 OPTIONS.framework_item_list = a
988 elif o == '--framework-item-list':
989 OPTIONS.framework_item_list = a
Daniel Norman2c99c5b2019-03-07 13:01:48 -0800990 elif o == '--system-misc-info-keys':
Daniel Norman4cc9df62019-07-18 10:11:07 -0700991 logger.warning('--system-misc-info-keys has been renamed to '
992 '--framework-misc-info-keys')
Daniel Normand5d70ea2019-06-05 15:13:43 -0700993 OPTIONS.framework_misc_info_keys = a
994 elif o == '--framework-misc-info-keys':
995 OPTIONS.framework_misc_info_keys = a
Bill Peckhamf753e152019-02-19 18:02:46 -0800996 elif o == '--other-target-files':
Daniel Normand5d70ea2019-06-05 15:13:43 -0700997 logger.warning(
998 '--other-target-files has been renamed to --vendor-target-files')
999 OPTIONS.vendor_target_files = a
1000 elif o == '--vendor-target-files':
1001 OPTIONS.vendor_target_files = a
Daniel Norman2c99c5b2019-03-07 13:01:48 -08001002 elif o == '--other-item-list':
Daniel Normand5d70ea2019-06-05 15:13:43 -07001003 logger.warning('--other-item-list has been renamed to --vendor-item-list')
1004 OPTIONS.vendor_item_list = a
1005 elif o == '--vendor-item-list':
1006 OPTIONS.vendor_item_list = a
Bill Peckhamf753e152019-02-19 18:02:46 -08001007 elif o == '--output-target-files':
1008 OPTIONS.output_target_files = a
Daniel Normanfdb38812019-04-15 09:47:24 -07001009 elif o == '--output-dir':
1010 OPTIONS.output_dir = a
1011 elif o == '--output-item-list':
1012 OPTIONS.output_item_list = a
Daniel Norman3b64ce12019-04-16 16:11:35 -07001013 elif o == '--output-ota':
1014 OPTIONS.output_ota = a
Daniel Norman1bd2a1d2019-04-18 12:32:18 -07001015 elif o == '--output-img':
1016 OPTIONS.output_img = a
Daniel Normanf0318252019-04-15 11:34:56 -07001017 elif o == '--output-super-empty':
1018 OPTIONS.output_super_empty = a
Daniel Normana4911da2019-03-15 14:36:21 -07001019 elif o == '--rebuild_recovery':
1020 OPTIONS.rebuild_recovery = True
Bill Peckham364c1cc2019-03-29 18:27:23 -07001021 elif o == '--keep-tmp':
Bill Peckhamf753e152019-02-19 18:02:46 -08001022 OPTIONS.keep_tmp = True
1023 else:
1024 return False
1025 return True
Bill Peckhame9eb5f92019-02-01 15:52:10 -08001026
Bill Peckhamf753e152019-02-19 18:02:46 -08001027 args = common.ParseOptions(
Daniel Normane5b134a2019-04-17 14:54:06 -07001028 sys.argv[1:],
1029 __doc__,
Bill Peckhamf753e152019-02-19 18:02:46 -08001030 extra_long_opts=[
1031 'system-target-files=',
Daniel Normand5d70ea2019-06-05 15:13:43 -07001032 'framework-target-files=',
Daniel Norman2c99c5b2019-03-07 13:01:48 -08001033 'system-item-list=',
Daniel Normand5d70ea2019-06-05 15:13:43 -07001034 'framework-item-list=',
Daniel Norman2c99c5b2019-03-07 13:01:48 -08001035 'system-misc-info-keys=',
Daniel Normand5d70ea2019-06-05 15:13:43 -07001036 'framework-misc-info-keys=',
Bill Peckhamf753e152019-02-19 18:02:46 -08001037 'other-target-files=',
Daniel Normand5d70ea2019-06-05 15:13:43 -07001038 'vendor-target-files=',
Daniel Norman2c99c5b2019-03-07 13:01:48 -08001039 'other-item-list=',
Daniel Normand5d70ea2019-06-05 15:13:43 -07001040 'vendor-item-list=',
Bill Peckhamf753e152019-02-19 18:02:46 -08001041 'output-target-files=',
Daniel Normanfdb38812019-04-15 09:47:24 -07001042 'output-dir=',
1043 'output-item-list=',
Daniel Norman3b64ce12019-04-16 16:11:35 -07001044 'output-ota=',
Daniel Norman1bd2a1d2019-04-18 12:32:18 -07001045 'output-img=',
Daniel Normanf0318252019-04-15 11:34:56 -07001046 'output-super-empty=',
Daniel Normana4911da2019-03-15 14:36:21 -07001047 'rebuild_recovery',
Bill Peckham364c1cc2019-03-29 18:27:23 -07001048 'keep-tmp',
Bill Peckhamf753e152019-02-19 18:02:46 -08001049 ],
1050 extra_option_handler=option_handler)
Bill Peckhame9eb5f92019-02-01 15:52:10 -08001051
Tao Bao2ad4b822019-06-27 16:52:12 -07001052 # pylint: disable=too-many-boolean-expressions
Daniel Normand5d70ea2019-06-05 15:13:43 -07001053 if (args or OPTIONS.framework_target_files is None or
1054 OPTIONS.vendor_target_files is None or
Daniel Normane5b134a2019-04-17 14:54:06 -07001055 (OPTIONS.output_target_files is None and OPTIONS.output_dir is None) or
1056 (OPTIONS.output_dir is not None and OPTIONS.output_item_list is None)):
Bill Peckhamf753e152019-02-19 18:02:46 -08001057 common.Usage(__doc__)
Bill Peckham889b0c62019-02-21 18:53:37 -08001058 sys.exit(1)
Bill Peckhame9eb5f92019-02-01 15:52:10 -08001059
Daniel Normand5d70ea2019-06-05 15:13:43 -07001060 if OPTIONS.framework_item_list:
Daniel Norman4cc9df62019-07-18 10:11:07 -07001061 framework_item_list = common.LoadListFromFile(OPTIONS.framework_item_list)
Daniel Norman2c99c5b2019-03-07 13:01:48 -08001062 else:
Daniel Normand5d70ea2019-06-05 15:13:43 -07001063 framework_item_list = DEFAULT_FRAMEWORK_ITEM_LIST
Daniel Norman2c99c5b2019-03-07 13:01:48 -08001064
Daniel Normand5d70ea2019-06-05 15:13:43 -07001065 if OPTIONS.framework_misc_info_keys:
Daniel Norman4cc9df62019-07-18 10:11:07 -07001066 framework_misc_info_keys = common.LoadListFromFile(
Daniel Normand5d70ea2019-06-05 15:13:43 -07001067 OPTIONS.framework_misc_info_keys)
Daniel Norman2c99c5b2019-03-07 13:01:48 -08001068 else:
Daniel Normand5d70ea2019-06-05 15:13:43 -07001069 framework_misc_info_keys = DEFAULT_FRAMEWORK_MISC_INFO_KEYS
Daniel Norman2c99c5b2019-03-07 13:01:48 -08001070
Daniel Normand5d70ea2019-06-05 15:13:43 -07001071 if OPTIONS.vendor_item_list:
Daniel Norman4cc9df62019-07-18 10:11:07 -07001072 vendor_item_list = common.LoadListFromFile(OPTIONS.vendor_item_list)
Daniel Norman2c99c5b2019-03-07 13:01:48 -08001073 else:
Daniel Normand5d70ea2019-06-05 15:13:43 -07001074 vendor_item_list = DEFAULT_VENDOR_ITEM_LIST
Daniel Norman2c99c5b2019-03-07 13:01:48 -08001075
Daniel Normanfdb38812019-04-15 09:47:24 -07001076 if OPTIONS.output_item_list:
Daniel Norman4cc9df62019-07-18 10:11:07 -07001077 output_item_list = common.LoadListFromFile(OPTIONS.output_item_list)
Daniel Normanfdb38812019-04-15 09:47:24 -07001078 else:
1079 output_item_list = None
1080
Daniel Normane5964522019-03-19 10:32:03 -07001081 if not validate_config_lists(
Daniel Normand5d70ea2019-06-05 15:13:43 -07001082 framework_item_list=framework_item_list,
1083 framework_misc_info_keys=framework_misc_info_keys,
1084 vendor_item_list=vendor_item_list):
Daniel Normane5964522019-03-19 10:32:03 -07001085 sys.exit(1)
1086
Daniel Norman2c99c5b2019-03-07 13:01:48 -08001087 call_func_with_temp_dir(
1088 lambda temp_dir: merge_target_files(
1089 temp_dir=temp_dir,
Daniel Normand5d70ea2019-06-05 15:13:43 -07001090 framework_target_files=OPTIONS.framework_target_files,
1091 framework_item_list=framework_item_list,
1092 framework_misc_info_keys=framework_misc_info_keys,
1093 vendor_target_files=OPTIONS.vendor_target_files,
1094 vendor_item_list=vendor_item_list,
Daniel Normana4911da2019-03-15 14:36:21 -07001095 output_target_files=OPTIONS.output_target_files,
Daniel Normanfdb38812019-04-15 09:47:24 -07001096 output_dir=OPTIONS.output_dir,
1097 output_item_list=output_item_list,
Daniel Norman3b64ce12019-04-16 16:11:35 -07001098 output_ota=OPTIONS.output_ota,
Daniel Norman1bd2a1d2019-04-18 12:32:18 -07001099 output_img=OPTIONS.output_img,
Daniel Normanf0318252019-04-15 11:34:56 -07001100 output_super_empty=OPTIONS.output_super_empty,
Daniel Normane5b134a2019-04-17 14:54:06 -07001101 rebuild_recovery=OPTIONS.rebuild_recovery), OPTIONS.keep_tmp)
Bill Peckhame9eb5f92019-02-01 15:52:10 -08001102
1103
1104if __name__ == '__main__':
Bill Peckham889b0c62019-02-21 18:53:37 -08001105 main()