blob: e692fafb22b2f83b9747a5bf32a3f5184b902cf9 [file] [log] [blame]
Colin Cross7f64b6d2015-07-09 13:57:48 -07001// 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 Cross7f64b6d2015-07-09 13:57:48 -070016
17import (
18 "fmt"
19 "reflect"
Dan Willemsen490fd492015-11-24 17:53:15 -080020 "runtime"
Colin Cross7f64b6d2015-07-09 13:57:48 -070021 "strings"
22
Colin Cross7f64b6d2015-07-09 13:57:48 -070023 "github.com/google/blueprint/proptools"
24)
25
26func init() {
Colin Cross1e676be2016-10-12 14:38:15 -070027 PreDepsMutators(func(ctx RegisterMutatorsContext) {
28 ctx.BottomUp("variable", variableMutator).Parallel()
29 })
Colin Cross7f64b6d2015-07-09 13:57:48 -070030}
31
32type variableProperties struct {
33 Product_variables struct {
Dan Willemsen47cf66b2015-09-16 16:48:54 -070034 Platform_sdk_version struct {
35 Asflags []string
36 }
Colin Cross83163192015-12-01 16:31:16 -080037
38 // unbundled_build is a catch-all property to annotate modules that don't build in one or
39 // more unbundled branches, usually due to dependencies missing from the manifest.
40 Unbundled_build struct {
41 Enabled *bool `android:"arch_variant"`
42 } `android:"arch_variant"`
Dan Willemsen496e3952016-01-05 14:27:55 -080043
44 Brillo struct {
Dan Willemsen7b872832016-10-03 23:36:47 -070045 Cflags []string
Dan Willemsen496e3952016-01-05 14:27:55 -080046 Version_script *string `android:"arch_variant"`
47 } `android:"arch_variant"`
Dan Willemsen97cae012016-03-01 14:08:42 -080048
49 Malloc_not_svelte struct {
50 Cflags []string
51 }
Evgenii Stepanov8391efa2016-05-12 18:04:18 -070052
53 Safestack struct {
54 Cflags []string `android:"arch_variant"`
55 } `android:"arch_variant"`
Dan Willemsena6f7d152016-07-12 14:57:40 -070056
Dan Willemsen1be35382016-07-25 17:42:18 -070057 Binder32bit struct {
58 Cflags []string
59 }
Dan Willemsenfcebcd52016-09-22 14:49:10 -070060
Colin Cross0e409a22017-05-02 10:58:50 -070061 Device_uses_hwc2 struct {
62 Cflags []string
63 }
64
Colin Cross95436422017-05-04 13:57:05 -070065 Override_rs_driver struct {
66 Cflags []string
67 }
68
Colin Cross6bc59ef2016-12-08 09:46:35 -080069 // debuggable is true for eng and userdebug builds, and can be used to turn on additional
70 // debugging features that don't significantly impact runtime behavior. userdebug builds
71 // are used for dogfooding and performance testing, and should be as similar to user builds
72 // as possible.
Dan Willemsenfcebcd52016-09-22 14:49:10 -070073 Debuggable struct {
Dan Willemsen7b872832016-10-03 23:36:47 -070074 Cflags []string
75 Cppflags []string
Dan Willemsenfcebcd52016-09-22 14:49:10 -070076 }
Colin Cross6bc59ef2016-12-08 09:46:35 -080077
78 // eng is true for -eng builds, and can be used to turn on additionaly heavyweight debugging
79 // features.
80 Eng struct {
81 Cflags []string
82 Cppflags []string
83 }
Colin Crossb2123aa2017-05-05 13:37:11 -070084
85 Pdk struct {
86 Enabled *bool
87 }
Colin Cross83163192015-12-01 16:31:16 -080088 } `android:"arch_variant"`
Colin Cross7f64b6d2015-07-09 13:57:48 -070089}
90
91var zeroProductVariables variableProperties
92
Colin Cross485e5722015-08-27 13:28:01 -070093type productVariables struct {
Dan Willemsen174978c2016-05-11 00:27:49 -070094 // Suffix to add to generated Makefiles
95 Make_suffix *string `json:",omitempty"`
96
Dan Albert30c9d6e2017-03-28 14:54:55 -070097 Platform_sdk_version *int `json:",omitempty"`
98 Platform_version_all_codenames []string `json:",omitempty"`
Colin Cross4225f652015-09-17 14:33:42 -070099
Dan Willemsen1d31ec32015-09-22 16:56:09 -0700100 DeviceName *string `json:",omitempty"`
Colin Cross4225f652015-09-17 14:33:42 -0700101 DeviceArch *string `json:",omitempty"`
102 DeviceArchVariant *string `json:",omitempty"`
103 DeviceCpuVariant *string `json:",omitempty"`
104 DeviceAbi *[]string `json:",omitempty"`
Dan Willemsendd0e2c32015-10-20 14:29:35 -0700105 DeviceUsesClang *bool `json:",omitempty"`
Dan Willemsend2ede872016-11-18 14:54:24 -0800106 DeviceVndkVersion *string `json:",omitempty"`
Colin Cross4225f652015-09-17 14:33:42 -0700107
108 DeviceSecondaryArch *string `json:",omitempty"`
109 DeviceSecondaryArchVariant *string `json:",omitempty"`
110 DeviceSecondaryCpuVariant *string `json:",omitempty"`
111 DeviceSecondaryAbi *[]string `json:",omitempty"`
112
113 HostArch *string `json:",omitempty"`
114 HostSecondaryArch *string `json:",omitempty"`
Dan Willemsen490fd492015-11-24 17:53:15 -0800115
116 CrossHost *string `json:",omitempty"`
117 CrossHostArch *string `json:",omitempty"`
118 CrossHostSecondaryArch *string `json:",omitempty"`
Colin Cross83163192015-12-01 16:31:16 -0800119
Dan Willemsenb5038162016-03-16 12:35:33 -0700120 Allow_missing_dependencies *bool `json:",omitempty"`
121 Unbundled_build *bool `json:",omitempty"`
122 Brillo *bool `json:",omitempty"`
123 Malloc_not_svelte *bool `json:",omitempty"`
Evgenii Stepanov8391efa2016-05-12 18:04:18 -0700124 Safestack *bool `json:",omitempty"`
Dan Willemsen36cff8b2016-05-17 16:35:02 -0700125 HostStaticBinaries *bool `json:",omitempty"`
Dan Willemsen1be35382016-07-25 17:42:18 -0700126 Binder32bit *bool `json:",omitempty"`
Colin Cross9d45bb72016-08-29 16:14:13 -0700127 UseGoma *bool `json:",omitempty"`
Dan Willemsenfcebcd52016-09-22 14:49:10 -0700128 Debuggable *bool `json:",omitempty"`
Colin Cross6bc59ef2016-12-08 09:46:35 -0800129 Eng *bool `json:",omitempty"`
Vishwath Mohan1b017a72017-01-19 13:54:55 -0800130 EnableCFI *bool `json:",omitempty"`
Colin Cross0e409a22017-05-02 10:58:50 -0700131 Device_uses_hwc2 *bool `json:",omitempty"`
Colin Cross16b23492016-01-06 14:41:07 -0800132
Dan Willemsen4353bc42016-12-05 17:16:02 -0800133 VendorPath *string `json:",omitempty"`
134
Dan Willemsena03cf6d2016-09-26 15:45:04 -0700135 ClangTidy *bool `json:",omitempty"`
136 TidyChecks *string `json:",omitempty"`
137
Ryan Campbell469a18a2017-02-27 09:01:54 -0800138 NativeCoverage *bool `json:",omitempty"`
139 CoveragePaths *[]string `json:",omitempty"`
140 CoverageExcludePaths *[]string `json:",omitempty"`
Dan Willemsen581341d2017-02-09 16:16:31 -0800141
Colin Cross1e7d3702016-08-24 15:25:47 -0700142 DevicePrefer32BitExecutables *bool `json:",omitempty"`
143 HostPrefer32BitExecutables *bool `json:",omitempty"`
144
Colin Cross23ae82a2016-11-02 14:34:39 -0700145 SanitizeHost []string `json:",omitempty"`
146 SanitizeDevice []string `json:",omitempty"`
147 SanitizeDeviceArch []string `json:",omitempty"`
Hiroshi Yamauchie2a10632016-12-19 13:44:41 -0800148
149 ArtUseReadBarrier *bool `json:",omitempty"`
Jack He8cc71432016-12-08 15:45:07 -0800150
151 BtConfigIncludeDir *string `json:",omitempty"`
Colin Cross95436422017-05-04 13:57:05 -0700152
153 Override_rs_driver *string `json:",omitempty"`
Colin Crossa6bc19e2015-09-16 13:53:42 -0700154}
155
156func boolPtr(v bool) *bool {
157 return &v
Colin Cross485e5722015-08-27 13:28:01 -0700158}
159
Dan Willemsen47cf66b2015-09-16 16:48:54 -0700160func intPtr(v int) *int {
161 return &v
162}
163
Colin Cross4225f652015-09-17 14:33:42 -0700164func stringPtr(v string) *string {
165 return &v
166}
167
Colin Cross27385972015-09-18 10:57:10 -0700168func (v *productVariables) SetDefaultConfig() {
169 *v = productVariables{
Dan Albert22c0b412016-11-10 11:08:00 -0800170 Platform_sdk_version: intPtr(24),
Colin Cross4225f652015-09-17 14:33:42 -0700171 HostArch: stringPtr("x86_64"),
172 HostSecondaryArch: stringPtr("x86"),
Dan Willemsen1d31ec32015-09-22 16:56:09 -0700173 DeviceName: stringPtr("flounder"),
Colin Cross4225f652015-09-17 14:33:42 -0700174 DeviceArch: stringPtr("arm64"),
Dan Willemsena91d1272016-03-29 22:06:05 -0700175 DeviceArchVariant: stringPtr("armv8-a"),
Colin Cross4225f652015-09-17 14:33:42 -0700176 DeviceCpuVariant: stringPtr("denver64"),
177 DeviceAbi: &[]string{"arm64-v8a"},
Dan Willemsendd0e2c32015-10-20 14:29:35 -0700178 DeviceUsesClang: boolPtr(true),
Colin Cross4225f652015-09-17 14:33:42 -0700179 DeviceSecondaryArch: stringPtr("arm"),
180 DeviceSecondaryArchVariant: stringPtr("armv7-a-neon"),
Dan Willemsen1d31ec32015-09-22 16:56:09 -0700181 DeviceSecondaryCpuVariant: stringPtr("denver"),
Colin Cross4225f652015-09-17 14:33:42 -0700182 DeviceSecondaryAbi: &[]string{"armeabi-v7a"},
Dan Willemsen97cae012016-03-01 14:08:42 -0800183 Malloc_not_svelte: boolPtr(false),
Evgenii Stepanov8391efa2016-05-12 18:04:18 -0700184 Safestack: boolPtr(false),
Colin Cross485e5722015-08-27 13:28:01 -0700185 }
Dan Willemsen490fd492015-11-24 17:53:15 -0800186
187 if runtime.GOOS == "linux" {
188 v.CrossHost = stringPtr("windows")
189 v.CrossHostArch = stringPtr("x86")
Dan Willemsen07cd0512016-02-03 23:16:33 -0800190 v.CrossHostSecondaryArch = stringPtr("x86_64")
Dan Willemsen490fd492015-11-24 17:53:15 -0800191 }
Colin Cross7f64b6d2015-07-09 13:57:48 -0700192}
193
Colin Cross635c3b02016-05-18 15:37:25 -0700194func variableMutator(mctx BottomUpMutatorContext) {
195 var module Module
Colin Cross7f64b6d2015-07-09 13:57:48 -0700196 var ok bool
Colin Cross635c3b02016-05-18 15:37:25 -0700197 if module, ok = mctx.Module().(Module); !ok {
Colin Cross7f64b6d2015-07-09 13:57:48 -0700198 return
199 }
200
201 // TODO: depend on config variable, create variants, propagate variants up tree
202 a := module.base()
Colin Cross06a931b2015-10-28 17:23:31 -0700203 variableValues := reflect.ValueOf(&a.variableProperties.Product_variables).Elem()
Colin Cross7f64b6d2015-07-09 13:57:48 -0700204 zeroValues := reflect.ValueOf(zeroProductVariables.Product_variables)
205
206 for i := 0; i < variableValues.NumField(); i++ {
207 variableValue := variableValues.Field(i)
208 zeroValue := zeroValues.Field(i)
Colin Cross485e5722015-08-27 13:28:01 -0700209 name := variableValues.Type().Field(i).Name
210 property := "product_variables." + proptools.PropertyNameForField(name)
Colin Cross7f64b6d2015-07-09 13:57:48 -0700211
Colin Crossa6bc19e2015-09-16 13:53:42 -0700212 // Check that the variable was set for the product
Colin Cross485e5722015-08-27 13:28:01 -0700213 val := reflect.ValueOf(mctx.Config().(Config).ProductVariables).FieldByName(name)
Colin Crossa6bc19e2015-09-16 13:53:42 -0700214 if !val.IsValid() || val.Kind() != reflect.Ptr || val.IsNil() {
215 continue
Colin Cross7f64b6d2015-07-09 13:57:48 -0700216 }
Colin Crossa6bc19e2015-09-16 13:53:42 -0700217
218 val = val.Elem()
219
220 // For bools, check that the value is true
221 if val.Kind() == reflect.Bool && val.Bool() == false {
222 continue
223 }
224
225 // Check if any properties were set for the module
226 if reflect.DeepEqual(variableValue.Interface(), zeroValue.Interface()) {
227 continue
228 }
229
230 a.setVariableProperties(mctx, property, variableValue, val.Interface())
Colin Cross7f64b6d2015-07-09 13:57:48 -0700231 }
232}
233
Colin Cross635c3b02016-05-18 15:37:25 -0700234func (a *ModuleBase) setVariableProperties(ctx BottomUpMutatorContext,
Colin Cross7f64b6d2015-07-09 13:57:48 -0700235 prefix string, productVariablePropertyValue reflect.Value, variableValue interface{}) {
236
Colin Cross7e0eaf12017-05-05 16:16:24 -0700237 printfIntoProperties(ctx, prefix, productVariablePropertyValue, variableValue)
Colin Cross7f64b6d2015-07-09 13:57:48 -0700238
Colin Cross06a931b2015-10-28 17:23:31 -0700239 err := proptools.AppendMatchingProperties(a.generalProperties,
240 productVariablePropertyValue.Addr().Interface(), nil)
241 if err != nil {
242 if propertyErr, ok := err.(*proptools.ExtendPropertyError); ok {
243 ctx.PropertyErrorf(propertyErr.Property, "%s", propertyErr.Err.Error())
244 } else {
245 panic(err)
246 }
247 }
Colin Cross7f64b6d2015-07-09 13:57:48 -0700248}
249
Colin Cross7e0eaf12017-05-05 16:16:24 -0700250func printfIntoPropertiesError(ctx BottomUpMutatorContext, prefix string,
251 productVariablePropertyValue reflect.Value, i int, err error) {
252
253 field := productVariablePropertyValue.Type().Field(i).Name
254 property := prefix + "." + proptools.PropertyNameForField(field)
255 ctx.PropertyErrorf(property, "%s", err)
256}
257
258func printfIntoProperties(ctx BottomUpMutatorContext, prefix string,
259 productVariablePropertyValue reflect.Value, variableValue interface{}) {
260
Colin Cross7f64b6d2015-07-09 13:57:48 -0700261 for i := 0; i < productVariablePropertyValue.NumField(); i++ {
262 propertyValue := productVariablePropertyValue.Field(i)
Colin Crossdd0dbe62015-12-02 15:24:38 -0800263 kind := propertyValue.Kind()
264 if kind == reflect.Ptr {
265 if propertyValue.IsNil() {
266 continue
267 }
268 propertyValue = propertyValue.Elem()
269 }
Colin Cross7f64b6d2015-07-09 13:57:48 -0700270 switch propertyValue.Kind() {
271 case reflect.String:
Colin Cross7e0eaf12017-05-05 16:16:24 -0700272 err := printfIntoProperty(propertyValue, variableValue)
273 if err != nil {
274 printfIntoPropertiesError(ctx, prefix, productVariablePropertyValue, i, err)
275 }
Colin Cross7f64b6d2015-07-09 13:57:48 -0700276 case reflect.Slice:
277 for j := 0; j < propertyValue.Len(); j++ {
Colin Cross7e0eaf12017-05-05 16:16:24 -0700278 err := printfIntoProperty(propertyValue.Index(j), variableValue)
279 if err != nil {
280 printfIntoPropertiesError(ctx, prefix, productVariablePropertyValue, i, err)
281 }
Colin Cross7f64b6d2015-07-09 13:57:48 -0700282 }
Colin Crossdd0dbe62015-12-02 15:24:38 -0800283 case reflect.Bool:
284 // Nothing
Colin Cross7f64b6d2015-07-09 13:57:48 -0700285 case reflect.Struct:
Colin Cross7e0eaf12017-05-05 16:16:24 -0700286 printfIntoProperties(ctx, prefix, propertyValue, variableValue)
Colin Cross7f64b6d2015-07-09 13:57:48 -0700287 default:
288 panic(fmt.Errorf("unsupported field kind %q", propertyValue.Kind()))
289 }
290 }
291}
292
Colin Cross7e0eaf12017-05-05 16:16:24 -0700293func printfIntoProperty(propertyValue reflect.Value, variableValue interface{}) error {
Colin Cross7f64b6d2015-07-09 13:57:48 -0700294 s := propertyValue.String()
Colin Cross7e0eaf12017-05-05 16:16:24 -0700295
296 count := strings.Count(s, "%")
297 if count == 0 {
298 return nil
299 }
300
301 if count > 1 {
302 return fmt.Errorf("product variable properties only support a single '%%'")
303 }
304
Colin Cross7f64b6d2015-07-09 13:57:48 -0700305 if strings.Contains(s, "%d") {
306 switch v := variableValue.(type) {
307 case int:
Colin Cross7e0eaf12017-05-05 16:16:24 -0700308 // Nothing
Colin Cross7f64b6d2015-07-09 13:57:48 -0700309 case bool:
310 if v {
Colin Cross7e0eaf12017-05-05 16:16:24 -0700311 variableValue = 1
312 } else {
313 variableValue = 0
Colin Cross7f64b6d2015-07-09 13:57:48 -0700314 }
315 default:
Colin Cross7e0eaf12017-05-05 16:16:24 -0700316 return fmt.Errorf("unsupported type %T for %%d", variableValue)
Colin Cross7f64b6d2015-07-09 13:57:48 -0700317 }
Colin Cross7e0eaf12017-05-05 16:16:24 -0700318 } else if strings.Contains(s, "%s") {
319 switch variableValue.(type) {
320 case string:
321 // Nothing
322 default:
323 return fmt.Errorf("unsupported type %T for %%s", variableValue)
324 }
325 } else {
326 return fmt.Errorf("unsupported %% in product variable property")
Colin Cross7f64b6d2015-07-09 13:57:48 -0700327 }
Colin Cross7e0eaf12017-05-05 16:16:24 -0700328
329 propertyValue.Set(reflect.ValueOf(fmt.Sprintf(s, variableValue)))
330
331 return nil
Colin Cross7f64b6d2015-07-09 13:57:48 -0700332}