blob: 2f8cb57f1106a3ad1a0c0f5cd5914550f8737f15 [file] [log] [blame]
Colin Cross3f40fa42015-01-30 17:27:36 -08001// Copyright 2015 Google Inc. All rights reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
Colin Cross635c3b02016-05-18 15:37:25 -070015package android
Colin Cross3f40fa42015-01-30 17:27:36 -080016
Jingwen Chenc711fec2020-11-22 23:52:50 -050017// This is the primary location to write and read all configuration values and
18// product variables necessary for soong_build's operation.
19
Colin Cross3f40fa42015-01-30 17:27:36 -080020import (
Colin Cross3f40fa42015-01-30 17:27:36 -080021 "encoding/json"
Lukacs T. Berki720b3962021-03-17 13:34:30 +010022 "errors"
Colin Cross3f40fa42015-01-30 17:27:36 -080023 "fmt"
Colin Crossd8f20142016-11-03 09:43:26 -070024 "io/ioutil"
Colin Cross3f40fa42015-01-30 17:27:36 -080025 "os"
Colin Cross35cec122015-04-02 14:37:16 -070026 "path/filepath"
Colin Cross3f40fa42015-01-30 17:27:36 -080027 "runtime"
Inseob Kim60c32f02020-12-21 22:53:05 +090028 "strconv"
Dan Willemsen34cc69e2015-09-23 15:26:20 -070029 "strings"
Colin Crossc1e86a32015-04-15 12:33:28 -070030 "sync"
Colin Cross6ff51382015-12-17 16:39:19 -080031
Colin Cross98be1bb2019-12-13 20:41:13 -080032 "github.com/google/blueprint"
Colin Crosse87040b2017-12-11 15:52:26 -080033 "github.com/google/blueprint/bootstrap"
Colin Cross98be1bb2019-12-13 20:41:13 -080034 "github.com/google/blueprint/pathtools"
Colin Cross6ff51382015-12-17 16:39:19 -080035 "github.com/google/blueprint/proptools"
Colin Cross9d34f352019-11-22 16:03:51 -080036
37 "android/soong/android/soongconfig"
Colin Cross3f40fa42015-01-30 17:27:36 -080038)
39
Jingwen Chenc711fec2020-11-22 23:52:50 -050040// Bool re-exports proptools.Bool for the android package.
Colin Cross6ff51382015-12-17 16:39:19 -080041var Bool = proptools.Bool
Jingwen Chenc711fec2020-11-22 23:52:50 -050042
43// String re-exports proptools.String for the android package.
Jack He8cc71432016-12-08 15:45:07 -080044var String = proptools.String
Jingwen Chenc711fec2020-11-22 23:52:50 -050045
46// StringDefault re-exports proptools.StringDefault for the android package.
Jeongik Cha219141c2020-08-06 23:00:37 +090047var StringDefault = proptools.StringDefault
Jiyong Park6a927c42020-01-21 02:03:43 +090048
Jingwen Chenc711fec2020-11-22 23:52:50 -050049// FutureApiLevelInt is a placeholder constant for unreleased API levels.
Dan Albert0b176c82020-07-23 16:43:25 -070050const FutureApiLevelInt = 10000
51
Jingwen Chenc711fec2020-11-22 23:52:50 -050052// FutureApiLevel represents unreleased API levels.
Dan Albert0b176c82020-07-23 16:43:25 -070053var FutureApiLevel = ApiLevel{
54 value: "current",
55 number: FutureApiLevelInt,
56 isPreview: true,
57}
Colin Cross6ff51382015-12-17 16:39:19 -080058
Jingwen Chenc4d91bc2020-11-24 22:59:26 -050059// The product variables file name, containing product config from Kati.
Dan Willemsen87b17d12015-07-14 00:39:06 -070060const productVariablesFileName = "soong.variables"
Colin Cross3f40fa42015-01-30 17:27:36 -080061
Colin Cross9272ade2016-08-17 15:24:12 -070062// A Config object represents the entire build configuration for Android.
Colin Crossc3c0a492015-04-10 15:43:55 -070063type Config struct {
64 *config
65}
66
Jingwen Chenc711fec2020-11-22 23:52:50 -050067// BuildDir returns the build output directory for the configuration.
Jeff Gastonefc1b412017-03-29 17:29:06 -070068func (c Config) BuildDir() string {
69 return c.buildDir
70}
71
Lukacs T. Berki89e9a162021-03-12 08:31:32 +010072func (c Config) NinjaBuildDir() string {
73 return c.buildDir
74}
75
76func (c Config) SrcDir() string {
77 return c.srcDir
78}
79
Jingwen Chenc711fec2020-11-22 23:52:50 -050080// A DeviceConfig object represents the configuration for a particular device
81// being built. For now there will only be one of these, but in the future there
82// may be multiple devices being built.
Colin Cross9272ade2016-08-17 15:24:12 -070083type DeviceConfig struct {
84 *deviceConfig
85}
86
Jingwen Chenc711fec2020-11-22 23:52:50 -050087// VendorConfig represents the configuration for vendor-specific behavior.
Colin Cross9d34f352019-11-22 16:03:51 -080088type VendorConfig soongconfig.SoongConfig
Dan Willemsen0fe78662018-03-26 12:41:18 -070089
Jingwen Chenc711fec2020-11-22 23:52:50 -050090// Definition of general build configuration for soong_build. Some of these
Jingwen Chenc4d91bc2020-11-24 22:59:26 -050091// product configuration values are read from Kati-generated soong.variables.
Colin Cross1332b002015-04-07 17:11:30 -070092type config struct {
Jingwen Chenc711fec2020-11-22 23:52:50 -050093 // Options configurable with soong.variables
Dan Willemsen45133ac2018-03-09 21:22:06 -080094 productVariables productVariables
Colin Cross3f40fa42015-01-30 17:27:36 -080095
Dan Willemsen674dc7f2018-03-12 18:06:05 -070096 // Only available on configs created by TestConfig
97 TestProductVariables *productVariables
98
Jingwen Chenc711fec2020-11-22 23:52:50 -050099 // A specialized context object for Bazel/Soong mixed builds and migration
100 // purposes.
Chris Parsonsf3c96ef2020-09-29 02:23:17 -0400101 BazelContext BazelContext
102
Dan Willemsen87b17d12015-07-14 00:39:06 -0700103 ProductVariablesFileName string
104
Jaewoong Jung642916f2020-10-09 17:25:15 -0700105 Targets map[OsType][]Target
106 BuildOSTarget Target // the Target for tools run on the build machine
107 BuildOSCommonTarget Target // the Target for common (java) tools run on the build machine
108 AndroidCommonTarget Target // the Target for common modules for the Android device
109 AndroidFirstDeviceTarget Target // the first Target for modules for the Android device
Dan Willemsen218f6562015-07-08 18:13:11 -0700110
Jingwen Chenc711fec2020-11-22 23:52:50 -0500111 // multilibConflicts for an ArchType is true if there is earlier configured
112 // device architecture with the same multilib value.
Colin Cross3b19f5d2019-09-17 14:45:31 -0700113 multilibConflicts map[ArchType]bool
114
Colin Cross9272ade2016-08-17 15:24:12 -0700115 deviceConfig *deviceConfig
116
Chris Parsons8f232a22020-06-23 17:37:05 -0400117 srcDir string // the path of the root source directory
118 buildDir string // the path of the build output directory
119 moduleListFile string // the path to the file which lists blueprint files to parse.
Colin Crossc1e86a32015-04-15 12:33:28 -0700120
Colin Cross6ccbc912017-10-10 23:07:38 -0700121 env map[string]string
Dan Willemsene7680ba2015-09-11 17:06:19 -0700122 envLock sync.Mutex
123 envDeps map[string]string
124 envFrozen bool
Dan Willemsen5ba07e82015-12-11 13:51:06 -0800125
Jingwen Chencda22c92020-11-23 00:22:30 -0500126 // Changes behavior based on whether Kati runs after soong_build, or if soong_build
127 // runs standalone.
128 katiEnabled bool
Colin Cross1e7d3702016-08-24 15:25:47 -0700129
Colin Cross32616ed2017-09-05 21:56:44 -0700130 captureBuild bool // true for tests, saves build parameters for each module
131 ignoreEnvironment bool // true for tests, returns empty from all Getenv calls
Colin Crosscec81712017-07-13 14:43:27 -0700132
Colin Crosse87040b2017-12-11 15:52:26 -0800133 stopBefore bootstrap.StopBefore
134
Colin Cross98be1bb2019-12-13 20:41:13 -0800135 fs pathtools.FileSystem
136 mockBpList string
137
Colin Cross5e6a7972020-06-07 16:56:32 -0700138 // If testAllowNonExistentPaths is true then PathForSource and PathForModuleSrc won't error
139 // in tests when a path doesn't exist.
Pedro Loureiro5d190cc2021-02-15 15:41:33 +0000140 TestAllowNonExistentPaths bool
Colin Cross5e6a7972020-06-07 16:56:32 -0700141
Jingwen Chenc711fec2020-11-22 23:52:50 -0500142 // The list of files that when changed, must invalidate soong_build to
143 // regenerate build.ninja.
Colin Cross12129292020-10-29 18:23:58 -0700144 ninjaFileDepsSet sync.Map
145
Colin Cross9272ade2016-08-17 15:24:12 -0700146 OncePer
147}
148
149type deviceConfig struct {
Dan Willemsen00269f22017-07-06 16:59:48 -0700150 config *config
Colin Cross9272ade2016-08-17 15:24:12 -0700151 OncePer
Colin Cross3f40fa42015-01-30 17:27:36 -0800152}
153
Colin Cross485e5722015-08-27 13:28:01 -0700154type jsonConfigurable interface {
Colin Cross27385972015-09-18 10:57:10 -0700155 SetDefaultConfig()
Colin Cross485e5722015-08-27 13:28:01 -0700156}
Colin Cross3f40fa42015-01-30 17:27:36 -0800157
Colin Cross485e5722015-08-27 13:28:01 -0700158func loadConfig(config *config) error {
Colin Cross988414c2020-01-11 01:11:46 +0000159 return loadFromConfigFile(&config.productVariables, absolutePath(config.ProductVariablesFileName))
Colin Cross485e5722015-08-27 13:28:01 -0700160}
161
Jingwen Chenc711fec2020-11-22 23:52:50 -0500162// loadFromConfigFile loads and decodes configuration options from a JSON file
163// in the current working directory.
Colin Cross485e5722015-08-27 13:28:01 -0700164func loadFromConfigFile(configurable jsonConfigurable, filename string) error {
Colin Cross3f40fa42015-01-30 17:27:36 -0800165 // Try to open the file
Colin Cross485e5722015-08-27 13:28:01 -0700166 configFileReader, err := os.Open(filename)
Colin Cross3f40fa42015-01-30 17:27:36 -0800167 defer configFileReader.Close()
168 if os.IsNotExist(err) {
169 // Need to create a file, so that blueprint & ninja don't get in
170 // a dependency tracking loop.
171 // Make a file-configurable-options with defaults, write it out using
172 // a json writer.
Colin Cross27385972015-09-18 10:57:10 -0700173 configurable.SetDefaultConfig()
174 err = saveToConfigFile(configurable, filename)
Colin Cross3f40fa42015-01-30 17:27:36 -0800175 if err != nil {
176 return err
177 }
Colin Cross15cd21a2018-02-27 11:26:02 -0800178 } else if err != nil {
179 return fmt.Errorf("config file: could not open %s: %s", filename, err.Error())
Colin Cross3f40fa42015-01-30 17:27:36 -0800180 } else {
181 // Make a decoder for it
182 jsonDecoder := json.NewDecoder(configFileReader)
Colin Cross485e5722015-08-27 13:28:01 -0700183 err = jsonDecoder.Decode(configurable)
Colin Cross3f40fa42015-01-30 17:27:36 -0800184 if err != nil {
Colin Cross15cd21a2018-02-27 11:26:02 -0800185 return fmt.Errorf("config file: %s did not parse correctly: %s", filename, err.Error())
Colin Cross3f40fa42015-01-30 17:27:36 -0800186 }
187 }
188
Colin Cross3f40fa42015-01-30 17:27:36 -0800189 // No error
190 return nil
191}
192
Colin Crossd8f20142016-11-03 09:43:26 -0700193// atomically writes the config file in case two copies of soong_build are running simultaneously
194// (for example, docs generation and ninja manifest generation)
Colin Cross485e5722015-08-27 13:28:01 -0700195func saveToConfigFile(config jsonConfigurable, filename string) error {
Colin Cross3f40fa42015-01-30 17:27:36 -0800196 data, err := json.MarshalIndent(&config, "", " ")
197 if err != nil {
198 return fmt.Errorf("cannot marshal config data: %s", err.Error())
199 }
200
Colin Crossd8f20142016-11-03 09:43:26 -0700201 f, err := ioutil.TempFile(filepath.Dir(filename), "config")
Colin Cross3f40fa42015-01-30 17:27:36 -0800202 if err != nil {
Jingwen Chenc711fec2020-11-22 23:52:50 -0500203 return fmt.Errorf("cannot create empty config file %s: %s", filename, err.Error())
Colin Cross3f40fa42015-01-30 17:27:36 -0800204 }
Colin Crossd8f20142016-11-03 09:43:26 -0700205 defer os.Remove(f.Name())
206 defer f.Close()
Colin Cross3f40fa42015-01-30 17:27:36 -0800207
Colin Crossd8f20142016-11-03 09:43:26 -0700208 _, err = f.Write(data)
Colin Cross3f40fa42015-01-30 17:27:36 -0800209 if err != nil {
Colin Cross485e5722015-08-27 13:28:01 -0700210 return fmt.Errorf("default config file: %s could not be written: %s", filename, err.Error())
211 }
212
Colin Crossd8f20142016-11-03 09:43:26 -0700213 _, err = f.WriteString("\n")
Colin Cross485e5722015-08-27 13:28:01 -0700214 if err != nil {
215 return fmt.Errorf("default config file: %s could not be written: %s", filename, err.Error())
Colin Cross3f40fa42015-01-30 17:27:36 -0800216 }
217
Colin Crossd8f20142016-11-03 09:43:26 -0700218 f.Close()
219 os.Rename(f.Name(), filename)
220
Colin Cross3f40fa42015-01-30 17:27:36 -0800221 return nil
222}
223
Colin Cross988414c2020-01-11 01:11:46 +0000224// NullConfig returns a mostly empty Config for use by standalone tools like dexpreopt_gen that
225// use the android package.
226func NullConfig(buildDir string) Config {
227 return Config{
228 config: &config{
229 buildDir: buildDir,
230 fs: pathtools.OsFs,
231 },
232 }
233}
234
Jingwen Chenc711fec2020-11-22 23:52:50 -0500235// TestConfig returns a Config object for testing.
Colin Cross98be1bb2019-12-13 20:41:13 -0800236func TestConfig(buildDir string, env map[string]string, bp string, fs map[string][]byte) Config {
Colin Cross9c6241f2019-04-22 15:51:26 -0700237 envCopy := make(map[string]string)
238 for k, v := range env {
239 envCopy[k] = v
240 }
241
Jingwen Chen2838c812020-11-23 01:06:40 -0500242 // Copy the real PATH value to the test environment, it's needed by
243 // NonHermeticHostSystemTool() used in x86_darwin_host.go
Lukacs T. Berkideba7212021-03-04 10:50:10 +0100244 envCopy["PATH"] = os.Getenv("PATH")
Colin Cross9c6241f2019-04-22 15:51:26 -0700245
Dan Willemsen00269f22017-07-06 16:59:48 -0700246 config := &config{
Dan Willemsen45133ac2018-03-09 21:22:06 -0800247 productVariables: productVariables{
Dan Albert4f378d72020-07-23 17:32:15 -0700248 DeviceName: stringPtr("test_device"),
249 Platform_sdk_version: intPtr(30),
250 Platform_sdk_codename: stringPtr("S"),
251 Platform_version_active_codenames: []string{"S"},
252 DeviceSystemSdkVersions: []string{"14", "15"},
253 Platform_systemsdk_versions: []string{"29", "30"},
254 AAPTConfig: []string{"normal", "large", "xlarge", "hdpi", "xhdpi", "xxhdpi"},
255 AAPTPreferredConfig: stringPtr("xhdpi"),
256 AAPTCharacteristics: stringPtr("nosdcard"),
257 AAPTPrebuiltDPI: []string{"xhdpi", "xxhdpi"},
258 UncompressPrivAppDex: boolPtr(true),
Inseob Kim60c32f02020-12-21 22:53:05 +0900259 ShippingApiLevel: stringPtr("30"),
Dan Willemsen00269f22017-07-06 16:59:48 -0700260 },
261
Colin Cross6ccbc912017-10-10 23:07:38 -0700262 buildDir: buildDir,
263 captureBuild: true,
Colin Cross9c6241f2019-04-22 15:51:26 -0700264 env: envCopy,
Colin Cross5e6a7972020-06-07 16:56:32 -0700265
266 // Set testAllowNonExistentPaths so that test contexts don't need to specify every path
267 // passed to PathForSource or PathForModuleSrc.
Pedro Loureiro5d190cc2021-02-15 15:41:33 +0000268 TestAllowNonExistentPaths: true,
Chris Parsonsf3c96ef2020-09-29 02:23:17 -0400269
270 BazelContext: noopBazelContext{},
Dan Willemsen00269f22017-07-06 16:59:48 -0700271 }
272 config.deviceConfig = &deviceConfig{
273 config: config,
274 }
Dan Willemsen45133ac2018-03-09 21:22:06 -0800275 config.TestProductVariables = &config.productVariables
Dan Willemsen00269f22017-07-06 16:59:48 -0700276
Colin Cross98be1bb2019-12-13 20:41:13 -0800277 config.mockFileSystem(bp, fs)
278
Dan Willemsen00269f22017-07-06 16:59:48 -0700279 return Config{config}
Colin Crossce75d2c2016-10-06 16:12:58 -0700280}
281
Jingwen Chenc711fec2020-11-22 23:52:50 -0500282// TestArchConfigNativeBridge returns a Config object suitable for using
283// for tests that need to run the arch mutator for native bridge supported
284// archs.
Colin Cross98be1bb2019-12-13 20:41:13 -0800285func TestArchConfigNativeBridge(buildDir string, env map[string]string, bp string, fs map[string][]byte) Config {
286 testConfig := TestArchConfig(buildDir, env, bp, fs)
dimitry1f33e402019-03-26 12:39:31 +0100287 config := testConfig.config
288
Colin Cross0d99f7c2019-05-14 16:01:24 -0700289 config.Targets[Android] = []Target{
Jiyong Park1613e552020-09-14 19:43:17 +0900290 {Android, Arch{ArchType: X86_64, ArchVariant: "silvermont", Abi: []string{"arm64-v8a"}}, NativeBridgeDisabled, "", "", false},
291 {Android, Arch{ArchType: X86, ArchVariant: "silvermont", Abi: []string{"armeabi-v7a"}}, NativeBridgeDisabled, "", "", false},
292 {Android, Arch{ArchType: Arm64, ArchVariant: "armv8-a", Abi: []string{"arm64-v8a"}}, NativeBridgeEnabled, "x86_64", "arm64", false},
293 {Android, Arch{ArchType: Arm, ArchVariant: "armv7-a-neon", Abi: []string{"armeabi-v7a"}}, NativeBridgeEnabled, "x86", "arm", false},
dimitry1f33e402019-03-26 12:39:31 +0100294 }
295
296 return testConfig
297}
298
Paul Duffinecdac8a2021-02-24 19:18:42 +0000299func fuchsiaTargets() map[OsType][]Target {
300 return map[OsType][]Target{
301 Fuchsia: {
Jiyong Park1613e552020-09-14 19:43:17 +0900302 {Fuchsia, Arch{ArchType: Arm64, ArchVariant: "", Abi: []string{"arm64-v8a"}}, NativeBridgeDisabled, "", "", false},
Doug Hornc32c6b02019-01-17 14:44:05 -0800303 },
Paul Duffinecdac8a2021-02-24 19:18:42 +0000304 BuildOs: {
Jiyong Park1613e552020-09-14 19:43:17 +0900305 {BuildOs, Arch{ArchType: X86_64}, NativeBridgeDisabled, "", "", false},
Doug Hornc32c6b02019-01-17 14:44:05 -0800306 },
307 }
Doug Hornc32c6b02019-01-17 14:44:05 -0800308}
309
Paul Duffinecdac8a2021-02-24 19:18:42 +0000310var PrepareForTestSetDeviceToFuchsia = FixtureModifyConfig(func(config Config) {
311 config.Targets = fuchsiaTargets()
312})
313
Paul Duffin35816122021-02-24 01:49:52 +0000314func modifyTestConfigToSupportArchMutator(testConfig Config) {
Colin Crossae4c6182017-09-15 17:33:55 -0700315 config := testConfig.config
316
Dan Willemsen0ef639b2018-10-10 17:02:29 -0700317 config.Targets = map[OsType][]Target{
318 Android: []Target{
Jiyong Park1613e552020-09-14 19:43:17 +0900319 {Android, Arch{ArchType: Arm64, ArchVariant: "armv8-a", Abi: []string{"arm64-v8a"}}, NativeBridgeDisabled, "", "", false},
320 {Android, Arch{ArchType: Arm, ArchVariant: "armv7-a-neon", Abi: []string{"armeabi-v7a"}}, NativeBridgeDisabled, "", "", false},
Colin Crossae4c6182017-09-15 17:33:55 -0700321 },
Dan Willemsen0ef639b2018-10-10 17:02:29 -0700322 BuildOs: []Target{
Jiyong Park1613e552020-09-14 19:43:17 +0900323 {BuildOs, Arch{ArchType: X86_64}, NativeBridgeDisabled, "", "", false},
324 {BuildOs, Arch{ArchType: X86}, NativeBridgeDisabled, "", "", false},
Colin Crossae4c6182017-09-15 17:33:55 -0700325 },
326 }
327
Colin Cross0d99f7c2019-05-14 16:01:24 -0700328 if runtime.GOOS == "darwin" {
329 config.Targets[BuildOs] = config.Targets[BuildOs][:1]
330 }
331
Colin Cross0f7d2ef2019-10-16 11:03:10 -0700332 config.BuildOSTarget = config.Targets[BuildOs][0]
333 config.BuildOSCommonTarget = getCommonTargets(config.Targets[BuildOs])[0]
334 config.AndroidCommonTarget = getCommonTargets(config.Targets[Android])[0]
Jaewoong Jung642916f2020-10-09 17:25:15 -0700335 config.AndroidFirstDeviceTarget = firstTarget(config.Targets[Android], "lib64", "lib32")[0]
Inseob Kim1f086e22019-05-09 13:29:15 +0900336 config.TestProductVariables.DeviceArch = proptools.StringPtr("arm64")
337 config.TestProductVariables.DeviceArchVariant = proptools.StringPtr("armv8-a")
338 config.TestProductVariables.DeviceSecondaryArch = proptools.StringPtr("arm")
339 config.TestProductVariables.DeviceSecondaryArchVariant = proptools.StringPtr("armv7-a-neon")
Paul Duffin35816122021-02-24 01:49:52 +0000340}
Colin Cross2a076922018-10-04 23:28:25 -0700341
Paul Duffin35816122021-02-24 01:49:52 +0000342// TestArchConfig returns a Config object suitable for using for tests that
343// need to run the arch mutator.
344func TestArchConfig(buildDir string, env map[string]string, bp string, fs map[string][]byte) Config {
345 testConfig := TestConfig(buildDir, env, bp, fs)
346 modifyTestConfigToSupportArchMutator(testConfig)
Colin Crossae4c6182017-09-15 17:33:55 -0700347 return testConfig
348}
349
Jingwen Chenc711fec2020-11-22 23:52:50 -0500350// ConfigForAdditionalRun is a config object which is "reset" for another
351// bootstrap run. Only per-run data is reset. Data which needs to persist across
352// multiple runs in the same program execution is carried over (such as Bazel
353// context or environment deps).
Chris Parsonsf3c96ef2020-09-29 02:23:17 -0400354func ConfigForAdditionalRun(c Config) (Config, error) {
355 newConfig, err := NewConfig(c.srcDir, c.buildDir, c.moduleListFile)
356 if err != nil {
357 return Config{}, err
358 }
359 newConfig.BazelContext = c.BazelContext
360 newConfig.envDeps = c.envDeps
361 return newConfig, nil
362}
363
Jingwen Chenc711fec2020-11-22 23:52:50 -0500364// NewConfig creates a new Config object. The srcDir argument specifies the path
365// to the root source directory. It also loads the config file, if found.
Chris Parsons8f232a22020-06-23 17:37:05 -0400366func NewConfig(srcDir, buildDir string, moduleListFile string) (Config, error) {
Jingwen Chenc711fec2020-11-22 23:52:50 -0500367 // Make a config with default options.
Colin Cross9272ade2016-08-17 15:24:12 -0700368 config := &config{
Colin Cross9272ade2016-08-17 15:24:12 -0700369 ProductVariablesFileName: filepath.Join(buildDir, productVariablesFileName),
Dan Willemsen87b17d12015-07-14 00:39:06 -0700370
Colin Cross6ccbc912017-10-10 23:07:38 -0700371 env: originalEnv,
372
Colin Cross3b19f5d2019-09-17 14:45:31 -0700373 srcDir: srcDir,
374 buildDir: buildDir,
375 multilibConflicts: make(map[ArchType]bool),
Colin Cross98be1bb2019-12-13 20:41:13 -0800376
Chris Parsons8f232a22020-06-23 17:37:05 -0400377 moduleListFile: moduleListFile,
378 fs: pathtools.NewOsFs(absSrcDir),
Colin Cross68f55102015-03-25 14:43:57 -0700379 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800380
Dan Willemsen00269f22017-07-06 16:59:48 -0700381 config.deviceConfig = &deviceConfig{
Colin Cross9272ade2016-08-17 15:24:12 -0700382 config: config,
383 }
384
Liz Kammer7941b302020-07-28 13:27:34 -0700385 // Soundness check of the build and source directories. This won't catch strange
386 // configurations with symlinks, but at least checks the obvious case.
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700387 absBuildDir, err := filepath.Abs(buildDir)
388 if err != nil {
389 return Config{}, err
390 }
391
392 absSrcDir, err := filepath.Abs(srcDir)
393 if err != nil {
394 return Config{}, err
395 }
396
397 if strings.HasPrefix(absSrcDir, absBuildDir) {
398 return Config{}, fmt.Errorf("Build dir must not contain source directory")
399 }
400
Colin Cross3f40fa42015-01-30 17:27:36 -0800401 // Load any configurable options from the configuration file
Colin Cross9272ade2016-08-17 15:24:12 -0700402 err = loadConfig(config)
Colin Cross3f40fa42015-01-30 17:27:36 -0800403 if err != nil {
Colin Crossc3c0a492015-04-10 15:43:55 -0700404 return Config{}, err
Colin Cross3f40fa42015-01-30 17:27:36 -0800405 }
406
Jingwen Chencda22c92020-11-23 00:22:30 -0500407 KatiEnabledMarkerFile := filepath.Join(buildDir, ".soong.kati_enabled")
408 if _, err := os.Stat(absolutePath(KatiEnabledMarkerFile)); err == nil {
409 config.katiEnabled = true
Dan Willemsen5ba07e82015-12-11 13:51:06 -0800410 }
411
Jingwen Chenc711fec2020-11-22 23:52:50 -0500412 // Sets up the map of target OSes to the finer grained compilation targets
413 // that are configured from the product variables.
Colin Crossa1ad8d12016-06-01 17:09:44 -0700414 targets, err := decodeTargetProductVariables(config)
Dan Willemsen218f6562015-07-08 18:13:11 -0700415 if err != nil {
416 return Config{}, err
417 }
418
Paul Duffin1356d8c2020-02-25 19:26:33 +0000419 // Make the CommonOS OsType available for all products.
420 targets[CommonOS] = []Target{commonTargetMap[CommonOS.Name]}
421
Dan Albert4098deb2016-10-19 14:04:41 -0700422 var archConfig []archConfig
Jingwen Chenc4d91bc2020-11-24 22:59:26 -0500423 if config.NdkAbis() {
Dan Albert4098deb2016-10-19 14:04:41 -0700424 archConfig = getNdkAbisConfig()
Martin Stjernholmc1ecc432019-11-15 15:00:31 +0000425 } else if config.AmlAbis() {
426 archConfig = getAmlAbisConfig()
Dan Albert4098deb2016-10-19 14:04:41 -0700427 }
428
429 if archConfig != nil {
Dan Willemsen01a3c252019-01-11 19:02:16 -0800430 androidTargets, err := decodeArchSettings(Android, archConfig)
Dan Willemsen322acaf2016-01-12 23:07:05 -0800431 if err != nil {
432 return Config{}, err
433 }
Dan Willemsen0ef639b2018-10-10 17:02:29 -0700434 targets[Android] = androidTargets
Dan Willemsen322acaf2016-01-12 23:07:05 -0800435 }
436
Colin Cross3b19f5d2019-09-17 14:45:31 -0700437 multilib := make(map[string]bool)
438 for _, target := range targets[Android] {
439 if seen := multilib[target.Arch.ArchType.Multilib]; seen {
440 config.multilibConflicts[target.Arch.ArchType] = true
441 }
442 multilib[target.Arch.ArchType.Multilib] = true
443 }
444
Jingwen Chenc711fec2020-11-22 23:52:50 -0500445 // Map of OS to compilation targets.
Colin Crossa1ad8d12016-06-01 17:09:44 -0700446 config.Targets = targets
Jingwen Chenc711fec2020-11-22 23:52:50 -0500447
448 // Compilation targets for host tools.
Colin Cross0f7d2ef2019-10-16 11:03:10 -0700449 config.BuildOSTarget = config.Targets[BuildOs][0]
450 config.BuildOSCommonTarget = getCommonTargets(config.Targets[BuildOs])[0]
Jingwen Chenc711fec2020-11-22 23:52:50 -0500451
452 // Compilation targets for Android.
Colin Cross0f7d2ef2019-10-16 11:03:10 -0700453 if len(config.Targets[Android]) > 0 {
454 config.AndroidCommonTarget = getCommonTargets(config.Targets[Android])[0]
Jaewoong Jung642916f2020-10-09 17:25:15 -0700455 config.AndroidFirstDeviceTarget = firstTarget(config.Targets[Android], "lib64", "lib32")[0]
Colin Cross0f7d2ef2019-10-16 11:03:10 -0700456 }
Dan Willemsen218f6562015-07-08 18:13:11 -0700457
Colin Cross1a6acd42020-06-16 17:51:46 -0700458 if Bool(config.productVariables.GcovCoverage) && Bool(config.productVariables.ClangCoverage) {
459 return Config{}, fmt.Errorf("GcovCoverage and ClangCoverage cannot both be set")
460 }
461
462 config.productVariables.Native_coverage = proptools.BoolPtr(
463 Bool(config.productVariables.GcovCoverage) ||
464 Bool(config.productVariables.ClangCoverage))
465
Chris Parsonsf3c96ef2020-09-29 02:23:17 -0400466 config.BazelContext, err = NewBazelContext(config)
Colin Cross3f40fa42015-01-30 17:27:36 -0800467
Jingwen Chenc711fec2020-11-22 23:52:50 -0500468 return Config{config}, err
469}
Colin Cross988414c2020-01-11 01:11:46 +0000470
Colin Cross98be1bb2019-12-13 20:41:13 -0800471// mockFileSystem replaces all reads with accesses to the provided map of
472// filenames to contents stored as a byte slice.
473func (c *config) mockFileSystem(bp string, fs map[string][]byte) {
474 mockFS := map[string][]byte{}
475
476 if _, exists := mockFS["Android.bp"]; !exists {
477 mockFS["Android.bp"] = []byte(bp)
478 }
479
480 for k, v := range fs {
481 mockFS[k] = v
482 }
483
484 // no module list file specified; find every file named Blueprints or Android.bp
485 pathsToParse := []string{}
486 for candidate := range mockFS {
487 base := filepath.Base(candidate)
488 if base == "Blueprints" || base == "Android.bp" {
489 pathsToParse = append(pathsToParse, candidate)
490 }
491 }
492 if len(pathsToParse) < 1 {
493 panic(fmt.Sprintf("No Blueprint or Android.bp files found in mock filesystem: %v\n", mockFS))
494 }
495 mockFS[blueprint.MockModuleListFile] = []byte(strings.Join(pathsToParse, "\n"))
496
497 c.fs = pathtools.MockFs(mockFS)
498 c.mockBpList = blueprint.MockModuleListFile
499}
500
Colin Crosse87040b2017-12-11 15:52:26 -0800501func (c *config) StopBefore() bootstrap.StopBefore {
502 return c.stopBefore
Dan Willemsen218f6562015-07-08 18:13:11 -0700503}
504
Jingwen Chenc711fec2020-11-22 23:52:50 -0500505// SetStopBefore configures soong_build to exit earlier at a specific point.
Colin Crosse87040b2017-12-11 15:52:26 -0800506func (c *config) SetStopBefore(stopBefore bootstrap.StopBefore) {
507 c.stopBefore = stopBefore
508}
509
Lukacs T. Berkid1e3f1f2021-03-16 08:55:23 +0100510func (c *config) SetAllowMissingDependencies() {
511 c.productVariables.Allow_missing_dependencies = proptools.BoolPtr(true)
512}
513
Colin Crosse87040b2017-12-11 15:52:26 -0800514var _ bootstrap.ConfigStopBefore = (*config)(nil)
515
Jingwen Chenc711fec2020-11-22 23:52:50 -0500516// BlueprintToolLocation returns the directory containing build system tools
517// from Blueprint, like soong_zip and merge_zips.
Dan Willemsenc2aa4a92016-05-26 15:13:03 -0700518func (c *config) BlueprintToolLocation() string {
519 return filepath.Join(c.buildDir, "host", c.PrebuiltOS(), "bin")
520}
521
Colin Crosse87040b2017-12-11 15:52:26 -0800522var _ bootstrap.ConfigBlueprintToolLocation = (*config)(nil)
523
Dan Willemsen60e62f02018-11-16 21:05:32 -0800524func (c *config) HostToolPath(ctx PathContext, tool string) Path {
525 return PathForOutput(ctx, "host", c.PrebuiltOS(), "bin", tool)
526}
527
Martin Stjernholm7260d062019-12-09 21:47:14 +0000528func (c *config) HostJNIToolPath(ctx PathContext, path string) Path {
529 ext := ".so"
530 if runtime.GOOS == "darwin" {
531 ext = ".dylib"
532 }
533 return PathForOutput(ctx, "host", c.PrebuiltOS(), "lib64", path+ext)
534}
535
536func (c *config) HostJavaToolPath(ctx PathContext, path string) Path {
537 return PathForOutput(ctx, "host", c.PrebuiltOS(), "framework", path)
538}
539
Jingwen Chenc711fec2020-11-22 23:52:50 -0500540// PrebuiltOS returns the name of the host OS used in prebuilts directories.
Colin Cross1332b002015-04-07 17:11:30 -0700541func (c *config) PrebuiltOS() string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800542 switch runtime.GOOS {
543 case "linux":
544 return "linux-x86"
545 case "darwin":
546 return "darwin-x86"
547 default:
548 panic("Unknown GOOS")
549 }
550}
551
552// GoRoot returns the path to the root directory of the Go toolchain.
Colin Cross1332b002015-04-07 17:11:30 -0700553func (c *config) GoRoot() string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800554 return fmt.Sprintf("%s/prebuilts/go/%s", c.srcDir, c.PrebuiltOS())
555}
556
Jingwen Chenc711fec2020-11-22 23:52:50 -0500557// PrebuiltBuildTool returns the path to a tool in the prebuilts directory containing
558// checked-in tools, like Kati, Ninja or Toybox, for the current host OS.
Dan Willemsen4e0aa232019-04-10 22:59:54 -0700559func (c *config) PrebuiltBuildTool(ctx PathContext, tool string) Path {
560 return PathForSource(ctx, "prebuilts/build-tools", c.PrebuiltOS(), "bin", tool)
561}
562
Jingwen Chenc711fec2020-11-22 23:52:50 -0500563// CpPreserveSymlinksFlags returns the host-specific flag for the cp(1) command
564// to preserve symlinks.
Colin Cross1332b002015-04-07 17:11:30 -0700565func (c *config) CpPreserveSymlinksFlags() string {
Colin Cross485e5722015-08-27 13:28:01 -0700566 switch runtime.GOOS {
Colin Cross3f40fa42015-01-30 17:27:36 -0800567 case "darwin":
568 return "-R"
569 case "linux":
570 return "-d"
571 default:
572 return ""
573 }
574}
Colin Cross68f55102015-03-25 14:43:57 -0700575
Colin Cross1332b002015-04-07 17:11:30 -0700576func (c *config) Getenv(key string) string {
Colin Cross68f55102015-03-25 14:43:57 -0700577 var val string
578 var exists bool
Colin Crossc1e86a32015-04-15 12:33:28 -0700579 c.envLock.Lock()
Colin Crossc0d58b42017-02-06 15:40:41 -0800580 defer c.envLock.Unlock()
581 if c.envDeps == nil {
582 c.envDeps = make(map[string]string)
583 }
Colin Cross68f55102015-03-25 14:43:57 -0700584 if val, exists = c.envDeps[key]; !exists {
Dan Willemsene7680ba2015-09-11 17:06:19 -0700585 if c.envFrozen {
586 panic("Cannot access new environment variables after envdeps are frozen")
587 }
Colin Cross6ccbc912017-10-10 23:07:38 -0700588 val, _ = c.env[key]
Colin Cross68f55102015-03-25 14:43:57 -0700589 c.envDeps[key] = val
590 }
591 return val
592}
593
Colin Cross99d7c232016-11-23 16:52:04 -0800594func (c *config) GetenvWithDefault(key string, defaultValue string) string {
595 ret := c.Getenv(key)
596 if ret == "" {
597 return defaultValue
598 }
599 return ret
600}
601
602func (c *config) IsEnvTrue(key string) bool {
603 value := c.Getenv(key)
604 return value == "1" || value == "y" || value == "yes" || value == "on" || value == "true"
605}
606
607func (c *config) IsEnvFalse(key string) bool {
608 value := c.Getenv(key)
609 return value == "0" || value == "n" || value == "no" || value == "off" || value == "false"
610}
611
Jingwen Chenc711fec2020-11-22 23:52:50 -0500612// EnvDeps returns the environment variables this build depends on. The first
613// call to this function blocks future reads from the environment.
Colin Cross1332b002015-04-07 17:11:30 -0700614func (c *config) EnvDeps() map[string]string {
Dan Willemsene7680ba2015-09-11 17:06:19 -0700615 c.envLock.Lock()
Colin Crossc0d58b42017-02-06 15:40:41 -0800616 defer c.envLock.Unlock()
Dan Willemsene7680ba2015-09-11 17:06:19 -0700617 c.envFrozen = true
Colin Cross68f55102015-03-25 14:43:57 -0700618 return c.envDeps
619}
Colin Cross35cec122015-04-02 14:37:16 -0700620
Jingwen Chencda22c92020-11-23 00:22:30 -0500621func (c *config) KatiEnabled() bool {
622 return c.katiEnabled
Dan Willemsen5ba07e82015-12-11 13:51:06 -0800623}
624
Nan Zhang581fd212018-01-10 16:06:12 -0800625func (c *config) BuildId() string {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800626 return String(c.productVariables.BuildId)
Nan Zhang581fd212018-01-10 16:06:12 -0800627}
628
Jingwen Chenc711fec2020-11-22 23:52:50 -0500629// BuildNumberFile returns the path to a text file containing metadata
630// representing the current build's number.
631//
632// Rules that want to reference the build number should read from this file
633// without depending on it. They will run whenever their other dependencies
634// require them to run and get the current build number. This ensures they don't
635// rebuild on every incremental build when the build number changes.
Colin Cross2a2e0db2020-02-21 16:55:46 -0800636func (c *config) BuildNumberFile(ctx PathContext) Path {
637 return PathForOutput(ctx, String(c.productVariables.BuildNumberFile))
Nan Zhang581fd212018-01-10 16:06:12 -0800638}
639
Jingwen Chenc711fec2020-11-22 23:52:50 -0500640// DeviceName returns the name of the current device target.
Colin Cross35cec122015-04-02 14:37:16 -0700641// TODO: take an AndroidModuleContext to select the device name for multi-device builds
Colin Cross1332b002015-04-07 17:11:30 -0700642func (c *config) DeviceName() string {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800643 return *c.productVariables.DeviceName
Colin Cross35cec122015-04-02 14:37:16 -0700644}
645
Anton Hansson53c88442019-03-18 15:53:16 +0000646func (c *config) DeviceResourceOverlays() []string {
647 return c.productVariables.DeviceResourceOverlays
648}
649
650func (c *config) ProductResourceOverlays() []string {
651 return c.productVariables.ProductResourceOverlays
Colin Cross30e076a2015-04-13 13:58:27 -0700652}
653
Colin Crossbfd347d2018-05-09 11:11:35 -0700654func (c *config) PlatformVersionName() string {
655 return String(c.productVariables.Platform_version_name)
656}
657
Dan Albert4f378d72020-07-23 17:32:15 -0700658func (c *config) PlatformSdkVersion() ApiLevel {
659 return uncheckedFinalApiLevel(*c.productVariables.Platform_sdk_version)
Colin Cross30e076a2015-04-13 13:58:27 -0700660}
661
Colin Crossd09b0b62018-04-18 11:06:47 -0700662func (c *config) PlatformSdkCodename() string {
663 return String(c.productVariables.Platform_sdk_codename)
664}
665
Colin Cross092c9da2019-04-02 22:56:43 -0700666func (c *config) PlatformSecurityPatch() string {
667 return String(c.productVariables.Platform_security_patch)
668}
669
670func (c *config) PlatformPreviewSdkVersion() string {
671 return String(c.productVariables.Platform_preview_sdk_version)
672}
673
674func (c *config) PlatformMinSupportedTargetSdkVersion() string {
675 return String(c.productVariables.Platform_min_supported_target_sdk_version)
676}
677
678func (c *config) PlatformBaseOS() string {
679 return String(c.productVariables.Platform_base_os)
680}
681
Dan Albert1a246272020-07-06 14:49:35 -0700682func (c *config) MinSupportedSdkVersion() ApiLevel {
683 return uncheckedFinalApiLevel(16)
684}
685
686func (c *config) FinalApiLevels() []ApiLevel {
687 var levels []ApiLevel
Dan Albert4f378d72020-07-23 17:32:15 -0700688 for i := 1; i <= c.PlatformSdkVersion().FinalOrFutureInt(); i++ {
Dan Albert1a246272020-07-06 14:49:35 -0700689 levels = append(levels, uncheckedFinalApiLevel(i))
690 }
691 return levels
692}
693
694func (c *config) PreviewApiLevels() []ApiLevel {
695 var levels []ApiLevel
696 for i, codename := range c.PlatformVersionActiveCodenames() {
697 levels = append(levels, ApiLevel{
698 value: codename,
699 number: i,
700 isPreview: true,
701 })
702 }
703 return levels
704}
705
706func (c *config) AllSupportedApiLevels() []ApiLevel {
707 var levels []ApiLevel
708 levels = append(levels, c.FinalApiLevels()...)
709 return append(levels, c.PreviewApiLevels()...)
Dan Albertf5415d72017-08-17 16:19:59 -0700710}
711
Jingwen Chenc711fec2020-11-22 23:52:50 -0500712// DefaultAppTargetSdk returns the API level that platform apps are targeting.
713// This converts a codename to the exact ApiLevel it represents.
Dan Albert4f378d72020-07-23 17:32:15 -0700714func (c *config) DefaultAppTargetSdk(ctx EarlyModuleContext) ApiLevel {
Colin Crossd09b0b62018-04-18 11:06:47 -0700715 if Bool(c.productVariables.Platform_sdk_final) {
716 return c.PlatformSdkVersion()
Colin Crossd09b0b62018-04-18 11:06:47 -0700717 }
Jingwen Chenc711fec2020-11-22 23:52:50 -0500718 codename := c.PlatformSdkCodename()
719 if codename == "" {
720 return NoneApiLevel
721 }
722 if codename == "REL" {
723 panic("Platform_sdk_codename should not be REL when Platform_sdk_final is true")
724 }
725 return ApiLevelOrPanic(ctx, codename)
Colin Crossd09b0b62018-04-18 11:06:47 -0700726}
727
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800728func (c *config) AppsDefaultVersionName() string {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800729 return String(c.productVariables.AppsDefaultVersionName)
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800730}
731
Dan Albert31384de2017-07-28 12:39:46 -0700732// Codenames that are active in the current lunch target.
733func (c *config) PlatformVersionActiveCodenames() []string {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800734 return c.productVariables.Platform_version_active_codenames
Dan Albert31384de2017-07-28 12:39:46 -0700735}
736
Colin Crossface4e42017-10-30 17:32:15 -0700737func (c *config) ProductAAPTConfig() []string {
Colin Crossa74ca042019-01-31 14:31:51 -0800738 return c.productVariables.AAPTConfig
Colin Cross30e076a2015-04-13 13:58:27 -0700739}
740
Colin Crossface4e42017-10-30 17:32:15 -0700741func (c *config) ProductAAPTPreferredConfig() string {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800742 return String(c.productVariables.AAPTPreferredConfig)
Colin Cross30e076a2015-04-13 13:58:27 -0700743}
744
Colin Crossface4e42017-10-30 17:32:15 -0700745func (c *config) ProductAAPTCharacteristics() string {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800746 return String(c.productVariables.AAPTCharacteristics)
Colin Crossface4e42017-10-30 17:32:15 -0700747}
748
749func (c *config) ProductAAPTPrebuiltDPI() []string {
Colin Crossa74ca042019-01-31 14:31:51 -0800750 return c.productVariables.AAPTPrebuiltDPI
Colin Cross30e076a2015-04-13 13:58:27 -0700751}
752
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700753func (c *config) DefaultAppCertificateDir(ctx PathContext) SourcePath {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800754 defaultCert := String(c.productVariables.DefaultAppCertificate)
Colin Cross61ae0b72017-12-01 17:16:02 -0800755 if defaultCert != "" {
756 return PathForSource(ctx, filepath.Dir(defaultCert))
Colin Cross61ae0b72017-12-01 17:16:02 -0800757 }
Jingwen Chenc711fec2020-11-22 23:52:50 -0500758 return PathForSource(ctx, "build/make/target/product/security")
Colin Cross30e076a2015-04-13 13:58:27 -0700759}
760
Colin Crosse1731a52017-12-14 11:22:55 -0800761func (c *config) DefaultAppCertificate(ctx PathContext) (pem, key SourcePath) {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800762 defaultCert := String(c.productVariables.DefaultAppCertificate)
Colin Cross61ae0b72017-12-01 17:16:02 -0800763 if defaultCert != "" {
Colin Crosse1731a52017-12-14 11:22:55 -0800764 return PathForSource(ctx, defaultCert+".x509.pem"), PathForSource(ctx, defaultCert+".pk8")
Colin Cross61ae0b72017-12-01 17:16:02 -0800765 }
Jingwen Chenc711fec2020-11-22 23:52:50 -0500766 defaultDir := c.DefaultAppCertificateDir(ctx)
767 return defaultDir.Join(ctx, "testkey.x509.pem"), defaultDir.Join(ctx, "testkey.pk8")
Colin Cross30e076a2015-04-13 13:58:27 -0700768}
Colin Cross6ff51382015-12-17 16:39:19 -0800769
Jiyong Park9335a262018-12-24 11:31:58 +0900770func (c *config) ApexKeyDir(ctx ModuleContext) SourcePath {
771 // TODO(b/121224311): define another variable such as TARGET_APEX_KEY_OVERRIDE
772 defaultCert := String(c.productVariables.DefaultAppCertificate)
Dan Willemsen412160e2019-04-09 21:36:26 -0700773 if defaultCert == "" || filepath.Dir(defaultCert) == "build/make/target/product/security" {
Jiyong Park9335a262018-12-24 11:31:58 +0900774 // When defaultCert is unset or is set to the testkeys path, use the APEX keys
775 // that is under the module dir
Colin Cross07e51612019-03-05 12:46:40 -0800776 return pathForModuleSrc(ctx)
Jiyong Park9335a262018-12-24 11:31:58 +0900777 }
Jingwen Chenc711fec2020-11-22 23:52:50 -0500778 // If not, APEX keys are under the specified directory
779 return PathForSource(ctx, filepath.Dir(defaultCert))
Jiyong Park9335a262018-12-24 11:31:58 +0900780}
781
Jingwen Chenc711fec2020-11-22 23:52:50 -0500782// AllowMissingDependencies configures Blueprint/Soong to not fail when modules
783// are configured to depend on non-existent modules. Note that this does not
784// affect missing input dependencies at the Ninja level.
Colin Cross6ff51382015-12-17 16:39:19 -0800785func (c *config) AllowMissingDependencies() bool {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800786 return Bool(c.productVariables.Allow_missing_dependencies)
Colin Cross6ff51382015-12-17 16:39:19 -0800787}
Dan Willemsen322acaf2016-01-12 23:07:05 -0800788
Jeongik Cha816a23a2020-07-08 01:09:23 +0900789// Returns true if a full platform source tree cannot be assumed.
Colin Crossfc3674a2017-09-18 17:41:52 -0700790func (c *config) UnbundledBuild() bool {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800791 return Bool(c.productVariables.Unbundled_build)
Colin Crossfc3674a2017-09-18 17:41:52 -0700792}
793
Martin Stjernholmfd9eb4b2020-06-17 01:13:15 +0100794// Returns true if building apps that aren't bundled with the platform.
795// UnbundledBuild() is always true when this is true.
796func (c *config) UnbundledBuildApps() bool {
797 return Bool(c.productVariables.Unbundled_build_apps)
798}
799
Jeongik Cha816a23a2020-07-08 01:09:23 +0900800// Returns true if building modules against prebuilt SDKs.
801func (c *config) AlwaysUsePrebuiltSdks() bool {
802 return Bool(c.productVariables.Always_use_prebuilt_sdks)
Colin Cross1f367bf2018-12-18 22:46:24 -0800803}
804
Paul Duffin9a89a2a2020-10-28 19:20:06 +0000805// Returns true if the boot jars check should be skipped.
806func (c *config) SkipBootJarsCheck() bool {
807 return Bool(c.productVariables.Skip_boot_jars_check)
808}
809
Doug Horn21b94272019-01-16 12:06:11 -0800810func (c *config) Fuchsia() bool {
811 return Bool(c.productVariables.Fuchsia)
812}
813
Colin Cross126a25c2017-10-31 13:55:34 -0700814func (c *config) MinimizeJavaDebugInfo() bool {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800815 return Bool(c.productVariables.MinimizeJavaDebugInfo) && !Bool(c.productVariables.Eng)
Colin Cross126a25c2017-10-31 13:55:34 -0700816}
817
Colin Crossed064c02018-09-05 16:28:13 -0700818func (c *config) Debuggable() bool {
819 return Bool(c.productVariables.Debuggable)
820}
821
Jaewoong Jung1d6eb682018-11-29 15:08:44 -0800822func (c *config) Eng() bool {
823 return Bool(c.productVariables.Eng)
824}
825
Jiyong Park8d52f862018-07-07 18:02:07 +0900826func (c *config) DevicePrimaryArchType() ArchType {
Dan Willemsen0ef639b2018-10-10 17:02:29 -0700827 return c.Targets[Android][0].Arch.ArchType
Jiyong Park8d52f862018-07-07 18:02:07 +0900828}
829
Colin Cross16b23492016-01-06 14:41:07 -0800830func (c *config) SanitizeHost() []string {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800831 return append([]string(nil), c.productVariables.SanitizeHost...)
Colin Cross16b23492016-01-06 14:41:07 -0800832}
833
834func (c *config) SanitizeDevice() []string {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800835 return append([]string(nil), c.productVariables.SanitizeDevice...)
Colin Cross23ae82a2016-11-02 14:34:39 -0700836}
837
Ivan Lozano0c3a1ef2017-06-28 09:10:48 -0700838func (c *config) SanitizeDeviceDiag() []string {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800839 return append([]string(nil), c.productVariables.SanitizeDeviceDiag...)
Ivan Lozano0c3a1ef2017-06-28 09:10:48 -0700840}
841
Colin Cross23ae82a2016-11-02 14:34:39 -0700842func (c *config) SanitizeDeviceArch() []string {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800843 return append([]string(nil), c.productVariables.SanitizeDeviceArch...)
Colin Cross16b23492016-01-06 14:41:07 -0800844}
Colin Crossa1ad8d12016-06-01 17:09:44 -0700845
Vishwath Mohan1b017a72017-01-19 13:54:55 -0800846func (c *config) EnableCFI() bool {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800847 if c.productVariables.EnableCFI == nil {
Vishwath Mohanc32c3eb2017-01-24 14:20:54 -0800848 return true
Vishwath Mohanc32c3eb2017-01-24 14:20:54 -0800849 }
Jingwen Chenc711fec2020-11-22 23:52:50 -0500850 return *c.productVariables.EnableCFI
Vishwath Mohan1b017a72017-01-19 13:54:55 -0800851}
852
Kostya Kortchinskyd5275c82019-02-01 08:42:56 -0800853func (c *config) DisableScudo() bool {
854 return Bool(c.productVariables.DisableScudo)
855}
856
Colin Crossa1ad8d12016-06-01 17:09:44 -0700857func (c *config) Android64() bool {
Dan Willemsen0ef639b2018-10-10 17:02:29 -0700858 for _, t := range c.Targets[Android] {
Colin Crossa1ad8d12016-06-01 17:09:44 -0700859 if t.Arch.ArchType.Multilib == "lib64" {
860 return true
861 }
862 }
863
864 return false
865}
Colin Cross9272ade2016-08-17 15:24:12 -0700866
Colin Cross9d45bb72016-08-29 16:14:13 -0700867func (c *config) UseGoma() bool {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800868 return Bool(c.productVariables.UseGoma)
Colin Cross9d45bb72016-08-29 16:14:13 -0700869}
870
Ramy Medhatbbf25672019-07-17 12:30:04 +0000871func (c *config) UseRBE() bool {
872 return Bool(c.productVariables.UseRBE)
873}
874
Ramy Medhat8ea054a2020-01-27 14:19:44 -0500875func (c *config) UseRBEJAVAC() bool {
876 return Bool(c.productVariables.UseRBEJAVAC)
877}
878
879func (c *config) UseRBER8() bool {
880 return Bool(c.productVariables.UseRBER8)
881}
882
883func (c *config) UseRBED8() bool {
884 return Bool(c.productVariables.UseRBED8)
885}
886
Colin Cross8b8bec32019-11-15 13:18:43 -0800887func (c *config) UseRemoteBuild() bool {
888 return c.UseGoma() || c.UseRBE()
889}
890
Colin Cross66548102018-06-19 22:47:35 -0700891func (c *config) RunErrorProne() bool {
892 return c.IsEnvTrue("RUN_ERROR_PRONE")
893}
894
Jingwen Chenc711fec2020-11-22 23:52:50 -0500895// XrefCorpusName returns the Kythe cross-reference corpus name.
Sasha Smundak2a4549e2018-11-05 16:49:08 -0800896func (c *config) XrefCorpusName() string {
897 return c.Getenv("XREF_CORPUS")
898}
899
Jingwen Chenc711fec2020-11-22 23:52:50 -0500900// XrefCuEncoding returns the compilation unit encoding to use for Kythe code
901// xrefs. Can be 'json' (default), 'proto' or 'all'.
Sasha Smundak6c2d4f92020-01-09 17:34:23 -0800902func (c *config) XrefCuEncoding() string {
903 if enc := c.Getenv("KYTHE_KZIP_ENCODING"); enc != "" {
904 return enc
905 }
906 return "json"
907}
908
Sasha Smundakb0addaf2021-02-16 10:39:40 -0800909// XrefCuJavaSourceMax returns the maximum number of the Java source files
910// in a single compilation unit
911const xrefJavaSourceFileMaxDefault = "1000"
912
913func (c Config) XrefCuJavaSourceMax() string {
914 v := c.Getenv("KYTHE_JAVA_SOURCE_BATCH_SIZE")
915 if v == "" {
916 return xrefJavaSourceFileMaxDefault
917 }
918 if _, err := strconv.ParseUint(v, 0, 0); err != nil {
919 fmt.Fprintf(os.Stderr,
920 "bad KYTHE_JAVA_SOURCE_BATCH_SIZE value: %s, will use %s",
921 err, xrefJavaSourceFileMaxDefault)
922 return xrefJavaSourceFileMaxDefault
923 }
924 return v
925
926}
927
Sasha Smundak2a4549e2018-11-05 16:49:08 -0800928func (c *config) EmitXrefRules() bool {
929 return c.XrefCorpusName() != ""
930}
931
Dan Willemsena03cf6d2016-09-26 15:45:04 -0700932func (c *config) ClangTidy() bool {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800933 return Bool(c.productVariables.ClangTidy)
Dan Willemsena03cf6d2016-09-26 15:45:04 -0700934}
935
936func (c *config) TidyChecks() string {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800937 if c.productVariables.TidyChecks == nil {
Dan Willemsena03cf6d2016-09-26 15:45:04 -0700938 return ""
939 }
Dan Willemsen45133ac2018-03-09 21:22:06 -0800940 return *c.productVariables.TidyChecks
Dan Willemsena03cf6d2016-09-26 15:45:04 -0700941}
942
Colin Cross0f4e0d62016-07-27 10:56:55 -0700943func (c *config) LibartImgHostBaseAddress() string {
944 return "0x60000000"
945}
946
947func (c *config) LibartImgDeviceBaseAddress() string {
Elliott Hughesda3a0712020-03-06 16:55:28 -0800948 return "0x70000000"
Colin Cross0f4e0d62016-07-27 10:56:55 -0700949}
950
Hiroshi Yamauchie2a10632016-12-19 13:44:41 -0800951func (c *config) ArtUseReadBarrier() bool {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800952 return Bool(c.productVariables.ArtUseReadBarrier)
Hiroshi Yamauchie2a10632016-12-19 13:44:41 -0800953}
954
Jingwen Chenc711fec2020-11-22 23:52:50 -0500955// Enforce Runtime Resource Overlays for a module. RROs supersede static RROs,
956// but some modules still depend on it.
957//
958// More info: https://source.android.com/devices/architecture/rros
Dan Willemsen3fb1fae2018-03-12 15:30:26 -0700959func (c *config) EnforceRROForModule(name string) bool {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800960 enforceList := c.productVariables.EnforceRROTargets
Jeongik Chacee5ba92021-02-19 12:11:51 +0900961
Roland Levillainf6cc2612020-07-09 16:58:14 +0100962 if len(enforceList) > 0 {
Yo Chiang4ebd06a2019-10-01 13:13:41 +0800963 if InList("*", enforceList) {
Dan Willemsen3fb1fae2018-03-12 15:30:26 -0700964 return true
965 }
Colin Crossa74ca042019-01-31 14:31:51 -0800966 return InList(name, enforceList)
Dan Willemsen3fb1fae2018-03-12 15:30:26 -0700967 }
968 return false
969}
Dan Willemsen3fb1fae2018-03-12 15:30:26 -0700970func (c *config) EnforceRROExcludedOverlay(path string) bool {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800971 excluded := c.productVariables.EnforceRROExcludedOverlays
Roland Levillainf6cc2612020-07-09 16:58:14 +0100972 if len(excluded) > 0 {
Jaewoong Jung3aff5782020-02-11 07:54:35 -0800973 return HasAnyPrefix(path, excluded)
Dan Willemsen3fb1fae2018-03-12 15:30:26 -0700974 }
975 return false
976}
977
978func (c *config) ExportedNamespaces() []string {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800979 return append([]string(nil), c.productVariables.NamespacesToExport...)
Dan Willemsen3fb1fae2018-03-12 15:30:26 -0700980}
981
982func (c *config) HostStaticBinaries() bool {
Dan Willemsen45133ac2018-03-09 21:22:06 -0800983 return Bool(c.productVariables.HostStaticBinaries)
Dan Willemsen3fb1fae2018-03-12 15:30:26 -0700984}
985
Colin Cross5a0dcd52018-10-05 14:20:06 -0700986func (c *config) UncompressPrivAppDex() bool {
987 return Bool(c.productVariables.UncompressPrivAppDex)
988}
989
990func (c *config) ModulesLoadedByPrivilegedModules() []string {
991 return c.productVariables.ModulesLoadedByPrivilegedModules
992}
993
Jingwen Chenc711fec2020-11-22 23:52:50 -0500994// DexpreoptGlobalConfigPath returns the path to the dexpreopt.config file in
995// the output directory, if it was created during the product configuration
996// phase by Kati.
Jingwen Chenebb0b572020-11-02 00:24:57 -0500997func (c *config) DexpreoptGlobalConfigPath(ctx PathContext) OptionalPath {
Colin Cross988414c2020-01-11 01:11:46 +0000998 if c.productVariables.DexpreoptGlobalConfig == nil {
Jingwen Chenebb0b572020-11-02 00:24:57 -0500999 return OptionalPathForPath(nil)
1000 }
1001 return OptionalPathForPath(
1002 pathForBuildToolDep(ctx, *c.productVariables.DexpreoptGlobalConfig))
1003}
1004
Jingwen Chenc711fec2020-11-22 23:52:50 -05001005// DexpreoptGlobalConfig returns the raw byte contents of the dexpreopt global
1006// configuration. Since the configuration file was created by Kati during
1007// product configuration (externally of soong_build), it's not tracked, so we
1008// also manually add a Ninja file dependency on the configuration file to the
1009// rule that creates the main build.ninja file. This ensures that build.ninja is
1010// regenerated correctly if dexpreopt.config changes.
Jingwen Chenebb0b572020-11-02 00:24:57 -05001011func (c *config) DexpreoptGlobalConfig(ctx PathContext) ([]byte, error) {
1012 path := c.DexpreoptGlobalConfigPath(ctx)
1013 if !path.Valid() {
Colin Cross988414c2020-01-11 01:11:46 +00001014 return nil, nil
1015 }
Jingwen Chenebb0b572020-11-02 00:24:57 -05001016 ctx.AddNinjaFileDeps(path.String())
1017 return ioutil.ReadFile(absolutePath(path.String()))
Colin Cross43f08db2018-11-12 10:13:39 -08001018}
1019
David Brazdil91b4e3e2019-01-23 21:04:05 +00001020func (c *config) FrameworksBaseDirExists(ctx PathContext) bool {
1021 return ExistentPathForSource(ctx, "frameworks", "base").Valid()
1022}
1023
Inseob Kimae553032019-05-14 18:52:49 +09001024func (c *config) VndkSnapshotBuildArtifacts() bool {
1025 return Bool(c.productVariables.VndkSnapshotBuildArtifacts)
1026}
1027
Colin Cross3b19f5d2019-09-17 14:45:31 -07001028func (c *config) HasMultilibConflict(arch ArchType) bool {
1029 return c.multilibConflicts[arch]
1030}
1031
Bill Peckhambae47492021-01-08 09:34:44 -08001032func (c *config) PrebuiltHiddenApiDir(ctx PathContext) string {
1033 return String(c.productVariables.PrebuiltHiddenApiDir)
1034}
1035
Colin Cross9272ade2016-08-17 15:24:12 -07001036func (c *deviceConfig) Arches() []Arch {
1037 var arches []Arch
Dan Willemsen0ef639b2018-10-10 17:02:29 -07001038 for _, target := range c.config.Targets[Android] {
Colin Cross9272ade2016-08-17 15:24:12 -07001039 arches = append(arches, target.Arch)
1040 }
1041 return arches
1042}
Dan Willemsend2ede872016-11-18 14:54:24 -08001043
Jayant Chowdhary34ce67d2018-03-08 11:00:50 -08001044func (c *deviceConfig) BinderBitness() string {
Dan Willemsen45133ac2018-03-09 21:22:06 -08001045 is32BitBinder := c.config.productVariables.Binder32bit
Jayant Chowdhary34ce67d2018-03-08 11:00:50 -08001046 if is32BitBinder != nil && *is32BitBinder {
1047 return "32"
1048 }
1049 return "64"
1050}
1051
Dan Willemsen4353bc42016-12-05 17:16:02 -08001052func (c *deviceConfig) VendorPath() string {
Dan Willemsen45133ac2018-03-09 21:22:06 -08001053 if c.config.productVariables.VendorPath != nil {
1054 return *c.config.productVariables.VendorPath
Dan Willemsen4353bc42016-12-05 17:16:02 -08001055 }
1056 return "vendor"
1057}
1058
Justin Yun71549282017-11-17 12:10:28 +09001059func (c *deviceConfig) VndkVersion() string {
Dan Willemsen45133ac2018-03-09 21:22:06 -08001060 return String(c.config.productVariables.DeviceVndkVersion)
Justin Yun71549282017-11-17 12:10:28 +09001061}
1062
Jose Galmes6f843bc2020-12-11 13:36:29 -08001063func (c *deviceConfig) RecoverySnapshotVersion() string {
1064 return String(c.config.productVariables.RecoverySnapshotVersion)
1065}
1066
Jeongik Cha219141c2020-08-06 23:00:37 +09001067func (c *deviceConfig) CurrentApiLevelForVendorModules() string {
1068 return StringDefault(c.config.productVariables.DeviceCurrentApiLevelForVendorModules, "current")
1069}
1070
Justin Yun8fe12122017-12-07 17:18:15 +09001071func (c *deviceConfig) PlatformVndkVersion() string {
Dan Willemsen45133ac2018-03-09 21:22:06 -08001072 return String(c.config.productVariables.Platform_vndk_version)
Justin Yun8fe12122017-12-07 17:18:15 +09001073}
1074
Justin Yun5f7f7e82019-11-18 19:52:14 +09001075func (c *deviceConfig) ProductVndkVersion() string {
1076 return String(c.config.productVariables.ProductVndkVersion)
1077}
1078
Justin Yun71549282017-11-17 12:10:28 +09001079func (c *deviceConfig) ExtraVndkVersions() []string {
Dan Willemsen45133ac2018-03-09 21:22:06 -08001080 return c.config.productVariables.ExtraVndkVersions
Dan Willemsend2ede872016-11-18 14:54:24 -08001081}
Jack He8cc71432016-12-08 15:45:07 -08001082
Vic Yangefd249e2018-11-12 20:19:56 -08001083func (c *deviceConfig) VndkUseCoreVariant() bool {
1084 return Bool(c.config.productVariables.VndkUseCoreVariant)
1085}
1086
Jiyong Park1a5d7b12018-01-15 15:05:10 +09001087func (c *deviceConfig) SystemSdkVersions() []string {
Colin Crossa74ca042019-01-31 14:31:51 -08001088 return c.config.productVariables.DeviceSystemSdkVersions
Jiyong Park1a5d7b12018-01-15 15:05:10 +09001089}
1090
1091func (c *deviceConfig) PlatformSystemSdkVersions() []string {
Dan Willemsen45133ac2018-03-09 21:22:06 -08001092 return c.config.productVariables.Platform_systemsdk_versions
Jiyong Park1a5d7b12018-01-15 15:05:10 +09001093}
1094
Jiyong Park2db76922017-11-08 16:03:48 +09001095func (c *deviceConfig) OdmPath() string {
Dan Willemsen45133ac2018-03-09 21:22:06 -08001096 if c.config.productVariables.OdmPath != nil {
1097 return *c.config.productVariables.OdmPath
Jiyong Park2db76922017-11-08 16:03:48 +09001098 }
1099 return "odm"
1100}
1101
Jaekyun Seok5cfbfbb2018-01-10 19:00:15 +09001102func (c *deviceConfig) ProductPath() string {
Dan Willemsen45133ac2018-03-09 21:22:06 -08001103 if c.config.productVariables.ProductPath != nil {
1104 return *c.config.productVariables.ProductPath
Jiyong Park2db76922017-11-08 16:03:48 +09001105 }
Jaekyun Seok5cfbfbb2018-01-10 19:00:15 +09001106 return "product"
Jiyong Park2db76922017-11-08 16:03:48 +09001107}
1108
Justin Yund5f6c822019-06-25 16:47:17 +09001109func (c *deviceConfig) SystemExtPath() string {
1110 if c.config.productVariables.SystemExtPath != nil {
1111 return *c.config.productVariables.SystemExtPath
Dario Frenifd05a742018-05-29 13:28:54 +01001112 }
Justin Yund5f6c822019-06-25 16:47:17 +09001113 return "system_ext"
Dario Frenifd05a742018-05-29 13:28:54 +01001114}
1115
Jack He8cc71432016-12-08 15:45:07 -08001116func (c *deviceConfig) BtConfigIncludeDir() string {
Dan Willemsen45133ac2018-03-09 21:22:06 -08001117 return String(c.config.productVariables.BtConfigIncludeDir)
Jack He8cc71432016-12-08 15:45:07 -08001118}
Dan Willemsen581341d2017-02-09 16:16:31 -08001119
Jiyong Parkd773eb32017-07-03 13:18:12 +09001120func (c *deviceConfig) DeviceKernelHeaderDirs() []string {
Dan Willemsen45133ac2018-03-09 21:22:06 -08001121 return c.config.productVariables.DeviceKernelHeaders
Jiyong Parkd773eb32017-07-03 13:18:12 +09001122}
1123
Yi Kongceb5b762020-03-20 15:22:27 +08001124func (c *deviceConfig) SamplingPGO() bool {
1125 return Bool(c.config.productVariables.SamplingPGO)
1126}
1127
Roland Levillainada12702020-06-09 13:07:36 +01001128// JavaCoverageEnabledForPath returns whether Java code coverage is enabled for
1129// path. Coverage is enabled by default when the product variable
1130// JavaCoveragePaths is empty. If JavaCoveragePaths is not empty, coverage is
1131// enabled for any path which is part of this variable (and not part of the
1132// JavaCoverageExcludePaths product variable). Value "*" in JavaCoveragePaths
1133// represents any path.
1134func (c *deviceConfig) JavaCoverageEnabledForPath(path string) bool {
1135 coverage := false
Chris Gross2f748692020-06-24 20:36:59 +00001136 if len(c.config.productVariables.JavaCoveragePaths) == 0 ||
Roland Levillainada12702020-06-09 13:07:36 +01001137 InList("*", c.config.productVariables.JavaCoveragePaths) ||
1138 HasAnyPrefix(path, c.config.productVariables.JavaCoveragePaths) {
1139 coverage = true
1140 }
Roland Levillainf6cc2612020-07-09 16:58:14 +01001141 if coverage && len(c.config.productVariables.JavaCoverageExcludePaths) > 0 {
Roland Levillainada12702020-06-09 13:07:36 +01001142 if HasAnyPrefix(path, c.config.productVariables.JavaCoverageExcludePaths) {
1143 coverage = false
1144 }
1145 }
1146 return coverage
1147}
1148
Colin Cross1a6acd42020-06-16 17:51:46 -07001149// Returns true if gcov or clang coverage is enabled.
Dan Willemsen581341d2017-02-09 16:16:31 -08001150func (c *deviceConfig) NativeCoverageEnabled() bool {
Colin Cross1a6acd42020-06-16 17:51:46 -07001151 return Bool(c.config.productVariables.GcovCoverage) ||
1152 Bool(c.config.productVariables.ClangCoverage)
Dan Willemsen581341d2017-02-09 16:16:31 -08001153}
1154
Oliver Nguyen1382ab62019-12-06 15:22:41 -08001155func (c *deviceConfig) ClangCoverageEnabled() bool {
1156 return Bool(c.config.productVariables.ClangCoverage)
1157}
1158
Colin Cross1a6acd42020-06-16 17:51:46 -07001159func (c *deviceConfig) GcovCoverageEnabled() bool {
1160 return Bool(c.config.productVariables.GcovCoverage)
1161}
1162
Roland Levillain4f5297b2020-06-09 12:44:06 +01001163// NativeCoverageEnabledForPath returns whether (GCOV- or Clang-based) native
1164// code coverage is enabled for path. By default, coverage is not enabled for a
1165// given path unless it is part of the NativeCoveragePaths product variable (and
1166// not part of the NativeCoverageExcludePaths product variable). Value "*" in
1167// NativeCoveragePaths represents any path.
1168func (c *deviceConfig) NativeCoverageEnabledForPath(path string) bool {
Ryan Campbell469a18a2017-02-27 09:01:54 -08001169 coverage := false
Roland Levillainf6cc2612020-07-09 16:58:14 +01001170 if len(c.config.productVariables.NativeCoveragePaths) > 0 {
Roland Levillain4f5297b2020-06-09 12:44:06 +01001171 if InList("*", c.config.productVariables.NativeCoveragePaths) || HasAnyPrefix(path, c.config.productVariables.NativeCoveragePaths) {
Ivan Lozano5f595532017-07-13 14:46:05 -07001172 coverage = true
Dan Willemsen581341d2017-02-09 16:16:31 -08001173 }
1174 }
Roland Levillainf6cc2612020-07-09 16:58:14 +01001175 if coverage && len(c.config.productVariables.NativeCoverageExcludePaths) > 0 {
Roland Levillain4f5297b2020-06-09 12:44:06 +01001176 if HasAnyPrefix(path, c.config.productVariables.NativeCoverageExcludePaths) {
Ivan Lozano5f595532017-07-13 14:46:05 -07001177 coverage = false
Ryan Campbell469a18a2017-02-27 09:01:54 -08001178 }
1179 }
1180 return coverage
Dan Willemsen581341d2017-02-09 16:16:31 -08001181}
Ivan Lozano5f595532017-07-13 14:46:05 -07001182
Pirama Arumuga Nainar49540802018-01-29 23:11:42 -08001183func (c *deviceConfig) PgoAdditionalProfileDirs() []string {
Dan Willemsen45133ac2018-03-09 21:22:06 -08001184 return c.config.productVariables.PgoAdditionalProfileDirs
Pirama Arumuga Nainar49540802018-01-29 23:11:42 -08001185}
1186
Tri Vo35a51432018-03-25 20:00:00 -07001187func (c *deviceConfig) VendorSepolicyDirs() []string {
1188 return c.config.productVariables.BoardVendorSepolicyDirs
1189}
1190
1191func (c *deviceConfig) OdmSepolicyDirs() []string {
1192 return c.config.productVariables.BoardOdmSepolicyDirs
1193}
1194
Felixa20a8752020-05-17 18:28:35 +02001195func (c *deviceConfig) SystemExtPublicSepolicyDirs() []string {
1196 return c.config.productVariables.SystemExtPublicSepolicyDirs
Tri Vo35a51432018-03-25 20:00:00 -07001197}
1198
Felixa20a8752020-05-17 18:28:35 +02001199func (c *deviceConfig) SystemExtPrivateSepolicyDirs() []string {
1200 return c.config.productVariables.SystemExtPrivateSepolicyDirs
Tri Vo35a51432018-03-25 20:00:00 -07001201}
1202
Inseob Kim0866b002019-04-15 20:21:29 +09001203func (c *deviceConfig) SepolicyM4Defs() []string {
1204 return c.config.productVariables.BoardSepolicyM4Defs
1205}
1206
Jiyong Park7f67f482019-01-05 12:57:48 +09001207func (c *deviceConfig) OverrideManifestPackageNameFor(name string) (manifestName string, overridden bool) {
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001208 return findOverrideValue(c.config.productVariables.ManifestPackageNameOverrides, name,
1209 "invalid override rule %q in PRODUCT_MANIFEST_PACKAGE_NAME_OVERRIDES should be <module_name>:<manifest_name>")
1210}
1211
1212func (c *deviceConfig) OverrideCertificateFor(name string) (certificatePath string, overridden bool) {
Jaewoong Jungacb6db32019-02-28 16:22:30 +00001213 return findOverrideValue(c.config.productVariables.CertificateOverrides, name,
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001214 "invalid override rule %q in PRODUCT_CERTIFICATE_OVERRIDES should be <module_name>:<certificate_module_name>")
1215}
1216
Jaewoong Jung9d22a912019-01-23 16:27:47 -08001217func (c *deviceConfig) OverridePackageNameFor(name string) string {
1218 newName, overridden := findOverrideValue(
1219 c.config.productVariables.PackageNameOverrides,
1220 name,
1221 "invalid override rule %q in PRODUCT_PACKAGE_NAME_OVERRIDES should be <module_name>:<package_name>")
1222 if overridden {
1223 return newName
1224 }
1225 return name
1226}
1227
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001228func findOverrideValue(overrides []string, name string, errorMsg string) (newValue string, overridden bool) {
Jiyong Park7f67f482019-01-05 12:57:48 +09001229 if overrides == nil || len(overrides) == 0 {
1230 return "", false
1231 }
1232 for _, o := range overrides {
1233 split := strings.Split(o, ":")
1234 if len(split) != 2 {
1235 // This shouldn't happen as this is first checked in make, but just in case.
Jaewoong Jung2ad817c2019-01-18 14:27:16 -08001236 panic(fmt.Errorf(errorMsg, o))
Jiyong Park7f67f482019-01-05 12:57:48 +09001237 }
1238 if matchPattern(split[0], name) {
1239 return substPattern(split[0], split[1], name), true
1240 }
1241 }
1242 return "", false
1243}
1244
Ivan Lozano5f595532017-07-13 14:46:05 -07001245func (c *config) IntegerOverflowDisabledForPath(path string) bool {
Roland Levillainf6cc2612020-07-09 16:58:14 +01001246 if len(c.productVariables.IntegerOverflowExcludePaths) == 0 {
Ivan Lozano5f595532017-07-13 14:46:05 -07001247 return false
1248 }
Jaewoong Jung3aff5782020-02-11 07:54:35 -08001249 return HasAnyPrefix(path, c.productVariables.IntegerOverflowExcludePaths)
Ivan Lozano5f595532017-07-13 14:46:05 -07001250}
Vishwath Mohan1fa3ac52017-10-31 02:26:14 -07001251
1252func (c *config) CFIDisabledForPath(path string) bool {
Roland Levillainf6cc2612020-07-09 16:58:14 +01001253 if len(c.productVariables.CFIExcludePaths) == 0 {
Vishwath Mohan1fa3ac52017-10-31 02:26:14 -07001254 return false
1255 }
Jaewoong Jung3aff5782020-02-11 07:54:35 -08001256 return HasAnyPrefix(path, c.productVariables.CFIExcludePaths)
Vishwath Mohan1fa3ac52017-10-31 02:26:14 -07001257}
1258
1259func (c *config) CFIEnabledForPath(path string) bool {
Roland Levillainf6cc2612020-07-09 16:58:14 +01001260 if len(c.productVariables.CFIIncludePaths) == 0 {
Vishwath Mohan1fa3ac52017-10-31 02:26:14 -07001261 return false
1262 }
Jaewoong Jung3aff5782020-02-11 07:54:35 -08001263 return HasAnyPrefix(path, c.productVariables.CFIIncludePaths)
Vishwath Mohan1fa3ac52017-10-31 02:26:14 -07001264}
Colin Crosse15ddaf2017-12-04 11:24:31 -08001265
Evgenii Stepanov4beaa0c2021-01-05 16:41:26 -08001266func (c *config) MemtagHeapDisabledForPath(path string) bool {
1267 if len(c.productVariables.MemtagHeapExcludePaths) == 0 {
1268 return false
1269 }
1270 return HasAnyPrefix(path, c.productVariables.MemtagHeapExcludePaths)
1271}
1272
1273func (c *config) MemtagHeapAsyncEnabledForPath(path string) bool {
1274 if len(c.productVariables.MemtagHeapAsyncIncludePaths) == 0 {
1275 return false
1276 }
1277 return HasAnyPrefix(path, c.productVariables.MemtagHeapAsyncIncludePaths)
1278}
1279
1280func (c *config) MemtagHeapSyncEnabledForPath(path string) bool {
1281 if len(c.productVariables.MemtagHeapSyncIncludePaths) == 0 {
1282 return false
1283 }
1284 return HasAnyPrefix(path, c.productVariables.MemtagHeapSyncIncludePaths)
1285}
1286
Dan Willemsen0fe78662018-03-26 12:41:18 -07001287func (c *config) VendorConfig(name string) VendorConfig {
Colin Cross9d34f352019-11-22 16:03:51 -08001288 return soongconfig.Config(c.productVariables.VendorVars[name])
Dan Willemsen0fe78662018-03-26 12:41:18 -07001289}
1290
Colin Cross395f2cf2018-10-24 16:10:32 -07001291func (c *config) NdkAbis() bool {
1292 return Bool(c.productVariables.Ndk_abis)
1293}
1294
Martin Stjernholmc1ecc432019-11-15 15:00:31 +00001295func (c *config) AmlAbis() bool {
1296 return Bool(c.productVariables.Aml_abis)
1297}
1298
Dan Albert23d37e02018-11-28 08:30:10 -08001299func (c *config) ExcludeDraftNdkApis() bool {
1300 return Bool(c.productVariables.Exclude_draft_ndk_apis)
1301}
1302
Jiyong Park8fd61922018-11-08 02:50:25 +09001303func (c *config) FlattenApex() bool {
Roland Levillaina3863212019-08-12 19:56:16 +01001304 return Bool(c.productVariables.Flatten_apex)
Jiyong Park8fd61922018-11-08 02:50:25 +09001305}
1306
Jiyong Park4da07972021-01-05 21:01:11 +09001307func (c *config) ForceApexSymlinkOptimization() bool {
1308 return Bool(c.productVariables.ForceApexSymlinkOptimization)
1309}
1310
Mohammad Samiul Islam3cd005d2020-11-26 13:32:26 +00001311func (c *config) CompressedApex() bool {
1312 return Bool(c.productVariables.CompressedApex)
1313}
1314
Jeongik Chac9464142019-01-07 12:07:27 +09001315func (c *config) EnforceSystemCertificate() bool {
1316 return Bool(c.productVariables.EnforceSystemCertificate)
1317}
1318
Colin Cross440e0d02020-06-11 11:32:11 -07001319func (c *config) EnforceSystemCertificateAllowList() []string {
1320 return c.productVariables.EnforceSystemCertificateAllowList
Jeongik Chac9464142019-01-07 12:07:27 +09001321}
1322
Jeongik Cha2cc570d2019-10-29 15:44:45 +09001323func (c *config) EnforceProductPartitionInterface() bool {
1324 return Bool(c.productVariables.EnforceProductPartitionInterface)
1325}
1326
JaeMan Parkff715562020-10-19 17:25:58 +09001327func (c *config) EnforceInterPartitionJavaSdkLibrary() bool {
1328 return Bool(c.productVariables.EnforceInterPartitionJavaSdkLibrary)
1329}
1330
1331func (c *config) InterPartitionJavaLibraryAllowList() []string {
1332 return c.productVariables.InterPartitionJavaLibraryAllowList
1333}
1334
Jooyung Han3ab2c3e2019-12-05 16:27:44 +09001335func (c *config) InstallExtraFlattenedApexes() bool {
1336 return Bool(c.productVariables.InstallExtraFlattenedApexes)
1337}
1338
Colin Crossf24a22a2019-01-31 14:12:44 -08001339func (c *config) ProductHiddenAPIStubs() []string {
1340 return c.productVariables.ProductHiddenAPIStubs
Colin Cross8faf8fc2019-01-16 15:15:52 -08001341}
1342
Colin Crossf24a22a2019-01-31 14:12:44 -08001343func (c *config) ProductHiddenAPIStubsSystem() []string {
1344 return c.productVariables.ProductHiddenAPIStubsSystem
Colin Cross8faf8fc2019-01-16 15:15:52 -08001345}
1346
Colin Crossf24a22a2019-01-31 14:12:44 -08001347func (c *config) ProductHiddenAPIStubsTest() []string {
1348 return c.productVariables.ProductHiddenAPIStubsTest
Colin Cross8faf8fc2019-01-16 15:15:52 -08001349}
Dan Willemsen71c74602019-04-10 12:27:35 -07001350
Dan Willemsen54879d12019-04-18 10:08:46 -07001351func (c *deviceConfig) TargetFSConfigGen() []string {
Dan Willemsen71c74602019-04-10 12:27:35 -07001352 return c.config.productVariables.TargetFSConfigGen
1353}
Inseob Kim0866b002019-04-15 20:21:29 +09001354
1355func (c *config) ProductPublicSepolicyDirs() []string {
1356 return c.productVariables.ProductPublicSepolicyDirs
1357}
1358
1359func (c *config) ProductPrivateSepolicyDirs() []string {
1360 return c.productVariables.ProductPrivateSepolicyDirs
1361}
1362
Colin Cross50ddcc42019-05-16 12:28:22 -07001363func (c *config) MissingUsesLibraries() []string {
1364 return c.productVariables.MissingUsesLibraries
1365}
1366
Inseob Kim1f086e22019-05-09 13:29:15 +09001367func (c *deviceConfig) DeviceArch() string {
1368 return String(c.config.productVariables.DeviceArch)
1369}
1370
1371func (c *deviceConfig) DeviceArchVariant() string {
1372 return String(c.config.productVariables.DeviceArchVariant)
1373}
1374
1375func (c *deviceConfig) DeviceSecondaryArch() string {
1376 return String(c.config.productVariables.DeviceSecondaryArch)
1377}
1378
1379func (c *deviceConfig) DeviceSecondaryArchVariant() string {
1380 return String(c.config.productVariables.DeviceSecondaryArchVariant)
1381}
Yifan Hong82db7352020-01-21 16:12:26 -08001382
1383func (c *deviceConfig) BoardUsesRecoveryAsBoot() bool {
1384 return Bool(c.config.productVariables.BoardUsesRecoveryAsBoot)
1385}
Yifan Hong97365ee2020-07-29 09:51:57 -07001386
1387func (c *deviceConfig) BoardKernelBinaries() []string {
1388 return c.config.productVariables.BoardKernelBinaries
1389}
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001390
Yifan Hong42bef8d2020-08-05 14:36:09 -07001391func (c *deviceConfig) BoardKernelModuleInterfaceVersions() []string {
1392 return c.config.productVariables.BoardKernelModuleInterfaceVersions
1393}
1394
Yifan Hongdd8dacc2020-10-21 15:40:17 -07001395func (c *deviceConfig) BoardMoveRecoveryResourcesToVendorBoot() bool {
1396 return Bool(c.config.productVariables.BoardMoveRecoveryResourcesToVendorBoot)
1397}
1398
Inseob Kim16ebd5a2020-12-09 23:08:17 +09001399func (c *deviceConfig) PlatformSepolicyVersion() string {
1400 return String(c.config.productVariables.PlatformSepolicyVersion)
1401}
1402
1403func (c *deviceConfig) BoardSepolicyVers() string {
1404 return String(c.config.productVariables.BoardSepolicyVers)
1405}
1406
1407func (c *deviceConfig) BoardReqdMaskPolicy() []string {
1408 return c.config.productVariables.BoardReqdMaskPolicy
1409}
1410
Inseob Kim7cf14652021-01-06 23:06:52 +09001411func (c *deviceConfig) DirectedVendorSnapshot() bool {
1412 return c.config.productVariables.DirectedVendorSnapshot
1413}
1414
1415func (c *deviceConfig) VendorSnapshotModules() map[string]bool {
1416 return c.config.productVariables.VendorSnapshotModules
1417}
1418
Jose Galmes4c6895e2021-02-09 07:44:30 -08001419func (c *deviceConfig) DirectedRecoverySnapshot() bool {
1420 return c.config.productVariables.DirectedRecoverySnapshot
1421}
1422
1423func (c *deviceConfig) RecoverySnapshotModules() map[string]bool {
1424 return c.config.productVariables.RecoverySnapshotModules
1425}
1426
Justin DeMartino383bfb32021-02-24 10:49:43 -08001427func createDirsMap(previous map[string]bool, dirs []string) (map[string]bool, error) {
1428 var ret = make(map[string]bool)
1429 for _, dir := range dirs {
1430 clean := filepath.Clean(dir)
1431 if previous[clean] || ret[clean] {
1432 return nil, fmt.Errorf("Duplicate entry %s", dir)
1433 }
1434 ret[clean] = true
1435 }
1436 return ret, nil
1437}
1438
1439func (c *deviceConfig) createDirsMapOnce(onceKey OnceKey, previous map[string]bool, dirs []string) map[string]bool {
1440 dirMap := c.Once(onceKey, func() interface{} {
1441 ret, err := createDirsMap(previous, dirs)
1442 if err != nil {
1443 panic(fmt.Errorf("%s: %w", onceKey.key, err))
1444 }
1445 return ret
1446 })
1447 if dirMap == nil {
1448 return nil
1449 }
1450 return dirMap.(map[string]bool)
1451}
1452
1453var vendorSnapshotDirsExcludedKey = NewOnceKey("VendorSnapshotDirsExcludedMap")
1454
1455func (c *deviceConfig) VendorSnapshotDirsExcludedMap() map[string]bool {
1456 return c.createDirsMapOnce(vendorSnapshotDirsExcludedKey, nil,
1457 c.config.productVariables.VendorSnapshotDirsExcluded)
1458}
1459
1460var vendorSnapshotDirsIncludedKey = NewOnceKey("VendorSnapshotDirsIncludedMap")
1461
1462func (c *deviceConfig) VendorSnapshotDirsIncludedMap() map[string]bool {
1463 excludedMap := c.VendorSnapshotDirsExcludedMap()
1464 return c.createDirsMapOnce(vendorSnapshotDirsIncludedKey, excludedMap,
1465 c.config.productVariables.VendorSnapshotDirsIncluded)
1466}
1467
1468var recoverySnapshotDirsExcludedKey = NewOnceKey("RecoverySnapshotDirsExcludedMap")
1469
1470func (c *deviceConfig) RecoverySnapshotDirsExcludedMap() map[string]bool {
1471 return c.createDirsMapOnce(recoverySnapshotDirsExcludedKey, nil,
1472 c.config.productVariables.RecoverySnapshotDirsExcluded)
1473}
1474
1475var recoverySnapshotDirsIncludedKey = NewOnceKey("RecoverySnapshotDirsIncludedMap")
1476
1477func (c *deviceConfig) RecoverySnapshotDirsIncludedMap() map[string]bool {
1478 excludedMap := c.RecoverySnapshotDirsExcludedMap()
1479 return c.createDirsMapOnce(recoverySnapshotDirsIncludedKey, excludedMap,
1480 c.config.productVariables.RecoverySnapshotDirsIncluded)
1481}
1482
Inseob Kim60c32f02020-12-21 22:53:05 +09001483func (c *deviceConfig) ShippingApiLevel() ApiLevel {
1484 if c.config.productVariables.ShippingApiLevel == nil {
1485 return NoneApiLevel
1486 }
1487 apiLevel, _ := strconv.Atoi(*c.config.productVariables.ShippingApiLevel)
1488 return uncheckedFinalApiLevel(apiLevel)
1489}
1490
Inseob Kim0cac7b42021-02-03 18:16:46 +09001491func (c *deviceConfig) BuildBrokenVendorPropertyNamespace() bool {
1492 return c.config.productVariables.BuildBrokenVendorPropertyNamespace
1493}
1494
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001495// The ConfiguredJarList struct provides methods for handling a list of (apex, jar) pairs.
1496// Such lists are used in the build system for things like bootclasspath jars or system server jars.
1497// The apex part is either an apex name, or a special names "platform" or "system_ext". Jar is a
1498// module name. The pairs come from Make product variables as a list of colon-separated strings.
1499//
1500// Examples:
1501// - "com.android.art:core-oj"
1502// - "platform:framework"
1503// - "system_ext:foo"
1504//
1505type ConfiguredJarList struct {
Jingwen Chenc711fec2020-11-22 23:52:50 -05001506 // A list of apex components, which can be an apex name,
1507 // or special names like "platform" or "system_ext".
1508 apexes []string
1509
1510 // A list of jar module name components.
1511 jars []string
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001512}
1513
Jingwen Chenc711fec2020-11-22 23:52:50 -05001514// Len returns the length of the list of jars.
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001515func (l *ConfiguredJarList) Len() int {
1516 return len(l.jars)
1517}
1518
Jingwen Chenc711fec2020-11-22 23:52:50 -05001519// Jar returns the idx-th jar component of (apex, jar) pairs.
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001520func (l *ConfiguredJarList) Jar(idx int) string {
1521 return l.jars[idx]
1522}
1523
Jingwen Chenc711fec2020-11-22 23:52:50 -05001524// Apex returns the idx-th apex component of (apex, jar) pairs.
Paul Duffin9a89a2a2020-10-28 19:20:06 +00001525func (l *ConfiguredJarList) Apex(idx int) string {
1526 return l.apexes[idx]
1527}
1528
Jingwen Chenc711fec2020-11-22 23:52:50 -05001529// ContainsJar returns true if the (apex, jar) pairs contains a pair with the
1530// given jar module name.
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001531func (l *ConfiguredJarList) ContainsJar(jar string) bool {
1532 return InList(jar, l.jars)
1533}
1534
1535// If the list contains the given (apex, jar) pair.
1536func (l *ConfiguredJarList) containsApexJarPair(apex, jar string) bool {
1537 for i := 0; i < l.Len(); i++ {
Paul Duffin1e8c6072020-10-23 18:28:55 +01001538 if apex == l.apexes[i] && jar == l.jars[i] {
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001539 return true
1540 }
1541 }
1542 return false
1543}
1544
Jingwen Chenc711fec2020-11-22 23:52:50 -05001545// IndexOfJar returns the first pair with the given jar name on the list, or -1
1546// if not found.
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001547func (l *ConfiguredJarList) IndexOfJar(jar string) int {
1548 return IndexList(jar, l.jars)
1549}
1550
Paul Duffin7d584e92020-10-23 18:26:03 +01001551func copyAndAppend(list []string, item string) []string {
1552 // Create the result list to be 1 longer than the input.
1553 result := make([]string, len(list)+1)
1554
1555 // Copy the whole input list into the result.
1556 count := copy(result, list)
1557
1558 // Insert the extra item at the end.
1559 result[count] = item
1560
1561 return result
1562}
1563
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001564// Append an (apex, jar) pair to the list.
Paul Duffin7d584e92020-10-23 18:26:03 +01001565func (l *ConfiguredJarList) Append(apex string, jar string) ConfiguredJarList {
1566 // Create a copy of the backing arrays before appending to avoid sharing backing
1567 // arrays that are mutated across instances.
1568 apexes := copyAndAppend(l.apexes, apex)
1569 jars := copyAndAppend(l.jars, jar)
1570
1571 return ConfiguredJarList{apexes, jars}
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001572}
1573
Jingwen Chenc711fec2020-11-22 23:52:50 -05001574// RemoveList filters out a list of (apex, jar) pairs from the receiving list of pairs.
Paul Duffin7d584e92020-10-23 18:26:03 +01001575func (l *ConfiguredJarList) RemoveList(list ConfiguredJarList) ConfiguredJarList {
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001576 apexes := make([]string, 0, l.Len())
1577 jars := make([]string, 0, l.Len())
1578
1579 for i, jar := range l.jars {
Paul Duffin1e8c6072020-10-23 18:28:55 +01001580 apex := l.apexes[i]
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001581 if !list.containsApexJarPair(apex, jar) {
1582 apexes = append(apexes, apex)
1583 jars = append(jars, jar)
1584 }
1585 }
1586
Paul Duffin7d584e92020-10-23 18:26:03 +01001587 return ConfiguredJarList{apexes, jars}
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001588}
1589
Jingwen Chenc711fec2020-11-22 23:52:50 -05001590// CopyOfJars returns a copy of the list of strings containing jar module name
1591// components.
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001592func (l *ConfiguredJarList) CopyOfJars() []string {
1593 return CopyOf(l.jars)
1594}
1595
Jingwen Chenc711fec2020-11-22 23:52:50 -05001596// CopyOfApexJarPairs returns a copy of the list of strings with colon-separated
1597// (apex, jar) pairs.
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001598func (l *ConfiguredJarList) CopyOfApexJarPairs() []string {
1599 pairs := make([]string, 0, l.Len())
1600
1601 for i, jar := range l.jars {
Paul Duffin1e8c6072020-10-23 18:28:55 +01001602 apex := l.apexes[i]
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001603 pairs = append(pairs, apex+":"+jar)
1604 }
1605
1606 return pairs
1607}
1608
Jingwen Chenc711fec2020-11-22 23:52:50 -05001609// BuildPaths returns a list of build paths based on the given directory prefix.
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001610func (l *ConfiguredJarList) BuildPaths(ctx PathContext, dir OutputPath) WritablePaths {
1611 paths := make(WritablePaths, l.Len())
1612 for i, jar := range l.jars {
1613 paths[i] = dir.Join(ctx, ModuleStem(jar)+".jar")
1614 }
1615 return paths
1616}
1617
Jingwen Chenc711fec2020-11-22 23:52:50 -05001618// UnmarshalJSON converts JSON configuration from raw bytes into a
1619// ConfiguredJarList structure.
Paul Duffin69d1fb12020-10-23 21:14:20 +01001620func (l *ConfiguredJarList) UnmarshalJSON(b []byte) error {
1621 // Try and unmarshal into a []string each item of which contains a pair
1622 // <apex>:<jar>.
1623 var list []string
1624 err := json.Unmarshal(b, &list)
1625 if err != nil {
1626 // Did not work so return
1627 return err
1628 }
1629
1630 apexes, jars, err := splitListOfPairsIntoPairOfLists(list)
1631 if err != nil {
1632 return err
1633 }
1634 l.apexes = apexes
1635 l.jars = jars
1636 return nil
1637}
1638
Lukacs T. Berki720b3962021-03-17 13:34:30 +01001639func (l *ConfiguredJarList) MarshalJSON() ([]byte, error) {
1640 if len(l.apexes) != len(l.jars) {
1641 return nil, errors.New(fmt.Sprintf("Inconsistent ConfiguredJarList: apexes: %q, jars: %q", l.apexes, l.jars))
1642 }
1643
1644 list := make([]string, 0, len(l.apexes))
1645
1646 for i := 0; i < len(l.apexes); i++ {
1647 list = append(list, l.apexes[i]+":"+l.jars[i])
1648 }
1649
1650 return json.Marshal(list)
1651}
1652
Jingwen Chenc711fec2020-11-22 23:52:50 -05001653// ModuleStem hardcodes the stem of framework-minus-apex to return "framework".
1654//
1655// TODO(b/139391334): hard coded until we find a good way to query the stem of a
1656// module before any other mutators are run.
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001657func ModuleStem(module string) string {
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001658 if module == "framework-minus-apex" {
1659 return "framework"
1660 }
1661 return module
1662}
1663
Jingwen Chenc711fec2020-11-22 23:52:50 -05001664// DevicePaths computes the on-device paths for the list of (apex, jar) pairs,
1665// based on the operating system.
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001666func (l *ConfiguredJarList) DevicePaths(cfg Config, ostype OsType) []string {
1667 paths := make([]string, l.Len())
1668 for i, jar := range l.jars {
1669 apex := l.apexes[i]
1670 name := ModuleStem(jar) + ".jar"
1671
1672 var subdir string
1673 if apex == "platform" {
1674 subdir = "system/framework"
1675 } else if apex == "system_ext" {
1676 subdir = "system_ext/framework"
1677 } else {
1678 subdir = filepath.Join("apex", apex, "javalib")
1679 }
1680
1681 if ostype.Class == Host {
1682 paths[i] = filepath.Join(cfg.Getenv("OUT_DIR"), "host", cfg.PrebuiltOS(), subdir, name)
1683 } else {
1684 paths[i] = filepath.Join("/", subdir, name)
1685 }
1686 }
1687 return paths
1688}
1689
Paul Duffin7d584e92020-10-23 18:26:03 +01001690func (l *ConfiguredJarList) String() string {
1691 var pairs []string
1692 for i := 0; i < l.Len(); i++ {
1693 pairs = append(pairs, l.apexes[i]+":"+l.jars[i])
1694 }
1695 return strings.Join(pairs, ",")
1696}
1697
Paul Duffin01416602020-10-23 21:04:03 +01001698func splitListOfPairsIntoPairOfLists(list []string) ([]string, []string, error) {
1699 // Now we need to populate this list by splitting each item in the slice of
1700 // pairs and appending them to the appropriate list of apexes or jars.
1701 apexes := make([]string, len(list))
1702 jars := make([]string, len(list))
1703
1704 for i, apexjar := range list {
1705 apex, jar, err := splitConfiguredJarPair(apexjar)
1706 if err != nil {
1707 return nil, nil, err
1708 }
1709 apexes[i] = apex
1710 jars[i] = jar
1711 }
1712
1713 return apexes, jars, nil
1714}
1715
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001716// Expected format for apexJarValue = <apex name>:<jar name>
Paul Duffin01416602020-10-23 21:04:03 +01001717func splitConfiguredJarPair(str string) (string, string, error) {
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001718 pair := strings.SplitN(str, ":", 2)
1719 if len(pair) == 2 {
Paul Duffin9c3ac962021-02-03 14:11:27 +00001720 apex := pair[0]
1721 jar := pair[1]
1722 if apex == "" {
1723 return apex, jar, fmt.Errorf("invalid apex '%s' in <apex>:<jar> pair '%s', expected format: <apex>:<jar>", apex, str)
1724 }
1725 return apex, jar, nil
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001726 } else {
Paul Duffin01416602020-10-23 21:04:03 +01001727 return "error-apex", "error-jar", fmt.Errorf("malformed (apex, jar) pair: '%s', expected format: <apex>:<jar>", str)
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001728 }
1729}
1730
Paul Duffin9c3ac962021-02-03 14:11:27 +00001731// CreateTestConfiguredJarList is a function to create ConfiguredJarList for tests.
Paul Duffine10dfa42020-10-23 21:23:44 +01001732func CreateTestConfiguredJarList(list []string) ConfiguredJarList {
Paul Duffin9c3ac962021-02-03 14:11:27 +00001733 // Create the ConfiguredJarList in as similar way as it is created at runtime by marshalling to
1734 // a json list of strings and then unmarshalling into a ConfiguredJarList instance.
1735 b, err := json.Marshal(list)
Paul Duffin01416602020-10-23 21:04:03 +01001736 if err != nil {
Paul Duffine10dfa42020-10-23 21:23:44 +01001737 panic(err)
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001738 }
1739
Paul Duffin9c3ac962021-02-03 14:11:27 +00001740 var jarList ConfiguredJarList
1741 err = json.Unmarshal(b, &jarList)
1742 if err != nil {
1743 panic(err)
1744 }
1745
1746 return jarList
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001747}
1748
Jingwen Chenc711fec2020-11-22 23:52:50 -05001749// EmptyConfiguredJarList returns an empty jar list.
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001750func EmptyConfiguredJarList() ConfiguredJarList {
1751 return ConfiguredJarList{}
1752}
1753
1754var earlyBootJarsKey = NewOnceKey("earlyBootJars")
1755
1756func (c *config) BootJars() []string {
1757 return c.Once(earlyBootJarsKey, func() interface{} {
Paul Duffin69d1fb12020-10-23 21:14:20 +01001758 list := c.productVariables.BootJars.CopyOfJars()
Jingwen Chenc711fec2020-11-22 23:52:50 -05001759 return append(list, c.productVariables.UpdatableBootJars.CopyOfJars()...)
Ulya Trafimovich249386a2020-07-01 14:31:13 +01001760 }).([]string)
1761}
Paul Duffin9a89a2a2020-10-28 19:20:06 +00001762
1763func (c *config) NonUpdatableBootJars() ConfiguredJarList {
1764 return c.productVariables.BootJars
1765}
1766
1767func (c *config) UpdatableBootJars() ConfiguredJarList {
1768 return c.productVariables.UpdatableBootJars
1769}