blob: 4d7e8df5520c77b06809f26d3d499869020712d4 [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
17import (
Colin Cross3f40fa42015-01-30 17:27:36 -080018 "encoding/json"
19 "fmt"
20 "os"
Colin Cross35cec122015-04-02 14:37:16 -070021 "path/filepath"
Colin Cross3f40fa42015-01-30 17:27:36 -080022 "runtime"
Dan Willemsen5951c8a2016-07-19 19:08:14 -070023 "strconv"
Dan Willemsen34cc69e2015-09-23 15:26:20 -070024 "strings"
Colin Crossc1e86a32015-04-15 12:33:28 -070025 "sync"
Colin Cross6ff51382015-12-17 16:39:19 -080026
27 "github.com/google/blueprint/proptools"
Colin Cross3f40fa42015-01-30 17:27:36 -080028)
29
Colin Cross6ff51382015-12-17 16:39:19 -080030var Bool = proptools.Bool
31
Colin Cross3f40fa42015-01-30 17:27:36 -080032// The configuration file name
Dan Willemsen87b17d12015-07-14 00:39:06 -070033const configFileName = "soong.config"
34const productVariablesFileName = "soong.variables"
Colin Cross3f40fa42015-01-30 17:27:36 -080035
36// A FileConfigurableOptions contains options which can be configured by the
37// config file. These will be included in the config struct.
38type FileConfigurableOptions struct {
Dan Willemsen322acaf2016-01-12 23:07:05 -080039 Mega_device *bool `json:",omitempty"`
Dan Albert4098deb2016-10-19 14:04:41 -070040 Ndk_abis *bool `json:",omitempty"`
Colin Cross3f40fa42015-01-30 17:27:36 -080041}
42
Colin Cross27385972015-09-18 10:57:10 -070043func (f *FileConfigurableOptions) SetDefaultConfig() {
44 *f = FileConfigurableOptions{}
Colin Cross3f40fa42015-01-30 17:27:36 -080045}
46
Colin Cross9272ade2016-08-17 15:24:12 -070047// A Config object represents the entire build configuration for Android.
Colin Crossc3c0a492015-04-10 15:43:55 -070048type Config struct {
49 *config
50}
51
Colin Cross9272ade2016-08-17 15:24:12 -070052// A DeviceConfig object represents the configuration for a particular device being built. For
53// now there will only be one of these, but in the future there may be multiple devices being
54// built
55type DeviceConfig struct {
56 *deviceConfig
57}
58
Colin Cross1332b002015-04-07 17:11:30 -070059type config struct {
Colin Cross3f40fa42015-01-30 17:27:36 -080060 FileConfigurableOptions
Colin Cross485e5722015-08-27 13:28:01 -070061 ProductVariables productVariables
Colin Cross3f40fa42015-01-30 17:27:36 -080062
Dan Willemsen87b17d12015-07-14 00:39:06 -070063 ConfigFileName string
64 ProductVariablesFileName string
65
Colin Crossa1ad8d12016-06-01 17:09:44 -070066 Targets map[OsClass][]Target
67 BuildOsVariant string
Dan Willemsen218f6562015-07-08 18:13:11 -070068
Colin Cross9272ade2016-08-17 15:24:12 -070069 deviceConfig *deviceConfig
70
Dan Willemsen87b17d12015-07-14 00:39:06 -070071 srcDir string // the path of the root source directory
72 buildDir string // the path of the build output directory
Colin Crossc1e86a32015-04-15 12:33:28 -070073
Dan Willemsene7680ba2015-09-11 17:06:19 -070074 envLock sync.Mutex
75 envDeps map[string]string
76 envFrozen bool
Dan Willemsen5ba07e82015-12-11 13:51:06 -080077
78 inMake bool
Colin Cross1e7d3702016-08-24 15:25:47 -070079
Colin Cross9272ade2016-08-17 15:24:12 -070080 OncePer
81}
82
83type deviceConfig struct {
84 config *config
85 targets []Arch
86 OncePer
Colin Cross3f40fa42015-01-30 17:27:36 -080087}
88
Colin Cross485e5722015-08-27 13:28:01 -070089type jsonConfigurable interface {
Colin Cross27385972015-09-18 10:57:10 -070090 SetDefaultConfig()
Colin Cross485e5722015-08-27 13:28:01 -070091}
Colin Cross3f40fa42015-01-30 17:27:36 -080092
Colin Cross485e5722015-08-27 13:28:01 -070093func loadConfig(config *config) error {
Dan Willemsen87b17d12015-07-14 00:39:06 -070094 err := loadFromConfigFile(&config.FileConfigurableOptions, config.ConfigFileName)
Colin Cross485e5722015-08-27 13:28:01 -070095 if err != nil {
96 return err
97 }
98
Dan Willemsen87b17d12015-07-14 00:39:06 -070099 return loadFromConfigFile(&config.ProductVariables, config.ProductVariablesFileName)
Colin Cross485e5722015-08-27 13:28:01 -0700100}
101
102// loads configuration options from a JSON file in the cwd.
103func loadFromConfigFile(configurable jsonConfigurable, filename string) error {
Colin Cross3f40fa42015-01-30 17:27:36 -0800104 // Try to open the file
Colin Cross485e5722015-08-27 13:28:01 -0700105 configFileReader, err := os.Open(filename)
Colin Cross3f40fa42015-01-30 17:27:36 -0800106 defer configFileReader.Close()
107 if os.IsNotExist(err) {
108 // Need to create a file, so that blueprint & ninja don't get in
109 // a dependency tracking loop.
110 // Make a file-configurable-options with defaults, write it out using
111 // a json writer.
Colin Cross27385972015-09-18 10:57:10 -0700112 configurable.SetDefaultConfig()
113 err = saveToConfigFile(configurable, filename)
Colin Cross3f40fa42015-01-30 17:27:36 -0800114 if err != nil {
115 return err
116 }
117 } else {
118 // Make a decoder for it
119 jsonDecoder := json.NewDecoder(configFileReader)
Colin Cross485e5722015-08-27 13:28:01 -0700120 err = jsonDecoder.Decode(configurable)
Colin Cross3f40fa42015-01-30 17:27:36 -0800121 if err != nil {
Colin Cross485e5722015-08-27 13:28:01 -0700122 return fmt.Errorf("config file: %s did not parse correctly: "+err.Error(), filename)
Colin Cross3f40fa42015-01-30 17:27:36 -0800123 }
124 }
125
Colin Cross3f40fa42015-01-30 17:27:36 -0800126 // No error
127 return nil
128}
129
Colin Cross485e5722015-08-27 13:28:01 -0700130func saveToConfigFile(config jsonConfigurable, filename string) error {
Colin Cross3f40fa42015-01-30 17:27:36 -0800131 data, err := json.MarshalIndent(&config, "", " ")
132 if err != nil {
133 return fmt.Errorf("cannot marshal config data: %s", err.Error())
134 }
135
Colin Cross485e5722015-08-27 13:28:01 -0700136 configFileWriter, err := os.Create(filename)
Colin Cross3f40fa42015-01-30 17:27:36 -0800137 if err != nil {
Colin Cross485e5722015-08-27 13:28:01 -0700138 return fmt.Errorf("cannot create empty config file %s: %s\n", filename, err.Error())
Colin Cross3f40fa42015-01-30 17:27:36 -0800139 }
140 defer configFileWriter.Close()
141
142 _, err = configFileWriter.Write(data)
143 if err != nil {
Colin Cross485e5722015-08-27 13:28:01 -0700144 return fmt.Errorf("default config file: %s could not be written: %s", filename, err.Error())
145 }
146
147 _, err = configFileWriter.WriteString("\n")
148 if err != nil {
149 return fmt.Errorf("default config file: %s could not be written: %s", filename, err.Error())
Colin Cross3f40fa42015-01-30 17:27:36 -0800150 }
151
152 return nil
153}
154
Colin Crossce75d2c2016-10-06 16:12:58 -0700155// TestConfig returns a Config object suitable for using for tests
Colin Cross0d614dd2016-10-14 15:38:43 -0700156func TestConfig(buildDir string) Config {
157 return Config{&config{
158 buildDir: buildDir,
159 }}
Colin Crossce75d2c2016-10-06 16:12:58 -0700160}
161
Colin Cross3f40fa42015-01-30 17:27:36 -0800162// New creates a new Config object. The srcDir argument specifies the path to
163// the root source directory. It also loads the config file, if found.
Dan Willemsen87b17d12015-07-14 00:39:06 -0700164func NewConfig(srcDir, buildDir string) (Config, error) {
Colin Cross3f40fa42015-01-30 17:27:36 -0800165 // Make a config with default options
Colin Cross9272ade2016-08-17 15:24:12 -0700166 config := &config{
167 ConfigFileName: filepath.Join(buildDir, configFileName),
168 ProductVariablesFileName: filepath.Join(buildDir, productVariablesFileName),
Dan Willemsen87b17d12015-07-14 00:39:06 -0700169
Colin Cross9272ade2016-08-17 15:24:12 -0700170 srcDir: srcDir,
171 buildDir: buildDir,
172 envDeps: make(map[string]string),
173
174 deviceConfig: &deviceConfig{},
Colin Cross68f55102015-03-25 14:43:57 -0700175 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800176
Colin Cross9272ade2016-08-17 15:24:12 -0700177 deviceConfig := &deviceConfig{
178 config: config,
179 }
180
181 config.deviceConfig = deviceConfig
182
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700183 // Sanity check the build and source directories. This won't catch strange
184 // configurations with symlinks, but at least checks the obvious cases.
185 absBuildDir, err := filepath.Abs(buildDir)
186 if err != nil {
187 return Config{}, err
188 }
189
190 absSrcDir, err := filepath.Abs(srcDir)
191 if err != nil {
192 return Config{}, err
193 }
194
195 if strings.HasPrefix(absSrcDir, absBuildDir) {
196 return Config{}, fmt.Errorf("Build dir must not contain source directory")
197 }
198
Colin Cross3f40fa42015-01-30 17:27:36 -0800199 // Load any configurable options from the configuration file
Colin Cross9272ade2016-08-17 15:24:12 -0700200 err = loadConfig(config)
Colin Cross3f40fa42015-01-30 17:27:36 -0800201 if err != nil {
Colin Crossc3c0a492015-04-10 15:43:55 -0700202 return Config{}, err
Colin Cross3f40fa42015-01-30 17:27:36 -0800203 }
204
Dan Willemsen5ba07e82015-12-11 13:51:06 -0800205 inMakeFile := filepath.Join(buildDir, ".soong.in_make")
206 if _, err := os.Stat(inMakeFile); err == nil {
207 config.inMake = true
208 }
209
Colin Crossa1ad8d12016-06-01 17:09:44 -0700210 targets, err := decodeTargetProductVariables(config)
Dan Willemsen218f6562015-07-08 18:13:11 -0700211 if err != nil {
212 return Config{}, err
213 }
214
Dan Albert4098deb2016-10-19 14:04:41 -0700215 var archConfig []archConfig
Dan Willemsen322acaf2016-01-12 23:07:05 -0800216 if Bool(config.Mega_device) {
Dan Albert4098deb2016-10-19 14:04:41 -0700217 archConfig = getMegaDeviceConfig()
218 } else if Bool(config.Ndk_abis) {
219 archConfig = getNdkAbisConfig()
220 }
221
222 if archConfig != nil {
223 deviceTargets, err := decodeArchSettings(archConfig)
Dan Willemsen322acaf2016-01-12 23:07:05 -0800224 if err != nil {
225 return Config{}, err
226 }
Colin Crossa1ad8d12016-06-01 17:09:44 -0700227 targets[Device] = deviceTargets
Dan Willemsen322acaf2016-01-12 23:07:05 -0800228 }
229
Colin Crossa1ad8d12016-06-01 17:09:44 -0700230 config.Targets = targets
231 config.BuildOsVariant = targets[Host][0].String()
Dan Willemsen218f6562015-07-08 18:13:11 -0700232
Colin Cross9272ade2016-08-17 15:24:12 -0700233 return Config{config}, nil
Colin Cross3f40fa42015-01-30 17:27:36 -0800234}
235
Dan Willemsen218f6562015-07-08 18:13:11 -0700236func (c *config) RemoveAbandonedFiles() bool {
237 return false
238}
239
Dan Willemsenc2aa4a92016-05-26 15:13:03 -0700240func (c *config) BlueprintToolLocation() string {
241 return filepath.Join(c.buildDir, "host", c.PrebuiltOS(), "bin")
242}
243
Colin Cross3f40fa42015-01-30 17:27:36 -0800244// PrebuiltOS returns the name of the host OS used in prebuilts directories
Colin Cross1332b002015-04-07 17:11:30 -0700245func (c *config) PrebuiltOS() string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800246 switch runtime.GOOS {
247 case "linux":
248 return "linux-x86"
249 case "darwin":
250 return "darwin-x86"
251 default:
252 panic("Unknown GOOS")
253 }
254}
255
256// GoRoot returns the path to the root directory of the Go toolchain.
Colin Cross1332b002015-04-07 17:11:30 -0700257func (c *config) GoRoot() string {
Colin Cross3f40fa42015-01-30 17:27:36 -0800258 return fmt.Sprintf("%s/prebuilts/go/%s", c.srcDir, c.PrebuiltOS())
259}
260
Colin Cross1332b002015-04-07 17:11:30 -0700261func (c *config) CpPreserveSymlinksFlags() string {
Colin Cross485e5722015-08-27 13:28:01 -0700262 switch runtime.GOOS {
Colin Cross3f40fa42015-01-30 17:27:36 -0800263 case "darwin":
264 return "-R"
265 case "linux":
266 return "-d"
267 default:
268 return ""
269 }
270}
Colin Cross68f55102015-03-25 14:43:57 -0700271
Colin Cross1332b002015-04-07 17:11:30 -0700272func (c *config) Getenv(key string) string {
Colin Cross68f55102015-03-25 14:43:57 -0700273 var val string
274 var exists bool
Colin Crossc1e86a32015-04-15 12:33:28 -0700275 c.envLock.Lock()
Colin Cross68f55102015-03-25 14:43:57 -0700276 if val, exists = c.envDeps[key]; !exists {
Dan Willemsene7680ba2015-09-11 17:06:19 -0700277 if c.envFrozen {
278 panic("Cannot access new environment variables after envdeps are frozen")
279 }
Colin Cross68f55102015-03-25 14:43:57 -0700280 val = os.Getenv(key)
281 c.envDeps[key] = val
282 }
Colin Crossc1e86a32015-04-15 12:33:28 -0700283 c.envLock.Unlock()
Colin Cross68f55102015-03-25 14:43:57 -0700284 return val
285}
286
Colin Cross1332b002015-04-07 17:11:30 -0700287func (c *config) EnvDeps() map[string]string {
Dan Willemsene7680ba2015-09-11 17:06:19 -0700288 c.envLock.Lock()
289 c.envFrozen = true
290 c.envLock.Unlock()
Colin Cross68f55102015-03-25 14:43:57 -0700291 return c.envDeps
292}
Colin Cross35cec122015-04-02 14:37:16 -0700293
Dan Willemsen5ba07e82015-12-11 13:51:06 -0800294func (c *config) EmbeddedInMake() bool {
295 return c.inMake
296}
297
Colin Cross35cec122015-04-02 14:37:16 -0700298// DeviceName returns the name of the current device target
299// TODO: take an AndroidModuleContext to select the device name for multi-device builds
Colin Cross1332b002015-04-07 17:11:30 -0700300func (c *config) DeviceName() string {
Dan Willemsen1d31ec32015-09-22 16:56:09 -0700301 return *c.ProductVariables.DeviceName
Colin Cross35cec122015-04-02 14:37:16 -0700302}
303
Dan Willemsendd0e2c32015-10-20 14:29:35 -0700304func (c *config) DeviceUsesClang() bool {
305 if c.ProductVariables.DeviceUsesClang != nil {
306 return *c.ProductVariables.DeviceUsesClang
307 }
Dan Willemsen0084d782016-03-01 14:54:24 -0800308 return true
Dan Willemsendd0e2c32015-10-20 14:29:35 -0700309}
310
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700311func (c *config) ResourceOverlays() []SourcePath {
Colin Cross30e076a2015-04-13 13:58:27 -0700312 return nil
313}
314
315func (c *config) PlatformVersion() string {
316 return "M"
317}
318
319func (c *config) PlatformSdkVersion() string {
Dan Willemsen5951c8a2016-07-19 19:08:14 -0700320 return strconv.Itoa(*c.ProductVariables.Platform_sdk_version)
Colin Cross30e076a2015-04-13 13:58:27 -0700321}
322
323func (c *config) BuildNumber() string {
324 return "000000"
325}
326
327func (c *config) ProductAaptConfig() []string {
328 return []string{"normal", "large", "xlarge", "hdpi", "xhdpi", "xxhdpi"}
329}
330
331func (c *config) ProductAaptPreferredConfig() string {
332 return "xhdpi"
333}
334
335func (c *config) ProductAaptCharacteristics() string {
336 return "nosdcard"
337}
338
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700339func (c *config) DefaultAppCertificateDir(ctx PathContext) SourcePath {
340 return PathForSource(ctx, "build/target/product/security")
Colin Cross30e076a2015-04-13 13:58:27 -0700341}
342
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700343func (c *config) DefaultAppCertificate(ctx PathContext) SourcePath {
344 return c.DefaultAppCertificateDir(ctx).Join(ctx, "testkey")
Colin Cross30e076a2015-04-13 13:58:27 -0700345}
Colin Cross6ff51382015-12-17 16:39:19 -0800346
347func (c *config) AllowMissingDependencies() bool {
Dan Willemsenb5038162016-03-16 12:35:33 -0700348 return Bool(c.ProductVariables.Allow_missing_dependencies)
Colin Cross6ff51382015-12-17 16:39:19 -0800349}
Dan Willemsen322acaf2016-01-12 23:07:05 -0800350
Colin Cross1e7d3702016-08-24 15:25:47 -0700351func (c *config) DevicePrefer32BitExecutables() bool {
352 return Bool(c.ProductVariables.DevicePrefer32BitExecutables)
353}
354
Dan Willemsen7f730fd2016-01-14 11:22:23 -0800355func (c *config) SkipDeviceInstall() bool {
Dan Willemsen322acaf2016-01-12 23:07:05 -0800356 return c.EmbeddedInMake() || Bool(c.Mega_device)
357}
Colin Cross16b23492016-01-06 14:41:07 -0800358
359func (c *config) SanitizeHost() []string {
360 if c.ProductVariables.SanitizeHost == nil {
361 return nil
362 }
Colin Crosscc85e682016-07-06 14:24:16 -0700363 return append([]string(nil), *c.ProductVariables.SanitizeHost...)
Colin Cross16b23492016-01-06 14:41:07 -0800364}
365
366func (c *config) SanitizeDevice() []string {
367 if c.ProductVariables.SanitizeDevice == nil {
368 return nil
369 }
Colin Crosscc85e682016-07-06 14:24:16 -0700370 return append([]string(nil), *c.ProductVariables.SanitizeDevice...)
Colin Cross16b23492016-01-06 14:41:07 -0800371}
Colin Crossa1ad8d12016-06-01 17:09:44 -0700372
373func (c *config) Android64() bool {
374 for _, t := range c.Targets[Device] {
375 if t.Arch.ArchType.Multilib == "lib64" {
376 return true
377 }
378 }
379
380 return false
381}
Colin Cross9272ade2016-08-17 15:24:12 -0700382
Colin Cross9d45bb72016-08-29 16:14:13 -0700383func (c *config) UseGoma() bool {
384 return Bool(c.ProductVariables.UseGoma)
385}
386
Colin Cross0f4e0d62016-07-27 10:56:55 -0700387func (c *config) LibartImgHostBaseAddress() string {
388 return "0x60000000"
389}
390
391func (c *config) LibartImgDeviceBaseAddress() string {
392 switch c.Targets[Device][0].Arch.ArchType {
393 default:
394 return "0x70000000"
395 case Mips, Mips64:
396 return "0x30000000"
397 }
398}
399
Colin Cross9272ade2016-08-17 15:24:12 -0700400func (c *deviceConfig) Arches() []Arch {
401 var arches []Arch
402 for _, target := range c.config.Targets[Device] {
403 arches = append(arches, target.Arch)
404 }
405 return arches
406}