# Copyright (C) 2023 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. """ Export build flags (with values) to make. """ load("//build/bazel/utils:schema_validation.bzl", "validate") # Partitions that get build system flag summaries _flag_partitions = [ "product", "system", "system_ext", "vendor", ] ALL = ["all"] PRODUCT = ["product"] SYSTEM = ["system"] SYSTEM_EXT = ["system_ext"] VENDOR = ["vendor"] _valid_types = ["NoneType", "bool", "list", "string", "int"] _all_flags_schema = { "type": "list", "of": { "type": "dict", "required_keys": { "name": {"type": "string"}, "partitions": { "type": "list", "of": { "type": "string", "choices": _flag_partitions + ["all"], }, "unique": True, }, "default": { "or": [ {"type": t} for t in _valid_types ], }, "declared_in": {"type": "string"}, }, }, } _all_values_schema = { "type": "list", "of": { "type": "dict", "required_keys": { "name": {"type": "string"}, "value": { "or": [ {"type": t} for t in _valid_types ], }, "set_in": {"type": "string"}, }, }, } def flag(name, partitions, default): """Declare a flag. Args: name: name of the flag partitions: the partitions where this should be recorded. default: the default value of the flag. Returns: A dictionary containing the flag declaration. """ if not partitions: fail("At least 1 partition is required") if not name.startswith("RELEASE_"): fail("Release flag names must start with RELEASE_") if " " in name or "\t" in name or "\n" in name: fail("Flag names must not contain whitespace: \"" + name + "\"") for partition in partitions: if partition == "all": if len(partitions) > 1: fail("\"all\" can't be combined with other partitions: " + str(partitions)) elif partition not in _flag_partitions: fail("Invalid partition: " + partition + ", allowed partitions: " + str(_flag_partitions)) if type(default) not in _valid_types: fail("Invalid type of default for flag \"" + name + "\" (" + type(default) + ")") return { "name": name, "partitions": partitions, "default": default, } def value(name, value): """Define the flag value for a particular configuration. Args: name: The name of the flag. value: The value for the flag. Returns: A dictionary containing the name and value to be used. """ return { "name": name, "value": value, } def _format_value(val): """Format the starlark type correctly for make. Args: val: The value to format Returns: The value, formatted correctly for make. """ if type(val) == "NoneType": return "" elif type(val) == "bool": return "true" if val else "" else: return val def release_config(all_flags, all_values): """Return the make variables that should be set for this release config. Args: all_flags: A list of flag objects (from flag() calls). all_values: A list of value objects (from value() calls). Returns: A dictionary of {name: value} variables for make. """ validate(all_flags, _all_flags_schema) validate(all_values, _all_values_schema) # Validate flags flag_names = [] for flag in all_flags: if flag["name"] in flag_names: fail(flag["declared_in"] + ": Duplicate declaration of flag " + flag["name"]) flag_names.append(flag["name"]) # Record which flags go on which partition partitions = {} for flag in all_flags: for partition in flag["partitions"]: if partition == "all": if len(flag["partitions"]) > 1: fail("\"all\" can't be combined with other partitions: " + str(flag["partitions"])) for partition in _flag_partitions: partitions.setdefault(partition, []).append(flag["name"]) else: partitions.setdefault(partition, []).append(flag["name"]) # Validate values # TODO(joeo): Disallow duplicate values after we've split AOSP and vendor flags. values = {} for value in all_values: if value["name"] not in flag_names: fail(value["set_in"] + ": Value set for undeclared build flag: " + value["name"]) values[value["name"]] = value # Collect values result = { "_ALL_RELEASE_FLAGS": sorted(flag_names), } for partition, names in partitions.items(): result["_ALL_RELEASE_FLAGS.PARTITIONS." + partition] = names for flag in all_flags: if flag["name"] in values: val = values[flag["name"]]["value"] set_in = values[flag["name"]]["set_in"] else: val = flag["default"] set_in = flag["declared_in"] val = _format_value(val) result[flag["name"]] = val result["_ALL_RELEASE_FLAGS." + flag["name"] + ".PARTITIONS"] = flag["partitions"] result["_ALL_RELEASE_FLAGS." + flag["name"] + ".DEFAULT"] = _format_value(flag["default"]) result["_ALL_RELEASE_FLAGS." + flag["name"] + ".VALUE"] = val result["_ALL_RELEASE_FLAGS." + flag["name"] + ".DECLARED_IN"] = flag["declared_in"] result["_ALL_RELEASE_FLAGS." + flag["name"] + ".SET_IN"] = set_in return result