| // Copyright 2015 Google Inc. All rights reserved. |
| // |
| // 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. |
| |
| package android |
| |
| // This is the primary location to write and read all configuration values and |
| // product variables necessary for soong_build's operation. |
| |
| import ( |
| "encoding/json" |
| "fmt" |
| "os" |
| "path/filepath" |
| "runtime" |
| "strconv" |
| "strings" |
| "sync" |
| "unicode" |
| |
| "android/soong/shared" |
| |
| "github.com/google/blueprint" |
| "github.com/google/blueprint/bootstrap" |
| "github.com/google/blueprint/pathtools" |
| "github.com/google/blueprint/proptools" |
| |
| "android/soong/android/soongconfig" |
| "android/soong/remoteexec" |
| ) |
| |
| // Bool re-exports proptools.Bool for the android package. |
| var Bool = proptools.Bool |
| |
| // String re-exports proptools.String for the android package. |
| var String = proptools.String |
| |
| // StringDefault re-exports proptools.StringDefault for the android package. |
| var StringDefault = proptools.StringDefault |
| |
| // FutureApiLevelInt is a placeholder constant for unreleased API levels. |
| const FutureApiLevelInt = 10000 |
| |
| // PrivateApiLevel represents the api level of SdkSpecPrivate (sdk_version: "") |
| // This api_level exists to differentiate user-provided "" from "current" sdk_version |
| // The differentiation is necessary to enable different validation rules for these two possible values. |
| var PrivateApiLevel = ApiLevel{ |
| value: "current", // The value is current since aidl expects `current` as the default (TestAidlFlagsWithMinSdkVersion) |
| number: FutureApiLevelInt + 1, // This is used to differentiate it from FutureApiLevel |
| isPreview: true, |
| } |
| |
| // FutureApiLevel represents unreleased API levels. |
| var FutureApiLevel = ApiLevel{ |
| value: "current", |
| number: FutureApiLevelInt, |
| isPreview: true, |
| } |
| |
| // The product variables file name, containing product config from Kati. |
| const productVariablesFileName = "soong.variables" |
| |
| // A Config object represents the entire build configuration for Android. |
| type Config struct { |
| *config |
| } |
| |
| type SoongBuildMode int |
| |
| type CmdArgs struct { |
| bootstrap.Args |
| RunGoTests bool |
| OutDir string |
| SoongOutDir string |
| SoongVariables string |
| |
| BazelQueryViewDir string |
| ModuleGraphFile string |
| ModuleActionsFile string |
| DocFile string |
| |
| BuildFromSourceStub bool |
| |
| EnsureAllowlistIntegrity bool |
| } |
| |
| // Build modes that soong_build can run as. |
| const ( |
| // Don't use bazel at all during module analysis. |
| AnalysisNoBazel SoongBuildMode = iota |
| |
| // Generate BUILD files which faithfully represent the dependency graph of |
| // blueprint modules. Individual BUILD targets will not, however, faitfhully |
| // express build semantics. |
| GenerateQueryView |
| |
| // Create a JSON representation of the module graph and exit. |
| GenerateModuleGraph |
| |
| // Generate a documentation file for module type definitions and exit. |
| GenerateDocFile |
| ) |
| |
| const testKeyDir = "build/make/target/product/security" |
| |
| // SoongOutDir returns the build output directory for the configuration. |
| func (c Config) SoongOutDir() string { |
| return c.soongOutDir |
| } |
| |
| // tempDir returns the path to out/soong/.temp, which is cleared at the beginning of every build. |
| func (c Config) tempDir() string { |
| return shared.TempDirForOutDir(c.soongOutDir) |
| } |
| |
| func (c Config) OutDir() string { |
| return c.outDir |
| } |
| |
| func (c Config) RunGoTests() bool { |
| return c.runGoTests |
| } |
| |
| func (c Config) DebugCompilation() bool { |
| return false // Never compile Go code in the main build for debugging |
| } |
| |
| func (c Config) Subninjas() []string { |
| return []string{} |
| } |
| |
| func (c Config) PrimaryBuilderInvocations() []bootstrap.PrimaryBuilderInvocation { |
| return []bootstrap.PrimaryBuilderInvocation{} |
| } |
| |
| // RunningInsideUnitTest returns true if this code is being run as part of a Soong unit test. |
| func (c Config) RunningInsideUnitTest() bool { |
| return c.config.TestProductVariables != nil |
| } |
| |
| // DisableHiddenApiChecks returns true if hiddenapi checks have been disabled. |
| // For 'eng' target variant hiddenapi checks are disabled by default for performance optimisation, |
| // but can be enabled by setting environment variable ENABLE_HIDDENAPI_FLAGS=true. |
| // For other target variants hiddenapi check are enabled by default but can be disabled by |
| // setting environment variable UNSAFE_DISABLE_HIDDENAPI_FLAGS=true. |
| // If both ENABLE_HIDDENAPI_FLAGS=true and UNSAFE_DISABLE_HIDDENAPI_FLAGS=true, then |
| // ENABLE_HIDDENAPI_FLAGS=true will be triggered and hiddenapi checks will be considered enabled. |
| func (c Config) DisableHiddenApiChecks() bool { |
| return !c.IsEnvTrue("ENABLE_HIDDENAPI_FLAGS") && |
| (c.IsEnvTrue("UNSAFE_DISABLE_HIDDENAPI_FLAGS") || |
| Bool(c.productVariables.Eng)) |
| } |
| |
| // DisableVerifyOverlaps returns true if verify_overlaps is skipped. |
| // Mismatch in version of apexes and module SDK is required for mainline prebuilts to work in |
| // trunk stable. |
| // Thus, verify_overlaps is disabled when RELEASE_DEFAULT_MODULE_BUILD_FROM_SOURCE is set to false. |
| // TODO(b/308174018): Re-enable verify_overlaps for both builr from source/mainline prebuilts. |
| func (c Config) DisableVerifyOverlaps() bool { |
| if c.IsEnvFalse("DISABLE_VERIFY_OVERLAPS") && c.ReleaseDisableVerifyOverlaps() { |
| panic("The current release configuration does not support verify_overlaps. DISABLE_VERIFY_OVERLAPS cannot be set to false") |
| } |
| return c.IsEnvTrue("DISABLE_VERIFY_OVERLAPS") || c.ReleaseDisableVerifyOverlaps() || !c.ReleaseDefaultModuleBuildFromSource() |
| } |
| |
| // MaxPageSizeSupported returns the max page size supported by the device. This |
| // value will define the ELF segment alignment for binaries (executables and |
| // shared libraries). |
| func (c Config) MaxPageSizeSupported() string { |
| return String(c.config.productVariables.DeviceMaxPageSizeSupported) |
| } |
| |
| // NoBionicPageSizeMacro returns true when AOSP is page size agnostic. |
| // This means that the bionic's macro PAGE_SIZE won't be defined. |
| // Returns false when AOSP is NOT page size agnostic. |
| // This means that bionic's macro PAGE_SIZE is defined. |
| func (c Config) NoBionicPageSizeMacro() bool { |
| return Bool(c.config.productVariables.DeviceNoBionicPageSizeMacro) |
| } |
| |
| // The release version passed to aconfig, derived from RELEASE_VERSION |
| func (c Config) ReleaseVersion() string { |
| return c.config.productVariables.ReleaseVersion |
| } |
| |
| // The aconfig value set passed to aconfig, derived from RELEASE_VERSION |
| func (c Config) ReleaseAconfigValueSets() []string { |
| return c.config.productVariables.ReleaseAconfigValueSets |
| } |
| |
| // The flag default permission value passed to aconfig |
| // derived from RELEASE_ACONFIG_FLAG_DEFAULT_PERMISSION |
| func (c Config) ReleaseAconfigFlagDefaultPermission() string { |
| return c.config.productVariables.ReleaseAconfigFlagDefaultPermission |
| } |
| |
| // The flag indicating behavior for the tree wrt building modules or using prebuilts |
| // derived from RELEASE_DEFAULT_MODULE_BUILD_FROM_SOURCE |
| func (c Config) ReleaseDefaultModuleBuildFromSource() bool { |
| return c.config.productVariables.ReleaseDefaultModuleBuildFromSource == nil || |
| Bool(c.config.productVariables.ReleaseDefaultModuleBuildFromSource) |
| } |
| |
| func (c Config) ReleaseDisableVerifyOverlaps() bool { |
| return c.config.productVariables.GetBuildFlagBool("RELEASE_DISABLE_VERIFY_OVERLAPS_CHECK") |
| } |
| |
| // Enables flagged apis annotated with READ_WRITE aconfig flags to be included in the stubs |
| // and hiddenapi flags so that they are accessible at runtime |
| func (c Config) ReleaseExportRuntimeApis() bool { |
| return Bool(c.config.productVariables.ExportRuntimeApis) |
| } |
| |
| // Enables ABI monitoring of NDK libraries |
| func (c Config) ReleaseNdkAbiMonitored() bool { |
| return c.config.productVariables.GetBuildFlagBool("RELEASE_NDK_ABI_MONITORED") |
| } |
| |
| // Enable read flag from new storage, for C/C++ |
| func (c Config) ReleaseReadFromNewStorageCc() bool { |
| return c.config.productVariables.GetBuildFlagBool("RELEASE_READ_FROM_NEW_STORAGE_CC") |
| } |
| |
| func (c Config) ReleaseHiddenApiExportableStubs() bool { |
| return c.config.productVariables.GetBuildFlagBool("RELEASE_HIDDEN_API_EXPORTABLE_STUBS") || |
| Bool(c.config.productVariables.HiddenapiExportableStubs) |
| } |
| |
| // Enable read flag from new storage |
| func (c Config) ReleaseReadFromNewStorage() bool { |
| return c.config.productVariables.GetBuildFlagBool("RELEASE_READ_FROM_NEW_STORAGE") |
| } |
| |
| // A DeviceConfig object represents the configuration for a particular device |
| // being built. For now there will only be one of these, but in the future there |
| // may be multiple devices being built. |
| type DeviceConfig struct { |
| *deviceConfig |
| } |
| |
| // VendorConfig represents the configuration for vendor-specific behavior. |
| type VendorConfig soongconfig.SoongConfig |
| |
| // Definition of general build configuration for soong_build. Some of these |
| // product configuration values are read from Kati-generated soong.variables. |
| type config struct { |
| // Options configurable with soong.variables |
| productVariables ProductVariables |
| |
| // Only available on configs created by TestConfig |
| TestProductVariables *ProductVariables |
| |
| ProductVariablesFileName string |
| |
| // BuildOS stores the OsType for the OS that the build is running on. |
| BuildOS OsType |
| |
| // BuildArch stores the ArchType for the CPU that the build is running on. |
| BuildArch ArchType |
| |
| Targets map[OsType][]Target |
| BuildOSTarget Target // the Target for tools run on the build machine |
| BuildOSCommonTarget Target // the Target for common (java) tools run on the build machine |
| AndroidCommonTarget Target // the Target for common modules for the Android device |
| AndroidFirstDeviceTarget Target // the first Target for modules for the Android device |
| |
| // multilibConflicts for an ArchType is true if there is earlier configured |
| // device architecture with the same multilib value. |
| multilibConflicts map[ArchType]bool |
| |
| deviceConfig *deviceConfig |
| |
| outDir string // The output directory (usually out/) |
| soongOutDir string |
| moduleListFile string // the path to the file which lists blueprint files to parse. |
| |
| runGoTests bool |
| |
| env map[string]string |
| envLock sync.Mutex |
| envDeps map[string]string |
| envFrozen bool |
| |
| // Changes behavior based on whether Kati runs after soong_build, or if soong_build |
| // runs standalone. |
| katiEnabled bool |
| |
| captureBuild bool // true for tests, saves build parameters for each module |
| ignoreEnvironment bool // true for tests, returns empty from all Getenv calls |
| |
| fs pathtools.FileSystem |
| mockBpList string |
| |
| BuildMode SoongBuildMode |
| |
| // If testAllowNonExistentPaths is true then PathForSource and PathForModuleSrc won't error |
| // in tests when a path doesn't exist. |
| TestAllowNonExistentPaths bool |
| |
| // The list of files that when changed, must invalidate soong_build to |
| // regenerate build.ninja. |
| ninjaFileDepsSet sync.Map |
| |
| OncePer |
| |
| // If buildFromSourceStub is true then the Java API stubs are |
| // built from the source Java files, not the signature text files. |
| buildFromSourceStub bool |
| |
| // If ensureAllowlistIntegrity is true, then the presence of any allowlisted |
| // modules that aren't mixed-built for at least one variant will cause a build |
| // failure |
| ensureAllowlistIntegrity bool |
| |
| // List of Api libraries that contribute to Api surfaces. |
| apiLibraries map[string]struct{} |
| } |
| |
| type deviceConfig struct { |
| config *config |
| OncePer |
| } |
| |
| type jsonConfigurable interface { |
| SetDefaultConfig() |
| } |
| |
| func loadConfig(config *config) error { |
| return loadFromConfigFile(&config.productVariables, absolutePath(config.ProductVariablesFileName)) |
| } |
| |
| // Checks if the string is a valid go identifier. This is equivalent to blueprint's definition |
| // of an identifier, so it will match the same identifiers as those that can be used in bp files. |
| func isGoIdentifier(ident string) bool { |
| for i, r := range ident { |
| valid := r == '_' || unicode.IsLetter(r) || unicode.IsDigit(r) && i > 0 |
| if !valid { |
| return false |
| } |
| } |
| return len(ident) > 0 |
| } |
| |
| // loadFromConfigFile loads and decodes configuration options from a JSON file |
| // in the current working directory. |
| func loadFromConfigFile(configurable *ProductVariables, filename string) error { |
| // Try to open the file |
| configFileReader, err := os.Open(filename) |
| defer configFileReader.Close() |
| if os.IsNotExist(err) { |
| // Need to create a file, so that blueprint & ninja don't get in |
| // a dependency tracking loop. |
| // Make a file-configurable-options with defaults, write it out using |
| // a json writer. |
| configurable.SetDefaultConfig() |
| err = saveToConfigFile(configurable, filename) |
| if err != nil { |
| return err |
| } |
| } else if err != nil { |
| return fmt.Errorf("config file: could not open %s: %s", filename, err.Error()) |
| } else { |
| // Make a decoder for it |
| jsonDecoder := json.NewDecoder(configFileReader) |
| jsonDecoder.DisallowUnknownFields() |
| err = jsonDecoder.Decode(configurable) |
| if err != nil { |
| return fmt.Errorf("config file: %s did not parse correctly: %s", filename, err.Error()) |
| } |
| } |
| |
| if Bool(configurable.GcovCoverage) && Bool(configurable.ClangCoverage) { |
| return fmt.Errorf("GcovCoverage and ClangCoverage cannot both be set") |
| } |
| |
| configurable.Native_coverage = proptools.BoolPtr( |
| Bool(configurable.GcovCoverage) || |
| Bool(configurable.ClangCoverage)) |
| |
| // The go scanner's definition of identifiers is c-style identifiers, but allowing unicode's |
| // definition of letters and digits. This is the same scanner that blueprint uses, so it |
| // will allow the same identifiers as are valid in bp files. |
| for namespace := range configurable.VendorVars { |
| if !isGoIdentifier(namespace) { |
| return fmt.Errorf("soong config namespaces must be valid identifiers: %q", namespace) |
| } |
| for variable := range configurable.VendorVars[namespace] { |
| if !isGoIdentifier(variable) { |
| return fmt.Errorf("soong config variables must be valid identifiers: %q", variable) |
| } |
| } |
| } |
| |
| // when Platform_sdk_final is true (or PLATFORM_VERSION_CODENAME is REL), use Platform_sdk_version; |
| // if false (pre-released version, for example), use Platform_sdk_codename. |
| if Bool(configurable.Platform_sdk_final) { |
| if configurable.Platform_sdk_version != nil { |
| configurable.Platform_sdk_version_or_codename = |
| proptools.StringPtr(strconv.Itoa(*(configurable.Platform_sdk_version))) |
| } else { |
| return fmt.Errorf("Platform_sdk_version cannot be pointed by a NULL pointer") |
| } |
| } else { |
| configurable.Platform_sdk_version_or_codename = |
| proptools.StringPtr(String(configurable.Platform_sdk_codename)) |
| } |
| |
| return nil |
| } |
| |
| // atomically writes the config file in case two copies of soong_build are running simultaneously |
| // (for example, docs generation and ninja manifest generation) |
| func saveToConfigFile(config *ProductVariables, filename string) error { |
| data, err := json.MarshalIndent(&config, "", " ") |
| if err != nil { |
| return fmt.Errorf("cannot marshal config data: %s", err.Error()) |
| } |
| |
| f, err := os.CreateTemp(filepath.Dir(filename), "config") |
| if err != nil { |
| return fmt.Errorf("cannot create empty config file %s: %s", filename, err.Error()) |
| } |
| defer os.Remove(f.Name()) |
| defer f.Close() |
| |
| _, err = f.Write(data) |
| if err != nil { |
| return fmt.Errorf("default config file: %s could not be written: %s", filename, err.Error()) |
| } |
| |
| _, err = f.WriteString("\n") |
| if err != nil { |
| return fmt.Errorf("default config file: %s could not be written: %s", filename, err.Error()) |
| } |
| |
| f.Close() |
| os.Rename(f.Name(), filename) |
| |
| return nil |
| } |
| |
| // NullConfig returns a mostly empty Config for use by standalone tools like dexpreopt_gen that |
| // use the android package. |
| func NullConfig(outDir, soongOutDir string) Config { |
| return Config{ |
| config: &config{ |
| outDir: outDir, |
| soongOutDir: soongOutDir, |
| fs: pathtools.OsFs, |
| }, |
| } |
| } |
| |
| // NewConfig creates a new Config object. The srcDir argument specifies the path |
| // to the root source directory. It also loads the config file, if found. |
| func NewConfig(cmdArgs CmdArgs, availableEnv map[string]string) (Config, error) { |
| // Make a config with default options. |
| config := &config{ |
| ProductVariablesFileName: cmdArgs.SoongVariables, |
| |
| env: availableEnv, |
| |
| outDir: cmdArgs.OutDir, |
| soongOutDir: cmdArgs.SoongOutDir, |
| runGoTests: cmdArgs.RunGoTests, |
| multilibConflicts: make(map[ArchType]bool), |
| |
| moduleListFile: cmdArgs.ModuleListFile, |
| fs: pathtools.NewOsFs(absSrcDir), |
| |
| buildFromSourceStub: cmdArgs.BuildFromSourceStub, |
| } |
| |
| config.deviceConfig = &deviceConfig{ |
| config: config, |
| } |
| |
| // Soundness check of the build and source directories. This won't catch strange |
| // configurations with symlinks, but at least checks the obvious case. |
| absBuildDir, err := filepath.Abs(cmdArgs.SoongOutDir) |
| if err != nil { |
| return Config{}, err |
| } |
| |
| absSrcDir, err := filepath.Abs(".") |
| if err != nil { |
| return Config{}, err |
| } |
| |
| if strings.HasPrefix(absSrcDir, absBuildDir) { |
| return Config{}, fmt.Errorf("Build dir must not contain source directory") |
| } |
| |
| // Load any configurable options from the configuration file |
| err = loadConfig(config) |
| if err != nil { |
| return Config{}, err |
| } |
| |
| KatiEnabledMarkerFile := filepath.Join(cmdArgs.SoongOutDir, ".soong.kati_enabled") |
| if _, err := os.Stat(absolutePath(KatiEnabledMarkerFile)); err == nil { |
| config.katiEnabled = true |
| } |
| |
| determineBuildOS(config) |
| |
| // Sets up the map of target OSes to the finer grained compilation targets |
| // that are configured from the product variables. |
| targets, err := decodeTargetProductVariables(config) |
| if err != nil { |
| return Config{}, err |
| } |
| |
| // Make the CommonOS OsType available for all products. |
| targets[CommonOS] = []Target{commonTargetMap[CommonOS.Name]} |
| |
| var archConfig []archConfig |
| if config.NdkAbis() { |
| archConfig = getNdkAbisConfig() |
| } else if config.AmlAbis() { |
| archConfig = getAmlAbisConfig() |
| } |
| |
| if archConfig != nil { |
| androidTargets, err := decodeAndroidArchSettings(archConfig) |
| if err != nil { |
| return Config{}, err |
| } |
| targets[Android] = androidTargets |
| } |
| |
| multilib := make(map[string]bool) |
| for _, target := range targets[Android] { |
| if seen := multilib[target.Arch.ArchType.Multilib]; seen { |
| config.multilibConflicts[target.Arch.ArchType] = true |
| } |
| multilib[target.Arch.ArchType.Multilib] = true |
| } |
| |
| // Map of OS to compilation targets. |
| config.Targets = targets |
| |
| // Compilation targets for host tools. |
| config.BuildOSTarget = config.Targets[config.BuildOS][0] |
| config.BuildOSCommonTarget = getCommonTargets(config.Targets[config.BuildOS])[0] |
| |
| // Compilation targets for Android. |
| if len(config.Targets[Android]) > 0 { |
| config.AndroidCommonTarget = getCommonTargets(config.Targets[Android])[0] |
| config.AndroidFirstDeviceTarget = FirstTarget(config.Targets[Android], "lib64", "lib32")[0] |
| } |
| |
| setBuildMode := func(arg string, mode SoongBuildMode) { |
| if arg != "" { |
| if config.BuildMode != AnalysisNoBazel { |
| fmt.Fprintf(os.Stderr, "buildMode is already set, illegal argument: %s", arg) |
| os.Exit(1) |
| } |
| config.BuildMode = mode |
| } |
| } |
| setBuildMode(cmdArgs.BazelQueryViewDir, GenerateQueryView) |
| setBuildMode(cmdArgs.ModuleGraphFile, GenerateModuleGraph) |
| setBuildMode(cmdArgs.DocFile, GenerateDocFile) |
| |
| // TODO(b/276958307): Replace the hardcoded list to a sdk_library local prop. |
| config.apiLibraries = map[string]struct{}{ |
| "android.net.ipsec.ike": {}, |
| "art.module.public.api": {}, |
| "conscrypt.module.public.api": {}, |
| "framework-adservices": {}, |
| "framework-appsearch": {}, |
| "framework-bluetooth": {}, |
| "framework-configinfrastructure": {}, |
| "framework-connectivity": {}, |
| "framework-connectivity-t": {}, |
| "framework-devicelock": {}, |
| "framework-graphics": {}, |
| "framework-healthfitness": {}, |
| "framework-location": {}, |
| "framework-media": {}, |
| "framework-mediaprovider": {}, |
| "framework-nfc": {}, |
| "framework-ondevicepersonalization": {}, |
| "framework-pdf": {}, |
| "framework-pdf-v": {}, |
| "framework-permission": {}, |
| "framework-permission-s": {}, |
| "framework-scheduling": {}, |
| "framework-sdkextensions": {}, |
| "framework-statsd": {}, |
| "framework-sdksandbox": {}, |
| "framework-tethering": {}, |
| "framework-uwb": {}, |
| "framework-virtualization": {}, |
| "framework-wifi": {}, |
| "i18n.module.public.api": {}, |
| } |
| |
| config.productVariables.Build_from_text_stub = boolPtr(config.BuildFromTextStub()) |
| |
| return Config{config}, err |
| } |
| |
| // mockFileSystem replaces all reads with accesses to the provided map of |
| // filenames to contents stored as a byte slice. |
| func (c *config) mockFileSystem(bp string, fs map[string][]byte) { |
| mockFS := map[string][]byte{} |
| |
| if _, exists := mockFS["Android.bp"]; !exists { |
| mockFS["Android.bp"] = []byte(bp) |
| } |
| |
| for k, v := range fs { |
| mockFS[k] = v |
| } |
| |
| // no module list file specified; find every file named Blueprints or Android.bp |
| pathsToParse := []string{} |
| for candidate := range mockFS { |
| base := filepath.Base(candidate) |
| if base == "Android.bp" { |
| pathsToParse = append(pathsToParse, candidate) |
| } |
| } |
| if len(pathsToParse) < 1 { |
| panic(fmt.Sprintf("No Blueprint or Android.bp files found in mock filesystem: %v\n", mockFS)) |
| } |
| mockFS[blueprint.MockModuleListFile] = []byte(strings.Join(pathsToParse, "\n")) |
| |
| c.fs = pathtools.MockFs(mockFS) |
| c.mockBpList = blueprint.MockModuleListFile |
| } |
| |
| func (c *config) SetAllowMissingDependencies() { |
| c.productVariables.Allow_missing_dependencies = proptools.BoolPtr(true) |
| } |
| |
| // BlueprintToolLocation returns the directory containing build system tools |
| // from Blueprint, like soong_zip and merge_zips. |
| func (c *config) HostToolDir() string { |
| if c.KatiEnabled() { |
| return filepath.Join(c.outDir, "host", c.PrebuiltOS(), "bin") |
| } else { |
| return filepath.Join(c.soongOutDir, "host", c.PrebuiltOS(), "bin") |
| } |
| } |
| |
| func (c *config) HostToolPath(ctx PathContext, tool string) Path { |
| path := pathForInstall(ctx, ctx.Config().BuildOS, ctx.Config().BuildArch, "bin", tool) |
| return path |
| } |
| |
| func (c *config) HostJNIToolPath(ctx PathContext, lib string) Path { |
| ext := ".so" |
| if runtime.GOOS == "darwin" { |
| ext = ".dylib" |
| } |
| path := pathForInstall(ctx, ctx.Config().BuildOS, ctx.Config().BuildArch, "lib64", lib+ext) |
| return path |
| } |
| |
| func (c *config) HostJavaToolPath(ctx PathContext, tool string) Path { |
| path := pathForInstall(ctx, ctx.Config().BuildOS, ctx.Config().BuildArch, "framework", tool) |
| return path |
| } |
| |
| func (c *config) HostCcSharedLibPath(ctx PathContext, lib string) Path { |
| libDir := "lib" |
| if ctx.Config().BuildArch.Multilib == "lib64" { |
| libDir = "lib64" |
| } |
| return pathForInstall(ctx, ctx.Config().BuildOS, ctx.Config().BuildArch, libDir, lib+".so") |
| } |
| |
| // PrebuiltOS returns the name of the host OS used in prebuilts directories. |
| func (c *config) PrebuiltOS() string { |
| switch runtime.GOOS { |
| case "linux": |
| return "linux-x86" |
| case "darwin": |
| return "darwin-x86" |
| default: |
| panic("Unknown GOOS") |
| } |
| } |
| |
| // GoRoot returns the path to the root directory of the Go toolchain. |
| func (c *config) GoRoot() string { |
| return fmt.Sprintf("prebuilts/go/%s", c.PrebuiltOS()) |
| } |
| |
| // PrebuiltBuildTool returns the path to a tool in the prebuilts directory containing |
| // checked-in tools, like Kati, Ninja or Toybox, for the current host OS. |
| func (c *config) PrebuiltBuildTool(ctx PathContext, tool string) Path { |
| return PathForSource(ctx, "prebuilts/build-tools", c.PrebuiltOS(), "bin", tool) |
| } |
| |
| // CpPreserveSymlinksFlags returns the host-specific flag for the cp(1) command |
| // to preserve symlinks. |
| func (c *config) CpPreserveSymlinksFlags() string { |
| switch runtime.GOOS { |
| case "darwin": |
| return "-R" |
| case "linux": |
| return "-d" |
| default: |
| return "" |
| } |
| } |
| |
| func (c *config) Getenv(key string) string { |
| var val string |
| var exists bool |
| c.envLock.Lock() |
| defer c.envLock.Unlock() |
| if c.envDeps == nil { |
| c.envDeps = make(map[string]string) |
| } |
| if val, exists = c.envDeps[key]; !exists { |
| if c.envFrozen { |
| panic("Cannot access new environment variables after envdeps are frozen") |
| } |
| val, _ = c.env[key] |
| c.envDeps[key] = val |
| } |
| return val |
| } |
| |
| func (c *config) GetenvWithDefault(key string, defaultValue string) string { |
| ret := c.Getenv(key) |
| if ret == "" { |
| return defaultValue |
| } |
| return ret |
| } |
| |
| func (c *config) IsEnvTrue(key string) bool { |
| value := strings.ToLower(c.Getenv(key)) |
| return value == "1" || value == "y" || value == "yes" || value == "on" || value == "true" |
| } |
| |
| func (c *config) IsEnvFalse(key string) bool { |
| value := strings.ToLower(c.Getenv(key)) |
| return value == "0" || value == "n" || value == "no" || value == "off" || value == "false" |
| } |
| |
| func (c *config) TargetsJava21() bool { |
| return c.IsEnvTrue("EXPERIMENTAL_TARGET_JAVA_VERSION_21") |
| } |
| |
| // EnvDeps returns the environment variables this build depends on. The first |
| // call to this function blocks future reads from the environment. |
| func (c *config) EnvDeps() map[string]string { |
| c.envLock.Lock() |
| defer c.envLock.Unlock() |
| c.envFrozen = true |
| return c.envDeps |
| } |
| |
| func (c *config) KatiEnabled() bool { |
| return c.katiEnabled |
| } |
| |
| func (c *config) ProductVariables() ProductVariables { |
| return c.productVariables |
| } |
| |
| func (c *config) BuildId() string { |
| return String(c.productVariables.BuildId) |
| } |
| |
| func (c *config) DisplayBuildNumber() bool { |
| return Bool(c.productVariables.DisplayBuildNumber) |
| } |
| |
| // BuildNumberFile returns the path to a text file containing metadata |
| // representing the current build's number. |
| // |
| // Rules that want to reference the build number should read from this file |
| // without depending on it. They will run whenever their other dependencies |
| // require them to run and get the current build number. This ensures they don't |
| // rebuild on every incremental build when the build number changes. |
| func (c *config) BuildNumberFile(ctx PathContext) Path { |
| return PathForOutput(ctx, String(c.productVariables.BuildNumberFile)) |
| } |
| |
| // BuildHostnameFile returns the path to a text file containing metadata |
| // representing the current build's host name. |
| func (c *config) BuildHostnameFile(ctx PathContext) Path { |
| return PathForOutput(ctx, String(c.productVariables.BuildHostnameFile)) |
| } |
| |
| // BuildThumbprintFile returns the path to a text file containing metadata |
| // representing the current build's thumbprint. |
| // |
| // Rules that want to reference the build thumbprint should read from this file |
| // without depending on it. They will run whenever their other dependencies |
| // require them to run and get the current build thumbprint. This ensures they |
| // don't rebuild on every incremental build when the build thumbprint changes. |
| func (c *config) BuildThumbprintFile(ctx PathContext) Path { |
| return PathForArbitraryOutput(ctx, "target", "product", c.DeviceName(), String(c.productVariables.BuildThumbprintFile)) |
| } |
| |
| // DeviceName returns the name of the current device target. |
| // TODO: take an AndroidModuleContext to select the device name for multi-device builds |
| func (c *config) DeviceName() string { |
| return *c.productVariables.DeviceName |
| } |
| |
| // DeviceProduct returns the current product target. There could be multiple of |
| // these per device type. |
| // |
| // NOTE: Do not base conditional logic on this value. It may break product inheritance. |
| func (c *config) DeviceProduct() string { |
| return *c.productVariables.DeviceProduct |
| } |
| |
| // HasDeviceProduct returns if the build has a product. A build will not |
| // necessarily have a product when --skip-config is passed to soong, like it is |
| // in prebuilts/build-tools/build-prebuilts.sh |
| func (c *config) HasDeviceProduct() bool { |
| return c.productVariables.DeviceProduct != nil |
| } |
| |
| func (c *config) DeviceAbi() []string { |
| return c.productVariables.DeviceAbi |
| } |
| |
| func (c *config) DeviceResourceOverlays() []string { |
| return c.productVariables.DeviceResourceOverlays |
| } |
| |
| func (c *config) ProductResourceOverlays() []string { |
| return c.productVariables.ProductResourceOverlays |
| } |
| |
| func (c *config) PlatformDisplayVersionName() string { |
| return String(c.productVariables.Platform_display_version_name) |
| } |
| |
| func (c *config) PlatformVersionName() string { |
| return String(c.productVariables.Platform_version_name) |
| } |
| |
| func (c *config) PlatformSdkVersion() ApiLevel { |
| return uncheckedFinalApiLevel(*c.productVariables.Platform_sdk_version) |
| } |
| |
| func (c *config) RawPlatformSdkVersion() *int { |
| return c.productVariables.Platform_sdk_version |
| } |
| |
| func (c *config) PlatformSdkFinal() bool { |
| return Bool(c.productVariables.Platform_sdk_final) |
| } |
| |
| func (c *config) PlatformSdkCodename() string { |
| return String(c.productVariables.Platform_sdk_codename) |
| } |
| |
| func (c *config) PlatformSdkExtensionVersion() int { |
| return *c.productVariables.Platform_sdk_extension_version |
| } |
| |
| func (c *config) PlatformBaseSdkExtensionVersion() int { |
| return *c.productVariables.Platform_base_sdk_extension_version |
| } |
| |
| func (c *config) PlatformSecurityPatch() string { |
| return String(c.productVariables.Platform_security_patch) |
| } |
| |
| func (c *config) PlatformPreviewSdkVersion() string { |
| return String(c.productVariables.Platform_preview_sdk_version) |
| } |
| |
| func (c *config) PlatformMinSupportedTargetSdkVersion() string { |
| var val, ok = c.productVariables.BuildFlags["RELEASE_PLATFORM_MIN_SUPPORTED_TARGET_SDK_VERSION"] |
| if !ok { |
| return "" |
| } |
| return val |
| } |
| |
| func (c *config) PlatformBaseOS() string { |
| return String(c.productVariables.Platform_base_os) |
| } |
| |
| func (c *config) PlatformVersionLastStable() string { |
| return String(c.productVariables.Platform_version_last_stable) |
| } |
| |
| func (c *config) PlatformVersionKnownCodenames() string { |
| return String(c.productVariables.Platform_version_known_codenames) |
| } |
| |
| func (c *config) MinSupportedSdkVersion() ApiLevel { |
| return uncheckedFinalApiLevel(21) |
| } |
| |
| func (c *config) FinalApiLevels() []ApiLevel { |
| var levels []ApiLevel |
| for i := 1; i <= c.PlatformSdkVersion().FinalOrFutureInt(); i++ { |
| levels = append(levels, uncheckedFinalApiLevel(i)) |
| } |
| return levels |
| } |
| |
| func (c *config) PreviewApiLevels() []ApiLevel { |
| var levels []ApiLevel |
| i := 0 |
| for _, codename := range c.PlatformVersionActiveCodenames() { |
| if codename == "REL" { |
| continue |
| } |
| |
| levels = append(levels, ApiLevel{ |
| value: codename, |
| number: i, |
| isPreview: true, |
| }) |
| i++ |
| } |
| return levels |
| } |
| |
| func (c *config) LatestPreviewApiLevel() ApiLevel { |
| level := NoneApiLevel |
| for _, l := range c.PreviewApiLevels() { |
| if l.GreaterThan(level) { |
| level = l |
| } |
| } |
| return level |
| } |
| |
| func (c *config) AllSupportedApiLevels() []ApiLevel { |
| var levels []ApiLevel |
| levels = append(levels, c.FinalApiLevels()...) |
| return append(levels, c.PreviewApiLevels()...) |
| } |
| |
| // DefaultAppTargetSdk returns the API level that platform apps are targeting. |
| // This converts a codename to the exact ApiLevel it represents. |
| func (c *config) DefaultAppTargetSdk(ctx EarlyModuleContext) ApiLevel { |
| if Bool(c.productVariables.Platform_sdk_final) { |
| return c.PlatformSdkVersion() |
| } |
| codename := c.PlatformSdkCodename() |
| hostOnlyBuild := c.productVariables.DeviceArch == nil |
| if codename == "" { |
| // There are some host-only builds (those are invoked by build-prebuilts.sh) which |
| // don't set platform sdk codename. Platform sdk codename makes sense only when we |
| // are building the platform. So we don't enforce the below panic for the host-only |
| // builds. |
| if hostOnlyBuild { |
| return NoneApiLevel |
| } |
| panic("Platform_sdk_codename must be set") |
| } |
| if codename == "REL" { |
| panic("Platform_sdk_codename should not be REL when Platform_sdk_final is true") |
| } |
| return ApiLevelOrPanic(ctx, codename) |
| } |
| |
| func (c *config) AppsDefaultVersionName() string { |
| return String(c.productVariables.AppsDefaultVersionName) |
| } |
| |
| // Codenames that are active in the current lunch target. |
| func (c *config) PlatformVersionActiveCodenames() []string { |
| return c.productVariables.Platform_version_active_codenames |
| } |
| |
| // All unreleased codenames. |
| func (c *config) PlatformVersionAllPreviewCodenames() []string { |
| return c.productVariables.Platform_version_all_preview_codenames |
| } |
| |
| func (c *config) ProductAAPTConfig() []string { |
| return c.productVariables.AAPTConfig |
| } |
| |
| func (c *config) ProductAAPTPreferredConfig() string { |
| return String(c.productVariables.AAPTPreferredConfig) |
| } |
| |
| func (c *config) ProductAAPTCharacteristics() string { |
| return String(c.productVariables.AAPTCharacteristics) |
| } |
| |
| func (c *config) ProductAAPTPrebuiltDPI() []string { |
| return c.productVariables.AAPTPrebuiltDPI |
| } |
| |
| func (c *config) DefaultAppCertificateDir(ctx PathContext) SourcePath { |
| defaultCert := String(c.productVariables.DefaultAppCertificate) |
| if defaultCert != "" { |
| return PathForSource(ctx, filepath.Dir(defaultCert)) |
| } |
| return PathForSource(ctx, testKeyDir) |
| } |
| |
| func (c *config) DefaultAppCertificate(ctx PathContext) (pem, key SourcePath) { |
| defaultCert := String(c.productVariables.DefaultAppCertificate) |
| if defaultCert != "" { |
| return PathForSource(ctx, defaultCert+".x509.pem"), PathForSource(ctx, defaultCert+".pk8") |
| } |
| defaultDir := c.DefaultAppCertificateDir(ctx) |
| return defaultDir.Join(ctx, "testkey.x509.pem"), defaultDir.Join(ctx, "testkey.pk8") |
| } |
| |
| func (c *config) BuildKeys() string { |
| defaultCert := String(c.productVariables.DefaultAppCertificate) |
| if defaultCert == "" || defaultCert == filepath.Join(testKeyDir, "testkey") { |
| return "test-keys" |
| } |
| return "dev-keys" |
| } |
| |
| func (c *config) ApexKeyDir(ctx ModuleContext) SourcePath { |
| // TODO(b/121224311): define another variable such as TARGET_APEX_KEY_OVERRIDE |
| defaultCert := String(c.productVariables.DefaultAppCertificate) |
| if defaultCert == "" || filepath.Dir(defaultCert) == testKeyDir { |
| // When defaultCert is unset or is set to the testkeys path, use the APEX keys |
| // that is under the module dir |
| return pathForModuleSrc(ctx) |
| } |
| // If not, APEX keys are under the specified directory |
| return PathForSource(ctx, filepath.Dir(defaultCert)) |
| } |
| |
| // Certificate for the NetworkStack sepolicy context |
| func (c *config) MainlineSepolicyDevCertificatesDir(ctx ModuleContext) SourcePath { |
| cert := String(c.productVariables.MainlineSepolicyDevCertificates) |
| if cert != "" { |
| return PathForSource(ctx, cert) |
| } |
| return c.DefaultAppCertificateDir(ctx) |
| } |
| |
| // AllowMissingDependencies configures Blueprint/Soong to not fail when modules |
| // are configured to depend on non-existent modules. Note that this does not |
| // affect missing input dependencies at the Ninja level. |
| func (c *config) AllowMissingDependencies() bool { |
| return Bool(c.productVariables.Allow_missing_dependencies) |
| } |
| |
| // Returns true if a full platform source tree cannot be assumed. |
| func (c *config) UnbundledBuild() bool { |
| return Bool(c.productVariables.Unbundled_build) |
| } |
| |
| // Returns true if building apps that aren't bundled with the platform. |
| // UnbundledBuild() is always true when this is true. |
| func (c *config) UnbundledBuildApps() bool { |
| return len(c.productVariables.Unbundled_build_apps) > 0 |
| } |
| |
| // Returns true if building image that aren't bundled with the platform. |
| // UnbundledBuild() is always true when this is true. |
| func (c *config) UnbundledBuildImage() bool { |
| return Bool(c.productVariables.Unbundled_build_image) |
| } |
| |
| // Returns true if building modules against prebuilt SDKs. |
| func (c *config) AlwaysUsePrebuiltSdks() bool { |
| return Bool(c.productVariables.Always_use_prebuilt_sdks) |
| } |
| |
| func (c *config) MinimizeJavaDebugInfo() bool { |
| return Bool(c.productVariables.MinimizeJavaDebugInfo) && !Bool(c.productVariables.Eng) |
| } |
| |
| func (c *config) Debuggable() bool { |
| return Bool(c.productVariables.Debuggable) |
| } |
| |
| func (c *config) Eng() bool { |
| return Bool(c.productVariables.Eng) |
| } |
| |
| func (c *config) BuildType() string { |
| return String(c.productVariables.BuildType) |
| } |
| |
| // DevicePrimaryArchType returns the ArchType for the first configured device architecture, or |
| // Common if there are no device architectures. |
| func (c *config) DevicePrimaryArchType() ArchType { |
| if androidTargets := c.Targets[Android]; len(androidTargets) > 0 { |
| return androidTargets[0].Arch.ArchType |
| } |
| return Common |
| } |
| |
| func (c *config) SanitizeHost() []string { |
| return append([]string(nil), c.productVariables.SanitizeHost...) |
| } |
| |
| func (c *config) SanitizeDevice() []string { |
| return append([]string(nil), c.productVariables.SanitizeDevice...) |
| } |
| |
| func (c *config) SanitizeDeviceDiag() []string { |
| return append([]string(nil), c.productVariables.SanitizeDeviceDiag...) |
| } |
| |
| func (c *config) SanitizeDeviceArch() []string { |
| return append([]string(nil), c.productVariables.SanitizeDeviceArch...) |
| } |
| |
| func (c *config) EnableCFI() bool { |
| if c.productVariables.EnableCFI == nil { |
| return true |
| } |
| return *c.productVariables.EnableCFI |
| } |
| |
| func (c *config) DisableScudo() bool { |
| return Bool(c.productVariables.DisableScudo) |
| } |
| |
| func (c *config) Android64() bool { |
| for _, t := range c.Targets[Android] { |
| if t.Arch.ArchType.Multilib == "lib64" { |
| return true |
| } |
| } |
| |
| return false |
| } |
| |
| func (c *config) UseGoma() bool { |
| return Bool(c.productVariables.UseGoma) |
| } |
| |
| func (c *config) UseRBE() bool { |
| return Bool(c.productVariables.UseRBE) |
| } |
| |
| func (c *config) UseRBEJAVAC() bool { |
| return Bool(c.productVariables.UseRBEJAVAC) |
| } |
| |
| func (c *config) UseRBER8() bool { |
| return Bool(c.productVariables.UseRBER8) |
| } |
| |
| func (c *config) UseRBED8() bool { |
| return Bool(c.productVariables.UseRBED8) |
| } |
| |
| func (c *config) UseRemoteBuild() bool { |
| return c.UseGoma() || c.UseRBE() |
| } |
| |
| func (c *config) RunErrorProne() bool { |
| return c.IsEnvTrue("RUN_ERROR_PRONE") |
| } |
| |
| // XrefCorpusName returns the Kythe cross-reference corpus name. |
| func (c *config) XrefCorpusName() string { |
| return c.Getenv("XREF_CORPUS") |
| } |
| |
| // XrefCuEncoding returns the compilation unit encoding to use for Kythe code |
| // xrefs. Can be 'json' (default), 'proto' or 'all'. |
| func (c *config) XrefCuEncoding() string { |
| if enc := c.Getenv("KYTHE_KZIP_ENCODING"); enc != "" { |
| return enc |
| } |
| return "json" |
| } |
| |
| // XrefCuJavaSourceMax returns the maximum number of the Java source files |
| // in a single compilation unit |
| const xrefJavaSourceFileMaxDefault = "1000" |
| |
| func (c Config) XrefCuJavaSourceMax() string { |
| v := c.Getenv("KYTHE_JAVA_SOURCE_BATCH_SIZE") |
| if v == "" { |
| return xrefJavaSourceFileMaxDefault |
| } |
| if _, err := strconv.ParseUint(v, 0, 0); err != nil { |
| fmt.Fprintf(os.Stderr, |
| "bad KYTHE_JAVA_SOURCE_BATCH_SIZE value: %s, will use %s", |
| err, xrefJavaSourceFileMaxDefault) |
| return xrefJavaSourceFileMaxDefault |
| } |
| return v |
| |
| } |
| |
| func (c *config) EmitXrefRules() bool { |
| return c.XrefCorpusName() != "" |
| } |
| |
| func (c *config) ClangTidy() bool { |
| return Bool(c.productVariables.ClangTidy) |
| } |
| |
| func (c *config) TidyChecks() string { |
| if c.productVariables.TidyChecks == nil { |
| return "" |
| } |
| return *c.productVariables.TidyChecks |
| } |
| |
| func (c *config) LibartImgHostBaseAddress() string { |
| return "0x60000000" |
| } |
| |
| func (c *config) LibartImgDeviceBaseAddress() string { |
| return "0x70000000" |
| } |
| |
| func (c *config) ArtUseReadBarrier() bool { |
| return Bool(c.productVariables.ArtUseReadBarrier) |
| } |
| |
| // Enforce Runtime Resource Overlays for a module. RROs supersede static RROs, |
| // but some modules still depend on it. |
| // |
| // More info: https://source.android.com/devices/architecture/rros |
| func (c *config) EnforceRROForModule(name string) bool { |
| enforceList := c.productVariables.EnforceRROTargets |
| |
| if len(enforceList) > 0 { |
| if InList("*", enforceList) { |
| return true |
| } |
| return InList(name, enforceList) |
| } |
| return false |
| } |
| func (c *config) EnforceRROExcludedOverlay(path string) bool { |
| excluded := c.productVariables.EnforceRROExcludedOverlays |
| if len(excluded) > 0 { |
| return HasAnyPrefix(path, excluded) |
| } |
| return false |
| } |
| |
| func (c *config) ExportedNamespaces() []string { |
| return append([]string(nil), c.productVariables.NamespacesToExport...) |
| } |
| |
| func (c *config) SourceRootDirs() []string { |
| return c.productVariables.SourceRootDirs |
| } |
| |
| func (c *config) HostStaticBinaries() bool { |
| return Bool(c.productVariables.HostStaticBinaries) |
| } |
| |
| func (c *config) UncompressPrivAppDex() bool { |
| return Bool(c.productVariables.UncompressPrivAppDex) |
| } |
| |
| func (c *config) ModulesLoadedByPrivilegedModules() []string { |
| return c.productVariables.ModulesLoadedByPrivilegedModules |
| } |
| |
| // DexpreoptGlobalConfigPath returns the path to the dexpreopt.config file in |
| // the output directory, if it was created during the product configuration |
| // phase by Kati. |
| func (c *config) DexpreoptGlobalConfigPath(ctx PathContext) OptionalPath { |
| if c.productVariables.DexpreoptGlobalConfig == nil { |
| return OptionalPathForPath(nil) |
| } |
| return OptionalPathForPath( |
| pathForBuildToolDep(ctx, *c.productVariables.DexpreoptGlobalConfig)) |
| } |
| |
| // DexpreoptGlobalConfig returns the raw byte contents of the dexpreopt global |
| // configuration. Since the configuration file was created by Kati during |
| // product configuration (externally of soong_build), it's not tracked, so we |
| // also manually add a Ninja file dependency on the configuration file to the |
| // rule that creates the main build.ninja file. This ensures that build.ninja is |
| // regenerated correctly if dexpreopt.config changes. |
| func (c *config) DexpreoptGlobalConfig(ctx PathContext) ([]byte, error) { |
| path := c.DexpreoptGlobalConfigPath(ctx) |
| if !path.Valid() { |
| return nil, nil |
| } |
| ctx.AddNinjaFileDeps(path.String()) |
| return os.ReadFile(absolutePath(path.String())) |
| } |
| |
| func (c *deviceConfig) WithDexpreopt() bool { |
| return c.config.productVariables.WithDexpreopt |
| } |
| |
| func (c *config) FrameworksBaseDirExists(ctx PathGlobContext) bool { |
| return ExistentPathForSource(ctx, "frameworks", "base", "Android.bp").Valid() |
| } |
| |
| func (c *config) VndkSnapshotBuildArtifacts() bool { |
| return Bool(c.productVariables.VndkSnapshotBuildArtifacts) |
| } |
| |
| func (c *config) HasMultilibConflict(arch ArchType) bool { |
| return c.multilibConflicts[arch] |
| } |
| |
| func (c *config) PrebuiltHiddenApiDir(_ PathContext) string { |
| return String(c.productVariables.PrebuiltHiddenApiDir) |
| } |
| |
| func (c *config) VendorApiLevel() string { |
| return String(c.productVariables.VendorApiLevel) |
| } |
| |
| func (c *config) PrevVendorApiLevel() string { |
| vendorApiLevel, err := strconv.Atoi(c.VendorApiLevel()) |
| if err != nil { |
| panic(fmt.Errorf("Cannot parse vendor API level %s to an integer: %s", |
| c.VendorApiLevel(), err)) |
| } |
| // The version before trunk stable is 34. |
| if vendorApiLevel == 202404 { |
| return "34" |
| } |
| if vendorApiLevel >= 1 && vendorApiLevel <= 34 { |
| return strconv.Itoa(vendorApiLevel - 1) |
| } |
| if vendorApiLevel < 202404 || vendorApiLevel%100 != 4 { |
| panic("Unknown vendor API level " + c.VendorApiLevel()) |
| } |
| return strconv.Itoa(vendorApiLevel - 100) |
| } |
| |
| func IsTrunkStableVendorApiLevel(level string) bool { |
| levelInt, err := strconv.Atoi(level) |
| return err == nil && levelInt >= 202404 |
| } |
| |
| func (c *config) VendorApiLevelFrozen() bool { |
| return c.productVariables.GetBuildFlagBool("RELEASE_BOARD_API_LEVEL_FROZEN") |
| } |
| |
| func (c *deviceConfig) Arches() []Arch { |
| var arches []Arch |
| for _, target := range c.config.Targets[Android] { |
| arches = append(arches, target.Arch) |
| } |
| return arches |
| } |
| |
| func (c *deviceConfig) BinderBitness() string { |
| is32BitBinder := c.config.productVariables.Binder32bit |
| if is32BitBinder != nil && *is32BitBinder { |
| return "32" |
| } |
| return "64" |
| } |
| |
| func (c *deviceConfig) VendorPath() string { |
| if c.config.productVariables.VendorPath != nil { |
| return *c.config.productVariables.VendorPath |
| } |
| return "vendor" |
| } |
| |
| func (c *deviceConfig) RecoverySnapshotVersion() string { |
| return String(c.config.productVariables.RecoverySnapshotVersion) |
| } |
| |
| func (c *deviceConfig) CurrentApiLevelForVendorModules() string { |
| return StringDefault(c.config.productVariables.DeviceCurrentApiLevelForVendorModules, "current") |
| } |
| |
| func (c *deviceConfig) ExtraVndkVersions() []string { |
| return c.config.productVariables.ExtraVndkVersions |
| } |
| |
| func (c *deviceConfig) SystemSdkVersions() []string { |
| return c.config.productVariables.DeviceSystemSdkVersions |
| } |
| |
| func (c *deviceConfig) PlatformSystemSdkVersions() []string { |
| return c.config.productVariables.Platform_systemsdk_versions |
| } |
| |
| func (c *deviceConfig) OdmPath() string { |
| if c.config.productVariables.OdmPath != nil { |
| return *c.config.productVariables.OdmPath |
| } |
| return "odm" |
| } |
| |
| func (c *deviceConfig) ProductPath() string { |
| if c.config.productVariables.ProductPath != nil { |
| return *c.config.productVariables.ProductPath |
| } |
| return "product" |
| } |
| |
| func (c *deviceConfig) SystemExtPath() string { |
| if c.config.productVariables.SystemExtPath != nil { |
| return *c.config.productVariables.SystemExtPath |
| } |
| return "system_ext" |
| } |
| |
| func (c *deviceConfig) BtConfigIncludeDir() string { |
| return String(c.config.productVariables.BtConfigIncludeDir) |
| } |
| |
| func (c *deviceConfig) DeviceKernelHeaderDirs() []string { |
| return c.config.productVariables.DeviceKernelHeaders |
| } |
| |
| func (c *deviceConfig) TargetSpecificHeaderPath() string { |
| return String(c.config.productVariables.TargetSpecificHeaderPath) |
| } |
| |
| // JavaCoverageEnabledForPath returns whether Java code coverage is enabled for |
| // path. Coverage is enabled by default when the product variable |
| // JavaCoveragePaths is empty. If JavaCoveragePaths is not empty, coverage is |
| // enabled for any path which is part of this variable (and not part of the |
| // JavaCoverageExcludePaths product variable). Value "*" in JavaCoveragePaths |
| // represents any path. |
| func (c *deviceConfig) JavaCoverageEnabledForPath(path string) bool { |
| coverage := false |
| if len(c.config.productVariables.JavaCoveragePaths) == 0 || |
| InList("*", c.config.productVariables.JavaCoveragePaths) || |
| HasAnyPrefix(path, c.config.productVariables.JavaCoveragePaths) { |
| coverage = true |
| } |
| if coverage && len(c.config.productVariables.JavaCoverageExcludePaths) > 0 { |
| if HasAnyPrefix(path, c.config.productVariables.JavaCoverageExcludePaths) { |
| coverage = false |
| } |
| } |
| return coverage |
| } |
| |
| // Returns true if gcov or clang coverage is enabled. |
| func (c *deviceConfig) NativeCoverageEnabled() bool { |
| return Bool(c.config.productVariables.GcovCoverage) || |
| Bool(c.config.productVariables.ClangCoverage) |
| } |
| |
| func (c *deviceConfig) ClangCoverageEnabled() bool { |
| return Bool(c.config.productVariables.ClangCoverage) |
| } |
| |
| func (c *deviceConfig) ClangCoverageContinuousMode() bool { |
| return Bool(c.config.productVariables.ClangCoverageContinuousMode) |
| } |
| |
| func (c *deviceConfig) GcovCoverageEnabled() bool { |
| return Bool(c.config.productVariables.GcovCoverage) |
| } |
| |
| // NativeCoverageEnabledForPath returns whether (GCOV- or Clang-based) native |
| // code coverage is enabled for path. By default, coverage is not enabled for a |
| // given path unless it is part of the NativeCoveragePaths product variable (and |
| // not part of the NativeCoverageExcludePaths product variable). Value "*" in |
| // NativeCoveragePaths represents any path. |
| func (c *deviceConfig) NativeCoverageEnabledForPath(path string) bool { |
| coverage := false |
| if len(c.config.productVariables.NativeCoveragePaths) > 0 { |
| if InList("*", c.config.productVariables.NativeCoveragePaths) || HasAnyPrefix(path, c.config.productVariables.NativeCoveragePaths) { |
| coverage = true |
| } |
| } |
| if coverage && len(c.config.productVariables.NativeCoverageExcludePaths) > 0 { |
| // Workaround coverage boot failure. |
| // http://b/269981180 |
| if strings.HasPrefix(path, "external/protobuf") { |
| coverage = false |
| } |
| if HasAnyPrefix(path, c.config.productVariables.NativeCoverageExcludePaths) { |
| coverage = false |
| } |
| } |
| return coverage |
| } |
| |
| func (c *deviceConfig) PgoAdditionalProfileDirs() []string { |
| return c.config.productVariables.PgoAdditionalProfileDirs |
| } |
| |
| // AfdoProfile returns fully qualified path associated to the given module name |
| func (c *deviceConfig) AfdoProfile(name string) (string, error) { |
| for _, afdoProfile := range c.config.productVariables.AfdoProfiles { |
| split := strings.Split(afdoProfile, ":") |
| if len(split) != 3 { |
| return "", fmt.Errorf("AFDO_PROFILES has invalid value: %s. "+ |
| "The expected format is <module>:<fully-qualified-path-to-fdo_profile>", afdoProfile) |
| } |
| if split[0] == name { |
| return strings.Join([]string{split[1], split[2]}, ":"), nil |
| } |
| } |
| return "", nil |
| } |
| |
| func (c *deviceConfig) VendorSepolicyDirs() []string { |
| return c.config.productVariables.BoardVendorSepolicyDirs |
| } |
| |
| func (c *deviceConfig) OdmSepolicyDirs() []string { |
| return c.config.productVariables.BoardOdmSepolicyDirs |
| } |
| |
| func (c *deviceConfig) SystemExtPublicSepolicyDirs() []string { |
| return c.config.productVariables.SystemExtPublicSepolicyDirs |
| } |
| |
| func (c *deviceConfig) SystemExtPrivateSepolicyDirs() []string { |
| return c.config.productVariables.SystemExtPrivateSepolicyDirs |
| } |
| |
| func (c *deviceConfig) SepolicyM4Defs() []string { |
| return c.config.productVariables.BoardSepolicyM4Defs |
| } |
| |
| func (c *deviceConfig) OverrideManifestPackageNameFor(name string) (manifestName string, overridden bool) { |
| return findOverrideValue(c.config.productVariables.ManifestPackageNameOverrides, name, |
| "invalid override rule %q in PRODUCT_MANIFEST_PACKAGE_NAME_OVERRIDES should be <module_name>:<manifest_name>") |
| } |
| |
| func (c *deviceConfig) OverrideCertificateFor(name string) (certificatePath string, overridden bool) { |
| return findOverrideValue(c.config.productVariables.CertificateOverrides, name, |
| "invalid override rule %q in PRODUCT_CERTIFICATE_OVERRIDES should be <module_name>:<certificate_module_name>") |
| } |
| |
| func (c *deviceConfig) OverridePackageNameFor(name string) string { |
| newName, overridden := findOverrideValue( |
| c.config.productVariables.PackageNameOverrides, |
| name, |
| "invalid override rule %q in PRODUCT_PACKAGE_NAME_OVERRIDES should be <module_name>:<package_name>") |
| if overridden { |
| return newName |
| } |
| return name |
| } |
| |
| func findOverrideValue(overrides []string, name string, errorMsg string) (newValue string, overridden bool) { |
| if overrides == nil || len(overrides) == 0 { |
| return "", false |
| } |
| for _, o := range overrides { |
| split := strings.Split(o, ":") |
| if len(split) != 2 { |
| // This shouldn't happen as this is first checked in make, but just in case. |
| panic(fmt.Errorf(errorMsg, o)) |
| } |
| if matchPattern(split[0], name) { |
| return substPattern(split[0], split[1], name), true |
| } |
| } |
| return "", false |
| } |
| |
| func (c *deviceConfig) ApexGlobalMinSdkVersionOverride() string { |
| return String(c.config.productVariables.ApexGlobalMinSdkVersionOverride) |
| } |
| |
| func (c *config) IntegerOverflowDisabledForPath(path string) bool { |
| if len(c.productVariables.IntegerOverflowExcludePaths) == 0 { |
| return false |
| } |
| return HasAnyPrefix(path, c.productVariables.IntegerOverflowExcludePaths) |
| } |
| |
| func (c *config) CFIDisabledForPath(path string) bool { |
| if len(c.productVariables.CFIExcludePaths) == 0 { |
| return false |
| } |
| return HasAnyPrefix(path, c.productVariables.CFIExcludePaths) |
| } |
| |
| func (c *config) CFIEnabledForPath(path string) bool { |
| if len(c.productVariables.CFIIncludePaths) == 0 { |
| return false |
| } |
| return HasAnyPrefix(path, c.productVariables.CFIIncludePaths) && !c.CFIDisabledForPath(path) |
| } |
| |
| func (c *config) MemtagHeapDisabledForPath(path string) bool { |
| if len(c.productVariables.MemtagHeapExcludePaths) == 0 { |
| return false |
| } |
| return HasAnyPrefix(path, c.productVariables.MemtagHeapExcludePaths) |
| } |
| |
| func (c *config) MemtagHeapAsyncEnabledForPath(path string) bool { |
| if len(c.productVariables.MemtagHeapAsyncIncludePaths) == 0 { |
| return false |
| } |
| return HasAnyPrefix(path, c.productVariables.MemtagHeapAsyncIncludePaths) && !c.MemtagHeapDisabledForPath(path) |
| } |
| |
| func (c *config) MemtagHeapSyncEnabledForPath(path string) bool { |
| if len(c.productVariables.MemtagHeapSyncIncludePaths) == 0 { |
| return false |
| } |
| return HasAnyPrefix(path, c.productVariables.MemtagHeapSyncIncludePaths) && !c.MemtagHeapDisabledForPath(path) |
| } |
| |
| func (c *config) HWASanDisabledForPath(path string) bool { |
| if len(c.productVariables.HWASanExcludePaths) == 0 { |
| return false |
| } |
| return HasAnyPrefix(path, c.productVariables.HWASanExcludePaths) |
| } |
| |
| func (c *config) HWASanEnabledForPath(path string) bool { |
| if len(c.productVariables.HWASanIncludePaths) == 0 { |
| return false |
| } |
| return HasAnyPrefix(path, c.productVariables.HWASanIncludePaths) && !c.HWASanDisabledForPath(path) |
| } |
| |
| func (c *config) VendorConfig(name string) VendorConfig { |
| return soongconfig.Config(c.productVariables.VendorVars[name]) |
| } |
| |
| func (c *config) NdkAbis() bool { |
| return Bool(c.productVariables.Ndk_abis) |
| } |
| |
| func (c *config) AmlAbis() bool { |
| return Bool(c.productVariables.Aml_abis) |
| } |
| |
| func (c *config) ForceApexSymlinkOptimization() bool { |
| return Bool(c.productVariables.ForceApexSymlinkOptimization) |
| } |
| |
| func (c *config) ApexCompressionEnabled() bool { |
| return Bool(c.productVariables.CompressedApex) && !c.UnbundledBuildApps() |
| } |
| |
| func (c *config) ApexTrimEnabled() bool { |
| return Bool(c.productVariables.TrimmedApex) |
| } |
| |
| func (c *config) EnforceSystemCertificate() bool { |
| return Bool(c.productVariables.EnforceSystemCertificate) |
| } |
| |
| func (c *config) EnforceSystemCertificateAllowList() []string { |
| return c.productVariables.EnforceSystemCertificateAllowList |
| } |
| |
| func (c *config) EnforceProductPartitionInterface() bool { |
| return Bool(c.productVariables.EnforceProductPartitionInterface) |
| } |
| |
| func (c *config) EnforceInterPartitionJavaSdkLibrary() bool { |
| return Bool(c.productVariables.EnforceInterPartitionJavaSdkLibrary) |
| } |
| |
| func (c *config) InterPartitionJavaLibraryAllowList() []string { |
| return c.productVariables.InterPartitionJavaLibraryAllowList |
| } |
| |
| func (c *config) ProductHiddenAPIStubs() []string { |
| return c.productVariables.ProductHiddenAPIStubs |
| } |
| |
| func (c *config) ProductHiddenAPIStubsSystem() []string { |
| return c.productVariables.ProductHiddenAPIStubsSystem |
| } |
| |
| func (c *config) ProductHiddenAPIStubsTest() []string { |
| return c.productVariables.ProductHiddenAPIStubsTest |
| } |
| |
| func (c *deviceConfig) TargetFSConfigGen() []string { |
| return c.config.productVariables.TargetFSConfigGen |
| } |
| |
| func (c *config) ProductPublicSepolicyDirs() []string { |
| return c.productVariables.ProductPublicSepolicyDirs |
| } |
| |
| func (c *config) ProductPrivateSepolicyDirs() []string { |
| return c.productVariables.ProductPrivateSepolicyDirs |
| } |
| |
| func (c *config) TargetMultitreeUpdateMeta() bool { |
| return c.productVariables.MultitreeUpdateMeta |
| } |
| |
| func (c *deviceConfig) DeviceArch() string { |
| return String(c.config.productVariables.DeviceArch) |
| } |
| |
| func (c *deviceConfig) DeviceArchVariant() string { |
| return String(c.config.productVariables.DeviceArchVariant) |
| } |
| |
| func (c *deviceConfig) DeviceSecondaryArch() string { |
| return String(c.config.productVariables.DeviceSecondaryArch) |
| } |
| |
| func (c *deviceConfig) DeviceSecondaryArchVariant() string { |
| return String(c.config.productVariables.DeviceSecondaryArchVariant) |
| } |
| |
| func (c *deviceConfig) BoardUsesRecoveryAsBoot() bool { |
| return Bool(c.config.productVariables.BoardUsesRecoveryAsBoot) |
| } |
| |
| func (c *deviceConfig) BoardKernelBinaries() []string { |
| return c.config.productVariables.BoardKernelBinaries |
| } |
| |
| func (c *deviceConfig) BoardKernelModuleInterfaceVersions() []string { |
| return c.config.productVariables.BoardKernelModuleInterfaceVersions |
| } |
| |
| func (c *deviceConfig) BoardMoveRecoveryResourcesToVendorBoot() bool { |
| return Bool(c.config.productVariables.BoardMoveRecoveryResourcesToVendorBoot) |
| } |
| |
| func (c *deviceConfig) PlatformSepolicyVersion() string { |
| return String(c.config.productVariables.PlatformSepolicyVersion) |
| } |
| |
| func (c *deviceConfig) PlatformSepolicyCompatVersions() []string { |
| return c.config.productVariables.PlatformSepolicyCompatVersions |
| } |
| |
| func (c *deviceConfig) BoardSepolicyVers() string { |
| if ver := String(c.config.productVariables.BoardSepolicyVers); ver != "" { |
| return ver |
| } |
| return c.PlatformSepolicyVersion() |
| } |
| |
| func (c *deviceConfig) SystemExtSepolicyPrebuiltApiDir() string { |
| return String(c.config.productVariables.SystemExtSepolicyPrebuiltApiDir) |
| } |
| |
| func (c *deviceConfig) ProductSepolicyPrebuiltApiDir() string { |
| return String(c.config.productVariables.ProductSepolicyPrebuiltApiDir) |
| } |
| |
| func (c *deviceConfig) IsPartnerTrebleSepolicyTestEnabled() bool { |
| return c.SystemExtSepolicyPrebuiltApiDir() != "" || c.ProductSepolicyPrebuiltApiDir() != "" |
| } |
| |
| func (c *deviceConfig) DirectedVendorSnapshot() bool { |
| return c.config.productVariables.DirectedVendorSnapshot |
| } |
| |
| func (c *deviceConfig) VendorSnapshotModules() map[string]bool { |
| return c.config.productVariables.VendorSnapshotModules |
| } |
| |
| func (c *deviceConfig) DirectedRecoverySnapshot() bool { |
| return c.config.productVariables.DirectedRecoverySnapshot |
| } |
| |
| func (c *deviceConfig) RecoverySnapshotModules() map[string]bool { |
| return c.config.productVariables.RecoverySnapshotModules |
| } |
| |
| func createDirsMap(previous map[string]bool, dirs []string) (map[string]bool, error) { |
| var ret = make(map[string]bool) |
| for _, dir := range dirs { |
| clean := filepath.Clean(dir) |
| if previous[clean] || ret[clean] { |
| return nil, fmt.Errorf("Duplicate entry %s", dir) |
| } |
| ret[clean] = true |
| } |
| return ret, nil |
| } |
| |
| func (c *deviceConfig) createDirsMapOnce(onceKey OnceKey, previous map[string]bool, dirs []string) map[string]bool { |
| dirMap := c.Once(onceKey, func() interface{} { |
| ret, err := createDirsMap(previous, dirs) |
| if err != nil { |
| panic(fmt.Errorf("%s: %w", onceKey.key, err)) |
| } |
| return ret |
| }) |
| if dirMap == nil { |
| return nil |
| } |
| return dirMap.(map[string]bool) |
| } |
| |
| var vendorSnapshotDirsExcludedKey = NewOnceKey("VendorSnapshotDirsExcludedMap") |
| |
| func (c *deviceConfig) VendorSnapshotDirsExcludedMap() map[string]bool { |
| return c.createDirsMapOnce(vendorSnapshotDirsExcludedKey, nil, |
| c.config.productVariables.VendorSnapshotDirsExcluded) |
| } |
| |
| var vendorSnapshotDirsIncludedKey = NewOnceKey("VendorSnapshotDirsIncludedMap") |
| |
| func (c *deviceConfig) VendorSnapshotDirsIncludedMap() map[string]bool { |
| excludedMap := c.VendorSnapshotDirsExcludedMap() |
| return c.createDirsMapOnce(vendorSnapshotDirsIncludedKey, excludedMap, |
| c.config.productVariables.VendorSnapshotDirsIncluded) |
| } |
| |
| var recoverySnapshotDirsExcludedKey = NewOnceKey("RecoverySnapshotDirsExcludedMap") |
| |
| func (c *deviceConfig) RecoverySnapshotDirsExcludedMap() map[string]bool { |
| return c.createDirsMapOnce(recoverySnapshotDirsExcludedKey, nil, |
| c.config.productVariables.RecoverySnapshotDirsExcluded) |
| } |
| |
| var recoverySnapshotDirsIncludedKey = NewOnceKey("RecoverySnapshotDirsIncludedMap") |
| |
| func (c *deviceConfig) RecoverySnapshotDirsIncludedMap() map[string]bool { |
| excludedMap := c.RecoverySnapshotDirsExcludedMap() |
| return c.createDirsMapOnce(recoverySnapshotDirsIncludedKey, excludedMap, |
| c.config.productVariables.RecoverySnapshotDirsIncluded) |
| } |
| |
| func (c *deviceConfig) HostFakeSnapshotEnabled() bool { |
| return c.config.productVariables.HostFakeSnapshotEnabled |
| } |
| |
| func (c *deviceConfig) ShippingApiLevel() ApiLevel { |
| if c.config.productVariables.Shipping_api_level == nil { |
| return NoneApiLevel |
| } |
| apiLevel, _ := strconv.Atoi(*c.config.productVariables.Shipping_api_level) |
| return uncheckedFinalApiLevel(apiLevel) |
| } |
| |
| func (c *deviceConfig) BuildBrokenPluginValidation() []string { |
| return c.config.productVariables.BuildBrokenPluginValidation |
| } |
| |
| func (c *deviceConfig) BuildBrokenClangAsFlags() bool { |
| return c.config.productVariables.BuildBrokenClangAsFlags |
| } |
| |
| func (c *deviceConfig) BuildBrokenClangCFlags() bool { |
| return c.config.productVariables.BuildBrokenClangCFlags |
| } |
| |
| func (c *deviceConfig) BuildBrokenClangProperty() bool { |
| return c.config.productVariables.BuildBrokenClangProperty |
| } |
| |
| func (c *deviceConfig) BuildBrokenEnforceSyspropOwner() bool { |
| return c.config.productVariables.BuildBrokenEnforceSyspropOwner |
| } |
| |
| func (c *deviceConfig) BuildBrokenTrebleSyspropNeverallow() bool { |
| return c.config.productVariables.BuildBrokenTrebleSyspropNeverallow |
| } |
| |
| func (c *deviceConfig) BuildBrokenUsesSoongPython2Modules() bool { |
| return c.config.productVariables.BuildBrokenUsesSoongPython2Modules |
| } |
| |
| func (c *deviceConfig) BuildDebugfsRestrictionsEnabled() bool { |
| return c.config.productVariables.BuildDebugfsRestrictionsEnabled |
| } |
| |
| func (c *deviceConfig) BuildBrokenVendorPropertyNamespace() bool { |
| return c.config.productVariables.BuildBrokenVendorPropertyNamespace |
| } |
| |
| func (c *deviceConfig) BuildBrokenInputDir(name string) bool { |
| return InList(name, c.config.productVariables.BuildBrokenInputDirModules) |
| } |
| |
| func (c *deviceConfig) BuildBrokenDontCheckSystemSdk() bool { |
| return c.config.productVariables.BuildBrokenDontCheckSystemSdk |
| } |
| |
| func (c *config) BuildWarningBadOptionalUsesLibsAllowlist() []string { |
| return c.productVariables.BuildWarningBadOptionalUsesLibsAllowlist |
| } |
| |
| func (c *deviceConfig) GenruleSandboxing() bool { |
| return Bool(c.config.productVariables.GenruleSandboxing) |
| } |
| |
| func (c *deviceConfig) RequiresInsecureExecmemForSwiftshader() bool { |
| return c.config.productVariables.RequiresInsecureExecmemForSwiftshader |
| } |
| |
| func (c *deviceConfig) Release_aidl_use_unfrozen() bool { |
| return Bool(c.config.productVariables.Release_aidl_use_unfrozen) |
| } |
| |
| func (c *config) SelinuxIgnoreNeverallows() bool { |
| return c.productVariables.SelinuxIgnoreNeverallows |
| } |
| |
| func (c *deviceConfig) SepolicyFreezeTestExtraDirs() []string { |
| return c.config.productVariables.SepolicyFreezeTestExtraDirs |
| } |
| |
| func (c *deviceConfig) SepolicyFreezeTestExtraPrebuiltDirs() []string { |
| return c.config.productVariables.SepolicyFreezeTestExtraPrebuiltDirs |
| } |
| |
| func (c *deviceConfig) GenerateAidlNdkPlatformBackend() bool { |
| return c.config.productVariables.GenerateAidlNdkPlatformBackend |
| } |
| |
| func (c *deviceConfig) AconfigContainerValidation() string { |
| return c.config.productVariables.AconfigContainerValidation |
| } |
| |
| func (c *config) IgnorePrefer32OnDevice() bool { |
| return c.productVariables.IgnorePrefer32OnDevice |
| } |
| |
| func (c *config) BootJars() []string { |
| return c.Once(earlyBootJarsKey, func() interface{} { |
| list := c.productVariables.BootJars.CopyOfJars() |
| return append(list, c.productVariables.ApexBootJars.CopyOfJars()...) |
| }).([]string) |
| } |
| |
| func (c *config) NonApexBootJars() ConfiguredJarList { |
| return c.productVariables.BootJars |
| } |
| |
| func (c *config) ApexBootJars() ConfiguredJarList { |
| return c.productVariables.ApexBootJars |
| } |
| |
| func (c *config) RBEWrapper() string { |
| return c.GetenvWithDefault("RBE_WRAPPER", remoteexec.DefaultWrapperPath) |
| } |
| |
| // UseHostMusl returns true if the host target has been configured to build against musl libc. |
| func (c *config) UseHostMusl() bool { |
| return Bool(c.productVariables.HostMusl) |
| } |
| |
| // ApiSurfaces directory returns the source path inside the api_surfaces repo |
| // (relative to workspace root). |
| func (c *config) ApiSurfacesDir(s ApiSurface, version string) string { |
| return filepath.Join( |
| "build", |
| "bazel", |
| "api_surfaces", |
| s.String(), |
| version) |
| } |
| |
| func (c *config) JavaCoverageEnabled() bool { |
| return c.IsEnvTrue("EMMA_INSTRUMENT") || c.IsEnvTrue("EMMA_INSTRUMENT_STATIC") || c.IsEnvTrue("EMMA_INSTRUMENT_FRAMEWORK") |
| } |
| |
| func (c *deviceConfig) BuildFromSourceStub() bool { |
| return Bool(c.config.productVariables.BuildFromSourceStub) |
| } |
| |
| func (c *config) BuildFromTextStub() bool { |
| // TODO: b/302320354 - Remove the coverage build specific logic once the |
| // robust solution for handling native properties in from-text stub build |
| // is implemented. |
| return !c.buildFromSourceStub && |
| !c.JavaCoverageEnabled() && |
| !c.deviceConfig.BuildFromSourceStub() |
| } |
| |
| func (c *config) SetBuildFromTextStub(b bool) { |
| c.buildFromSourceStub = !b |
| c.productVariables.Build_from_text_stub = boolPtr(b) |
| } |
| |
| func (c *config) SetApiLibraries(libs []string) { |
| c.apiLibraries = make(map[string]struct{}) |
| for _, lib := range libs { |
| c.apiLibraries[lib] = struct{}{} |
| } |
| } |
| |
| func (c *config) GetApiLibraries() map[string]struct{} { |
| return c.apiLibraries |
| } |
| |
| func (c *deviceConfig) CheckVendorSeappViolations() bool { |
| return Bool(c.config.productVariables.CheckVendorSeappViolations) |
| } |
| |
| func (c *config) GetBuildFlag(name string) (string, bool) { |
| val, ok := c.productVariables.BuildFlags[name] |
| return val, ok |
| } |
| |
| func (c *config) UseResourceProcessorByDefault() bool { |
| return c.productVariables.GetBuildFlagBool("RELEASE_USE_RESOURCE_PROCESSOR_BY_DEFAULT") |
| } |
| |
| var ( |
| mainlineApexContributionBuildFlagsToApexNames = map[string]string{ |
| "RELEASE_APEX_CONTRIBUTIONS_ADBD": "com.android.adbd", |
| "RELEASE_APEX_CONTRIBUTIONS_ADSERVICES": "com.android.adservices", |
| "RELEASE_APEX_CONTRIBUTIONS_APPSEARCH": "com.android.appsearch", |
| "RELEASE_APEX_CONTRIBUTIONS_ART": "com.android.art", |
| "RELEASE_APEX_CONTRIBUTIONS_BLUETOOTH": "com.android.btservices", |
| "RELEASE_APEX_CONTRIBUTIONS_CAPTIVEPORTALLOGIN": "", |
| "RELEASE_APEX_CONTRIBUTIONS_CELLBROADCAST": "com.android.cellbroadcast", |
| "RELEASE_APEX_CONTRIBUTIONS_CONFIGINFRASTRUCTURE": "com.android.configinfrastructure", |
| "RELEASE_APEX_CONTRIBUTIONS_CONNECTIVITY": "com.android.tethering", |
| "RELEASE_APEX_CONTRIBUTIONS_CONSCRYPT": "com.android.conscrypt", |
| "RELEASE_APEX_CONTRIBUTIONS_CRASHRECOVERY": "", |
| "RELEASE_APEX_CONTRIBUTIONS_DEVICELOCK": "com.android.devicelock", |
| "RELEASE_APEX_CONTRIBUTIONS_DOCUMENTSUIGOOGLE": "", |
| "RELEASE_APEX_CONTRIBUTIONS_EXTSERVICES": "com.android.extservices", |
| "RELEASE_APEX_CONTRIBUTIONS_HEALTHFITNESS": "com.android.healthfitness", |
| "RELEASE_APEX_CONTRIBUTIONS_IPSEC": "com.android.ipsec", |
| "RELEASE_APEX_CONTRIBUTIONS_MEDIA": "com.android.media", |
| "RELEASE_APEX_CONTRIBUTIONS_MEDIAPROVIDER": "com.android.mediaprovider", |
| "RELEASE_APEX_CONTRIBUTIONS_MODULE_METADATA": "", |
| "RELEASE_APEX_CONTRIBUTIONS_NETWORKSTACKGOOGLE": "", |
| "RELEASE_APEX_CONTRIBUTIONS_NEURALNETWORKS": "com.android.neuralnetworks", |
| "RELEASE_APEX_CONTRIBUTIONS_ONDEVICEPERSONALIZATION": "com.android.ondevicepersonalization", |
| "RELEASE_APEX_CONTRIBUTIONS_PERMISSION": "com.android.permission", |
| "RELEASE_APEX_CONTRIBUTIONS_PRIMARY_LIBS": "", |
| "RELEASE_APEX_CONTRIBUTIONS_REMOTEKEYPROVISIONING": "com.android.rkpd", |
| "RELEASE_APEX_CONTRIBUTIONS_RESOLV": "com.android.resolv", |
| "RELEASE_APEX_CONTRIBUTIONS_SCHEDULING": "com.android.scheduling", |
| "RELEASE_APEX_CONTRIBUTIONS_SDKEXTENSIONS": "com.android.sdkext", |
| "RELEASE_APEX_CONTRIBUTIONS_SWCODEC": "com.android.media.swcodec", |
| "RELEASE_APEX_CONTRIBUTIONS_STATSD": "com.android.os.statsd", |
| "RELEASE_APEX_CONTRIBUTIONS_TELEMETRY_TVP": "", |
| "RELEASE_APEX_CONTRIBUTIONS_TZDATA": "com.android.tzdata", |
| "RELEASE_APEX_CONTRIBUTIONS_UWB": "com.android.uwb", |
| "RELEASE_APEX_CONTRIBUTIONS_WIFI": "com.android.wifi", |
| } |
| ) |
| |
| // Returns the list of _selected_ apex_contributions |
| // Each mainline module will have one entry in the list |
| func (c *config) AllApexContributions() []string { |
| ret := []string{} |
| for _, f := range SortedKeys(mainlineApexContributionBuildFlagsToApexNames) { |
| if val, exists := c.GetBuildFlag(f); exists && val != "" { |
| ret = append(ret, val) |
| } |
| } |
| return ret |
| } |
| |
| func (c *config) AllMainlineApexNames() []string { |
| return SortedStringValues(mainlineApexContributionBuildFlagsToApexNames) |
| } |
| |
| func (c *config) BuildIgnoreApexContributionContents() *bool { |
| return c.productVariables.BuildIgnoreApexContributionContents |
| } |
| |
| func (c *config) ProductLocales() []string { |
| return c.productVariables.ProductLocales |
| } |
| |
| func (c *config) ProductDefaultWifiChannels() []string { |
| return c.productVariables.ProductDefaultWifiChannels |
| } |
| |
| func (c *config) BoardUseVbmetaDigestInFingerprint() bool { |
| return Bool(c.productVariables.BoardUseVbmetaDigestInFingerprint) |
| } |
| |
| func (c *config) OemProperties() []string { |
| return c.productVariables.OemProperties |
| } |