blob: ac7c730a3ad227d6b2903cbbe8a2b6ce5dd6ed35 [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 {
45 Version_script *string `android:"arch_variant"`
46 } `android:"arch_variant"`
Dan Willemsen97cae012016-03-01 14:08:42 -080047
48 Malloc_not_svelte struct {
49 Cflags []string
50 }
Evgenii Stepanov8391efa2016-05-12 18:04:18 -070051
52 Safestack struct {
53 Cflags []string `android:"arch_variant"`
54 } `android:"arch_variant"`
Dan Willemsena6f7d152016-07-12 14:57:40 -070055
56 Cpusets struct {
57 Cflags []string
58 }
59
60 Schedboost struct {
61 Cflags []string
62 }
Dan Willemsen1be35382016-07-25 17:42:18 -070063
64 Binder32bit struct {
65 Cflags []string
66 }
Dan Willemsenfcebcd52016-09-22 14:49:10 -070067
68 Debuggable struct {
69 Cflags []string
70 }
Colin Cross83163192015-12-01 16:31:16 -080071 } `android:"arch_variant"`
Colin Cross7f64b6d2015-07-09 13:57:48 -070072}
73
74var zeroProductVariables variableProperties
75
Colin Cross485e5722015-08-27 13:28:01 -070076type productVariables struct {
Dan Willemsen174978c2016-05-11 00:27:49 -070077 // Suffix to add to generated Makefiles
78 Make_suffix *string `json:",omitempty"`
79
Dan Willemsen97cae012016-03-01 14:08:42 -080080 Platform_sdk_version *int `json:",omitempty"`
Colin Cross4225f652015-09-17 14:33:42 -070081
Dan Willemsen1d31ec32015-09-22 16:56:09 -070082 DeviceName *string `json:",omitempty"`
Colin Cross4225f652015-09-17 14:33:42 -070083 DeviceArch *string `json:",omitempty"`
84 DeviceArchVariant *string `json:",omitempty"`
85 DeviceCpuVariant *string `json:",omitempty"`
86 DeviceAbi *[]string `json:",omitempty"`
Dan Willemsendd0e2c32015-10-20 14:29:35 -070087 DeviceUsesClang *bool `json:",omitempty"`
Colin Cross4225f652015-09-17 14:33:42 -070088
89 DeviceSecondaryArch *string `json:",omitempty"`
90 DeviceSecondaryArchVariant *string `json:",omitempty"`
91 DeviceSecondaryCpuVariant *string `json:",omitempty"`
92 DeviceSecondaryAbi *[]string `json:",omitempty"`
93
94 HostArch *string `json:",omitempty"`
95 HostSecondaryArch *string `json:",omitempty"`
Dan Willemsen490fd492015-11-24 17:53:15 -080096
97 CrossHost *string `json:",omitempty"`
98 CrossHostArch *string `json:",omitempty"`
99 CrossHostSecondaryArch *string `json:",omitempty"`
Colin Cross83163192015-12-01 16:31:16 -0800100
Dan Willemsenb5038162016-03-16 12:35:33 -0700101 Allow_missing_dependencies *bool `json:",omitempty"`
102 Unbundled_build *bool `json:",omitempty"`
103 Brillo *bool `json:",omitempty"`
104 Malloc_not_svelte *bool `json:",omitempty"`
Evgenii Stepanov8391efa2016-05-12 18:04:18 -0700105 Safestack *bool `json:",omitempty"`
Dan Willemsen36cff8b2016-05-17 16:35:02 -0700106 HostStaticBinaries *bool `json:",omitempty"`
Dan Willemsena6f7d152016-07-12 14:57:40 -0700107 Cpusets *bool `json:",omitempty"`
108 Schedboost *bool `json:",omitempty"`
Dan Willemsen1be35382016-07-25 17:42:18 -0700109 Binder32bit *bool `json:",omitempty"`
Colin Cross9d45bb72016-08-29 16:14:13 -0700110 UseGoma *bool `json:",omitempty"`
Dan Willemsenfcebcd52016-09-22 14:49:10 -0700111 Debuggable *bool `json:",omitempty"`
Colin Cross16b23492016-01-06 14:41:07 -0800112
Colin Cross1e7d3702016-08-24 15:25:47 -0700113 DevicePrefer32BitExecutables *bool `json:",omitempty"`
114 HostPrefer32BitExecutables *bool `json:",omitempty"`
115
Colin Cross16b23492016-01-06 14:41:07 -0800116 SanitizeHost *[]string `json:",omitempty"`
117 SanitizeDevice *[]string `json:",omitempty"`
Colin Crossa6bc19e2015-09-16 13:53:42 -0700118}
119
120func boolPtr(v bool) *bool {
121 return &v
Colin Cross485e5722015-08-27 13:28:01 -0700122}
123
Dan Willemsen47cf66b2015-09-16 16:48:54 -0700124func intPtr(v int) *int {
125 return &v
126}
127
Colin Cross4225f652015-09-17 14:33:42 -0700128func stringPtr(v string) *string {
129 return &v
130}
131
Colin Cross27385972015-09-18 10:57:10 -0700132func (v *productVariables) SetDefaultConfig() {
133 *v = productVariables{
Dan Willemsen5951c8a2016-07-19 19:08:14 -0700134 Platform_sdk_version: intPtr(23),
Colin Cross4225f652015-09-17 14:33:42 -0700135 HostArch: stringPtr("x86_64"),
136 HostSecondaryArch: stringPtr("x86"),
Dan Willemsen1d31ec32015-09-22 16:56:09 -0700137 DeviceName: stringPtr("flounder"),
Colin Cross4225f652015-09-17 14:33:42 -0700138 DeviceArch: stringPtr("arm64"),
Dan Willemsena91d1272016-03-29 22:06:05 -0700139 DeviceArchVariant: stringPtr("armv8-a"),
Colin Cross4225f652015-09-17 14:33:42 -0700140 DeviceCpuVariant: stringPtr("denver64"),
141 DeviceAbi: &[]string{"arm64-v8a"},
Dan Willemsendd0e2c32015-10-20 14:29:35 -0700142 DeviceUsesClang: boolPtr(true),
Colin Cross4225f652015-09-17 14:33:42 -0700143 DeviceSecondaryArch: stringPtr("arm"),
144 DeviceSecondaryArchVariant: stringPtr("armv7-a-neon"),
Dan Willemsen1d31ec32015-09-22 16:56:09 -0700145 DeviceSecondaryCpuVariant: stringPtr("denver"),
Colin Cross4225f652015-09-17 14:33:42 -0700146 DeviceSecondaryAbi: &[]string{"armeabi-v7a"},
Dan Willemsen97cae012016-03-01 14:08:42 -0800147 Malloc_not_svelte: boolPtr(false),
Evgenii Stepanov8391efa2016-05-12 18:04:18 -0700148 Safestack: boolPtr(false),
Colin Cross485e5722015-08-27 13:28:01 -0700149 }
Dan Willemsen490fd492015-11-24 17:53:15 -0800150
151 if runtime.GOOS == "linux" {
152 v.CrossHost = stringPtr("windows")
153 v.CrossHostArch = stringPtr("x86")
Dan Willemsen07cd0512016-02-03 23:16:33 -0800154 v.CrossHostSecondaryArch = stringPtr("x86_64")
Dan Willemsen490fd492015-11-24 17:53:15 -0800155 }
Colin Cross7f64b6d2015-07-09 13:57:48 -0700156}
157
Colin Cross635c3b02016-05-18 15:37:25 -0700158func variableMutator(mctx BottomUpMutatorContext) {
159 var module Module
Colin Cross7f64b6d2015-07-09 13:57:48 -0700160 var ok bool
Colin Cross635c3b02016-05-18 15:37:25 -0700161 if module, ok = mctx.Module().(Module); !ok {
Colin Cross7f64b6d2015-07-09 13:57:48 -0700162 return
163 }
164
165 // TODO: depend on config variable, create variants, propagate variants up tree
166 a := module.base()
Colin Cross06a931b2015-10-28 17:23:31 -0700167 variableValues := reflect.ValueOf(&a.variableProperties.Product_variables).Elem()
Colin Cross7f64b6d2015-07-09 13:57:48 -0700168 zeroValues := reflect.ValueOf(zeroProductVariables.Product_variables)
169
170 for i := 0; i < variableValues.NumField(); i++ {
171 variableValue := variableValues.Field(i)
172 zeroValue := zeroValues.Field(i)
Colin Cross485e5722015-08-27 13:28:01 -0700173 name := variableValues.Type().Field(i).Name
174 property := "product_variables." + proptools.PropertyNameForField(name)
Colin Cross7f64b6d2015-07-09 13:57:48 -0700175
Colin Crossa6bc19e2015-09-16 13:53:42 -0700176 // Check that the variable was set for the product
Colin Cross485e5722015-08-27 13:28:01 -0700177 val := reflect.ValueOf(mctx.Config().(Config).ProductVariables).FieldByName(name)
Colin Crossa6bc19e2015-09-16 13:53:42 -0700178 if !val.IsValid() || val.Kind() != reflect.Ptr || val.IsNil() {
179 continue
Colin Cross7f64b6d2015-07-09 13:57:48 -0700180 }
Colin Crossa6bc19e2015-09-16 13:53:42 -0700181
182 val = val.Elem()
183
184 // For bools, check that the value is true
185 if val.Kind() == reflect.Bool && val.Bool() == false {
186 continue
187 }
188
189 // Check if any properties were set for the module
190 if reflect.DeepEqual(variableValue.Interface(), zeroValue.Interface()) {
191 continue
192 }
193
194 a.setVariableProperties(mctx, property, variableValue, val.Interface())
Colin Cross7f64b6d2015-07-09 13:57:48 -0700195 }
196}
197
Colin Cross635c3b02016-05-18 15:37:25 -0700198func (a *ModuleBase) setVariableProperties(ctx BottomUpMutatorContext,
Colin Cross7f64b6d2015-07-09 13:57:48 -0700199 prefix string, productVariablePropertyValue reflect.Value, variableValue interface{}) {
200
Colin Crossdd0dbe62015-12-02 15:24:38 -0800201 printfIntoProperties(productVariablePropertyValue, variableValue)
Colin Cross7f64b6d2015-07-09 13:57:48 -0700202
Colin Cross06a931b2015-10-28 17:23:31 -0700203 err := proptools.AppendMatchingProperties(a.generalProperties,
204 productVariablePropertyValue.Addr().Interface(), nil)
205 if err != nil {
206 if propertyErr, ok := err.(*proptools.ExtendPropertyError); ok {
207 ctx.PropertyErrorf(propertyErr.Property, "%s", propertyErr.Err.Error())
208 } else {
209 panic(err)
210 }
211 }
Colin Cross7f64b6d2015-07-09 13:57:48 -0700212}
213
214func printfIntoProperties(productVariablePropertyValue reflect.Value, variableValue interface{}) {
215 for i := 0; i < productVariablePropertyValue.NumField(); i++ {
216 propertyValue := productVariablePropertyValue.Field(i)
Colin Crossdd0dbe62015-12-02 15:24:38 -0800217 kind := propertyValue.Kind()
218 if kind == reflect.Ptr {
219 if propertyValue.IsNil() {
220 continue
221 }
222 propertyValue = propertyValue.Elem()
223 }
Colin Cross7f64b6d2015-07-09 13:57:48 -0700224 switch propertyValue.Kind() {
225 case reflect.String:
226 printfIntoProperty(propertyValue, variableValue)
227 case reflect.Slice:
228 for j := 0; j < propertyValue.Len(); j++ {
229 printfIntoProperty(propertyValue.Index(j), variableValue)
230 }
Colin Crossdd0dbe62015-12-02 15:24:38 -0800231 case reflect.Bool:
232 // Nothing
Colin Cross7f64b6d2015-07-09 13:57:48 -0700233 case reflect.Struct:
234 printfIntoProperties(propertyValue, variableValue)
235 default:
236 panic(fmt.Errorf("unsupported field kind %q", propertyValue.Kind()))
237 }
238 }
239}
240
241func printfIntoProperty(propertyValue reflect.Value, variableValue interface{}) {
242 s := propertyValue.String()
243 // For now, we only support int formats
244 var i int
245 if strings.Contains(s, "%d") {
246 switch v := variableValue.(type) {
247 case int:
248 i = v
249 case bool:
250 if v {
251 i = 1
252 }
253 default:
254 panic(fmt.Errorf("unsupported type %T", variableValue))
255 }
256 propertyValue.Set(reflect.ValueOf(fmt.Sprintf(s, i)))
257 }
258}