| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 1 | #!/usr/bin/env python | 
|  | 2 | # | 
|  | 3 | # Copyright (C) 2014 The Android Open Source Project | 
|  | 4 | # | 
|  | 5 | # Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | 6 | # you may not use this file except in compliance with the License. | 
|  | 7 | # You may obtain a copy of 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, | 
|  | 13 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | 14 | # See the License for the specific language governing permissions and | 
|  | 15 | # limitations under the License. | 
|  | 16 |  | 
|  | 17 | """ | 
|  | 18 | Given a target-files zipfile that does not contain images (ie, does | 
|  | 19 | not have an IMAGES/ top-level subdirectory), produce the images and | 
|  | 20 | add them to the zipfile. | 
|  | 21 |  | 
| Tianjie Xu | b48589a | 2016-08-03 19:21:52 -0700 | [diff] [blame] | 22 | Usage:  add_img_to_target_files [flag] target_files | 
|  | 23 |  | 
|  | 24 | -a  (--add_missing) | 
|  | 25 | Build and add missing images to "IMAGES/". If this option is | 
|  | 26 | not specified, this script will simply exit when "IMAGES/" | 
|  | 27 | directory exists in the target file. | 
|  | 28 |  | 
|  | 29 | -r  (--rebuild_recovery) | 
|  | 30 | Rebuild the recovery patch and write it to the system image. Only | 
| Tao Bao | 4978fa9 | 2019-06-04 16:26:45 -0700 | [diff] [blame] | 31 | meaningful when system image needs to be rebuilt and there're separate | 
|  | 32 | boot / recovery images. | 
| Tianjie Xu | b48589a | 2016-08-03 19:21:52 -0700 | [diff] [blame] | 33 |  | 
|  | 34 | --replace_verity_private_key | 
|  | 35 | Replace the private key used for verity signing. (same as the option | 
|  | 36 | in sign_target_files_apks) | 
|  | 37 |  | 
|  | 38 | --replace_verity_public_key | 
|  | 39 | Replace the certificate (public key) used for verity verification. (same | 
|  | 40 | as the option in sign_target_files_apks) | 
|  | 41 |  | 
|  | 42 | --is_signing | 
|  | 43 | Skip building & adding the images for "userdata" and "cache" if we | 
|  | 44 | are signing the target files. | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 45 | """ | 
|  | 46 |  | 
| Tao Bao | 89fbb0f | 2017-01-10 10:47:58 -0800 | [diff] [blame] | 47 | from __future__ import print_function | 
|  | 48 |  | 
| Tao Bao | 822f584 | 2015-09-30 16:01:14 -0700 | [diff] [blame] | 49 | import datetime | 
| Tao Bao | 32fcdab | 2018-10-12 10:30:39 -0700 | [diff] [blame] | 50 | import logging | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 51 | import os | 
| David Zeuthen | d995f4b | 2016-01-29 16:59:17 -0500 | [diff] [blame] | 52 | import shlex | 
| Ying Wang | 2a04839 | 2015-06-25 13:56:53 -0700 | [diff] [blame] | 53 | import shutil | 
| Rupert Shuttleworth | 7294274 | 2020-12-08 06:18:35 +0000 | [diff] [blame] | 54 | import stat | 
| Tao Bao | 6b9fef5 | 2017-12-01 16:13:22 -0800 | [diff] [blame] | 55 | import sys | 
| Tao Bao | d86e311 | 2017-09-22 15:45:33 -0700 | [diff] [blame] | 56 | import uuid | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 57 | import zipfile | 
|  | 58 |  | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 59 | import build_image | 
| Yifan Hong | 055e6cf | 2018-11-29 13:51:48 -0800 | [diff] [blame] | 60 | import build_super_image | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 61 | import common | 
| Tianjie Xu | f1a1318 | 2017-01-19 17:39:30 -0800 | [diff] [blame] | 62 | import rangelib | 
| Tianjie Xu | cfa8622 | 2016-03-07 16:31:19 -0800 | [diff] [blame] | 63 | import sparse_img | 
| Hongguang Chen | f23364d | 2020-04-27 18:36:36 -0700 | [diff] [blame] | 64 | import verity_utils | 
| Kelvin Zhang | 5f0fcee | 2021-01-19 15:30:46 -0500 | [diff] [blame] | 65 | import ota_metadata_pb2 | 
|  | 66 |  | 
|  | 67 | from apex_utils import GetApexInfoFromTargetFiles | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 68 |  | 
| Tao Bao | 6b9fef5 | 2017-12-01 16:13:22 -0800 | [diff] [blame] | 69 | if sys.hexversion < 0x02070000: | 
|  | 70 | print("Python 2.7 or newer is required.", file=sys.stderr) | 
|  | 71 | sys.exit(1) | 
|  | 72 |  | 
| Tao Bao | 32fcdab | 2018-10-12 10:30:39 -0700 | [diff] [blame] | 73 | logger = logging.getLogger(__name__) | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 74 |  | 
| Tao Bao | 32fcdab | 2018-10-12 10:30:39 -0700 | [diff] [blame] | 75 | OPTIONS = common.OPTIONS | 
| Michael Runge | 2e0d8fc | 2014-11-13 21:41:08 -0800 | [diff] [blame] | 76 | OPTIONS.add_missing = False | 
|  | 77 | OPTIONS.rebuild_recovery = False | 
| Tianjie Xu | 9ac4cb0 | 2017-06-09 16:58:03 -0700 | [diff] [blame] | 78 | OPTIONS.replace_updated_files_list = [] | 
| Baligh Uddin | 59f4ff1 | 2015-09-16 21:20:30 -0700 | [diff] [blame] | 79 | OPTIONS.replace_verity_public_key = False | 
|  | 80 | OPTIONS.replace_verity_private_key = False | 
| Tianjie Xu | b48589a | 2016-08-03 19:21:52 -0700 | [diff] [blame] | 81 | OPTIONS.is_signing = False | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 82 |  | 
| Bryan Henry | e6d547d | 2018-07-31 18:32:00 -0700 | [diff] [blame] | 83 | # Use a fixed timestamp (01/01/2009 00:00:00 UTC) for files when packaging | 
|  | 84 | # images. (b/24377993, b/80600931) | 
| Tao Bao | e30a6a6 | 2018-08-27 10:57:19 -0700 | [diff] [blame] | 85 | FIXED_FILE_TIMESTAMP = int(( | 
|  | 86 | datetime.datetime(2009, 1, 1, 0, 0, 0, 0, None) - | 
|  | 87 | datetime.datetime.utcfromtimestamp(0)).total_seconds()) | 
| Tao Bao | a2ff4c9 | 2018-01-17 12:14:43 -0800 | [diff] [blame] | 88 |  | 
| Bowgo Tsai | d624fa6 | 2017-11-14 23:42:30 +0800 | [diff] [blame] | 89 |  | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 90 | class OutputFile(object): | 
| Tao Bao | 93e7ebe | 2019-01-13 23:23:01 -0800 | [diff] [blame] | 91 | """A helper class to write a generated file to the given dir or zip. | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 92 |  | 
| Tao Bao | 93e7ebe | 2019-01-13 23:23:01 -0800 | [diff] [blame] | 93 | When generating images, we want the outputs to go into the given zip file, or | 
|  | 94 | the given dir. | 
|  | 95 |  | 
|  | 96 | Attributes: | 
|  | 97 | name: The name of the output file, regardless of the final destination. | 
|  | 98 | """ | 
|  | 99 |  | 
| Kelvin Zhang | 5f0fcee | 2021-01-19 15:30:46 -0500 | [diff] [blame] | 100 | def __init__(self, output_zip, input_dir, *args): | 
| Tao Bao | 93e7ebe | 2019-01-13 23:23:01 -0800 | [diff] [blame] | 101 | # We write the intermediate output file under the given input_dir, even if | 
|  | 102 | # the final destination is a zip archive. | 
| Kelvin Zhang | 5f0fcee | 2021-01-19 15:30:46 -0500 | [diff] [blame] | 103 | self.name = os.path.join(input_dir, *args) | 
| Tao Bao | 93e7ebe | 2019-01-13 23:23:01 -0800 | [diff] [blame] | 104 | self._output_zip = output_zip | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 105 | if self._output_zip: | 
| Kelvin Zhang | 5f0fcee | 2021-01-19 15:30:46 -0500 | [diff] [blame] | 106 | self._zip_name = os.path.join(*args) | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 107 |  | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 108 | def Write(self): | 
|  | 109 | if self._output_zip: | 
|  | 110 | common.ZipWrite(self._output_zip, self.name, self._zip_name) | 
|  | 111 |  | 
| Tao Bao | e30a6a6 | 2018-08-27 10:57:19 -0700 | [diff] [blame] | 112 |  | 
| Tianjie Xu | cfa8622 | 2016-03-07 16:31:19 -0800 | [diff] [blame] | 113 | def GetCareMap(which, imgname): | 
| Tao Bao | 63c18fe | 2018-02-21 23:45:43 -0800 | [diff] [blame] | 114 | """Returns the care_map string for the given partition. | 
|  | 115 |  | 
|  | 116 | Args: | 
|  | 117 | which: The partition name, must be listed in PARTITIONS_WITH_CARE_MAP. | 
|  | 118 | imgname: The filename of the image. | 
|  | 119 |  | 
|  | 120 | Returns: | 
|  | 121 | (which, care_map_ranges): care_map_ranges is the raw string of the care_map | 
| Tao Bao | 77a4aa0 | 2019-05-01 22:08:43 -0700 | [diff] [blame] | 122 | RangeSet; or None. | 
| Tao Bao | 63c18fe | 2018-02-21 23:45:43 -0800 | [diff] [blame] | 123 | """ | 
| Tianjie Xu | 861f413 | 2018-09-12 11:49:33 -0700 | [diff] [blame] | 124 | assert which in common.PARTITIONS_WITH_CARE_MAP | 
| Tianjie Xu | cfa8622 | 2016-03-07 16:31:19 -0800 | [diff] [blame] | 125 |  | 
| Tao Bao | 8bdfb99 | 2019-04-25 00:42:33 -0700 | [diff] [blame] | 126 | # which + "_image_size" contains the size that the actual filesystem image | 
|  | 127 | # resides in, which is all that needs to be verified. The additional blocks in | 
|  | 128 | # the image file contain verity metadata, by reading which would trigger | 
|  | 129 | # invalid reads. | 
|  | 130 | image_size = OPTIONS.info_dict.get(which + "_image_size") | 
|  | 131 | if not image_size: | 
| Tao Bao | 77a4aa0 | 2019-05-01 22:08:43 -0700 | [diff] [blame] | 132 | return None | 
| Tao Bao | 8bdfb99 | 2019-04-25 00:42:33 -0700 | [diff] [blame] | 133 |  | 
| Tao Bao | a370545 | 2019-06-24 15:33:41 -0700 | [diff] [blame] | 134 | image_blocks = int(image_size) // 4096 - 1 | 
| Tao Bao | 8bdfb99 | 2019-04-25 00:42:33 -0700 | [diff] [blame] | 135 | assert image_blocks > 0, "blocks for {} must be positive".format(which) | 
|  | 136 |  | 
|  | 137 | # For sparse images, we will only check the blocks that are listed in the care | 
|  | 138 | # map, i.e. the ones with meaningful data. | 
|  | 139 | if "extfs_sparse_flag" in OPTIONS.info_dict: | 
|  | 140 | simg = sparse_img.SparseImage(imgname) | 
|  | 141 | care_map_ranges = simg.care_map.intersect( | 
| Tao Bao | 35f4ebc | 2018-09-27 15:31:11 -0700 | [diff] [blame] | 142 | rangelib.RangeSet("0-{}".format(image_blocks))) | 
| Tianjie Xu | f1a1318 | 2017-01-19 17:39:30 -0800 | [diff] [blame] | 143 |  | 
| Tao Bao | 8bdfb99 | 2019-04-25 00:42:33 -0700 | [diff] [blame] | 144 | # Otherwise for non-sparse images, we read all the blocks in the filesystem | 
|  | 145 | # image. | 
|  | 146 | else: | 
|  | 147 | care_map_ranges = rangelib.RangeSet("0-{}".format(image_blocks)) | 
|  | 148 |  | 
| Tao Bao | a2ff4c9 | 2018-01-17 12:14:43 -0800 | [diff] [blame] | 149 | return [which, care_map_ranges.to_string_raw()] | 
| Tianjie Xu | cfa8622 | 2016-03-07 16:31:19 -0800 | [diff] [blame] | 150 |  | 
|  | 151 |  | 
| Tao Bao | 886d883 | 2018-02-27 11:46:19 -0800 | [diff] [blame] | 152 | def AddSystem(output_zip, recovery_img=None, boot_img=None): | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 153 | """Turn the contents of SYSTEM into a system image and store it in | 
| David Zeuthen | d995f4b | 2016-01-29 16:59:17 -0500 | [diff] [blame] | 154 | output_zip. Returns the name of the system image file.""" | 
| Michael Runge | 2e0d8fc | 2014-11-13 21:41:08 -0800 | [diff] [blame] | 155 |  | 
| Tao Bao | 886d883 | 2018-02-27 11:46:19 -0800 | [diff] [blame] | 156 | img = OutputFile(output_zip, OPTIONS.input_tmp, "IMAGES", "system.img") | 
| Tao Bao | 93e7ebe | 2019-01-13 23:23:01 -0800 | [diff] [blame] | 157 | if os.path.exists(img.name): | 
| Tao Bao | 32fcdab | 2018-10-12 10:30:39 -0700 | [diff] [blame] | 158 | logger.info("system.img already exists; no need to rebuild...") | 
| Tao Bao | 93e7ebe | 2019-01-13 23:23:01 -0800 | [diff] [blame] | 159 | return img.name | 
| Michael Runge | 2e0d8fc | 2014-11-13 21:41:08 -0800 | [diff] [blame] | 160 |  | 
|  | 161 | def output_sink(fn, data): | 
| Tao Bao | a370545 | 2019-06-24 15:33:41 -0700 | [diff] [blame] | 162 | output_file = os.path.join(OPTIONS.input_tmp, "SYSTEM", fn) | 
|  | 163 | with open(output_file, "wb") as ofile: | 
|  | 164 | ofile.write(data) | 
| Michael Runge | 2e0d8fc | 2014-11-13 21:41:08 -0800 | [diff] [blame] | 165 |  | 
| Daniel Norman | a4911da | 2019-03-15 14:36:21 -0700 | [diff] [blame] | 166 | if output_zip: | 
|  | 167 | arc_name = "SYSTEM/" + fn | 
|  | 168 | if arc_name in output_zip.namelist(): | 
|  | 169 | OPTIONS.replace_updated_files_list.append(arc_name) | 
|  | 170 | else: | 
| Tao Bao | a370545 | 2019-06-24 15:33:41 -0700 | [diff] [blame] | 171 | common.ZipWrite(output_zip, output_file, arc_name) | 
| Tianjie Xu | 38af07f | 2017-05-25 17:38:53 -0700 | [diff] [blame] | 172 |  | 
| Bill Peckham | e868aec | 2019-09-17 17:06:47 -0700 | [diff] [blame] | 173 | board_uses_vendorimage = OPTIONS.info_dict.get( | 
|  | 174 | "board_uses_vendorimage") == "true" | 
|  | 175 |  | 
|  | 176 | if (OPTIONS.rebuild_recovery and not board_uses_vendorimage and | 
|  | 177 | recovery_img is not None and boot_img is not None): | 
|  | 178 | logger.info("Building new recovery patch on system at system/vendor") | 
| Dan Albert | 8b72aef | 2015-03-23 19:13:21 -0700 | [diff] [blame] | 179 | common.MakeRecoveryPatch(OPTIONS.input_tmp, output_sink, recovery_img, | 
|  | 180 | boot_img, info_dict=OPTIONS.info_dict) | 
| Michael Runge | 2e0d8fc | 2014-11-13 21:41:08 -0800 | [diff] [blame] | 181 |  | 
| Tao Bao | 886d883 | 2018-02-27 11:46:19 -0800 | [diff] [blame] | 182 | block_list = OutputFile(output_zip, OPTIONS.input_tmp, "IMAGES", "system.map") | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 183 | CreateImage(OPTIONS.input_tmp, OPTIONS.info_dict, "system", img, | 
|  | 184 | block_list=block_list) | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 185 | return img.name | 
| Doug Zongker | fc44a51 | 2014-08-26 13:10:25 -0700 | [diff] [blame] | 186 |  | 
|  | 187 |  | 
| Tao Bao | 886d883 | 2018-02-27 11:46:19 -0800 | [diff] [blame] | 188 | def AddSystemOther(output_zip): | 
| Alex Light | 4e358ab | 2016-06-16 14:47:10 -0700 | [diff] [blame] | 189 | """Turn the contents of SYSTEM_OTHER into a system_other image | 
|  | 190 | and store it in output_zip.""" | 
|  | 191 |  | 
| Tao Bao | 886d883 | 2018-02-27 11:46:19 -0800 | [diff] [blame] | 192 | img = OutputFile(output_zip, OPTIONS.input_tmp, "IMAGES", "system_other.img") | 
| Tao Bao | 93e7ebe | 2019-01-13 23:23:01 -0800 | [diff] [blame] | 193 | if os.path.exists(img.name): | 
| Tao Bao | 32fcdab | 2018-10-12 10:30:39 -0700 | [diff] [blame] | 194 | logger.info("system_other.img already exists; no need to rebuild...") | 
| Alex Light | 4e358ab | 2016-06-16 14:47:10 -0700 | [diff] [blame] | 195 | return | 
|  | 196 |  | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 197 | CreateImage(OPTIONS.input_tmp, OPTIONS.info_dict, "system_other", img) | 
| Alex Light | 4e358ab | 2016-06-16 14:47:10 -0700 | [diff] [blame] | 198 |  | 
|  | 199 |  | 
| Bill Peckham | e868aec | 2019-09-17 17:06:47 -0700 | [diff] [blame] | 200 | def AddVendor(output_zip, recovery_img=None, boot_img=None): | 
| Doug Zongker | fc44a51 | 2014-08-26 13:10:25 -0700 | [diff] [blame] | 201 | """Turn the contents of VENDOR into a vendor image and store in it | 
|  | 202 | output_zip.""" | 
| Michael Runge | 2e0d8fc | 2014-11-13 21:41:08 -0800 | [diff] [blame] | 203 |  | 
| Tao Bao | 886d883 | 2018-02-27 11:46:19 -0800 | [diff] [blame] | 204 | img = OutputFile(output_zip, OPTIONS.input_tmp, "IMAGES", "vendor.img") | 
| Tao Bao | 93e7ebe | 2019-01-13 23:23:01 -0800 | [diff] [blame] | 205 | if os.path.exists(img.name): | 
| Tao Bao | 32fcdab | 2018-10-12 10:30:39 -0700 | [diff] [blame] | 206 | logger.info("vendor.img already exists; no need to rebuild...") | 
| Tao Bao | 93e7ebe | 2019-01-13 23:23:01 -0800 | [diff] [blame] | 207 | return img.name | 
| Michael Runge | 2e0d8fc | 2014-11-13 21:41:08 -0800 | [diff] [blame] | 208 |  | 
| Bill Peckham | e868aec | 2019-09-17 17:06:47 -0700 | [diff] [blame] | 209 | def output_sink(fn, data): | 
|  | 210 | ofile = open(os.path.join(OPTIONS.input_tmp, "VENDOR", fn), "w") | 
|  | 211 | ofile.write(data) | 
|  | 212 | ofile.close() | 
|  | 213 |  | 
|  | 214 | if output_zip: | 
|  | 215 | arc_name = "VENDOR/" + fn | 
|  | 216 | if arc_name in output_zip.namelist(): | 
|  | 217 | OPTIONS.replace_updated_files_list.append(arc_name) | 
|  | 218 | else: | 
|  | 219 | common.ZipWrite(output_zip, ofile.name, arc_name) | 
|  | 220 |  | 
|  | 221 | board_uses_vendorimage = OPTIONS.info_dict.get( | 
|  | 222 | "board_uses_vendorimage") == "true" | 
|  | 223 |  | 
|  | 224 | if (OPTIONS.rebuild_recovery and board_uses_vendorimage and | 
|  | 225 | recovery_img is not None and boot_img is not None): | 
|  | 226 | logger.info("Building new recovery patch on vendor") | 
|  | 227 | common.MakeRecoveryPatch(OPTIONS.input_tmp, output_sink, recovery_img, | 
|  | 228 | boot_img, info_dict=OPTIONS.info_dict) | 
|  | 229 |  | 
| Tao Bao | 886d883 | 2018-02-27 11:46:19 -0800 | [diff] [blame] | 230 | block_list = OutputFile(output_zip, OPTIONS.input_tmp, "IMAGES", "vendor.map") | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 231 | CreateImage(OPTIONS.input_tmp, OPTIONS.info_dict, "vendor", img, | 
|  | 232 | block_list=block_list) | 
|  | 233 | return img.name | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 234 |  | 
| Yueyao Zhu | 889ee5e | 2017-05-12 17:50:46 -0700 | [diff] [blame] | 235 |  | 
| Tao Bao | 886d883 | 2018-02-27 11:46:19 -0800 | [diff] [blame] | 236 | def AddProduct(output_zip): | 
|  | 237 | """Turn the contents of PRODUCT into a product image and store it in | 
|  | 238 | output_zip.""" | 
| Jaekyun Seok | b7735d8 | 2017-11-27 17:04:47 +0900 | [diff] [blame] | 239 |  | 
| Tao Bao | 886d883 | 2018-02-27 11:46:19 -0800 | [diff] [blame] | 240 | img = OutputFile(output_zip, OPTIONS.input_tmp, "IMAGES", "product.img") | 
| Tao Bao | 93e7ebe | 2019-01-13 23:23:01 -0800 | [diff] [blame] | 241 | if os.path.exists(img.name): | 
| Tao Bao | 32fcdab | 2018-10-12 10:30:39 -0700 | [diff] [blame] | 242 | logger.info("product.img already exists; no need to rebuild...") | 
| Tao Bao | 93e7ebe | 2019-01-13 23:23:01 -0800 | [diff] [blame] | 243 | return img.name | 
| Jaekyun Seok | b7735d8 | 2017-11-27 17:04:47 +0900 | [diff] [blame] | 244 |  | 
| Tao Bao | 886d883 | 2018-02-27 11:46:19 -0800 | [diff] [blame] | 245 | block_list = OutputFile( | 
|  | 246 | output_zip, OPTIONS.input_tmp, "IMAGES", "product.map") | 
|  | 247 | CreateImage( | 
|  | 248 | OPTIONS.input_tmp, OPTIONS.info_dict, "product", img, | 
|  | 249 | block_list=block_list) | 
| Jaekyun Seok | b7735d8 | 2017-11-27 17:04:47 +0900 | [diff] [blame] | 250 | return img.name | 
|  | 251 |  | 
|  | 252 |  | 
| Justin Yun | 6151e3f | 2019-06-25 15:58:13 +0900 | [diff] [blame] | 253 | def AddSystemExt(output_zip): | 
|  | 254 | """Turn the contents of SYSTEM_EXT into a system_ext image and store it in | 
|  | 255 | output_zip.""" | 
| Dario Freni | 5f681e1 | 2018-05-29 13:09:01 +0100 | [diff] [blame] | 256 |  | 
|  | 257 | img = OutputFile(output_zip, OPTIONS.input_tmp, "IMAGES", | 
| Justin Yun | 6151e3f | 2019-06-25 15:58:13 +0900 | [diff] [blame] | 258 | "system_ext.img") | 
| Tao Bao | 93e7ebe | 2019-01-13 23:23:01 -0800 | [diff] [blame] | 259 | if os.path.exists(img.name): | 
| Justin Yun | 6151e3f | 2019-06-25 15:58:13 +0900 | [diff] [blame] | 260 | logger.info("system_ext.img already exists; no need to rebuild...") | 
| Tao Bao | 93e7ebe | 2019-01-13 23:23:01 -0800 | [diff] [blame] | 261 | return img.name | 
| Dario Freni | 5f681e1 | 2018-05-29 13:09:01 +0100 | [diff] [blame] | 262 |  | 
|  | 263 | block_list = OutputFile( | 
| Justin Yun | 6151e3f | 2019-06-25 15:58:13 +0900 | [diff] [blame] | 264 | output_zip, OPTIONS.input_tmp, "IMAGES", "system_ext.map") | 
| Dario Freni | 5f681e1 | 2018-05-29 13:09:01 +0100 | [diff] [blame] | 265 | CreateImage( | 
| Justin Yun | 6151e3f | 2019-06-25 15:58:13 +0900 | [diff] [blame] | 266 | OPTIONS.input_tmp, OPTIONS.info_dict, "system_ext", img, | 
| Dario Freni | 5f681e1 | 2018-05-29 13:09:01 +0100 | [diff] [blame] | 267 | block_list=block_list) | 
|  | 268 | return img.name | 
|  | 269 |  | 
|  | 270 |  | 
| Bowgo Tsai | d624fa6 | 2017-11-14 23:42:30 +0800 | [diff] [blame] | 271 | def AddOdm(output_zip): | 
|  | 272 | """Turn the contents of ODM into an odm image and store it in output_zip.""" | 
|  | 273 |  | 
|  | 274 | img = OutputFile(output_zip, OPTIONS.input_tmp, "IMAGES", "odm.img") | 
| Tao Bao | 93e7ebe | 2019-01-13 23:23:01 -0800 | [diff] [blame] | 275 | if os.path.exists(img.name): | 
| Tao Bao | 32fcdab | 2018-10-12 10:30:39 -0700 | [diff] [blame] | 276 | logger.info("odm.img already exists; no need to rebuild...") | 
| Tao Bao | 93e7ebe | 2019-01-13 23:23:01 -0800 | [diff] [blame] | 277 | return img.name | 
| Bowgo Tsai | d624fa6 | 2017-11-14 23:42:30 +0800 | [diff] [blame] | 278 |  | 
|  | 279 | block_list = OutputFile( | 
|  | 280 | output_zip, OPTIONS.input_tmp, "IMAGES", "odm.map") | 
|  | 281 | CreateImage( | 
|  | 282 | OPTIONS.input_tmp, OPTIONS.info_dict, "odm", img, | 
|  | 283 | block_list=block_list) | 
|  | 284 | return img.name | 
|  | 285 |  | 
|  | 286 |  | 
| Yifan Hong | cfb917a | 2020-05-07 14:58:20 -0700 | [diff] [blame] | 287 | def AddVendorDlkm(output_zip): | 
|  | 288 | """Turn the contents of VENDOR_DLKM into an vendor_dlkm image and store it in output_zip.""" | 
|  | 289 |  | 
|  | 290 | img = OutputFile(output_zip, OPTIONS.input_tmp, "IMAGES", "vendor_dlkm.img") | 
|  | 291 | if os.path.exists(img.name): | 
|  | 292 | logger.info("vendor_dlkm.img already exists; no need to rebuild...") | 
|  | 293 | return img.name | 
|  | 294 |  | 
|  | 295 | block_list = OutputFile( | 
|  | 296 | output_zip, OPTIONS.input_tmp, "IMAGES", "vendor_dlkm.map") | 
|  | 297 | CreateImage( | 
|  | 298 | OPTIONS.input_tmp, OPTIONS.info_dict, "vendor_dlkm", img, | 
|  | 299 | block_list=block_list) | 
|  | 300 | return img.name | 
|  | 301 |  | 
| Yifan Hong | f496f1b | 2020-07-15 16:52:59 -0700 | [diff] [blame] | 302 | def AddOdmDlkm(output_zip): | 
|  | 303 | """Turn the contents of OdmDlkm into an odm_dlkm image and store it in output_zip.""" | 
|  | 304 |  | 
|  | 305 | img = OutputFile(output_zip, OPTIONS.input_tmp, "IMAGES", "odm_dlkm.img") | 
|  | 306 | if os.path.exists(img.name): | 
|  | 307 | logger.info("odm_dlkm.img already exists; no need to rebuild...") | 
|  | 308 | return img.name | 
|  | 309 |  | 
|  | 310 | block_list = OutputFile( | 
|  | 311 | output_zip, OPTIONS.input_tmp, "IMAGES", "odm_dlkm.map") | 
|  | 312 | CreateImage( | 
|  | 313 | OPTIONS.input_tmp, OPTIONS.info_dict, "odm_dlkm", img, | 
|  | 314 | block_list=block_list) | 
|  | 315 | return img.name | 
|  | 316 |  | 
| Yifan Hong | cfb917a | 2020-05-07 14:58:20 -0700 | [diff] [blame] | 317 |  | 
| Tao Bao | 886d883 | 2018-02-27 11:46:19 -0800 | [diff] [blame] | 318 | def AddDtbo(output_zip): | 
| Tao Bao | c633ed0 | 2017-05-30 21:46:33 -0700 | [diff] [blame] | 319 | """Adds the DTBO image. | 
|  | 320 |  | 
| Tao Bao | 886d883 | 2018-02-27 11:46:19 -0800 | [diff] [blame] | 321 | Uses the image under IMAGES/ if it already exists. Otherwise looks for the | 
| Tao Bao | c633ed0 | 2017-05-30 21:46:33 -0700 | [diff] [blame] | 322 | image under PREBUILT_IMAGES/, signs it as needed, and returns the image name. | 
|  | 323 | """ | 
| Tao Bao | 886d883 | 2018-02-27 11:46:19 -0800 | [diff] [blame] | 324 | img = OutputFile(output_zip, OPTIONS.input_tmp, "IMAGES", "dtbo.img") | 
| Tao Bao | 93e7ebe | 2019-01-13 23:23:01 -0800 | [diff] [blame] | 325 | if os.path.exists(img.name): | 
| Tao Bao | 32fcdab | 2018-10-12 10:30:39 -0700 | [diff] [blame] | 326 | logger.info("dtbo.img already exists; no need to rebuild...") | 
| Tao Bao | 93e7ebe | 2019-01-13 23:23:01 -0800 | [diff] [blame] | 327 | return img.name | 
| Tao Bao | c633ed0 | 2017-05-30 21:46:33 -0700 | [diff] [blame] | 328 |  | 
|  | 329 | dtbo_prebuilt_path = os.path.join( | 
|  | 330 | OPTIONS.input_tmp, "PREBUILT_IMAGES", "dtbo.img") | 
|  | 331 | assert os.path.exists(dtbo_prebuilt_path) | 
|  | 332 | shutil.copy(dtbo_prebuilt_path, img.name) | 
|  | 333 |  | 
|  | 334 | # AVB-sign the image as needed. | 
| Bowgo Tsai | 3e599ea | 2017-05-26 18:30:04 +0800 | [diff] [blame] | 335 | if OPTIONS.info_dict.get("avb_enable") == "true": | 
| Rupert Shuttleworth | 7294274 | 2020-12-08 06:18:35 +0000 | [diff] [blame] | 336 | # Signing requires +w | 
|  | 337 | os.chmod(img.name, os.stat(img.name).st_mode | stat.S_IWUSR) | 
|  | 338 |  | 
| Tao Bao | f88e0ce | 2019-03-18 14:01:38 -0700 | [diff] [blame] | 339 | avbtool = OPTIONS.info_dict["avb_avbtool"] | 
| Tao Bao | 3ebfdde | 2017-05-23 23:06:55 -0700 | [diff] [blame] | 340 | part_size = OPTIONS.info_dict["dtbo_size"] | 
| Tao Bao | c633ed0 | 2017-05-30 21:46:33 -0700 | [diff] [blame] | 341 | # The AVB hash footer will be replaced if already present. | 
|  | 342 | cmd = [avbtool, "add_hash_footer", "--image", img.name, | 
|  | 343 | "--partition_size", str(part_size), "--partition_name", "dtbo"] | 
| Bowgo Tsai | 3e599ea | 2017-05-26 18:30:04 +0800 | [diff] [blame] | 344 | common.AppendAVBSigningArgs(cmd, "dtbo") | 
|  | 345 | args = OPTIONS.info_dict.get("avb_dtbo_add_hash_footer_args") | 
| Tao Bao | c633ed0 | 2017-05-30 21:46:33 -0700 | [diff] [blame] | 346 | if args and args.strip(): | 
|  | 347 | cmd.extend(shlex.split(args)) | 
| Tao Bao | 2764aee | 2018-11-21 11:02:48 -0800 | [diff] [blame] | 348 | common.RunAndCheckOutput(cmd) | 
| Tao Bao | c633ed0 | 2017-05-30 21:46:33 -0700 | [diff] [blame] | 349 |  | 
|  | 350 | img.Write() | 
|  | 351 | return img.name | 
|  | 352 |  | 
| Hongguang Chen | f23364d | 2020-04-27 18:36:36 -0700 | [diff] [blame] | 353 | def AddCustomImages(output_zip, partition_name): | 
|  | 354 | """Adds and signs custom images in IMAGES/. | 
|  | 355 |  | 
|  | 356 | Args: | 
|  | 357 | output_zip: The output zip file (needs to be already open), or None to | 
|  | 358 | write images to OPTIONS.input_tmp/. | 
|  | 359 |  | 
|  | 360 | Uses the image under IMAGES/ if it already exists. Otherwise looks for the | 
|  | 361 | image under PREBUILT_IMAGES/, signs it as needed, and returns the image name. | 
|  | 362 |  | 
|  | 363 | Raises: | 
|  | 364 | AssertionError: If image can't be found. | 
|  | 365 | """ | 
|  | 366 |  | 
|  | 367 | partition_size = OPTIONS.info_dict.get( | 
|  | 368 | "avb_{}_partition_size".format(partition_name)) | 
|  | 369 | key_path = OPTIONS.info_dict.get("avb_{}_key_path".format(partition_name)) | 
|  | 370 | algorithm = OPTIONS.info_dict.get("avb_{}_algorithm".format(partition_name)) | 
|  | 371 | extra_args = OPTIONS.info_dict.get( | 
|  | 372 | "avb_{}_add_hashtree_footer_args".format(partition_name)) | 
|  | 373 | partition_size = OPTIONS.info_dict.get( | 
|  | 374 | "avb_{}_partition_size".format(partition_name)) | 
|  | 375 |  | 
|  | 376 | builder = verity_utils.CreateCustomImageBuilder( | 
|  | 377 | OPTIONS.info_dict, partition_name, partition_size, | 
|  | 378 | key_path, algorithm, extra_args) | 
|  | 379 |  | 
|  | 380 | for img_name in OPTIONS.info_dict.get( | 
|  | 381 | "avb_{}_image_list".format(partition_name)).split(): | 
|  | 382 | custom_image = OutputFile(output_zip, OPTIONS.input_tmp, "IMAGES", img_name) | 
|  | 383 | if os.path.exists(custom_image.name): | 
|  | 384 | continue | 
|  | 385 |  | 
|  | 386 | custom_image_prebuilt_path = os.path.join( | 
|  | 387 | OPTIONS.input_tmp, "PREBUILT_IMAGES", img_name) | 
|  | 388 | assert os.path.exists(custom_image_prebuilt_path), \ | 
|  | 389 | "Failed to find %s at %s" % (img_name, custom_image_prebuilt_path) | 
|  | 390 |  | 
|  | 391 | shutil.copy(custom_image_prebuilt_path, custom_image.name) | 
|  | 392 |  | 
|  | 393 | if builder is not None: | 
|  | 394 | builder.Build(custom_image.name) | 
|  | 395 |  | 
|  | 396 | custom_image.Write() | 
|  | 397 |  | 
|  | 398 | default = os.path.join(OPTIONS.input_tmp, "IMAGES", partition_name + ".img") | 
|  | 399 | assert os.path.exists(default), \ | 
|  | 400 | "There should be one %s.img" % (partition_name) | 
|  | 401 | return default | 
|  | 402 |  | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 403 |  | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 404 | def CreateImage(input_dir, info_dict, what, output_file, block_list=None): | 
| Tao Bao | a370545 | 2019-06-24 15:33:41 -0700 | [diff] [blame] | 405 | logger.info("creating %s.img...", what) | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 406 |  | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 407 | image_props = build_image.ImagePropFromGlobalDict(info_dict, what) | 
| Bryan Henry | e6d547d | 2018-07-31 18:32:00 -0700 | [diff] [blame] | 408 | image_props["timestamp"] = FIXED_FILE_TIMESTAMP | 
| Tao Bao | 822f584 | 2015-09-30 16:01:14 -0700 | [diff] [blame] | 409 |  | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 410 | if what == "system": | 
|  | 411 | fs_config_prefix = "" | 
|  | 412 | else: | 
|  | 413 | fs_config_prefix = what + "_" | 
|  | 414 |  | 
|  | 415 | fs_config = os.path.join( | 
|  | 416 | input_dir, "META/" + fs_config_prefix + "filesystem_config.txt") | 
| Dan Albert | 8b72aef | 2015-03-23 19:13:21 -0700 | [diff] [blame] | 417 | if not os.path.exists(fs_config): | 
|  | 418 | fs_config = None | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 419 |  | 
| Ying Wang | a2292c9 | 2015-03-24 19:07:40 -0700 | [diff] [blame] | 420 | # Override values loaded from info_dict. | 
|  | 421 | if fs_config: | 
|  | 422 | image_props["fs_config"] = fs_config | 
| Ying Wang | a2292c9 | 2015-03-24 19:07:40 -0700 | [diff] [blame] | 423 | if block_list: | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 424 | image_props["block_list"] = block_list.name | 
| Ying Wang | a2292c9 | 2015-03-24 19:07:40 -0700 | [diff] [blame] | 425 |  | 
| Tao Bao | d86e311 | 2017-09-22 15:45:33 -0700 | [diff] [blame] | 426 | # Use repeatable ext4 FS UUID and hash_seed UUID (based on partition name and | 
|  | 427 | # build fingerprint). | 
| Tao Bao | 3ed35d3 | 2019-10-07 20:48:48 -0700 | [diff] [blame] | 428 | build_info = common.BuildInfo(info_dict) | 
| Yifan Hong | c08cbf0 | 2020-09-15 19:07:39 +0000 | [diff] [blame] | 429 | uuid_seed = what + "-" + build_info.GetPartitionFingerprint(what) | 
| Tao Bao | d86e311 | 2017-09-22 15:45:33 -0700 | [diff] [blame] | 430 | image_props["uuid"] = str(uuid.uuid5(uuid.NAMESPACE_URL, uuid_seed)) | 
|  | 431 | hash_seed = "hash_seed-" + uuid_seed | 
|  | 432 | image_props["hash_seed"] = str(uuid.uuid5(uuid.NAMESPACE_URL, hash_seed)) | 
|  | 433 |  | 
| Tao Bao | c6bd70a | 2018-09-27 16:58:00 -0700 | [diff] [blame] | 434 | build_image.BuildImage( | 
|  | 435 | os.path.join(input_dir, what.upper()), image_props, output_file.name) | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 436 |  | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 437 | output_file.Write() | 
|  | 438 | if block_list: | 
|  | 439 | block_list.Write() | 
|  | 440 |  | 
| Shashikant Baviskar | 16a7389 | 2019-02-07 10:57:21 +0900 | [diff] [blame] | 441 | # Set the '_image_size' for given image size. | 
| Tianjie Xu | f1a1318 | 2017-01-19 17:39:30 -0800 | [diff] [blame] | 442 | is_verity_partition = "verity_block_device" in image_props | 
| Tianjie Xu | 6b2e155 | 2017-06-01 11:32:32 -0700 | [diff] [blame] | 443 | verity_supported = (image_props.get("verity") == "true" or | 
| Bowgo Tsai | 3e599ea | 2017-05-26 18:30:04 +0800 | [diff] [blame] | 444 | image_props.get("avb_enable") == "true") | 
| Tianjie Xu | 6b2e155 | 2017-06-01 11:32:32 -0700 | [diff] [blame] | 445 | is_avb_enable = image_props.get("avb_hashtree_enable") == "true" | 
|  | 446 | if verity_supported and (is_verity_partition or is_avb_enable): | 
| Tao Bao | 35f4ebc | 2018-09-27 15:31:11 -0700 | [diff] [blame] | 447 | image_size = image_props.get("image_size") | 
|  | 448 | if image_size: | 
| Shashikant Baviskar | 16a7389 | 2019-02-07 10:57:21 +0900 | [diff] [blame] | 449 | image_size_key = what + "_image_size" | 
|  | 450 | info_dict[image_size_key] = int(image_size) | 
| Tianjie Xu | f1a1318 | 2017-01-19 17:39:30 -0800 | [diff] [blame] | 451 |  | 
| Yifan Hong | c767f7c | 2018-11-08 15:41:24 -0800 | [diff] [blame] | 452 | use_dynamic_size = ( | 
| Tao Bao | 2764aee | 2018-11-21 11:02:48 -0800 | [diff] [blame] | 453 | info_dict.get("use_dynamic_partition_size") == "true" and | 
|  | 454 | what in shlex.split(info_dict.get("dynamic_partition_list", "").strip())) | 
| Yifan Hong | c767f7c | 2018-11-08 15:41:24 -0800 | [diff] [blame] | 455 | if use_dynamic_size: | 
|  | 456 | info_dict.update(build_image.GlobalDictFromImageProp(image_props, what)) | 
|  | 457 |  | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 458 |  | 
| Tao Bao | 886d883 | 2018-02-27 11:46:19 -0800 | [diff] [blame] | 459 | def AddUserdata(output_zip): | 
| Ying Wang | 2a04839 | 2015-06-25 13:56:53 -0700 | [diff] [blame] | 460 | """Create a userdata image and store it in output_zip. | 
|  | 461 |  | 
|  | 462 | In most case we just create and store an empty userdata.img; | 
|  | 463 | But the invoker can also request to create userdata.img with real | 
|  | 464 | data from the target files, by setting "userdata_img_with_data=true" | 
|  | 465 | in OPTIONS.info_dict. | 
|  | 466 | """ | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 467 |  | 
| Tao Bao | 886d883 | 2018-02-27 11:46:19 -0800 | [diff] [blame] | 468 | img = OutputFile(output_zip, OPTIONS.input_tmp, "IMAGES", "userdata.img") | 
| Tao Bao | 93e7ebe | 2019-01-13 23:23:01 -0800 | [diff] [blame] | 469 | if os.path.exists(img.name): | 
| Tao Bao | 32fcdab | 2018-10-12 10:30:39 -0700 | [diff] [blame] | 470 | logger.info("userdata.img already exists; no need to rebuild...") | 
| Michael Runge | 2e0d8fc | 2014-11-13 21:41:08 -0800 | [diff] [blame] | 471 | return | 
|  | 472 |  | 
| Elliott Hughes | 305b088 | 2016-06-15 17:04:54 -0700 | [diff] [blame] | 473 | # Skip userdata.img if no size. | 
| Tao Bao | 2c15d9e | 2015-07-09 11:51:16 -0700 | [diff] [blame] | 474 | image_props = build_image.ImagePropFromGlobalDict(OPTIONS.info_dict, "data") | 
| Elliott Hughes | 305b088 | 2016-06-15 17:04:54 -0700 | [diff] [blame] | 475 | if not image_props.get("partition_size"): | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 476 | return | 
|  | 477 |  | 
| Tao Bao | 32fcdab | 2018-10-12 10:30:39 -0700 | [diff] [blame] | 478 | logger.info("creating userdata.img...") | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 479 |  | 
| Bryan Henry | e6d547d | 2018-07-31 18:32:00 -0700 | [diff] [blame] | 480 | image_props["timestamp"] = FIXED_FILE_TIMESTAMP | 
| Tao Bao | 822f584 | 2015-09-30 16:01:14 -0700 | [diff] [blame] | 481 |  | 
| Tao Bao | fa863c8 | 2017-05-23 23:49:03 -0700 | [diff] [blame] | 482 | if OPTIONS.info_dict.get("userdata_img_with_data") == "true": | 
|  | 483 | user_dir = os.path.join(OPTIONS.input_tmp, "DATA") | 
| Ying Wang | 2a04839 | 2015-06-25 13:56:53 -0700 | [diff] [blame] | 484 | else: | 
| Tao Bao | 1c830bf | 2017-12-25 10:43:47 -0800 | [diff] [blame] | 485 | user_dir = common.MakeTempDir() | 
| Ying Wang | 2a04839 | 2015-06-25 13:56:53 -0700 | [diff] [blame] | 486 |  | 
| Tao Bao | c6bd70a | 2018-09-27 16:58:00 -0700 | [diff] [blame] | 487 | build_image.BuildImage(user_dir, image_props, img.name) | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 488 |  | 
|  | 489 | common.CheckSize(img.name, "userdata.img", OPTIONS.info_dict) | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 490 | img.Write() | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 491 |  | 
|  | 492 |  | 
| Tao Bao | 744c4c7 | 2018-08-20 21:09:07 -0700 | [diff] [blame] | 493 | def AddVBMeta(output_zip, partitions, name, needed_partitions): | 
|  | 494 | """Creates a VBMeta image and stores it in output_zip. | 
|  | 495 |  | 
|  | 496 | It generates the requested VBMeta image. The requested image could be for | 
|  | 497 | top-level or chained VBMeta image, which is determined based on the name. | 
| Tao Bao | bf70c31 | 2017-07-11 17:27:55 -0700 | [diff] [blame] | 498 |  | 
|  | 499 | Args: | 
|  | 500 | output_zip: The output zip file, which needs to be already open. | 
|  | 501 | partitions: A dict that's keyed by partition names with image paths as | 
| Hongguang Chen | f23364d | 2020-04-27 18:36:36 -0700 | [diff] [blame] | 502 | values. Only valid partition names are accepted, as partitions listed | 
|  | 503 | in common.AVB_PARTITIONS and custom partitions listed in | 
|  | 504 | OPTIONS.info_dict.get("avb_custom_images_partition_list") | 
| David Anderson | 7709ab2 | 2018-10-15 14:41:34 -0700 | [diff] [blame] | 505 | name: Name of the VBMeta partition, e.g. 'vbmeta', 'vbmeta_system'. | 
| Tao Bao | 744c4c7 | 2018-08-20 21:09:07 -0700 | [diff] [blame] | 506 | needed_partitions: Partitions whose descriptors should be included into the | 
|  | 507 | generated VBMeta image. | 
|  | 508 |  | 
| Tao Bao | 7106420 | 2018-10-22 15:08:02 -0700 | [diff] [blame] | 509 | Returns: | 
|  | 510 | Path to the created image. | 
|  | 511 |  | 
| Tao Bao | 744c4c7 | 2018-08-20 21:09:07 -0700 | [diff] [blame] | 512 | Raises: | 
|  | 513 | AssertionError: On invalid input args. | 
| Tao Bao | bf70c31 | 2017-07-11 17:27:55 -0700 | [diff] [blame] | 514 | """ | 
| Tao Bao | 744c4c7 | 2018-08-20 21:09:07 -0700 | [diff] [blame] | 515 | assert needed_partitions, "Needed partitions must be specified" | 
|  | 516 |  | 
|  | 517 | img = OutputFile( | 
|  | 518 | output_zip, OPTIONS.input_tmp, "IMAGES", "{}.img".format(name)) | 
| Tao Bao | 93e7ebe | 2019-01-13 23:23:01 -0800 | [diff] [blame] | 519 | if os.path.exists(img.name): | 
| Tao Bao | 32fcdab | 2018-10-12 10:30:39 -0700 | [diff] [blame] | 520 | logger.info("%s.img already exists; not rebuilding...", name) | 
| Tao Bao | 93e7ebe | 2019-01-13 23:23:01 -0800 | [diff] [blame] | 521 | return img.name | 
| Tao Bao | 262bf3f | 2017-07-11 17:27:55 -0700 | [diff] [blame] | 522 |  | 
| Daniel Norman | 276f062 | 2019-07-26 14:13:51 -0700 | [diff] [blame] | 523 | common.BuildVBMeta(img.name, partitions, name, needed_partitions) | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 524 | img.Write() | 
| Tao Bao | 7106420 | 2018-10-22 15:08:02 -0700 | [diff] [blame] | 525 | return img.name | 
| David Zeuthen | 2ce63ed | 2016-09-15 13:43:54 -0400 | [diff] [blame] | 526 |  | 
|  | 527 |  | 
| Tao Bao | 886d883 | 2018-02-27 11:46:19 -0800 | [diff] [blame] | 528 | def AddPartitionTable(output_zip): | 
| David Zeuthen | 2532862 | 2016-04-08 15:08:03 -0400 | [diff] [blame] | 529 | """Create a partition table image and store it in output_zip.""" | 
|  | 530 |  | 
| Tao Bao | 886d883 | 2018-02-27 11:46:19 -0800 | [diff] [blame] | 531 | img = OutputFile( | 
|  | 532 | output_zip, OPTIONS.input_tmp, "IMAGES", "partition-table.img") | 
|  | 533 | bpt = OutputFile( | 
| Bryan Henry | f130a23 | 2018-04-26 11:59:33 -0700 | [diff] [blame] | 534 | output_zip, OPTIONS.input_tmp, "META", "partition-table.bpt") | 
| David Zeuthen | 2532862 | 2016-04-08 15:08:03 -0400 | [diff] [blame] | 535 |  | 
|  | 536 | # use BPTTOOL from environ, or "bpttool" if empty or not set. | 
|  | 537 | bpttool = os.getenv("BPTTOOL") or "bpttool" | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 538 | cmd = [bpttool, "make_table", "--output_json", bpt.name, | 
|  | 539 | "--output_gpt", img.name] | 
| David Zeuthen | 2532862 | 2016-04-08 15:08:03 -0400 | [diff] [blame] | 540 | input_files_str = OPTIONS.info_dict["board_bpt_input_files"] | 
|  | 541 | input_files = input_files_str.split(" ") | 
|  | 542 | for i in input_files: | 
|  | 543 | cmd.extend(["--input", i]) | 
|  | 544 | disk_size = OPTIONS.info_dict.get("board_bpt_disk_size") | 
|  | 545 | if disk_size: | 
|  | 546 | cmd.extend(["--disk_size", disk_size]) | 
|  | 547 | args = OPTIONS.info_dict.get("board_bpt_make_table_args") | 
|  | 548 | if args: | 
|  | 549 | cmd.extend(shlex.split(args)) | 
| Tao Bao | 2764aee | 2018-11-21 11:02:48 -0800 | [diff] [blame] | 550 | common.RunAndCheckOutput(cmd) | 
| David Zeuthen | 2532862 | 2016-04-08 15:08:03 -0400 | [diff] [blame] | 551 |  | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 552 | img.Write() | 
|  | 553 | bpt.Write() | 
| David Zeuthen | 2532862 | 2016-04-08 15:08:03 -0400 | [diff] [blame] | 554 |  | 
|  | 555 |  | 
| Tao Bao | 886d883 | 2018-02-27 11:46:19 -0800 | [diff] [blame] | 556 | def AddCache(output_zip): | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 557 | """Create an empty cache image and store it in output_zip.""" | 
|  | 558 |  | 
| Tao Bao | 886d883 | 2018-02-27 11:46:19 -0800 | [diff] [blame] | 559 | img = OutputFile(output_zip, OPTIONS.input_tmp, "IMAGES", "cache.img") | 
| Tao Bao | 93e7ebe | 2019-01-13 23:23:01 -0800 | [diff] [blame] | 560 | if os.path.exists(img.name): | 
| Tao Bao | 32fcdab | 2018-10-12 10:30:39 -0700 | [diff] [blame] | 561 | logger.info("cache.img already exists; no need to rebuild...") | 
| Michael Runge | 2e0d8fc | 2014-11-13 21:41:08 -0800 | [diff] [blame] | 562 | return | 
|  | 563 |  | 
| Tao Bao | 2c15d9e | 2015-07-09 11:51:16 -0700 | [diff] [blame] | 564 | image_props = build_image.ImagePropFromGlobalDict(OPTIONS.info_dict, "cache") | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 565 | # The build system has to explicitly request for cache.img. | 
|  | 566 | if "fs_type" not in image_props: | 
|  | 567 | return | 
|  | 568 |  | 
| Tao Bao | 32fcdab | 2018-10-12 10:30:39 -0700 | [diff] [blame] | 569 | logger.info("creating cache.img...") | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 570 |  | 
| Bryan Henry | e6d547d | 2018-07-31 18:32:00 -0700 | [diff] [blame] | 571 | image_props["timestamp"] = FIXED_FILE_TIMESTAMP | 
| Tao Bao | 822f584 | 2015-09-30 16:01:14 -0700 | [diff] [blame] | 572 |  | 
| Tao Bao | 1c830bf | 2017-12-25 10:43:47 -0800 | [diff] [blame] | 573 | user_dir = common.MakeTempDir() | 
| Tao Bao | c6bd70a | 2018-09-27 16:58:00 -0700 | [diff] [blame] | 574 | build_image.BuildImage(user_dir, image_props, img.name) | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 575 |  | 
|  | 576 | common.CheckSize(img.name, "cache.img", OPTIONS.info_dict) | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 577 | img.Write() | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 578 |  | 
|  | 579 |  | 
| Tao Bao | 5277d10 | 2018-04-17 23:47:21 -0700 | [diff] [blame] | 580 | def CheckAbOtaImages(output_zip, ab_partitions): | 
|  | 581 | """Checks that all the listed A/B partitions have their images available. | 
| Tao Bao | bea20ac | 2018-01-17 17:57:49 -0800 | [diff] [blame] | 582 |  | 
| Tao Bao | 5277d10 | 2018-04-17 23:47:21 -0700 | [diff] [blame] | 583 | The images need to be available under IMAGES/ or RADIO/, with the former takes | 
|  | 584 | a priority. | 
| Tao Bao | bea20ac | 2018-01-17 17:57:49 -0800 | [diff] [blame] | 585 |  | 
|  | 586 | Args: | 
|  | 587 | output_zip: The output zip file (needs to be already open), or None to | 
| Tao Bao | 5277d10 | 2018-04-17 23:47:21 -0700 | [diff] [blame] | 588 | find images in OPTIONS.input_tmp/. | 
| Tao Bao | bea20ac | 2018-01-17 17:57:49 -0800 | [diff] [blame] | 589 | ab_partitions: The list of A/B partitions. | 
|  | 590 |  | 
|  | 591 | Raises: | 
|  | 592 | AssertionError: If it can't find an image. | 
|  | 593 | """ | 
|  | 594 | for partition in ab_partitions: | 
|  | 595 | img_name = partition.strip() + ".img" | 
| Tao Bao | bea20ac | 2018-01-17 17:57:49 -0800 | [diff] [blame] | 596 |  | 
| Tao Bao | a2ff4c9 | 2018-01-17 12:14:43 -0800 | [diff] [blame] | 597 | # Assert that the image is present under IMAGES/ now. | 
| Tao Bao | bea20ac | 2018-01-17 17:57:49 -0800 | [diff] [blame] | 598 | if output_zip: | 
|  | 599 | # Zip spec says: All slashes MUST be forward slashes. | 
| Tao Bao | 5277d10 | 2018-04-17 23:47:21 -0700 | [diff] [blame] | 600 | images_path = "IMAGES/" + img_name | 
|  | 601 | radio_path = "RADIO/" + img_name | 
|  | 602 | available = (images_path in output_zip.namelist() or | 
|  | 603 | radio_path in output_zip.namelist()) | 
| Tao Bao | bea20ac | 2018-01-17 17:57:49 -0800 | [diff] [blame] | 604 | else: | 
| Tao Bao | 5277d10 | 2018-04-17 23:47:21 -0700 | [diff] [blame] | 605 | images_path = os.path.join(OPTIONS.input_tmp, "IMAGES", img_name) | 
|  | 606 | radio_path = os.path.join(OPTIONS.input_tmp, "RADIO", img_name) | 
|  | 607 | available = os.path.exists(images_path) or os.path.exists(radio_path) | 
|  | 608 |  | 
|  | 609 | assert available, "Failed to find " + img_name | 
| Tao Bao | bea20ac | 2018-01-17 17:57:49 -0800 | [diff] [blame] | 610 |  | 
|  | 611 |  | 
| Tianjie Xu | 4c05f4a | 2018-09-14 16:24:41 -0700 | [diff] [blame] | 612 | def AddCareMapForAbOta(output_zip, ab_partitions, image_paths): | 
| Tianjie Xu | 861f413 | 2018-09-12 11:49:33 -0700 | [diff] [blame] | 613 | """Generates and adds care_map.pb for a/b partition that has care_map. | 
| Tao Bao | bea20ac | 2018-01-17 17:57:49 -0800 | [diff] [blame] | 614 |  | 
|  | 615 | Args: | 
|  | 616 | output_zip: The output zip file (needs to be already open), or None to | 
| Tianjie Xu | 4c05f4a | 2018-09-14 16:24:41 -0700 | [diff] [blame] | 617 | write care_map.pb to OPTIONS.input_tmp/. | 
| Tao Bao | bea20ac | 2018-01-17 17:57:49 -0800 | [diff] [blame] | 618 | ab_partitions: The list of A/B partitions. | 
|  | 619 | image_paths: A map from the partition name to the image path. | 
|  | 620 | """ | 
|  | 621 | care_map_list = [] | 
|  | 622 | for partition in ab_partitions: | 
|  | 623 | partition = partition.strip() | 
| Tianjie Xu | 861f413 | 2018-09-12 11:49:33 -0700 | [diff] [blame] | 624 | if partition not in common.PARTITIONS_WITH_CARE_MAP: | 
| Tao Bao | a2ff4c9 | 2018-01-17 12:14:43 -0800 | [diff] [blame] | 625 | continue | 
|  | 626 |  | 
|  | 627 | verity_block_device = "{}_verity_block_device".format(partition) | 
|  | 628 | avb_hashtree_enable = "avb_{}_hashtree_enable".format(partition) | 
|  | 629 | if (verity_block_device in OPTIONS.info_dict or | 
|  | 630 | OPTIONS.info_dict.get(avb_hashtree_enable) == "true"): | 
|  | 631 | image_path = image_paths[partition] | 
|  | 632 | assert os.path.exists(image_path) | 
| Tao Bao | 77a4aa0 | 2019-05-01 22:08:43 -0700 | [diff] [blame] | 633 |  | 
|  | 634 | care_map = GetCareMap(partition, image_path) | 
|  | 635 | if not care_map: | 
|  | 636 | continue | 
|  | 637 | care_map_list += care_map | 
| Tao Bao | bea20ac | 2018-01-17 17:57:49 -0800 | [diff] [blame] | 638 |  | 
| Tianjie Xu | 861f413 | 2018-09-12 11:49:33 -0700 | [diff] [blame] | 639 | # adds fingerprint field to the care_map | 
| Tianjie Xu | 0fde41e | 2020-05-09 05:24:18 +0000 | [diff] [blame] | 640 | # TODO(xunchang) revisit the fingerprint calculation for care_map. | 
|  | 641 | partition_props = OPTIONS.info_dict.get(partition + ".build.prop") | 
| Tianjie Xu | 861f413 | 2018-09-12 11:49:33 -0700 | [diff] [blame] | 642 | prop_name_list = ["ro.{}.build.fingerprint".format(partition), | 
|  | 643 | "ro.{}.build.thumbprint".format(partition)] | 
|  | 644 |  | 
| Tianjie Xu | 0fde41e | 2020-05-09 05:24:18 +0000 | [diff] [blame] | 645 | present_props = [x for x in prop_name_list if | 
|  | 646 | partition_props and partition_props.GetProp(x)] | 
| Tianjie Xu | 861f413 | 2018-09-12 11:49:33 -0700 | [diff] [blame] | 647 | if not present_props: | 
| Tao Bao | 32fcdab | 2018-10-12 10:30:39 -0700 | [diff] [blame] | 648 | logger.warning("fingerprint is not present for partition %s", partition) | 
| Tianjie Xu | 861f413 | 2018-09-12 11:49:33 -0700 | [diff] [blame] | 649 | property_id, fingerprint = "unknown", "unknown" | 
|  | 650 | else: | 
|  | 651 | property_id = present_props[0] | 
| Tianjie Xu | 0fde41e | 2020-05-09 05:24:18 +0000 | [diff] [blame] | 652 | fingerprint = partition_props.GetProp(property_id) | 
| Tianjie Xu | 861f413 | 2018-09-12 11:49:33 -0700 | [diff] [blame] | 653 | care_map_list += [property_id, fingerprint] | 
|  | 654 |  | 
| Tianjie Xu | ccbae48 | 2018-08-15 14:28:27 -0700 | [diff] [blame] | 655 | if not care_map_list: | 
|  | 656 | return | 
|  | 657 |  | 
|  | 658 | # Converts the list into proto buf message by calling care_map_generator; and | 
|  | 659 | # writes the result to a temp file. | 
|  | 660 | temp_care_map_text = common.MakeTempFile(prefix="caremap_text-", | 
|  | 661 | suffix=".txt") | 
|  | 662 | with open(temp_care_map_text, 'w') as text_file: | 
|  | 663 | text_file.write('\n'.join(care_map_list)) | 
|  | 664 |  | 
| Tianjie Xu | 861f413 | 2018-09-12 11:49:33 -0700 | [diff] [blame] | 665 | temp_care_map = common.MakeTempFile(prefix="caremap-", suffix=".pb") | 
|  | 666 | care_map_gen_cmd = ["care_map_generator", temp_care_map_text, temp_care_map] | 
| Tao Bao | 2764aee | 2018-11-21 11:02:48 -0800 | [diff] [blame] | 667 | common.RunAndCheckOutput(care_map_gen_cmd) | 
| Tianjie Xu | ccbae48 | 2018-08-15 14:28:27 -0700 | [diff] [blame] | 668 |  | 
| Tianjie Xu | 4c05f4a | 2018-09-14 16:24:41 -0700 | [diff] [blame] | 669 | care_map_path = "META/care_map.pb" | 
| Tianjie Xu | ccbae48 | 2018-08-15 14:28:27 -0700 | [diff] [blame] | 670 | if output_zip and care_map_path not in output_zip.namelist(): | 
|  | 671 | common.ZipWrite(output_zip, temp_care_map, arcname=care_map_path) | 
|  | 672 | else: | 
|  | 673 | shutil.copy(temp_care_map, os.path.join(OPTIONS.input_tmp, care_map_path)) | 
|  | 674 | if output_zip: | 
|  | 675 | OPTIONS.replace_updated_files_list.append(care_map_path) | 
| Tao Bao | bea20ac | 2018-01-17 17:57:49 -0800 | [diff] [blame] | 676 |  | 
|  | 677 |  | 
|  | 678 | def AddPackRadioImages(output_zip, images): | 
|  | 679 | """Copies images listed in META/pack_radioimages.txt from RADIO/ to IMAGES/. | 
|  | 680 |  | 
|  | 681 | Args: | 
|  | 682 | output_zip: The output zip file (needs to be already open), or None to | 
|  | 683 | write images to OPTIONS.input_tmp/. | 
|  | 684 | images: A list of image names. | 
|  | 685 |  | 
|  | 686 | Raises: | 
|  | 687 | AssertionError: If a listed image can't be found. | 
|  | 688 | """ | 
|  | 689 | for image in images: | 
|  | 690 | img_name = image.strip() | 
|  | 691 | _, ext = os.path.splitext(img_name) | 
|  | 692 | if not ext: | 
|  | 693 | img_name += ".img" | 
| Tao Bao | a2ff4c9 | 2018-01-17 12:14:43 -0800 | [diff] [blame] | 694 |  | 
| Tao Bao | bea20ac | 2018-01-17 17:57:49 -0800 | [diff] [blame] | 695 | prebuilt_path = os.path.join(OPTIONS.input_tmp, "IMAGES", img_name) | 
|  | 696 | if os.path.exists(prebuilt_path): | 
| Tao Bao | 32fcdab | 2018-10-12 10:30:39 -0700 | [diff] [blame] | 697 | logger.info("%s already exists, no need to overwrite...", img_name) | 
| Tao Bao | bea20ac | 2018-01-17 17:57:49 -0800 | [diff] [blame] | 698 | continue | 
|  | 699 |  | 
|  | 700 | img_radio_path = os.path.join(OPTIONS.input_tmp, "RADIO", img_name) | 
|  | 701 | assert os.path.exists(img_radio_path), \ | 
|  | 702 | "Failed to find %s at %s" % (img_name, img_radio_path) | 
| Tao Bao | a2ff4c9 | 2018-01-17 12:14:43 -0800 | [diff] [blame] | 703 |  | 
| Tao Bao | bea20ac | 2018-01-17 17:57:49 -0800 | [diff] [blame] | 704 | if output_zip: | 
| Tao Bao | a2ff4c9 | 2018-01-17 12:14:43 -0800 | [diff] [blame] | 705 | common.ZipWrite(output_zip, img_radio_path, "IMAGES/" + img_name) | 
| Tao Bao | bea20ac | 2018-01-17 17:57:49 -0800 | [diff] [blame] | 706 | else: | 
|  | 707 | shutil.copy(img_radio_path, prebuilt_path) | 
|  | 708 |  | 
|  | 709 |  | 
| David Anderson | 1ef03e2 | 2018-08-30 13:11:47 -0700 | [diff] [blame] | 710 | def AddSuperEmpty(output_zip): | 
|  | 711 | """Create a super_empty.img and store it in output_zip.""" | 
|  | 712 |  | 
|  | 713 | img = OutputFile(output_zip, OPTIONS.input_tmp, "IMAGES", "super_empty.img") | 
| Yifan Hong | 055e6cf | 2018-11-29 13:51:48 -0800 | [diff] [blame] | 714 | build_super_image.BuildSuperImage(OPTIONS.info_dict, img.name) | 
| David Anderson | 1ef03e2 | 2018-08-30 13:11:47 -0700 | [diff] [blame] | 715 | img.Write() | 
|  | 716 |  | 
|  | 717 |  | 
| Yifan Hong | c767f7c | 2018-11-08 15:41:24 -0800 | [diff] [blame] | 718 | def AddSuperSplit(output_zip): | 
|  | 719 | """Create split super_*.img and store it in output_zip.""" | 
|  | 720 |  | 
| Yifan Hong | 055e6cf | 2018-11-29 13:51:48 -0800 | [diff] [blame] | 721 | outdir = os.path.join(OPTIONS.input_tmp, "OTA") | 
| Yifan Hong | e98427a | 2018-12-07 10:08:27 -0800 | [diff] [blame] | 722 | built = build_super_image.BuildSuperImage(OPTIONS.input_tmp, outdir) | 
| Yifan Hong | c767f7c | 2018-11-08 15:41:24 -0800 | [diff] [blame] | 723 |  | 
| Yifan Hong | e98427a | 2018-12-07 10:08:27 -0800 | [diff] [blame] | 724 | if built: | 
|  | 725 | for dev in OPTIONS.info_dict['super_block_devices'].strip().split(): | 
|  | 726 | img = OutputFile(output_zip, OPTIONS.input_tmp, "OTA", | 
|  | 727 | "super_" + dev + ".img") | 
|  | 728 | img.Write() | 
| Yifan Hong | c767f7c | 2018-11-08 15:41:24 -0800 | [diff] [blame] | 729 |  | 
|  | 730 |  | 
| Tianjie Xu | 9ac4cb0 | 2017-06-09 16:58:03 -0700 | [diff] [blame] | 731 | def ReplaceUpdatedFiles(zip_filename, files_list): | 
| Tao Bao | 89d7ab2 | 2017-12-14 17:05:33 -0800 | [diff] [blame] | 732 | """Updates all the ZIP entries listed in files_list. | 
| Tianjie Xu | 38af07f | 2017-05-25 17:38:53 -0700 | [diff] [blame] | 733 |  | 
| Tianjie Xu | 4c05f4a | 2018-09-14 16:24:41 -0700 | [diff] [blame] | 734 | For now the list includes META/care_map.pb, and the related files under | 
| Tianjie Xu | 9ac4cb0 | 2017-06-09 16:58:03 -0700 | [diff] [blame] | 735 | SYSTEM/ after rebuilding recovery. | 
|  | 736 | """ | 
| Tao Bao | 89d7ab2 | 2017-12-14 17:05:33 -0800 | [diff] [blame] | 737 | common.ZipDelete(zip_filename, files_list) | 
| Tianjie Xu | 38af07f | 2017-05-25 17:38:53 -0700 | [diff] [blame] | 738 | output_zip = zipfile.ZipFile(zip_filename, "a", | 
|  | 739 | compression=zipfile.ZIP_DEFLATED, | 
|  | 740 | allowZip64=True) | 
| Tianjie Xu | 9ac4cb0 | 2017-06-09 16:58:03 -0700 | [diff] [blame] | 741 | for item in files_list: | 
| Tianjie Xu | 38af07f | 2017-05-25 17:38:53 -0700 | [diff] [blame] | 742 | file_path = os.path.join(OPTIONS.input_tmp, item) | 
|  | 743 | assert os.path.exists(file_path) | 
|  | 744 | common.ZipWrite(output_zip, file_path, arcname=item) | 
|  | 745 | common.ZipClose(output_zip) | 
|  | 746 |  | 
|  | 747 |  | 
| Chris Gross | 435b8fe | 2020-09-15 09:53:44 -0700 | [diff] [blame] | 748 | def HasPartition(partition_name): | 
|  | 749 | """Determines if the target files archive should build a given partition.""" | 
|  | 750 |  | 
|  | 751 | return ((os.path.isdir( | 
|  | 752 | os.path.join(OPTIONS.input_tmp, partition_name.upper())) and | 
|  | 753 | OPTIONS.info_dict.get( | 
|  | 754 | "building_{}_image".format(partition_name)) == "true") or | 
|  | 755 | os.path.exists( | 
|  | 756 | os.path.join(OPTIONS.input_tmp, "IMAGES", | 
|  | 757 | "{}.img".format(partition_name)))) | 
|  | 758 |  | 
| Kelvin Zhang | 5f0fcee | 2021-01-19 15:30:46 -0500 | [diff] [blame] | 759 | def AddApexInfo(output_zip): | 
|  | 760 | apex_infos = GetApexInfoFromTargetFiles(OPTIONS.input_tmp) | 
|  | 761 | apex_metadata_proto = ota_metadata_pb2.ApexMetadata() | 
|  | 762 | apex_metadata_proto.apex_info.extend(apex_infos) | 
|  | 763 | apex_info_bytes = apex_metadata_proto.SerializeToString() | 
|  | 764 |  | 
|  | 765 | output_file = os.path.join(OPTIONS.input_tmp, "META", "apex_info.pb") | 
|  | 766 | with open(output_file, "wb") as ofile: | 
|  | 767 | ofile.write(apex_info_bytes) | 
|  | 768 | if output_zip: | 
|  | 769 | arc_name = "META/apex_info.pb" | 
|  | 770 | if arc_name in output_zip.namelist(): | 
|  | 771 | OPTIONS.replace_updated_files_list.append(arc_name) | 
|  | 772 | else: | 
|  | 773 | common.ZipWrite(output_zip, output_file, arc_name) | 
|  | 774 |  | 
| Chris Gross | 435b8fe | 2020-09-15 09:53:44 -0700 | [diff] [blame] | 775 |  | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 776 | def AddImagesToTargetFiles(filename): | 
| Tao Bao | ae396d9 | 2017-11-20 11:56:43 -0800 | [diff] [blame] | 777 | """Creates and adds images (boot/recovery/system/...) to a target_files.zip. | 
|  | 778 |  | 
|  | 779 | It works with either a zip file (zip mode), or a directory that contains the | 
|  | 780 | files to be packed into a target_files.zip (dir mode). The latter is used when | 
|  | 781 | being called from build/make/core/Makefile. | 
|  | 782 |  | 
|  | 783 | The images will be created under IMAGES/ in the input target_files.zip. | 
|  | 784 |  | 
|  | 785 | Args: | 
| Tao Bao | dba59ee | 2018-01-09 13:21:02 -0800 | [diff] [blame] | 786 | filename: the target_files.zip, or the zip root directory. | 
| Tao Bao | ae396d9 | 2017-11-20 11:56:43 -0800 | [diff] [blame] | 787 | """ | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 788 | if os.path.isdir(filename): | 
|  | 789 | OPTIONS.input_tmp = os.path.abspath(filename) | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 790 | else: | 
| Tao Bao | dba59ee | 2018-01-09 13:21:02 -0800 | [diff] [blame] | 791 | OPTIONS.input_tmp = common.UnzipTemp(filename) | 
| Doug Zongker | fc44a51 | 2014-08-26 13:10:25 -0700 | [diff] [blame] | 792 |  | 
| Michael Runge | 2e0d8fc | 2014-11-13 21:41:08 -0800 | [diff] [blame] | 793 | if not OPTIONS.add_missing: | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 794 | if os.path.isdir(os.path.join(OPTIONS.input_tmp, "IMAGES")): | 
| Tao Bao | 32fcdab | 2018-10-12 10:30:39 -0700 | [diff] [blame] | 795 | logger.warning("target_files appears to already contain images.") | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 796 | sys.exit(1) | 
| Doug Zongker | fc44a51 | 2014-08-26 13:10:25 -0700 | [diff] [blame] | 797 |  | 
| Tao Bao | 410ad8b | 2018-08-24 12:08:38 -0700 | [diff] [blame] | 798 | OPTIONS.info_dict = common.LoadInfoDict(OPTIONS.input_tmp, repacking=True) | 
| Tao Bao | dba59ee | 2018-01-09 13:21:02 -0800 | [diff] [blame] | 799 |  | 
|  | 800 | has_recovery = OPTIONS.info_dict.get("no_recovery") != "true" | 
| Chris Gross | a784ef1 | 2019-04-22 11:09:57 -0700 | [diff] [blame] | 801 | has_boot = OPTIONS.info_dict.get("no_boot") != "true" | 
| Steve Muckle | e1b1086 | 2019-07-10 10:49:37 -0700 | [diff] [blame] | 802 | has_vendor_boot = OPTIONS.info_dict.get("vendor_boot") == "true" | 
| Tao Bao | dba59ee | 2018-01-09 13:21:02 -0800 | [diff] [blame] | 803 |  | 
| Chris Gross | 435b8fe | 2020-09-15 09:53:44 -0700 | [diff] [blame] | 804 | # {vendor,odm,product,system_ext,vendor_dlkm,odm_dlkm, system, system_other}.img | 
|  | 805 | # can be built from source, or  dropped into target_files.zip as a prebuilt blob. | 
|  | 806 | has_vendor = HasPartition("vendor") | 
|  | 807 | has_odm = HasPartition("odm") | 
|  | 808 | has_vendor_dlkm = HasPartition("vendor_dlkm") | 
|  | 809 | has_odm_dlkm = HasPartition("odm_dlkm") | 
|  | 810 | has_product = HasPartition("product") | 
|  | 811 | has_system_ext = HasPartition("system_ext") | 
|  | 812 | has_system = HasPartition("system") | 
|  | 813 | has_system_other = HasPartition("system_other") | 
| Chris Gross | 203191b | 2020-05-30 02:39:12 +0000 | [diff] [blame] | 814 | has_userdata = OPTIONS.info_dict.get("building_userdata_image") == "true" | 
|  | 815 | has_cache = OPTIONS.info_dict.get("building_cache_image") == "true" | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 816 |  | 
| Tao Bao | dba59ee | 2018-01-09 13:21:02 -0800 | [diff] [blame] | 817 | # Set up the output destination. It writes to the given directory for dir | 
|  | 818 | # mode; otherwise appends to the given ZIP. | 
|  | 819 | if os.path.isdir(filename): | 
|  | 820 | output_zip = None | 
|  | 821 | else: | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 822 | output_zip = zipfile.ZipFile(filename, "a", | 
|  | 823 | compression=zipfile.ZIP_DEFLATED, | 
|  | 824 | allowZip64=True) | 
| Tao Bao | ae396d9 | 2017-11-20 11:56:43 -0800 | [diff] [blame] | 825 |  | 
|  | 826 | # Always make input_tmp/IMAGES available, since we may stage boot / recovery | 
|  | 827 | # images there even under zip mode. The directory will be cleaned up as part | 
|  | 828 | # of OPTIONS.input_tmp. | 
|  | 829 | images_dir = os.path.join(OPTIONS.input_tmp, "IMAGES") | 
|  | 830 | if not os.path.isdir(images_dir): | 
|  | 831 | os.makedirs(images_dir) | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 832 |  | 
| Tao Bao | bf70c31 | 2017-07-11 17:27:55 -0700 | [diff] [blame] | 833 | # A map between partition names and their paths, which could be used when | 
|  | 834 | # generating AVB vbmeta image. | 
| Tao Bao | 3ed35d3 | 2019-10-07 20:48:48 -0700 | [diff] [blame] | 835 | partitions = {} | 
| Tao Bao | bf70c31 | 2017-07-11 17:27:55 -0700 | [diff] [blame] | 836 |  | 
| Doug Zongker | fc44a51 | 2014-08-26 13:10:25 -0700 | [diff] [blame] | 837 | def banner(s): | 
| Tao Bao | a370545 | 2019-06-24 15:33:41 -0700 | [diff] [blame] | 838 | logger.info("\n\n++++ %s  ++++\n\n", s) | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 839 |  | 
| Chris Gross | a784ef1 | 2019-04-22 11:09:57 -0700 | [diff] [blame] | 840 | boot_image = None | 
|  | 841 | if has_boot: | 
|  | 842 | banner("boot") | 
| Steve Muckle | 9793cf6 | 2020-04-08 18:27:00 -0700 | [diff] [blame] | 843 | boot_images = OPTIONS.info_dict.get("boot_images") | 
|  | 844 | if boot_images is None: | 
|  | 845 | boot_images = "boot.img" | 
| Roopesh Nataraja | 3e15f6e | 2020-06-08 19:54:13 -0700 | [diff] [blame] | 846 | for index,b in enumerate(boot_images.split()): | 
| Steve Muckle | 9793cf6 | 2020-04-08 18:27:00 -0700 | [diff] [blame] | 847 | # common.GetBootableImage() returns the image directly if present. | 
|  | 848 | boot_image = common.GetBootableImage( | 
|  | 849 | "IMAGES/" + b, b, OPTIONS.input_tmp, "BOOT") | 
|  | 850 | # boot.img may be unavailable in some targets (e.g. aosp_arm64). | 
|  | 851 | if boot_image: | 
|  | 852 | boot_image_path = os.path.join(OPTIONS.input_tmp, "IMAGES", b) | 
| Roopesh Nataraja | 3e15f6e | 2020-06-08 19:54:13 -0700 | [diff] [blame] | 853 | # Although multiple boot images can be generated, include the image | 
|  | 854 | # descriptor of only the first boot image in vbmeta | 
|  | 855 | if index == 0: | 
| Steve Muckle | 9793cf6 | 2020-04-08 18:27:00 -0700 | [diff] [blame] | 856 | partitions['boot'] = boot_image_path | 
|  | 857 | if not os.path.exists(boot_image_path): | 
|  | 858 | boot_image.WriteToDir(OPTIONS.input_tmp) | 
|  | 859 | if output_zip: | 
|  | 860 | boot_image.AddToZip(output_zip) | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 861 |  | 
| Steve Muckle | e1b1086 | 2019-07-10 10:49:37 -0700 | [diff] [blame] | 862 | if has_vendor_boot: | 
|  | 863 | banner("vendor_boot") | 
|  | 864 | vendor_boot_image = common.GetVendorBootImage( | 
|  | 865 | "IMAGES/vendor_boot.img", "vendor_boot.img", OPTIONS.input_tmp, | 
|  | 866 | "VENDOR_BOOT") | 
|  | 867 | if vendor_boot_image: | 
|  | 868 | partitions['vendor_boot'] = os.path.join(OPTIONS.input_tmp, "IMAGES", | 
|  | 869 | "vendor_boot.img") | 
|  | 870 | if not os.path.exists(partitions['vendor_boot']): | 
|  | 871 | vendor_boot_image.WriteToDir(OPTIONS.input_tmp) | 
|  | 872 | if output_zip: | 
|  | 873 | vendor_boot_image.AddToZip(output_zip) | 
|  | 874 |  | 
| Michael Runge | 2e0d8fc | 2014-11-13 21:41:08 -0800 | [diff] [blame] | 875 | recovery_image = None | 
| Tao Bao | db45efa | 2015-10-27 19:25:18 -0700 | [diff] [blame] | 876 | if has_recovery: | 
|  | 877 | banner("recovery") | 
| Tao Bao | 262bf3f | 2017-07-11 17:27:55 -0700 | [diff] [blame] | 878 | recovery_image = common.GetBootableImage( | 
|  | 879 | "IMAGES/recovery.img", "recovery.img", OPTIONS.input_tmp, "RECOVERY") | 
|  | 880 | assert recovery_image, "Failed to create recovery.img." | 
| Tao Bao | bf70c31 | 2017-07-11 17:27:55 -0700 | [diff] [blame] | 881 | partitions['recovery'] = os.path.join( | 
| Tao Bao | 262bf3f | 2017-07-11 17:27:55 -0700 | [diff] [blame] | 882 | OPTIONS.input_tmp, "IMAGES", "recovery.img") | 
| Tao Bao | bf70c31 | 2017-07-11 17:27:55 -0700 | [diff] [blame] | 883 | if not os.path.exists(partitions['recovery']): | 
| Tao Bao | 262bf3f | 2017-07-11 17:27:55 -0700 | [diff] [blame] | 884 | recovery_image.WriteToDir(OPTIONS.input_tmp) | 
|  | 885 | if output_zip: | 
|  | 886 | recovery_image.AddToZip(output_zip) | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 887 |  | 
| Tao Bao | d42e97e | 2016-11-30 12:11:57 -0800 | [diff] [blame] | 888 | banner("recovery (two-step image)") | 
|  | 889 | # The special recovery.img for two-step package use. | 
|  | 890 | recovery_two_step_image = common.GetBootableImage( | 
| Tao Bao | 0480850 | 2019-07-25 23:11:41 -0700 | [diff] [blame] | 891 | "OTA/recovery-two-step.img", "recovery-two-step.img", | 
| Tao Bao | d42e97e | 2016-11-30 12:11:57 -0800 | [diff] [blame] | 892 | OPTIONS.input_tmp, "RECOVERY", two_step_image=True) | 
| Tao Bao | 262bf3f | 2017-07-11 17:27:55 -0700 | [diff] [blame] | 893 | assert recovery_two_step_image, "Failed to create recovery-two-step.img." | 
|  | 894 | recovery_two_step_image_path = os.path.join( | 
| Tao Bao | 0480850 | 2019-07-25 23:11:41 -0700 | [diff] [blame] | 895 | OPTIONS.input_tmp, "OTA", "recovery-two-step.img") | 
| Tao Bao | 262bf3f | 2017-07-11 17:27:55 -0700 | [diff] [blame] | 896 | if not os.path.exists(recovery_two_step_image_path): | 
|  | 897 | recovery_two_step_image.WriteToDir(OPTIONS.input_tmp) | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 898 | if output_zip: | 
|  | 899 | recovery_two_step_image.AddToZip(output_zip) | 
| Tao Bao | d42e97e | 2016-11-30 12:11:57 -0800 | [diff] [blame] | 900 |  | 
| Bill Peckham | cc57de3 | 2019-01-29 11:01:46 -0800 | [diff] [blame] | 901 | if has_system: | 
|  | 902 | banner("system") | 
|  | 903 | partitions['system'] = AddSystem( | 
|  | 904 | output_zip, recovery_img=recovery_image, boot_img=boot_image) | 
| Tao Bao | bf70c31 | 2017-07-11 17:27:55 -0700 | [diff] [blame] | 905 |  | 
| Doug Zongker | fc44a51 | 2014-08-26 13:10:25 -0700 | [diff] [blame] | 906 | if has_vendor: | 
|  | 907 | banner("vendor") | 
| Bill Peckham | e868aec | 2019-09-17 17:06:47 -0700 | [diff] [blame] | 908 | partitions['vendor'] = AddVendor( | 
|  | 909 | output_zip, recovery_img=recovery_image, boot_img=boot_image) | 
| Tao Bao | bf70c31 | 2017-07-11 17:27:55 -0700 | [diff] [blame] | 910 |  | 
| Jaekyun Seok | b7735d8 | 2017-11-27 17:04:47 +0900 | [diff] [blame] | 911 | if has_product: | 
|  | 912 | banner("product") | 
|  | 913 | partitions['product'] = AddProduct(output_zip) | 
|  | 914 |  | 
| Justin Yun | 6151e3f | 2019-06-25 15:58:13 +0900 | [diff] [blame] | 915 | if has_system_ext: | 
|  | 916 | banner("system_ext") | 
|  | 917 | partitions['system_ext'] = AddSystemExt(output_zip) | 
| Dario Freni | 5f681e1 | 2018-05-29 13:09:01 +0100 | [diff] [blame] | 918 |  | 
| Bowgo Tsai | d624fa6 | 2017-11-14 23:42:30 +0800 | [diff] [blame] | 919 | if has_odm: | 
|  | 920 | banner("odm") | 
|  | 921 | partitions['odm'] = AddOdm(output_zip) | 
|  | 922 |  | 
| Yifan Hong | cfb917a | 2020-05-07 14:58:20 -0700 | [diff] [blame] | 923 | if has_vendor_dlkm: | 
|  | 924 | banner("vendor_dlkm") | 
|  | 925 | partitions['vendor_dlkm'] = AddVendorDlkm(output_zip) | 
|  | 926 |  | 
| Yifan Hong | f496f1b | 2020-07-15 16:52:59 -0700 | [diff] [blame] | 927 | if has_odm_dlkm: | 
|  | 928 | banner("odm_dlkm") | 
|  | 929 | partitions['odm_dlkm'] = AddOdmDlkm(output_zip) | 
|  | 930 |  | 
| Alex Light | 4e358ab | 2016-06-16 14:47:10 -0700 | [diff] [blame] | 931 | if has_system_other: | 
|  | 932 | banner("system_other") | 
|  | 933 | AddSystemOther(output_zip) | 
| Tao Bao | bf70c31 | 2017-07-11 17:27:55 -0700 | [diff] [blame] | 934 |  | 
| Kelvin Zhang | 5f0fcee | 2021-01-19 15:30:46 -0500 | [diff] [blame] | 935 | AddApexInfo(output_zip) | 
|  | 936 |  | 
| Tianjie Xu | b48589a | 2016-08-03 19:21:52 -0700 | [diff] [blame] | 937 | if not OPTIONS.is_signing: | 
|  | 938 | banner("userdata") | 
|  | 939 | AddUserdata(output_zip) | 
|  | 940 | banner("cache") | 
|  | 941 | AddCache(output_zip) | 
| Tao Bao | c633ed0 | 2017-05-30 21:46:33 -0700 | [diff] [blame] | 942 |  | 
|  | 943 | if OPTIONS.info_dict.get("board_bpt_enable") == "true": | 
| David Zeuthen | 2532862 | 2016-04-08 15:08:03 -0400 | [diff] [blame] | 944 | banner("partition-table") | 
|  | 945 | AddPartitionTable(output_zip) | 
| Tao Bao | c633ed0 | 2017-05-30 21:46:33 -0700 | [diff] [blame] | 946 |  | 
| Tao Bao | c633ed0 | 2017-05-30 21:46:33 -0700 | [diff] [blame] | 947 | if OPTIONS.info_dict.get("has_dtbo") == "true": | 
|  | 948 | banner("dtbo") | 
| Tao Bao | bf70c31 | 2017-07-11 17:27:55 -0700 | [diff] [blame] | 949 | partitions['dtbo'] = AddDtbo(output_zip) | 
| Tao Bao | c633ed0 | 2017-05-30 21:46:33 -0700 | [diff] [blame] | 950 |  | 
| Hongguang Chen | f23364d | 2020-04-27 18:36:36 -0700 | [diff] [blame] | 951 | # Custom images. | 
|  | 952 | custom_partitions = OPTIONS.info_dict.get( | 
|  | 953 | "avb_custom_images_partition_list", "").strip().split() | 
|  | 954 | for partition_name in custom_partitions: | 
|  | 955 | partition_name = partition_name.strip() | 
|  | 956 | banner("custom images for " + partition_name) | 
|  | 957 | partitions[partition_name] = AddCustomImages(output_zip, partition_name) | 
|  | 958 |  | 
| Bowgo Tsai | 3e599ea | 2017-05-26 18:30:04 +0800 | [diff] [blame] | 959 | if OPTIONS.info_dict.get("avb_enable") == "true": | 
| Tao Bao | 744c4c7 | 2018-08-20 21:09:07 -0700 | [diff] [blame] | 960 | # vbmeta_partitions includes the partitions that should be included into | 
|  | 961 | # top-level vbmeta.img, which are the ones that are not included in any | 
|  | 962 | # chained VBMeta image plus the chained VBMeta images themselves. | 
| Hongguang Chen | f23364d | 2020-04-27 18:36:36 -0700 | [diff] [blame] | 963 | # Currently custom_partitions are all chained to VBMeta image. | 
|  | 964 | vbmeta_partitions = common.AVB_PARTITIONS[:] + tuple(custom_partitions) | 
| Tao Bao | 744c4c7 | 2018-08-20 21:09:07 -0700 | [diff] [blame] | 965 |  | 
| David Anderson | 7709ab2 | 2018-10-15 14:41:34 -0700 | [diff] [blame] | 966 | vbmeta_system = OPTIONS.info_dict.get("avb_vbmeta_system", "").strip() | 
|  | 967 | if vbmeta_system: | 
|  | 968 | banner("vbmeta_system") | 
| Tao Bao | 7106420 | 2018-10-22 15:08:02 -0700 | [diff] [blame] | 969 | partitions["vbmeta_system"] = AddVBMeta( | 
| David Anderson | 7709ab2 | 2018-10-15 14:41:34 -0700 | [diff] [blame] | 970 | output_zip, partitions, "vbmeta_system", vbmeta_system.split()) | 
| Tao Bao | 744c4c7 | 2018-08-20 21:09:07 -0700 | [diff] [blame] | 971 | vbmeta_partitions = [ | 
|  | 972 | item for item in vbmeta_partitions | 
| David Anderson | 7709ab2 | 2018-10-15 14:41:34 -0700 | [diff] [blame] | 973 | if item not in vbmeta_system.split()] | 
|  | 974 | vbmeta_partitions.append("vbmeta_system") | 
| Tao Bao | 744c4c7 | 2018-08-20 21:09:07 -0700 | [diff] [blame] | 975 |  | 
|  | 976 | vbmeta_vendor = OPTIONS.info_dict.get("avb_vbmeta_vendor", "").strip() | 
|  | 977 | if vbmeta_vendor: | 
|  | 978 | banner("vbmeta_vendor") | 
| Tao Bao | 7106420 | 2018-10-22 15:08:02 -0700 | [diff] [blame] | 979 | partitions["vbmeta_vendor"] = AddVBMeta( | 
| Tao Bao | 744c4c7 | 2018-08-20 21:09:07 -0700 | [diff] [blame] | 980 | output_zip, partitions, "vbmeta_vendor", vbmeta_vendor.split()) | 
|  | 981 | vbmeta_partitions = [ | 
|  | 982 | item for item in vbmeta_partitions | 
|  | 983 | if item not in vbmeta_vendor.split()] | 
|  | 984 | vbmeta_partitions.append("vbmeta_vendor") | 
|  | 985 |  | 
| Bowgo Tsai | 8218225 | 2020-11-13 11:28:17 +0800 | [diff] [blame] | 986 | if OPTIONS.info_dict.get("avb_building_vbmeta_image") == "true": | 
|  | 987 | banner("vbmeta") | 
|  | 988 | AddVBMeta(output_zip, partitions, "vbmeta", vbmeta_partitions) | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 989 |  | 
| Tao Bao | 48a2feb | 2019-06-28 11:00:05 -0700 | [diff] [blame] | 990 | if OPTIONS.info_dict.get("use_dynamic_partitions") == "true": | 
| David Anderson | 1ef03e2 | 2018-08-30 13:11:47 -0700 | [diff] [blame] | 991 | banner("super_empty") | 
|  | 992 | AddSuperEmpty(output_zip) | 
|  | 993 |  | 
| Tao Bao | 48a2feb | 2019-06-28 11:00:05 -0700 | [diff] [blame] | 994 | if OPTIONS.info_dict.get("build_super_partition") == "true": | 
| Tao Bao | 519d182 | 2018-12-27 12:47:23 -0800 | [diff] [blame] | 995 | if OPTIONS.info_dict.get( | 
|  | 996 | "build_retrofit_dynamic_partitions_ota_package") == "true": | 
| Yifan Hong | c767f7c | 2018-11-08 15:41:24 -0800 | [diff] [blame] | 997 | banner("super split images") | 
|  | 998 | AddSuperSplit(output_zip) | 
| Yifan Hong | c767f7c | 2018-11-08 15:41:24 -0800 | [diff] [blame] | 999 |  | 
| Tianjie Xu | aaca421 | 2016-06-28 14:34:03 -0700 | [diff] [blame] | 1000 | banner("radio") | 
| Tao Bao | bea20ac | 2018-01-17 17:57:49 -0800 | [diff] [blame] | 1001 | ab_partitions_txt = os.path.join(OPTIONS.input_tmp, "META", | 
|  | 1002 | "ab_partitions.txt") | 
|  | 1003 | if os.path.exists(ab_partitions_txt): | 
| Tao Bao | a370545 | 2019-06-24 15:33:41 -0700 | [diff] [blame] | 1004 | with open(ab_partitions_txt) as f: | 
| Tao Bao | bea20ac | 2018-01-17 17:57:49 -0800 | [diff] [blame] | 1005 | ab_partitions = f.readlines() | 
| Tianjie Xu | cfa8622 | 2016-03-07 16:31:19 -0800 | [diff] [blame] | 1006 |  | 
| Tao Bao | 5277d10 | 2018-04-17 23:47:21 -0700 | [diff] [blame] | 1007 | # For devices using A/B update, make sure we have all the needed images | 
|  | 1008 | # ready under IMAGES/ or RADIO/. | 
|  | 1009 | CheckAbOtaImages(output_zip, ab_partitions) | 
| Tianjie Xu | aaca421 | 2016-06-28 14:34:03 -0700 | [diff] [blame] | 1010 |  | 
| Tianjie Xu | 861f413 | 2018-09-12 11:49:33 -0700 | [diff] [blame] | 1011 | # Generate care_map.pb for ab_partitions, then write this file to | 
|  | 1012 | # target_files package. | 
| Tianjie Xu | 4c05f4a | 2018-09-14 16:24:41 -0700 | [diff] [blame] | 1013 | AddCareMapForAbOta(output_zip, ab_partitions, partitions) | 
| Tianjie Xu | cfa8622 | 2016-03-07 16:31:19 -0800 | [diff] [blame] | 1014 |  | 
| Tao Bao | 95a95c3 | 2017-06-16 15:30:23 -0700 | [diff] [blame] | 1015 | # Radio images that need to be packed into IMAGES/, and product-img.zip. | 
| Tao Bao | bea20ac | 2018-01-17 17:57:49 -0800 | [diff] [blame] | 1016 | pack_radioimages_txt = os.path.join( | 
| Tao Bao | 95a95c3 | 2017-06-16 15:30:23 -0700 | [diff] [blame] | 1017 | OPTIONS.input_tmp, "META", "pack_radioimages.txt") | 
| Tao Bao | bea20ac | 2018-01-17 17:57:49 -0800 | [diff] [blame] | 1018 | if os.path.exists(pack_radioimages_txt): | 
| Tao Bao | a370545 | 2019-06-24 15:33:41 -0700 | [diff] [blame] | 1019 | with open(pack_radioimages_txt) as f: | 
| Tao Bao | bea20ac | 2018-01-17 17:57:49 -0800 | [diff] [blame] | 1020 | AddPackRadioImages(output_zip, f.readlines()) | 
| Tao Bao | 95a95c3 | 2017-06-16 15:30:23 -0700 | [diff] [blame] | 1021 |  | 
| Dan Willemsen | 2ee00d5 | 2017-03-05 19:51:56 -0800 | [diff] [blame] | 1022 | if output_zip: | 
|  | 1023 | common.ZipClose(output_zip) | 
| Tianjie Xu | 9ac4cb0 | 2017-06-09 16:58:03 -0700 | [diff] [blame] | 1024 | if OPTIONS.replace_updated_files_list: | 
|  | 1025 | ReplaceUpdatedFiles(output_zip.filename, | 
|  | 1026 | OPTIONS.replace_updated_files_list) | 
| Tianjie Xu | 38af07f | 2017-05-25 17:38:53 -0700 | [diff] [blame] | 1027 |  | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 1028 |  | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 1029 | def main(argv): | 
| Baligh Uddin | 59f4ff1 | 2015-09-16 21:20:30 -0700 | [diff] [blame] | 1030 | def option_handler(o, a): | 
| Michael Runge | 2e0d8fc | 2014-11-13 21:41:08 -0800 | [diff] [blame] | 1031 | if o in ("-a", "--add_missing"): | 
|  | 1032 | OPTIONS.add_missing = True | 
|  | 1033 | elif o in ("-r", "--rebuild_recovery",): | 
|  | 1034 | OPTIONS.rebuild_recovery = True | 
| Baligh Uddin | 59f4ff1 | 2015-09-16 21:20:30 -0700 | [diff] [blame] | 1035 | elif o == "--replace_verity_private_key": | 
|  | 1036 | OPTIONS.replace_verity_private_key = (True, a) | 
|  | 1037 | elif o == "--replace_verity_public_key": | 
|  | 1038 | OPTIONS.replace_verity_public_key = (True, a) | 
| Tianjie Xu | b48589a | 2016-08-03 19:21:52 -0700 | [diff] [blame] | 1039 | elif o == "--is_signing": | 
|  | 1040 | OPTIONS.is_signing = True | 
| Michael Runge | 2e0d8fc | 2014-11-13 21:41:08 -0800 | [diff] [blame] | 1041 | else: | 
|  | 1042 | return False | 
|  | 1043 | return True | 
|  | 1044 |  | 
| Dan Albert | 8b72aef | 2015-03-23 19:13:21 -0700 | [diff] [blame] | 1045 | args = common.ParseOptions( | 
|  | 1046 | argv, __doc__, extra_opts="ar", | 
| Baligh Uddin | 59f4ff1 | 2015-09-16 21:20:30 -0700 | [diff] [blame] | 1047 | extra_long_opts=["add_missing", "rebuild_recovery", | 
|  | 1048 | "replace_verity_public_key=", | 
|  | 1049 | "replace_verity_private_key=", | 
| Tao Bao | 4581042 | 2016-10-17 16:20:12 -0700 | [diff] [blame] | 1050 | "is_signing"], | 
| Dan Albert | 8b72aef | 2015-03-23 19:13:21 -0700 | [diff] [blame] | 1051 | extra_option_handler=option_handler) | 
| Michael Runge | 2e0d8fc | 2014-11-13 21:41:08 -0800 | [diff] [blame] | 1052 |  | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 1053 | if len(args) != 1: | 
|  | 1054 | common.Usage(__doc__) | 
|  | 1055 | sys.exit(1) | 
|  | 1056 |  | 
| Tao Bao | 32fcdab | 2018-10-12 10:30:39 -0700 | [diff] [blame] | 1057 | common.InitLogging() | 
|  | 1058 |  | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 1059 | AddImagesToTargetFiles(args[0]) | 
| Tao Bao | 32fcdab | 2018-10-12 10:30:39 -0700 | [diff] [blame] | 1060 | logger.info("done.") | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 1061 |  | 
|  | 1062 | if __name__ == '__main__': | 
|  | 1063 | try: | 
|  | 1064 | common.CloseInheritedPipes() | 
|  | 1065 | main(sys.argv[1:]) | 
| Tao Bao | 32fcdab | 2018-10-12 10:30:39 -0700 | [diff] [blame] | 1066 | except common.ExternalError: | 
|  | 1067 | logger.exception("\n   ERROR:\n") | 
| Doug Zongker | 3c84f56 | 2014-07-31 11:06:30 -0700 | [diff] [blame] | 1068 | sys.exit(1) | 
| Doug Zongker | fc44a51 | 2014-08-26 13:10:25 -0700 | [diff] [blame] | 1069 | finally: | 
|  | 1070 | common.Cleanup() |