| #!/usr/bin/env python3 |
| # |
| # Copyright (C) 2024 The Android Open Source Project |
| # |
| # Licensed under the Apache License, Version 2.0 (the "License"); |
| # you may not use this file except in compliance with the License. |
| # You may obtain a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| # See the License for the specific language governing permissions and |
| # limitations under the License. |
| # |
| """A tool for generating {partition}/build.prop""" |
| |
| import argparse |
| import contextlib |
| import json |
| import os |
| import subprocess |
| import sys |
| |
| TEST_KEY_DIR = "build/make/target/product/security" |
| |
| def get_build_variant(product_config): |
| if product_config["Eng"]: |
| return "eng" |
| elif product_config["Debuggable"]: |
| return "userdebug" |
| else: |
| return "user" |
| |
| def get_build_flavor(product_config): |
| build_flavor = product_config["DeviceProduct"] + "-" + get_build_variant(product_config) |
| if "address" in product_config.get("SanitizeDevice", []) and "_asan" not in build_flavor: |
| build_flavor += "_asan" |
| return build_flavor |
| |
| def get_build_keys(product_config): |
| default_cert = product_config.get("DefaultAppCertificate", "") |
| if default_cert == "" or default_cert == os.path.join(TEST_KEY_DIR, "testKey"): |
| return "test-keys" |
| return "dev-keys" |
| |
| def override_config(config): |
| if "PRODUCT_BUILD_PROP_OVERRIDES" in config: |
| current_key = None |
| props_overrides = {} |
| |
| for var in config["PRODUCT_BUILD_PROP_OVERRIDES"]: |
| if "=" in var: |
| current_key, value = var.split("=") |
| props_overrides[current_key] = value |
| else: |
| props_overrides[current_key] += f" {var}" |
| |
| for key, value in props_overrides.items(): |
| if key not in config: |
| print(f"Key \"{key}\" isn't a valid prop override", file=sys.stderr) |
| sys.exit(1) |
| config[key] = value |
| |
| def parse_args(): |
| """Parse commandline arguments.""" |
| parser = argparse.ArgumentParser() |
| parser.add_argument("--build-fingerprint-file", required=True, type=argparse.FileType("r")) |
| parser.add_argument("--build-hostname-file", required=True, type=argparse.FileType("r")) |
| parser.add_argument("--build-number-file", required=True, type=argparse.FileType("r")) |
| parser.add_argument("--build-thumbprint-file", type=argparse.FileType("r")) |
| parser.add_argument("--build-username", required=True) |
| parser.add_argument("--date-file", required=True, type=argparse.FileType("r")) |
| parser.add_argument("--platform-preview-sdk-fingerprint-file", required=True, type=argparse.FileType("r")) |
| parser.add_argument("--prop-files", action="append", type=argparse.FileType("r"), default=[]) |
| parser.add_argument("--product-config", required=True, type=argparse.FileType("r")) |
| parser.add_argument("--partition", required=True) |
| parser.add_argument("--build-broken-dup-sysprop", action="store_true", default=False) |
| |
| parser.add_argument("--out", required=True, type=argparse.FileType("w")) |
| |
| args = parser.parse_args() |
| |
| # post process parse_args requiring manual handling |
| args.config = json.load(args.product_config) |
| config = args.config |
| |
| config["BuildFlavor"] = get_build_flavor(config) |
| config["BuildKeys"] = get_build_keys(config) |
| config["BuildVariant"] = get_build_variant(config) |
| |
| config["BuildFingerprint"] = args.build_fingerprint_file.read().strip() |
| config["BuildHostname"] = args.build_hostname_file.read().strip() |
| config["BuildNumber"] = args.build_number_file.read().strip() |
| config["BuildUsername"] = args.build_username |
| |
| build_version_tags_list = config["BuildVersionTags"] |
| if config["BuildType"] == "debug": |
| build_version_tags_list.append("debug") |
| build_version_tags_list.append(config["BuildKeys"]) |
| build_version_tags = ",".join(sorted(set(build_version_tags_list))) |
| config["BuildVersionTags"] = build_version_tags |
| |
| raw_date = args.date_file.read().strip() |
| config["Date"] = subprocess.check_output(["date", "-d", f"@{raw_date}"], text=True).strip() |
| config["DateUtc"] = subprocess.check_output(["date", "-d", f"@{raw_date}", "+%s"], text=True).strip() |
| |
| # build_desc is human readable strings that describe this build. This has the same info as the |
| # build fingerprint. |
| # e.g. "aosp_cf_x86_64_phone-userdebug VanillaIceCream MAIN eng.20240319.143939 test-keys" |
| config["BuildDesc"] = f"{config['DeviceProduct']}-{config['BuildVariant']} " \ |
| f"{config['Platform_version_name']} {config['BuildId']} " \ |
| f"{config['BuildNumber']} {config['BuildVersionTags']}" |
| |
| config["PlatformPreviewSdkFingerprint"] = args.platform_preview_sdk_fingerprint_file.read().strip() |
| |
| if args.build_thumbprint_file: |
| config["BuildThumbprint"] = args.build_thumbprint_file.read().strip() |
| |
| config["LineageDesc"] = config["BuildDesc"] |
| config["LineageDevice"] = config["DeviceName"] |
| |
| override_config(config) |
| |
| append_additional_system_props(args) |
| append_additional_vendor_props(args) |
| append_additional_product_props(args) |
| |
| return args |
| |
| def generate_common_build_props(args): |
| print("####################################") |
| print("# from generate_common_build_props") |
| print("# These properties identify this partition image.") |
| print("####################################") |
| |
| config = args.config |
| partition = args.partition |
| |
| if partition == "system": |
| print(f"ro.product.{partition}.brand={config['SystemBrand']}") |
| print(f"ro.product.{partition}.device={config['SystemDevice']}") |
| print(f"ro.product.{partition}.manufacturer={config['SystemManufacturer']}") |
| print(f"ro.product.{partition}.model={config['SystemModel']}") |
| print(f"ro.product.{partition}.name={config['SystemName']}") |
| else: |
| print(f"ro.product.{partition}.brand={config['ProductBrand']}") |
| print(f"ro.product.{partition}.device={config['DeviceName']}") |
| print(f"ro.product.{partition}.manufacturer={config['ProductManufacturer']}") |
| print(f"ro.product.{partition}.model={config['ProductModel']}") |
| print(f"ro.product.{partition}.name={config['DeviceProduct']}") |
| |
| if partition != "system": |
| if config["ProductModelForAttestation"]: |
| print(f"ro.product.model_for_attestation={config['ProductModelForAttestation']}") |
| if config["ProductBrandForAttestation"]: |
| print(f"ro.product.brand_for_attestation={config['ProductBrandForAttestation']}") |
| if config["ProductNameForAttestation"]: |
| print(f"ro.product.name_for_attestation={config['ProductNameForAttestation']}") |
| if config["ProductDeviceForAttestation"]: |
| print(f"ro.product.device_for_attestation={config['ProductDeviceForAttestation']}") |
| if config["ProductManufacturerForAttestation"]: |
| print(f"ro.product.manufacturer_for_attestation={config['ProductManufacturerForAttestation']}") |
| |
| if config["ZygoteForce64"]: |
| if partition == "vendor": |
| print(f"ro.{partition}.product.cpu.abilist={config['DeviceAbiList64']}") |
| print(f"ro.{partition}.product.cpu.abilist32=") |
| print(f"ro.{partition}.product.cpu.abilist64={config['DeviceAbiList64']}") |
| else: |
| if partition == "system" or partition == "vendor" or partition == "odm": |
| print(f"ro.{partition}.product.cpu.abilist={config['DeviceAbiList']}") |
| print(f"ro.{partition}.product.cpu.abilist32={config['DeviceAbiList32']}") |
| print(f"ro.{partition}.product.cpu.abilist64={config['DeviceAbiList64']}") |
| |
| print(f"ro.{partition}.build.date={config['Date']}") |
| print(f"ro.{partition}.build.date.utc={config['DateUtc']}") |
| # Allow optional assignments for ARC forward-declarations (b/249168657) |
| # TODO: Remove any tag-related inconsistencies once the goals from |
| # go/arc-android-sigprop-changes have been achieved. |
| print(f"ro.{partition}.build.fingerprint?={config['BuildFingerprint']}") |
| print(f"ro.{partition}.build.id?={config['BuildId']}") |
| print(f"ro.{partition}.build.tags?={config['BuildVersionTags']}") |
| print(f"ro.{partition}.build.type={config['BuildVariant']}") |
| print(f"ro.{partition}.build.version.incremental={config['BuildNumber']}") |
| print(f"ro.{partition}.build.version.release={config['Platform_version_last_stable']}") |
| print(f"ro.{partition}.build.version.release_or_codename={config['Platform_version_name']}") |
| print(f"ro.{partition}.build.version.sdk={config['Platform_sdk_version']}") |
| |
| def generate_build_info(args): |
| print() |
| print("####################################") |
| print("# from gen_build_prop.py:generate_build_info") |
| print("####################################") |
| print("# begin build properties") |
| |
| config = args.config |
| build_flags = config["BuildFlags"] |
| |
| print(f"ro.build.fingerprint?={config['BuildFingerprint']}") |
| |
| # The ro.build.id will be set dynamically by init, by appending the unique vbmeta digest. |
| if config["BoardUseVbmetaDigestInFingerprint"]: |
| print(f"ro.build.legacy.id={config['BuildId']}") |
| else: |
| print(f"ro.build.id?={config['BuildId']}") |
| |
| # ro.build.display.id is shown under Settings -> About Phone |
| if config["BuildVariant"] == "user": |
| # User builds should show: |
| # release build number or branch.buld_number non-release builds |
| |
| # Dev. branches should have DISPLAY_BUILD_NUMBER set |
| if config["DisplayBuildNumber"]: |
| print(f"ro.build.display.id?={config['BuildId']}.{config['BuildNumber']} {config['BuildKeys']}") |
| else: |
| print(f"ro.build.display.id?={config['BuildId']} {config['BuildKeys']}") |
| else: |
| # Non-user builds should show detailed build information (See build desc above) |
| print(f"ro.build.display.id?={config['LineageDesc']}") |
| print(f"ro.build.version.incremental={config['BuildNumber']}") |
| print(f"ro.build.version.sdk={config['Platform_sdk_version']}") |
| print(f"ro.build.version.preview_sdk={config['Platform_preview_sdk_version']}") |
| print(f"ro.build.version.preview_sdk_fingerprint={config['PlatformPreviewSdkFingerprint']}") |
| print(f"ro.build.version.codename={config['Platform_sdk_codename']}") |
| print(f"ro.build.version.all_codenames={','.join(config['Platform_version_active_codenames'])}") |
| print(f"ro.build.version.known_codenames={config['Platform_version_known_codenames']}") |
| print(f"ro.build.version.release={config['Platform_version_last_stable']}") |
| print(f"ro.build.version.release_or_codename={config['Platform_version_name']}") |
| print(f"ro.build.version.release_or_preview_display={config['Platform_display_version_name']}") |
| print(f"ro.build.version.security_patch={config['Platform_security_patch']}") |
| print(f"ro.build.version.base_os={config['Platform_base_os']}") |
| print(f"ro.build.version.min_supported_target_sdk={build_flags['RELEASE_PLATFORM_MIN_SUPPORTED_TARGET_SDK_VERSION']}") |
| print(f"ro.build.date={config['Date']}") |
| print(f"ro.build.date.utc={config['DateUtc']}") |
| print(f"ro.build.type={config['BuildVariant']}") |
| print(f"ro.build.user={config['BuildUsername']}") |
| print(f"ro.build.host={config['BuildHostname']}") |
| # TODO: Remove any tag-related optional property declarations once the goals |
| # from go/arc-android-sigprop-changes have been achieved. |
| print(f"ro.build.tags?={config['BuildVersionTags']}") |
| # ro.build.flavor are used only by the test harness to distinguish builds. |
| # Only add _asan for a sanitized build if it isn't already a part of the |
| # flavor (via a dedicated lunch config for example). |
| print(f"ro.build.flavor={config['BuildFlavor']}") |
| |
| print(f"ro.lineage.device={config['LineageDevice']}") |
| |
| # These values are deprecated, use "ro.product.cpu.abilist" |
| # instead (see below). |
| print(f"# ro.product.cpu.abi and ro.product.cpu.abi2 are obsolete,") |
| print(f"# use ro.product.cpu.abilist instead.") |
| print(f"ro.product.cpu.abi={config['DeviceAbi'][0]}") |
| if len(config["DeviceAbi"]) > 1: |
| print(f"ro.product.cpu.abi2={config['DeviceAbi'][1]}") |
| |
| if config["ProductLocales"]: |
| print(f"ro.product.locale={config['ProductLocales'][0]}") |
| print(f"ro.wifi.channels={' '.join(config['ProductDefaultWifiChannels'])}") |
| |
| print(f"# ro.build.product is obsolete; use ro.product.device") |
| print(f"ro.build.product={config['DeviceName']}") |
| |
| print(f"# Do not try to parse description or thumbprint") |
| print(f"ro.build.description?={config['BuildDesc']}") |
| if "BuildThumbprint" in config: |
| print(f"ro.build.thumbprint={config['BuildThumbprint']}") |
| |
| print(f"# end build properties") |
| |
| def write_properties_from_file(file): |
| print() |
| print("####################################") |
| print(f"# from {file.name}") |
| print("####################################") |
| print(file.read(), end="") |
| |
| def write_properties_from_variable(name, props, build_broken_dup_sysprop): |
| print() |
| print("####################################") |
| print(f"# from variable {name}") |
| print("####################################") |
| |
| # Implement the legacy behavior when BUILD_BROKEN_DUP_SYSPROP is on. |
| # Optional assignments are all converted to normal assignments and |
| # when their duplicates the first one wins. |
| if build_broken_dup_sysprop: |
| processed_props = [] |
| seen_props = set() |
| for line in props: |
| line = line.replace("?=", "=") |
| key, value = line.split("=", 1) |
| if key in seen_props: |
| continue |
| seen_props.add(key) |
| processed_props.append(line) |
| props = processed_props |
| |
| for line in props: |
| print(line) |
| |
| def append_additional_system_props(args): |
| props = [] |
| |
| config = args.config |
| |
| # Add the product-defined properties to the build properties. |
| if not config["PropertySplitEnabled"] or not config["VendorImageFileSystemType"]: |
| if "PRODUCT_PROPERTY_OVERRIDES" in config: |
| props += config["PRODUCT_PROPERTY_OVERRIDES"] |
| |
| props.append(f"ro.treble.enabled={'true' if config['FullTreble'] else 'false'}") |
| # Set ro.llndk.api_level to show the maximum vendor API level that the LLNDK |
| # in the system partition supports. |
| if config["VendorApiLevel"]: |
| props.append(f"ro.llndk.api_level={config['VendorApiLevel']}") |
| |
| # Sets ro.actionable_compatible_property.enabled to know on runtime whether |
| # the allowed list of actionable compatible properties is enabled or not. |
| props.append("ro.actionable_compatible_property.enabled=true") |
| |
| # Enable core platform API violation warnings on userdebug and eng builds. |
| if config["BuildVariant"] != "user": |
| props.append("persist.debug.dalvik.vm.core_platform_api_policy=just-warn") |
| |
| # Define ro.sanitize.<name> properties for all global sanitizers. |
| for sanitize_target in config["SanitizeDevice"]: |
| props.append(f"ro.sanitize.{sanitize_target}=true") |
| |
| # Sets the default value of ro.postinstall.fstab.prefix to /system. |
| # Device board config should override the value to /product when needed by: |
| # |
| # PRODUCT_PRODUCT_PROPERTIES += ro.postinstall.fstab.prefix=/product |
| # |
| # It then uses ${ro.postinstall.fstab.prefix}/etc/fstab.postinstall to |
| # mount system_other partition. |
| props.append("ro.postinstall.fstab.prefix=/system") |
| |
| enable_target_debugging = True |
| enable_dalvik_lock_contention_logging = True |
| if config["BuildVariant"] == "user" or config["BuildVariant"] == "userdebug": |
| # Target is secure in user builds. |
| props.append("ro.secure=1") |
| props.append("security.perf_harden=1") |
| |
| if config["BuildVariant"] == "user": |
| # Disable debugging in plain user builds. |
| props.append("ro.adb.secure=1") |
| enable_target_debugging = False |
| enable_dalvik_lock_contention_logging = False |
| else: |
| # Disable debugging in userdebug builds if PRODUCT_NOT_DEBUGGABLE_IN_USERDEBUG |
| # is set. |
| if config["ProductNotDebuggableInUserdebug"]: |
| enable_target_debugging = False |
| |
| # Disallow mock locations by default for user builds |
| props.append("ro.allow.mock.location=0") |
| else: |
| # Turn on checkjni for non-user builds. |
| props.append("ro.kernel.android.checkjni=1") |
| # Set device insecure for non-user builds. |
| props.append("ro.secure=0") |
| # Allow mock locations by default for non user builds |
| props.append("ro.allow.mock.location=1") |
| |
| if enable_dalvik_lock_contention_logging: |
| # Enable Dalvik lock contention logging. |
| props.append("dalvik.vm.lockprof.threshold=500") |
| |
| if enable_target_debugging: |
| # Target is more debuggable and adbd is on by default |
| props.append("ro.debuggable=1") |
| else: |
| # Target is less debuggable and adbd is off by default |
| props.append("ro.debuggable=0") |
| |
| if config["BuildVariant"] == "eng": |
| if "ro.setupwizard.mode=ENABLED" in props: |
| # Don't require the setup wizard on eng builds |
| props = list(filter(lambda x: not x.startswith("ro.setupwizard.mode="), props)) |
| props.append("ro.setupwizard.mode=OPTIONAL") |
| |
| if not config["SdkBuild"]: |
| # To speedup startup of non-preopted builds, don't verify or compile the boot image. |
| props.append("dalvik.vm.image-dex2oat-filter=extract") |
| # b/323566535 |
| props.append("init.svc_debug.no_fatal.zygote=true") |
| |
| if config["SdkBuild"]: |
| props.append("xmpp.auto-presence=true") |
| props.append("ro.config.nocheckin=yes") |
| |
| props.append("net.bt.name=Android") |
| |
| # This property is set by flashing debug boot image, so default to false. |
| props.append("ro.force.debuggable=0") |
| |
| config["ADDITIONAL_SYSTEM_PROPERTIES"] = props |
| |
| def append_additional_vendor_props(args): |
| props = [] |
| |
| config = args.config |
| build_flags = config["BuildFlags"] |
| |
| # Add cpu properties for bionic and ART. |
| props.append(f"ro.bionic.arch={config['DeviceArch']}") |
| props.append(f"ro.bionic.cpu_variant={config['DeviceCpuVariantRuntime']}") |
| props.append(f"ro.bionic.2nd_arch={config['DeviceSecondaryArch']}") |
| props.append(f"ro.bionic.2nd_cpu_variant={config['DeviceSecondaryCpuVariantRuntime']}") |
| |
| props.append(f"persist.sys.dalvik.vm.lib.2=libart.so") |
| props.append(f"dalvik.vm.isa.{config['DeviceArch']}.variant={config['Dex2oatTargetCpuVariantRuntime']}") |
| if config["Dex2oatTargetInstructionSetFeatures"]: |
| props.append(f"dalvik.vm.isa.{config['DeviceArch']}.features={config['Dex2oatTargetInstructionSetFeatures']}") |
| |
| if config["DeviceSecondaryArch"]: |
| props.append(f"dalvik.vm.isa.{config['DeviceSecondaryArch']}.variant={config['SecondaryDex2oatCpuVariantRuntime']}") |
| if config["SecondaryDex2oatInstructionSetFeatures"]: |
| props.append(f"dalvik.vm.isa.{config['DeviceSecondaryArch']}.features={config['SecondaryDex2oatInstructionSetFeatures']}") |
| |
| # Although these variables are prefixed with TARGET_RECOVERY_, they are also needed under charger |
| # mode (via libminui). |
| if config["RecoveryDefaultRotation"]: |
| props.append(f"ro.minui.default_rotation={config['RecoveryDefaultRotation']}") |
| |
| if config["RecoveryDefaultTouchRotation"]: |
| props.append(f"ro.minui.default_touch_rotation={config['RecoveryDefaultTouchRotation']}") |
| |
| if config["RecoveryOverscanPercent"]: |
| props.append(f"ro.minui.overscan_percent={config['RecoveryOverscanPercent']}") |
| |
| if config["RecoveryPixelFormat"]: |
| props.append(f"ro.minui.pixel_format={config['RecoveryPixelFormat']}") |
| |
| if "UseDynamicPartitions" in config: |
| props.append(f"ro.boot.dynamic_partitions={'true' if config['UseDynamicPartitions'] else 'false'}") |
| |
| if "RetrofitDynamicPartitions" in config: |
| props.append(f"ro.boot.dynamic_partitions_retrofit={'true' if config['RetrofitDynamicPartitions'] else 'false'}") |
| |
| if config["ShippingApiLevel"]: |
| props.append(f"ro.product.first_api_level={config['ShippingApiLevel']}") |
| |
| if config["ShippingVendorApiLevel"]: |
| props.append(f"ro.vendor.api_level={config['ShippingVendorApiLevel']}") |
| |
| if config["BuildVariant"] != "user" and config["BuildDebugfsRestrictionsEnabled"]: |
| props.append(f"ro.product.debugfs_restrictions.enabled=true") |
| |
| # Vendors with GRF must define BOARD_SHIPPING_API_LEVEL for the vendor API level. |
| # This must not be defined for the non-GRF devices. |
| # The values of the GRF properties will be verified by post_process_props.py |
| if config["BoardShippingApiLevel"]: |
| props.append(f"ro.board.first_api_level={config['BoardShippingApiLevel']}") |
| |
| # Build system set BOARD_API_LEVEL to show the api level of the vendor API surface. |
| # This must not be altered outside of build system. |
| if config["VendorApiLevel"]: |
| props.append(f"ro.board.api_level={config['VendorApiLevel']}") |
| |
| # RELEASE_BOARD_API_LEVEL_FROZEN is true when the vendor API surface is frozen. |
| if build_flags["RELEASE_BOARD_API_LEVEL_FROZEN"]: |
| props.append(f"ro.board.api_frozen=true") |
| |
| # Set build prop. This prop is read by ota_from_target_files when generating OTA, |
| # to decide if VABC should be disabled. |
| if config["DontUseVabcOta"]: |
| props.append(f"ro.vendor.build.dont_use_vabc=true") |
| |
| # Set the flag in vendor. So VTS would know if the new fingerprint format is in use when |
| # the system images are replaced by GSI. |
| if config["BoardUseVbmetaDigestInFingerprint"]: |
| props.append(f"ro.vendor.build.fingerprint_has_digest=1") |
| |
| props.append(f"ro.vendor.build.security_patch={config['VendorSecurityPatch']}") |
| props.append(f"ro.product.board={config['BootloaderBoardName']}") |
| props.append(f"ro.board.platform={config['BoardPlatform']}") |
| props.append(f"ro.hwui.use_vulkan={'true' if config['UsesVulkan'] else 'false'}") |
| |
| if config["ScreenDensity"]: |
| props.append(f"ro.sf.lcd_density={config['ScreenDensity']}") |
| |
| if "AbOtaUpdater" in config: |
| props.append(f"ro.build.ab_update={'true' if config['AbOtaUpdater'] else 'false'}") |
| if config["AbOtaUpdater"]: |
| props.append(f"ro.vendor.build.ab_ota_partitions={config['AbOtaPartitions']}") |
| |
| config["ADDITIONAL_VENDOR_PROPERTIES"] = props |
| |
| def append_additional_product_props(args): |
| props = [] |
| |
| config = args.config |
| |
| # Add the system server compiler filter if they are specified for the product. |
| if config["SystemServerCompilerFilter"]: |
| props.append(f"dalvik.vm.systemservercompilerfilter={config['SystemServerCompilerFilter']}") |
| |
| # Add the 16K developer args if it is defined for the product. |
| props.append(f"ro.product.build.16k_page.enabled={'true' if config['Product16KDeveloperOption'] else 'false'}") |
| |
| props.append(f"ro.product.page_size={16384 if config['TargetBoots16K'] else 4096}") |
| |
| props.append(f"ro.build.characteristics={config['AAPTCharacteristics']}") |
| |
| if "AbOtaUpdater" in config and config["AbOtaUpdater"]: |
| props.append(f"ro.product.ab_ota_partitions={config['AbOtaPartitions']}") |
| |
| # Set this property for VTS to skip large page size tests on unsupported devices. |
| props.append(f"ro.product.cpu.pagesize.max={config['DeviceMaxPageSizeSupported']}") |
| |
| if config["NoBionicPageSizeMacro"]: |
| props.append(f"ro.product.build.no_bionic_page_size_macro=true") |
| |
| # This is a temporary system property that controls the ART module. The plan is |
| # to remove it by Aug 2025, at which time Mainline updates of the ART module |
| # will ignore it as well. |
| # If the value is "default", it will be mangled by post_process_props.py. |
| props.append(f"ro.dalvik.vm.enable_uffd_gc={config['EnableUffdGc']}") |
| |
| config["ADDITIONAL_PRODUCT_PROPERTIES"] = props |
| |
| def build_system_prop(args): |
| config = args.config |
| |
| # Order matters here. When there are duplicates, the last one wins. |
| # TODO(b/117892318): don't allow duplicates so that the ordering doesn't matter |
| variables = [ |
| "ADDITIONAL_SYSTEM_PROPERTIES", |
| "PRODUCT_SYSTEM_PROPERTIES", |
| # TODO(b/117892318): deprecate this |
| "PRODUCT_SYSTEM_DEFAULT_PROPERTIES", |
| ] |
| |
| if not config["PropertySplitEnabled"]: |
| variables += [ |
| "ADDITIONAL_VENDOR_PROPERTIES", |
| "PRODUCT_VENDOR_PROPERTIES", |
| ] |
| |
| build_prop(args, gen_build_info=True, gen_common_build_props=True, variables=variables) |
| |
| def build_system_ext_prop(args): |
| config = args.config |
| |
| # Order matters here. When there are duplicates, the last one wins. |
| # TODO(b/117892318): don't allow duplicates so that the ordering doesn't matter |
| variables = ["PRODUCT_SYSTEM_EXT_PROPERTIES"] |
| |
| build_prop(args, gen_build_info=False, gen_common_build_props=True, variables=variables) |
| |
| ''' |
| def build_vendor_prop(args): |
| config = args.config |
| |
| # Order matters here. When there are duplicates, the last one wins. |
| # TODO(b/117892318): don't allow duplicates so that the ordering doesn't matter |
| variables = [] |
| if config["PropertySplitEnabled"]: |
| variables += [ |
| "ADDITIONAL_VENDOR_PROPERTIES", |
| "PRODUCT_VENDOR_PROPERTIES", |
| # TODO(b/117892318): deprecate this |
| "PRODUCT_DEFAULT_PROPERTY_OVERRIDES", |
| "PRODUCT_PROPERTY_OVERRIDES", |
| ] |
| |
| build_prop(args, gen_build_info=False, gen_common_build_props=True, variables=variables) |
| ''' |
| |
| def build_product_prop(args): |
| config = args.config |
| |
| # Order matters here. When there are duplicates, the last one wins. |
| # TODO(b/117892318): don't allow duplicates so that the ordering doesn't matter |
| variables = [ |
| "ADDITIONAL_PRODUCT_PROPERTIES", |
| "PRODUCT_PRODUCT_PROPERTIES", |
| ] |
| |
| gen_common_build_props = True |
| |
| # Skip common /product properties generation if device released before R and |
| # has no product partition. This is the first part of the check. |
| if config["Shipping_api_level"] and int(config["Shipping_api_level"]) < 30: |
| gen_common_build_props = False |
| |
| # The second part of the check - always generate common properties for the |
| # devices with product partition regardless of shipping level. |
| if config["UsesProductImage"]: |
| gen_common_build_props = True |
| |
| build_prop(args, gen_build_info=False, gen_common_build_props=True, variables=variables) |
| |
| if config["OemProperties"]: |
| print("####################################") |
| print("# PRODUCT_OEM_PROPERTIES") |
| print("####################################") |
| |
| for prop in config["OemProperties"]: |
| print(f"import /oem/oem.prop {prop}") |
| |
| def build_odm_prop(args): |
| variables = ["ADDITIONAL_ODM_PROPERTIES", "PRODUCT_ODM_PROPERTIES"] |
| build_prop(args, gen_build_info=False, gen_common_build_props=True, variables=variables) |
| |
| def build_prop(args, gen_build_info, gen_common_build_props, variables): |
| config = args.config |
| |
| if gen_common_build_props: |
| generate_common_build_props(args) |
| |
| if gen_build_info: |
| generate_build_info(args) |
| |
| for prop_file in args.prop_files: |
| write_properties_from_file(prop_file) |
| |
| for variable in variables: |
| if variable in config: |
| write_properties_from_variable(variable, config[variable], args.build_broken_dup_sysprop) |
| |
| def main(): |
| args = parse_args() |
| |
| with contextlib.redirect_stdout(args.out): |
| match args.partition: |
| case "system": |
| build_system_prop(args) |
| case "system_ext": |
| build_system_ext_prop(args) |
| case "odm": |
| build_odm_prop(args) |
| case "product": |
| build_product_prop(args) |
| # case "vendor": # NOT IMPLEMENTED |
| # build_vendor_prop(args) |
| case _: |
| sys.exit(f"not supported partition {args.partition}") |
| |
| if __name__ == "__main__": |
| main() |