blob: 27292c1065f2e6c8a5bdb90a420ce4a3f06068c5 [file] [log] [blame]
Inseob Kimbac07552024-06-18 11:09:12 +09001#!/usr/bin/env python3
2#
3# Copyright (C) 2024 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"""A tool for generating {partition}/build.prop"""
18
19import argparse
20import contextlib
21import json
Justin Yun962f9112024-07-03 13:34:29 +090022import os
Inseob Kimbac07552024-06-18 11:09:12 +090023import subprocess
24import sys
25
Justin Yun962f9112024-07-03 13:34:29 +090026TEST_KEY_DIR = "build/make/target/product/security"
27
Inseob Kimbac07552024-06-18 11:09:12 +090028def get_build_variant(product_config):
29 if product_config["Eng"]:
30 return "eng"
31 elif product_config["Debuggable"]:
32 return "userdebug"
33 else:
34 return "user"
35
36def get_build_flavor(product_config):
37 build_flavor = product_config["DeviceProduct"] + "-" + get_build_variant(product_config)
38 if "address" in product_config.get("SanitizeDevice", []) and "_asan" not in build_flavor:
39 build_flavor += "_asan"
40 return build_flavor
41
42def get_build_keys(product_config):
43 default_cert = product_config.get("DefaultAppCertificate", "")
44 if default_cert == "" or default_cert == os.path.join(TEST_KEY_DIR, "testKey"):
45 return "test-keys"
46 return "dev-keys"
47
Luca Stefani74b53962024-09-07 11:49:03 +020048def override_config(config):
49 if "PRODUCT_BUILD_PROP_OVERRIDES" in config:
50 current_key = None
51 props_overrides = {}
52
53 for var in config["PRODUCT_BUILD_PROP_OVERRIDES"]:
54 if "=" in var:
55 current_key, value = var.split("=")
56 props_overrides[current_key] = value
57 else:
58 props_overrides[current_key] += f" {var}"
59
60 for key, value in props_overrides.items():
61 if key not in config:
62 print(f"Key \"{key}\" isn't a valid prop override", file=sys.stderr)
63 sys.exit(1)
64 config[key] = value
65
Inseob Kimbac07552024-06-18 11:09:12 +090066def parse_args():
67 """Parse commandline arguments."""
68 parser = argparse.ArgumentParser()
69 parser.add_argument("--build-fingerprint-file", required=True, type=argparse.FileType("r"))
70 parser.add_argument("--build-hostname-file", required=True, type=argparse.FileType("r"))
71 parser.add_argument("--build-number-file", required=True, type=argparse.FileType("r"))
72 parser.add_argument("--build-thumbprint-file", type=argparse.FileType("r"))
73 parser.add_argument("--build-username", required=True)
74 parser.add_argument("--date-file", required=True, type=argparse.FileType("r"))
75 parser.add_argument("--platform-preview-sdk-fingerprint-file", required=True, type=argparse.FileType("r"))
76 parser.add_argument("--prop-files", action="append", type=argparse.FileType("r"), default=[])
77 parser.add_argument("--product-config", required=True, type=argparse.FileType("r"))
78 parser.add_argument("--partition", required=True)
79 parser.add_argument("--build-broken-dup-sysprop", action="store_true", default=False)
80
81 parser.add_argument("--out", required=True, type=argparse.FileType("w"))
82
83 args = parser.parse_args()
84
85 # post process parse_args requiring manual handling
86 args.config = json.load(args.product_config)
87 config = args.config
88
89 config["BuildFlavor"] = get_build_flavor(config)
90 config["BuildKeys"] = get_build_keys(config)
91 config["BuildVariant"] = get_build_variant(config)
92
93 config["BuildFingerprint"] = args.build_fingerprint_file.read().strip()
94 config["BuildHostname"] = args.build_hostname_file.read().strip()
95 config["BuildNumber"] = args.build_number_file.read().strip()
96 config["BuildUsername"] = args.build_username
Inseob Kimb2b1b562024-06-25 19:09:12 +090097
98 build_version_tags_list = config["BuildVersionTags"]
Inseob Kimbac07552024-06-18 11:09:12 +090099 if config["BuildType"] == "debug":
Inseob Kimb2b1b562024-06-25 19:09:12 +0900100 build_version_tags_list.append("debug")
101 build_version_tags_list.append(config["BuildKeys"])
102 build_version_tags = ",".join(sorted(set(build_version_tags_list)))
103 config["BuildVersionTags"] = build_version_tags
Inseob Kimbac07552024-06-18 11:09:12 +0900104
105 raw_date = args.date_file.read().strip()
106 config["Date"] = subprocess.check_output(["date", "-d", f"@{raw_date}"], text=True).strip()
107 config["DateUtc"] = subprocess.check_output(["date", "-d", f"@{raw_date}", "+%s"], text=True).strip()
108
109 # build_desc is human readable strings that describe this build. This has the same info as the
110 # build fingerprint.
111 # e.g. "aosp_cf_x86_64_phone-userdebug VanillaIceCream MAIN eng.20240319.143939 test-keys"
112 config["BuildDesc"] = f"{config['DeviceProduct']}-{config['BuildVariant']} " \
113 f"{config['Platform_version_name']} {config['BuildId']} " \
114 f"{config['BuildNumber']} {config['BuildVersionTags']}"
115
116 config["PlatformPreviewSdkFingerprint"] = args.platform_preview_sdk_fingerprint_file.read().strip()
117
118 if args.build_thumbprint_file:
119 config["BuildThumbprint"] = args.build_thumbprint_file.read().strip()
120
Luca Stefani74b53962024-09-07 11:49:03 +0200121 override_config(config)
122
Inseob Kimbac07552024-06-18 11:09:12 +0900123 append_additional_system_props(args)
124 append_additional_vendor_props(args)
125 append_additional_product_props(args)
126
127 return args
128
129def generate_common_build_props(args):
130 print("####################################")
131 print("# from generate_common_build_props")
132 print("# These properties identify this partition image.")
133 print("####################################")
134
135 config = args.config
136 partition = args.partition
137
138 if partition == "system":
139 print(f"ro.product.{partition}.brand={config['SystemBrand']}")
140 print(f"ro.product.{partition}.device={config['SystemDevice']}")
141 print(f"ro.product.{partition}.manufacturer={config['SystemManufacturer']}")
142 print(f"ro.product.{partition}.model={config['SystemModel']}")
143 print(f"ro.product.{partition}.name={config['SystemName']}")
144 else:
145 print(f"ro.product.{partition}.brand={config['ProductBrand']}")
146 print(f"ro.product.{partition}.device={config['DeviceName']}")
147 print(f"ro.product.{partition}.manufacturer={config['ProductManufacturer']}")
148 print(f"ro.product.{partition}.model={config['ProductModel']}")
149 print(f"ro.product.{partition}.name={config['DeviceProduct']}")
150
151 if partition != "system":
Inseob Kimaaf72f22024-08-05 12:51:05 +0900152 if config["ProductModelForAttestation"]:
153 print(f"ro.product.model_for_attestation={config['ProductModelForAttestation']}")
154 if config["ProductBrandForAttestation"]:
155 print(f"ro.product.brand_for_attestation={config['ProductBrandForAttestation']}")
156 if config["ProductNameForAttestation"]:
157 print(f"ro.product.name_for_attestation={config['ProductNameForAttestation']}")
158 if config["ProductDeviceForAttestation"]:
159 print(f"ro.product.device_for_attestation={config['ProductDeviceForAttestation']}")
160 if config["ProductManufacturerForAttestation"]:
161 print(f"ro.product.manufacturer_for_attestation={config['ProductManufacturerForAttestation']}")
Inseob Kimbac07552024-06-18 11:09:12 +0900162
163 if config["ZygoteForce64"]:
164 if partition == "vendor":
165 print(f"ro.{partition}.product.cpu.abilist={config['DeviceAbiList64']}")
166 print(f"ro.{partition}.product.cpu.abilist32=")
167 print(f"ro.{partition}.product.cpu.abilist64={config['DeviceAbiList64']}")
168 else:
169 if partition == "system" or partition == "vendor" or partition == "odm":
170 print(f"ro.{partition}.product.cpu.abilist={config['DeviceAbiList']}")
171 print(f"ro.{partition}.product.cpu.abilist32={config['DeviceAbiList32']}")
172 print(f"ro.{partition}.product.cpu.abilist64={config['DeviceAbiList64']}")
173
174 print(f"ro.{partition}.build.date={config['Date']}")
175 print(f"ro.{partition}.build.date.utc={config['DateUtc']}")
176 # Allow optional assignments for ARC forward-declarations (b/249168657)
177 # TODO: Remove any tag-related inconsistencies once the goals from
178 # go/arc-android-sigprop-changes have been achieved.
179 print(f"ro.{partition}.build.fingerprint?={config['BuildFingerprint']}")
180 print(f"ro.{partition}.build.id?={config['BuildId']}")
181 print(f"ro.{partition}.build.tags?={config['BuildVersionTags']}")
182 print(f"ro.{partition}.build.type={config['BuildVariant']}")
183 print(f"ro.{partition}.build.version.incremental={config['BuildNumber']}")
184 print(f"ro.{partition}.build.version.release={config['Platform_version_last_stable']}")
185 print(f"ro.{partition}.build.version.release_or_codename={config['Platform_version_name']}")
186 print(f"ro.{partition}.build.version.sdk={config['Platform_sdk_version']}")
187
188def generate_build_info(args):
189 print()
190 print("####################################")
191 print("# from gen_build_prop.py:generate_build_info")
192 print("####################################")
193 print("# begin build properties")
194
195 config = args.config
196 build_flags = config["BuildFlags"]
197
Michael Bestas982a5042024-10-17 06:10:07 +0300198 print(f"ro.build.fingerprint?={config['BuildFingerprint']}")
199
Inseob Kimbac07552024-06-18 11:09:12 +0900200 # The ro.build.id will be set dynamically by init, by appending the unique vbmeta digest.
201 if config["BoardUseVbmetaDigestInFingerprint"]:
202 print(f"ro.build.legacy.id={config['BuildId']}")
203 else:
204 print(f"ro.build.id?={config['BuildId']}")
205
206 # ro.build.display.id is shown under Settings -> About Phone
207 if config["BuildVariant"] == "user":
208 # User builds should show:
209 # release build number or branch.buld_number non-release builds
210
211 # Dev. branches should have DISPLAY_BUILD_NUMBER set
212 if config["DisplayBuildNumber"]:
Inseob Kim643157d2024-07-19 09:25:38 +0900213 print(f"ro.build.display.id?={config['BuildId']}.{config['BuildNumber']} {config['BuildKeys']}")
Inseob Kimbac07552024-06-18 11:09:12 +0900214 else:
215 print(f"ro.build.display.id?={config['BuildId']} {config['BuildKeys']}")
216 else:
217 # Non-user builds should show detailed build information (See build desc above)
218 print(f"ro.build.display.id?={config['BuildDesc']}")
219 print(f"ro.build.version.incremental={config['BuildNumber']}")
220 print(f"ro.build.version.sdk={config['Platform_sdk_version']}")
221 print(f"ro.build.version.preview_sdk={config['Platform_preview_sdk_version']}")
222 print(f"ro.build.version.preview_sdk_fingerprint={config['PlatformPreviewSdkFingerprint']}")
223 print(f"ro.build.version.codename={config['Platform_sdk_codename']}")
224 print(f"ro.build.version.all_codenames={','.join(config['Platform_version_active_codenames'])}")
225 print(f"ro.build.version.known_codenames={config['Platform_version_known_codenames']}")
226 print(f"ro.build.version.release={config['Platform_version_last_stable']}")
227 print(f"ro.build.version.release_or_codename={config['Platform_version_name']}")
228 print(f"ro.build.version.release_or_preview_display={config['Platform_display_version_name']}")
229 print(f"ro.build.version.security_patch={config['Platform_security_patch']}")
230 print(f"ro.build.version.base_os={config['Platform_base_os']}")
231 print(f"ro.build.version.min_supported_target_sdk={build_flags['RELEASE_PLATFORM_MIN_SUPPORTED_TARGET_SDK_VERSION']}")
232 print(f"ro.build.date={config['Date']}")
233 print(f"ro.build.date.utc={config['DateUtc']}")
234 print(f"ro.build.type={config['BuildVariant']}")
235 print(f"ro.build.user={config['BuildUsername']}")
236 print(f"ro.build.host={config['BuildHostname']}")
237 # TODO: Remove any tag-related optional property declarations once the goals
238 # from go/arc-android-sigprop-changes have been achieved.
239 print(f"ro.build.tags?={config['BuildVersionTags']}")
240 # ro.build.flavor are used only by the test harness to distinguish builds.
241 # Only add _asan for a sanitized build if it isn't already a part of the
242 # flavor (via a dedicated lunch config for example).
243 print(f"ro.build.flavor={config['BuildFlavor']}")
244
245 # These values are deprecated, use "ro.product.cpu.abilist"
246 # instead (see below).
247 print(f"# ro.product.cpu.abi and ro.product.cpu.abi2 are obsolete,")
248 print(f"# use ro.product.cpu.abilist instead.")
249 print(f"ro.product.cpu.abi={config['DeviceAbi'][0]}")
250 if len(config["DeviceAbi"]) > 1:
251 print(f"ro.product.cpu.abi2={config['DeviceAbi'][1]}")
252
253 if config["ProductLocales"]:
254 print(f"ro.product.locale={config['ProductLocales'][0]}")
255 print(f"ro.wifi.channels={' '.join(config['ProductDefaultWifiChannels'])}")
256
257 print(f"# ro.build.product is obsolete; use ro.product.device")
258 print(f"ro.build.product={config['DeviceName']}")
259
260 print(f"# Do not try to parse description or thumbprint")
261 print(f"ro.build.description?={config['BuildDesc']}")
Inseob Kim1d052472024-07-31 07:58:23 +0900262 if "BuildThumbprint" in config:
Inseob Kimbac07552024-06-18 11:09:12 +0900263 print(f"ro.build.thumbprint={config['BuildThumbprint']}")
264
265 print(f"# end build properties")
266
267def write_properties_from_file(file):
268 print()
269 print("####################################")
270 print(f"# from {file.name}")
271 print("####################################")
272 print(file.read(), end="")
273
274def write_properties_from_variable(name, props, build_broken_dup_sysprop):
275 print()
276 print("####################################")
277 print(f"# from variable {name}")
278 print("####################################")
279
280 # Implement the legacy behavior when BUILD_BROKEN_DUP_SYSPROP is on.
281 # Optional assignments are all converted to normal assignments and
282 # when their duplicates the first one wins.
283 if build_broken_dup_sysprop:
284 processed_props = []
285 seen_props = set()
286 for line in props:
287 line = line.replace("?=", "=")
288 key, value = line.split("=", 1)
289 if key in seen_props:
290 continue
291 seen_props.add(key)
292 processed_props.append(line)
293 props = processed_props
294
295 for line in props:
296 print(line)
297
298def append_additional_system_props(args):
299 props = []
300
301 config = args.config
302
303 # Add the product-defined properties to the build properties.
Inseob Kim4a363392024-07-31 02:09:34 +0000304 if not config["PropertySplitEnabled"] or not config["VendorImageFileSystemType"]:
Inseob Kimbac07552024-06-18 11:09:12 +0900305 if "PRODUCT_PROPERTY_OVERRIDES" in config:
306 props += config["PRODUCT_PROPERTY_OVERRIDES"]
307
308 props.append(f"ro.treble.enabled={'true' if config['FullTreble'] else 'false'}")
309 # Set ro.llndk.api_level to show the maximum vendor API level that the LLNDK
310 # in the system partition supports.
311 if config["VendorApiLevel"]:
312 props.append(f"ro.llndk.api_level={config['VendorApiLevel']}")
313
314 # Sets ro.actionable_compatible_property.enabled to know on runtime whether
315 # the allowed list of actionable compatible properties is enabled or not.
316 props.append("ro.actionable_compatible_property.enabled=true")
317
318 # Enable core platform API violation warnings on userdebug and eng builds.
319 if config["BuildVariant"] != "user":
320 props.append("persist.debug.dalvik.vm.core_platform_api_policy=just-warn")
321
322 # Define ro.sanitize.<name> properties for all global sanitizers.
323 for sanitize_target in config["SanitizeDevice"]:
324 props.append(f"ro.sanitize.{sanitize_target}=true")
325
326 # Sets the default value of ro.postinstall.fstab.prefix to /system.
327 # Device board config should override the value to /product when needed by:
328 #
329 # PRODUCT_PRODUCT_PROPERTIES += ro.postinstall.fstab.prefix=/product
330 #
331 # It then uses ${ro.postinstall.fstab.prefix}/etc/fstab.postinstall to
332 # mount system_other partition.
333 props.append("ro.postinstall.fstab.prefix=/system")
334
335 enable_target_debugging = True
Inseob Kim4a363392024-07-31 02:09:34 +0000336 enable_dalvik_lock_contention_logging = True
Inseob Kimbac07552024-06-18 11:09:12 +0900337 if config["BuildVariant"] == "user" or config["BuildVariant"] == "userdebug":
338 # Target is secure in user builds.
339 props.append("ro.secure=1")
340 props.append("security.perf_harden=1")
341
342 if config["BuildVariant"] == "user":
343 # Disable debugging in plain user builds.
344 props.append("ro.adb.secure=1")
345 enable_target_debugging = False
Inseob Kim4a363392024-07-31 02:09:34 +0000346 enable_dalvik_lock_contention_logging = False
347 else:
348 # Disable debugging in userdebug builds if PRODUCT_NOT_DEBUGGABLE_IN_USERDEBUG
349 # is set.
350 if config["ProductNotDebuggableInUserdebug"]:
351 enable_target_debugging = False
Inseob Kimbac07552024-06-18 11:09:12 +0900352
353 # Disallow mock locations by default for user builds
354 props.append("ro.allow.mock.location=0")
355 else:
356 # Turn on checkjni for non-user builds.
357 props.append("ro.kernel.android.checkjni=1")
358 # Set device insecure for non-user builds.
359 props.append("ro.secure=0")
360 # Allow mock locations by default for non user builds
361 props.append("ro.allow.mock.location=1")
362
Inseob Kim4a363392024-07-31 02:09:34 +0000363 if enable_dalvik_lock_contention_logging:
Inseob Kimbac07552024-06-18 11:09:12 +0900364 # Enable Dalvik lock contention logging.
365 props.append("dalvik.vm.lockprof.threshold=500")
366
Inseob Kim4a363392024-07-31 02:09:34 +0000367 if enable_target_debugging:
Inseob Kimbac07552024-06-18 11:09:12 +0900368 # Target is more debuggable and adbd is on by default
369 props.append("ro.debuggable=1")
370 else:
371 # Target is less debuggable and adbd is off by default
372 props.append("ro.debuggable=0")
373
374 if config["BuildVariant"] == "eng":
375 if "ro.setupwizard.mode=ENABLED" in props:
376 # Don't require the setup wizard on eng builds
377 props = list(filter(lambda x: not x.startswith("ro.setupwizard.mode="), props))
378 props.append("ro.setupwizard.mode=OPTIONAL")
379
380 if not config["SdkBuild"]:
381 # To speedup startup of non-preopted builds, don't verify or compile the boot image.
382 props.append("dalvik.vm.image-dex2oat-filter=extract")
383 # b/323566535
384 props.append("init.svc_debug.no_fatal.zygote=true")
385
386 if config["SdkBuild"]:
387 props.append("xmpp.auto-presence=true")
388 props.append("ro.config.nocheckin=yes")
389
390 props.append("net.bt.name=Android")
391
392 # This property is set by flashing debug boot image, so default to false.
393 props.append("ro.force.debuggable=0")
394
395 config["ADDITIONAL_SYSTEM_PROPERTIES"] = props
396
397def append_additional_vendor_props(args):
398 props = []
399
400 config = args.config
401 build_flags = config["BuildFlags"]
402
403 # Add cpu properties for bionic and ART.
404 props.append(f"ro.bionic.arch={config['DeviceArch']}")
405 props.append(f"ro.bionic.cpu_variant={config['DeviceCpuVariantRuntime']}")
406 props.append(f"ro.bionic.2nd_arch={config['DeviceSecondaryArch']}")
407 props.append(f"ro.bionic.2nd_cpu_variant={config['DeviceSecondaryCpuVariantRuntime']}")
408
409 props.append(f"persist.sys.dalvik.vm.lib.2=libart.so")
410 props.append(f"dalvik.vm.isa.{config['DeviceArch']}.variant={config['Dex2oatTargetCpuVariantRuntime']}")
411 if config["Dex2oatTargetInstructionSetFeatures"]:
412 props.append(f"dalvik.vm.isa.{config['DeviceArch']}.features={config['Dex2oatTargetInstructionSetFeatures']}")
413
414 if config["DeviceSecondaryArch"]:
415 props.append(f"dalvik.vm.isa.{config['DeviceSecondaryArch']}.variant={config['SecondaryDex2oatCpuVariantRuntime']}")
416 if config["SecondaryDex2oatInstructionSetFeatures"]:
417 props.append(f"dalvik.vm.isa.{config['DeviceSecondaryArch']}.features={config['SecondaryDex2oatInstructionSetFeatures']}")
418
419 # Although these variables are prefixed with TARGET_RECOVERY_, they are also needed under charger
420 # mode (via libminui).
421 if config["RecoveryDefaultRotation"]:
422 props.append(f"ro.minui.default_rotation={config['RecoveryDefaultRotation']}")
423
424 if config["RecoveryOverscanPercent"]:
425 props.append(f"ro.minui.overscan_percent={config['RecoveryOverscanPercent']}")
426
427 if config["RecoveryPixelFormat"]:
428 props.append(f"ro.minui.pixel_format={config['RecoveryPixelFormat']}")
429
430 if "UseDynamicPartitions" in config:
431 props.append(f"ro.boot.dynamic_partitions={'true' if config['UseDynamicPartitions'] else 'false'}")
432
433 if "RetrofitDynamicPartitions" in config:
434 props.append(f"ro.boot.dynamic_partitions_retrofit={'true' if config['RetrofitDynamicPartitions'] else 'false'}")
435
436 if config["ShippingApiLevel"]:
437 props.append(f"ro.product.first_api_level={config['ShippingApiLevel']}")
438
439 if config["ShippingVendorApiLevel"]:
440 props.append(f"ro.vendor.api_level={config['ShippingVendorApiLevel']}")
441
442 if config["BuildVariant"] != "user" and config["BuildDebugfsRestrictionsEnabled"]:
443 props.append(f"ro.product.debugfs_restrictions.enabled=true")
444
445 # Vendors with GRF must define BOARD_SHIPPING_API_LEVEL for the vendor API level.
446 # This must not be defined for the non-GRF devices.
447 # The values of the GRF properties will be verified by post_process_props.py
448 if config["BoardShippingApiLevel"]:
Inseob Kim4a363392024-07-31 02:09:34 +0000449 props.append(f"ro.board.first_api_level={config['BoardShippingApiLevel']}")
Inseob Kimbac07552024-06-18 11:09:12 +0900450
451 # Build system set BOARD_API_LEVEL to show the api level of the vendor API surface.
452 # This must not be altered outside of build system.
453 if config["VendorApiLevel"]:
454 props.append(f"ro.board.api_level={config['VendorApiLevel']}")
455
456 # RELEASE_BOARD_API_LEVEL_FROZEN is true when the vendor API surface is frozen.
457 if build_flags["RELEASE_BOARD_API_LEVEL_FROZEN"]:
458 props.append(f"ro.board.api_frozen=true")
459
460 # Set build prop. This prop is read by ota_from_target_files when generating OTA,
461 # to decide if VABC should be disabled.
462 if config["DontUseVabcOta"]:
463 props.append(f"ro.vendor.build.dont_use_vabc=true")
464
465 # Set the flag in vendor. So VTS would know if the new fingerprint format is in use when
466 # the system images are replaced by GSI.
467 if config["BoardUseVbmetaDigestInFingerprint"]:
468 props.append(f"ro.vendor.build.fingerprint_has_digest=1")
469
470 props.append(f"ro.vendor.build.security_patch={config['VendorSecurityPatch']}")
471 props.append(f"ro.product.board={config['BootloaderBoardName']}")
472 props.append(f"ro.board.platform={config['BoardPlatform']}")
473 props.append(f"ro.hwui.use_vulkan={'true' if config['UsesVulkan'] else 'false'}")
474
475 if config["ScreenDensity"]:
476 props.append(f"ro.sf.lcd_density={config['ScreenDensity']}")
477
478 if "AbOtaUpdater" in config:
479 props.append(f"ro.build.ab_update={'true' if config['AbOtaUpdater'] else 'false'}")
480 if config["AbOtaUpdater"]:
481 props.append(f"ro.vendor.build.ab_ota_partitions={config['AbOtaPartitions']}")
482
483 config["ADDITIONAL_VENDOR_PROPERTIES"] = props
484
485def append_additional_product_props(args):
486 props = []
487
488 config = args.config
489
490 # Add the system server compiler filter if they are specified for the product.
491 if config["SystemServerCompilerFilter"]:
492 props.append(f"dalvik.vm.systemservercompilerfilter={config['SystemServerCompilerFilter']}")
493
494 # Add the 16K developer args if it is defined for the product.
495 props.append(f"ro.product.build.16k_page.enabled={'true' if config['Product16KDeveloperOption'] else 'false'}")
496
Inseob Kim6b3ec752024-08-08 17:47:14 +0900497 props.append(f"ro.product.page_size={16384 if config['TargetBoots16K'] else 4096}")
498
Inseob Kimbac07552024-06-18 11:09:12 +0900499 props.append(f"ro.build.characteristics={config['AAPTCharacteristics']}")
500
501 if "AbOtaUpdater" in config and config["AbOtaUpdater"]:
502 props.append(f"ro.product.ab_ota_partitions={config['AbOtaPartitions']}")
503
504 # Set this property for VTS to skip large page size tests on unsupported devices.
505 props.append(f"ro.product.cpu.pagesize.max={config['DeviceMaxPageSizeSupported']}")
506
507 if config["NoBionicPageSizeMacro"]:
508 props.append(f"ro.product.build.no_bionic_page_size_macro=true")
509
Inseob Kim4a363392024-07-31 02:09:34 +0000510 # This is a temporary system property that controls the ART module. The plan is
511 # to remove it by Aug 2025, at which time Mainline updates of the ART module
512 # will ignore it as well.
Inseob Kimbac07552024-06-18 11:09:12 +0900513 # If the value is "default", it will be mangled by post_process_props.py.
514 props.append(f"ro.dalvik.vm.enable_uffd_gc={config['EnableUffdGc']}")
515
516 config["ADDITIONAL_PRODUCT_PROPERTIES"] = props
517
518def build_system_prop(args):
519 config = args.config
520
521 # Order matters here. When there are duplicates, the last one wins.
522 # TODO(b/117892318): don't allow duplicates so that the ordering doesn't matter
523 variables = [
524 "ADDITIONAL_SYSTEM_PROPERTIES",
525 "PRODUCT_SYSTEM_PROPERTIES",
526 # TODO(b/117892318): deprecate this
527 "PRODUCT_SYSTEM_DEFAULT_PROPERTIES",
528 ]
529
530 if not config["PropertySplitEnabled"]:
531 variables += [
532 "ADDITIONAL_VENDOR_PROPERTIES",
533 "PRODUCT_VENDOR_PROPERTIES",
534 ]
535
536 build_prop(args, gen_build_info=True, gen_common_build_props=True, variables=variables)
537
Inseob Kimaaf72f22024-08-05 12:51:05 +0900538def build_system_ext_prop(args):
539 config = args.config
540
541 # Order matters here. When there are duplicates, the last one wins.
542 # TODO(b/117892318): don't allow duplicates so that the ordering doesn't matter
543 variables = ["PRODUCT_SYSTEM_EXT_PROPERTIES"]
544
545 build_prop(args, gen_build_info=False, gen_common_build_props=True, variables=variables)
546
Inseob Kimbac07552024-06-18 11:09:12 +0900547'''
548def build_vendor_prop(args):
549 config = args.config
550
551 # Order matters here. When there are duplicates, the last one wins.
552 # TODO(b/117892318): don't allow duplicates so that the ordering doesn't matter
553 variables = []
554 if config["PropertySplitEnabled"]:
555 variables += [
556 "ADDITIONAL_VENDOR_PROPERTIES",
557 "PRODUCT_VENDOR_PROPERTIES",
558 # TODO(b/117892318): deprecate this
559 "PRODUCT_DEFAULT_PROPERTY_OVERRIDES",
560 "PRODUCT_PROPERTY_OVERRIDES",
561 ]
562
563 build_prop(args, gen_build_info=False, gen_common_build_props=True, variables=variables)
Inseob Kim6b3ec752024-08-08 17:47:14 +0900564'''
Inseob Kimbac07552024-06-18 11:09:12 +0900565
566def build_product_prop(args):
567 config = args.config
568
569 # Order matters here. When there are duplicates, the last one wins.
570 # TODO(b/117892318): don't allow duplicates so that the ordering doesn't matter
571 variables = [
572 "ADDITIONAL_PRODUCT_PROPERTIES",
573 "PRODUCT_PRODUCT_PROPERTIES",
574 ]
Inseob Kim6b3ec752024-08-08 17:47:14 +0900575
576 gen_common_build_props = True
577
578 # Skip common /product properties generation if device released before R and
579 # has no product partition. This is the first part of the check.
580 if config["Shipping_api_level"] and int(config["Shipping_api_level"]) < 30:
581 gen_common_build_props = False
582
583 # The second part of the check - always generate common properties for the
584 # devices with product partition regardless of shipping level.
585 if config["UsesProductImage"]:
586 gen_common_build_props = True
587
Inseob Kimbac07552024-06-18 11:09:12 +0900588 build_prop(args, gen_build_info=False, gen_common_build_props=True, variables=variables)
Inseob Kim6b3ec752024-08-08 17:47:14 +0900589
590 if config["OemProperties"]:
591 print("####################################")
592 print("# PRODUCT_OEM_PROPERTIES")
593 print("####################################")
594
595 for prop in config["OemProperties"]:
596 print(f"import /oem/oem.prop {prop}")
597
598def build_odm_prop(args):
599 variables = ["ADDITIONAL_ODM_PROPERTIES", "PRODUCT_ODM_PROPERTIES"]
600 build_prop(args, gen_build_info=False, gen_common_build_props=True, variables=variables)
Inseob Kimbac07552024-06-18 11:09:12 +0900601
602def build_prop(args, gen_build_info, gen_common_build_props, variables):
603 config = args.config
604
605 if gen_common_build_props:
606 generate_common_build_props(args)
607
608 if gen_build_info:
609 generate_build_info(args)
610
611 for prop_file in args.prop_files:
612 write_properties_from_file(prop_file)
613
614 for variable in variables:
615 if variable in config:
616 write_properties_from_variable(variable, config[variable], args.build_broken_dup_sysprop)
617
618def main():
619 args = parse_args()
620
621 with contextlib.redirect_stdout(args.out):
Inseob Kim6b3ec752024-08-08 17:47:14 +0900622 match args.partition:
623 case "system":
624 build_system_prop(args)
625 case "system_ext":
626 build_system_ext_prop(args)
627 case "odm":
628 build_odm_prop(args)
629 case "product":
630 build_product_prop(args)
631 # case "vendor": # NOT IMPLEMENTED
632 # build_vendor_prop(args)
633 case _:
634 sys.exit(f"not supported partition {args.partition}")
Inseob Kimbac07552024-06-18 11:09:12 +0900635
636if __name__ == "__main__":
637 main()