| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 1 | // 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 |  | 
|  | 15 | package common | 
|  | 16 |  | 
|  | 17 | import ( | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 18 | "fmt" | 
|  | 19 | "reflect" | 
|  | 20 | "runtime" | 
|  | 21 | "strings" | 
| Colin Cross | f6566ed | 2015-03-24 11:13:38 -0700 | [diff] [blame] | 22 |  | 
| Colin Cross | 463a90e | 2015-06-17 14:20:06 -0700 | [diff] [blame] | 23 | "android/soong" | 
|  | 24 |  | 
| Colin Cross | f6566ed | 2015-03-24 11:13:38 -0700 | [diff] [blame] | 25 | "github.com/google/blueprint" | 
|  | 26 | "github.com/google/blueprint/proptools" | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 27 | ) | 
|  | 28 |  | 
| Colin Cross | 463a90e | 2015-06-17 14:20:06 -0700 | [diff] [blame] | 29 | func init() { | 
|  | 30 | soong.RegisterEarlyMutator("host_or_device", HostOrDeviceMutator) | 
|  | 31 | soong.RegisterEarlyMutator("arch", ArchMutator) | 
|  | 32 | } | 
|  | 33 |  | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 34 | var ( | 
| Colin Cross | ec19363 | 2015-07-06 17:49:43 -0700 | [diff] [blame] | 35 | Arm    = newArch("arm", "lib32") | 
|  | 36 | Arm64  = newArch("arm64", "lib64") | 
|  | 37 | Mips   = newArch("mips", "lib32") | 
|  | 38 | Mips64 = newArch("mips64", "lib64") | 
|  | 39 | X86    = newArch("x86", "lib32") | 
|  | 40 | X86_64 = newArch("x86_64", "lib64") | 
| Colin Cross | 2fe6687 | 2015-03-30 17:20:39 -0700 | [diff] [blame] | 41 |  | 
|  | 42 | Common = ArchType{ | 
|  | 43 | Name: "common", | 
|  | 44 | } | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 45 | ) | 
|  | 46 |  | 
|  | 47 | /* | 
|  | 48 | Example blueprints file containing all variant property groups, with comment listing what type | 
|  | 49 | of variants get properties in that group: | 
|  | 50 |  | 
|  | 51 | module { | 
|  | 52 | arch: { | 
|  | 53 | arm: { | 
|  | 54 | // Host or device variants with arm architecture | 
|  | 55 | }, | 
|  | 56 | arm64: { | 
|  | 57 | // Host or device variants with arm64 architecture | 
|  | 58 | }, | 
|  | 59 | mips: { | 
|  | 60 | // Host or device variants with mips architecture | 
|  | 61 | }, | 
|  | 62 | mips64: { | 
|  | 63 | // Host or device variants with mips64 architecture | 
|  | 64 | }, | 
|  | 65 | x86: { | 
|  | 66 | // Host or device variants with x86 architecture | 
|  | 67 | }, | 
|  | 68 | x86_64: { | 
|  | 69 | // Host or device variants with x86_64 architecture | 
|  | 70 | }, | 
|  | 71 | }, | 
|  | 72 | multilib: { | 
|  | 73 | lib32: { | 
|  | 74 | // Host or device variants for 32-bit architectures | 
|  | 75 | }, | 
|  | 76 | lib64: { | 
|  | 77 | // Host or device variants for 64-bit architectures | 
|  | 78 | }, | 
|  | 79 | }, | 
|  | 80 | target: { | 
|  | 81 | android: { | 
|  | 82 | // Device variants | 
|  | 83 | }, | 
|  | 84 | host: { | 
|  | 85 | // Host variants | 
|  | 86 | }, | 
|  | 87 | linux: { | 
|  | 88 | // Linux host variants | 
|  | 89 | }, | 
|  | 90 | darwin: { | 
|  | 91 | // Darwin host variants | 
|  | 92 | }, | 
|  | 93 | windows: { | 
|  | 94 | // Windows host variants | 
|  | 95 | }, | 
|  | 96 | not_windows: { | 
|  | 97 | // Non-windows host variants | 
|  | 98 | }, | 
|  | 99 | }, | 
|  | 100 | } | 
|  | 101 | */ | 
| Colin Cross | 7d5136f | 2015-05-11 13:39:40 -0700 | [diff] [blame] | 102 |  | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 103 | type archProperties struct { | 
| Colin Cross | 7d5136f | 2015-05-11 13:39:40 -0700 | [diff] [blame] | 104 | // Properties to vary by target architecture | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 105 | Arch struct { | 
| Colin Cross | 7d5136f | 2015-05-11 13:39:40 -0700 | [diff] [blame] | 106 | // Properties for module variants being built to run on arm (host or device) | 
|  | 107 | Arm interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 108 | // Properties for module variants being built to run on arm64 (host or device) | 
|  | 109 | Arm64 interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 110 | // Properties for module variants being built to run on mips (host or device) | 
|  | 111 | Mips interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 112 | // Properties for module variants being built to run on mips64 (host or device) | 
|  | 113 | Mips64 interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 114 | // Properties for module variants being built to run on x86 (host or device) | 
|  | 115 | X86 interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 116 | // Properties for module variants being built to run on x86_64 (host or device) | 
|  | 117 | X86_64 interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
| Colin Cross | ec19363 | 2015-07-06 17:49:43 -0700 | [diff] [blame] | 118 |  | 
|  | 119 | // Arm arch variants | 
|  | 120 | Armv5te      interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 121 | Armv7_a      interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 122 | Armv7_a_neon interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 123 |  | 
|  | 124 | // Arm cpu variants | 
|  | 125 | Cortex_a7  interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 126 | Cortex_a8  interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 127 | Cortex_a9  interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 128 | Cortex_a15 interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 129 | Krait      interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 130 | Denver     interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 131 |  | 
|  | 132 | // Arm64 cpu variants | 
| Dan Willemsen | 00faa6d | 2015-09-11 17:38:13 -0700 | [diff] [blame] | 133 | Cortex_a53 interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 134 | Denver64   interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
| Colin Cross | ec19363 | 2015-07-06 17:49:43 -0700 | [diff] [blame] | 135 |  | 
|  | 136 | // Mips arch variants | 
|  | 137 | Mips_rev6 interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 138 |  | 
| Colin Cross | 01432f6 | 2015-07-09 17:56:26 -0700 | [diff] [blame] | 139 | // X86 arch variants | 
|  | 140 | X86_sse3 interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 141 | X86_sse4 interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 142 |  | 
| Colin Cross | ec19363 | 2015-07-06 17:49:43 -0700 | [diff] [blame] | 143 | // X86 cpu variants | 
|  | 144 | Atom       interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 145 | Silvermont interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 146 | } | 
| Colin Cross | ec19363 | 2015-07-06 17:49:43 -0700 | [diff] [blame] | 147 |  | 
| Colin Cross | 7d5136f | 2015-05-11 13:39:40 -0700 | [diff] [blame] | 148 | // Properties to vary by 32-bit or 64-bit | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 149 | Multilib struct { | 
| Colin Cross | 7d5136f | 2015-05-11 13:39:40 -0700 | [diff] [blame] | 150 | // Properties for module variants being built to run on 32-bit devices | 
|  | 151 | Lib32 interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 152 | // Properties for module variants being built to run on 64-bit devices | 
|  | 153 | Lib64 interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 154 | } | 
| Colin Cross | 7d5136f | 2015-05-11 13:39:40 -0700 | [diff] [blame] | 155 | // Properties to vary by build target (host or device, os, os+archictecture) | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 156 | Target struct { | 
| Colin Cross | 7d5136f | 2015-05-11 13:39:40 -0700 | [diff] [blame] | 157 | // Properties for module variants being built to run on the host | 
|  | 158 | Host interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 159 | // Properties for module variants being built to run on the device | 
|  | 160 | Android interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 161 | // Properties for module variants being built to run on arm devices | 
|  | 162 | Android_arm interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 163 | // Properties for module variants being built to run on arm64 devices | 
|  | 164 | Android_arm64 interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 165 | // Properties for module variants being built to run on mips devices | 
|  | 166 | Android_mips interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 167 | // Properties for module variants being built to run on mips64 devices | 
|  | 168 | Android_mips64 interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 169 | // Properties for module variants being built to run on x86 devices | 
|  | 170 | Android_x86 interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 171 | // Properties for module variants being built to run on x86_64 devices | 
|  | 172 | Android_x86_64 interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 173 | // Properties for module variants being built to run on devices that support 64-bit | 
|  | 174 | Android64 interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 175 | // Properties for module variants being built to run on devices that do not support 64-bit | 
|  | 176 | Android32 interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 177 | // Properties for module variants being built to run on linux hosts | 
|  | 178 | Linux interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 179 | // Properties for module variants being built to run on linux x86 hosts | 
|  | 180 | Linux_x86 interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 181 | // Properties for module variants being built to run on linux x86_64 hosts | 
|  | 182 | Linux_x86_64 interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 183 | // Properties for module variants being built to run on darwin hosts | 
|  | 184 | Darwin interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 185 | // Properties for module variants being built to run on darwin x86 hosts | 
|  | 186 | Darwin_x86 interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 187 | // Properties for module variants being built to run on darwin x86_64 hosts | 
|  | 188 | Darwin_x86_64 interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 189 | // Properties for module variants being built to run on windows hosts | 
|  | 190 | Windows interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
|  | 191 | // Properties for module variants being built to run on linux or darwin hosts | 
|  | 192 | Not_windows interface{} `blueprint:"filter(android:\"arch_variant\")"` | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 193 | } | 
|  | 194 | } | 
|  | 195 |  | 
|  | 196 | // An Arch indicates a single CPU architecture. | 
|  | 197 | type Arch struct { | 
| Colin Cross | d3ba039 | 2015-05-07 14:11:29 -0700 | [diff] [blame] | 198 | ArchType    ArchType | 
|  | 199 | ArchVariant string | 
|  | 200 | CpuVariant  string | 
|  | 201 | Abi         string | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 202 | } | 
|  | 203 |  | 
|  | 204 | func (a Arch) String() string { | 
| Colin Cross | d3ba039 | 2015-05-07 14:11:29 -0700 | [diff] [blame] | 205 | s := a.ArchType.String() | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 206 | if a.ArchVariant != "" { | 
|  | 207 | s += "_" + a.ArchVariant | 
|  | 208 | } | 
|  | 209 | if a.CpuVariant != "" { | 
|  | 210 | s += "_" + a.CpuVariant | 
|  | 211 | } | 
|  | 212 | return s | 
|  | 213 | } | 
|  | 214 |  | 
|  | 215 | type ArchType struct { | 
| Colin Cross | ec19363 | 2015-07-06 17:49:43 -0700 | [diff] [blame] | 216 | Name     string | 
|  | 217 | Multilib string | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 218 | } | 
|  | 219 |  | 
| Colin Cross | ec19363 | 2015-07-06 17:49:43 -0700 | [diff] [blame] | 220 | func newArch(name, multilib string) ArchType { | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 221 | return ArchType{ | 
| Colin Cross | ec19363 | 2015-07-06 17:49:43 -0700 | [diff] [blame] | 222 | Name:     name, | 
|  | 223 | Multilib: multilib, | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 224 | } | 
|  | 225 | } | 
|  | 226 |  | 
|  | 227 | func (a ArchType) String() string { | 
|  | 228 | return a.Name | 
|  | 229 | } | 
|  | 230 |  | 
|  | 231 | type HostOrDeviceSupported int | 
|  | 232 |  | 
|  | 233 | const ( | 
|  | 234 | _ HostOrDeviceSupported = iota | 
|  | 235 | HostSupported | 
|  | 236 | DeviceSupported | 
|  | 237 | HostAndDeviceSupported | 
|  | 238 | ) | 
|  | 239 |  | 
|  | 240 | type HostOrDevice int | 
|  | 241 |  | 
|  | 242 | const ( | 
|  | 243 | _ HostOrDevice = iota | 
|  | 244 | Host | 
|  | 245 | Device | 
|  | 246 | ) | 
|  | 247 |  | 
|  | 248 | func (hod HostOrDevice) String() string { | 
|  | 249 | switch hod { | 
|  | 250 | case Device: | 
|  | 251 | return "device" | 
|  | 252 | case Host: | 
|  | 253 | return "host" | 
|  | 254 | default: | 
|  | 255 | panic(fmt.Sprintf("unexpected HostOrDevice value %d", hod)) | 
|  | 256 | } | 
|  | 257 | } | 
|  | 258 |  | 
| Colin Cross | ec19363 | 2015-07-06 17:49:43 -0700 | [diff] [blame] | 259 | func (hod HostOrDevice) Property() string { | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 260 | switch hod { | 
|  | 261 | case Device: | 
|  | 262 | return "android" | 
|  | 263 | case Host: | 
|  | 264 | return "host" | 
|  | 265 | default: | 
|  | 266 | panic(fmt.Sprintf("unexpected HostOrDevice value %d", hod)) | 
|  | 267 | } | 
|  | 268 | } | 
|  | 269 |  | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 270 | func (hod HostOrDevice) Host() bool { | 
|  | 271 | if hod == 0 { | 
|  | 272 | panic("HostOrDevice unset") | 
|  | 273 | } | 
|  | 274 | return hod == Host | 
|  | 275 | } | 
|  | 276 |  | 
|  | 277 | func (hod HostOrDevice) Device() bool { | 
|  | 278 | if hod == 0 { | 
|  | 279 | panic("HostOrDevice unset") | 
|  | 280 | } | 
|  | 281 | return hod == Device | 
|  | 282 | } | 
|  | 283 |  | 
|  | 284 | var hostOrDeviceName = map[HostOrDevice]string{ | 
|  | 285 | Device: "device", | 
|  | 286 | Host:   "host", | 
|  | 287 | } | 
|  | 288 |  | 
|  | 289 | var ( | 
|  | 290 | armArch = Arch{ | 
| Colin Cross | d3ba039 | 2015-05-07 14:11:29 -0700 | [diff] [blame] | 291 | ArchType:    Arm, | 
|  | 292 | ArchVariant: "armv7-a-neon", | 
|  | 293 | CpuVariant:  "cortex-a15", | 
|  | 294 | Abi:         "armeabi-v7a", | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 295 | } | 
|  | 296 | arm64Arch = Arch{ | 
| Colin Cross | ec19363 | 2015-07-06 17:49:43 -0700 | [diff] [blame] | 297 | ArchType:   Arm64, | 
|  | 298 | CpuVariant: "denver64", | 
|  | 299 | Abi:        "arm64-v8a", | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 300 | } | 
| Colin Cross | d3ba039 | 2015-05-07 14:11:29 -0700 | [diff] [blame] | 301 | x86Arch = Arch{ | 
|  | 302 | ArchType: X86, | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 303 | } | 
| Colin Cross | d3ba039 | 2015-05-07 14:11:29 -0700 | [diff] [blame] | 304 | x8664Arch = Arch{ | 
|  | 305 | ArchType: X86_64, | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 306 | } | 
| Colin Cross | d3ba039 | 2015-05-07 14:11:29 -0700 | [diff] [blame] | 307 | commonArch = Arch{ | 
|  | 308 | ArchType: Common, | 
| Colin Cross | 2fe6687 | 2015-03-30 17:20:39 -0700 | [diff] [blame] | 309 | } | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 310 | ) | 
|  | 311 |  | 
| Colin Cross | d3ba039 | 2015-05-07 14:11:29 -0700 | [diff] [blame] | 312 | func HostOrDeviceMutator(mctx blueprint.EarlyMutatorContext) { | 
|  | 313 | var module AndroidModule | 
|  | 314 | var ok bool | 
|  | 315 | if module, ok = mctx.Module().(AndroidModule); !ok { | 
|  | 316 | return | 
|  | 317 | } | 
|  | 318 |  | 
|  | 319 | hods := []HostOrDevice{} | 
|  | 320 |  | 
|  | 321 | if module.base().HostSupported() { | 
|  | 322 | hods = append(hods, Host) | 
|  | 323 | } | 
|  | 324 |  | 
|  | 325 | if module.base().DeviceSupported() { | 
|  | 326 | hods = append(hods, Device) | 
|  | 327 | } | 
|  | 328 |  | 
|  | 329 | if len(hods) == 0 { | 
|  | 330 | return | 
|  | 331 | } | 
|  | 332 |  | 
|  | 333 | hodNames := []string{} | 
|  | 334 | for _, hod := range hods { | 
|  | 335 | hodNames = append(hodNames, hod.String()) | 
|  | 336 | } | 
|  | 337 |  | 
|  | 338 | modules := mctx.CreateVariations(hodNames...) | 
|  | 339 | for i, m := range modules { | 
|  | 340 | m.(AndroidModule).base().SetHostOrDevice(hods[i]) | 
|  | 341 | } | 
|  | 342 | } | 
|  | 343 |  | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 344 | func ArchMutator(mctx blueprint.EarlyMutatorContext) { | 
|  | 345 | var module AndroidModule | 
|  | 346 | var ok bool | 
|  | 347 | if module, ok = mctx.Module().(AndroidModule); !ok { | 
|  | 348 | return | 
|  | 349 | } | 
|  | 350 |  | 
|  | 351 | // TODO: this is all hardcoded for arm64 primary, arm secondary for now | 
|  | 352 | // Replace with a configuration file written by lunch or bootstrap | 
|  | 353 |  | 
|  | 354 | arches := []Arch{} | 
|  | 355 |  | 
| Colin Cross | d3ba039 | 2015-05-07 14:11:29 -0700 | [diff] [blame] | 356 | if module.base().HostSupported() && module.base().HostOrDevice().Host() { | 
| Colin Cross | 2fe6687 | 2015-03-30 17:20:39 -0700 | [diff] [blame] | 357 | switch module.base().commonProperties.Compile_multilib { | 
|  | 358 | case "common": | 
| Colin Cross | d3ba039 | 2015-05-07 14:11:29 -0700 | [diff] [blame] | 359 | arches = append(arches, commonArch) | 
| Dan Willemsen | ffce3fc | 2015-07-08 13:02:19 -0700 | [diff] [blame] | 360 | case "both": | 
|  | 361 | arches = append(arches, x8664Arch, x86Arch) | 
|  | 362 | case "first", "64": | 
|  | 363 | arches = append(arches, x8664Arch) | 
|  | 364 | case "32": | 
|  | 365 | arches = append(arches, x86Arch) | 
| Colin Cross | 2fe6687 | 2015-03-30 17:20:39 -0700 | [diff] [blame] | 366 | default: | 
| Colin Cross | d3ba039 | 2015-05-07 14:11:29 -0700 | [diff] [blame] | 367 | arches = append(arches, x8664Arch) | 
| Colin Cross | 2fe6687 | 2015-03-30 17:20:39 -0700 | [diff] [blame] | 368 | } | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 369 | } | 
|  | 370 |  | 
| Colin Cross | d3ba039 | 2015-05-07 14:11:29 -0700 | [diff] [blame] | 371 | if module.base().DeviceSupported() && module.base().HostOrDevice().Device() { | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 372 | switch module.base().commonProperties.Compile_multilib { | 
| Colin Cross | 2fe6687 | 2015-03-30 17:20:39 -0700 | [diff] [blame] | 373 | case "common": | 
| Colin Cross | d3ba039 | 2015-05-07 14:11:29 -0700 | [diff] [blame] | 374 | arches = append(arches, commonArch) | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 375 | case "both": | 
|  | 376 | arches = append(arches, arm64Arch, armArch) | 
|  | 377 | case "first", "64": | 
|  | 378 | arches = append(arches, arm64Arch) | 
|  | 379 | case "32": | 
|  | 380 | arches = append(arches, armArch) | 
|  | 381 | default: | 
|  | 382 | mctx.ModuleErrorf(`compile_multilib must be "both", "first", "32", or "64", found %q`, | 
|  | 383 | module.base().commonProperties.Compile_multilib) | 
|  | 384 | } | 
|  | 385 | } | 
|  | 386 |  | 
| Colin Cross | 5049f02 | 2015-03-18 13:28:46 -0700 | [diff] [blame] | 387 | if len(arches) == 0 { | 
|  | 388 | return | 
|  | 389 | } | 
|  | 390 |  | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 391 | archNames := []string{} | 
|  | 392 | for _, arch := range arches { | 
|  | 393 | archNames = append(archNames, arch.String()) | 
|  | 394 | } | 
|  | 395 |  | 
|  | 396 | modules := mctx.CreateVariations(archNames...) | 
|  | 397 |  | 
|  | 398 | for i, m := range modules { | 
|  | 399 | m.(AndroidModule).base().SetArch(arches[i]) | 
| Colin Cross | d3ba039 | 2015-05-07 14:11:29 -0700 | [diff] [blame] | 400 | m.(AndroidModule).base().setArchProperties(mctx) | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 401 | } | 
|  | 402 | } | 
|  | 403 |  | 
| Colin Cross | c472d57 | 2015-03-17 15:06:21 -0700 | [diff] [blame] | 404 | func InitArchModule(m AndroidModule, defaultMultilib Multilib, | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 405 | propertyStructs ...interface{}) (blueprint.Module, []interface{}) { | 
|  | 406 |  | 
|  | 407 | base := m.base() | 
|  | 408 |  | 
| Colin Cross | c472d57 | 2015-03-17 15:06:21 -0700 | [diff] [blame] | 409 | base.commonProperties.Compile_multilib = string(defaultMultilib) | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 410 |  | 
|  | 411 | base.generalProperties = append(base.generalProperties, | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 412 | propertyStructs...) | 
|  | 413 |  | 
|  | 414 | for _, properties := range base.generalProperties { | 
|  | 415 | propertiesValue := reflect.ValueOf(properties) | 
|  | 416 | if propertiesValue.Kind() != reflect.Ptr { | 
|  | 417 | panic("properties must be a pointer to a struct") | 
|  | 418 | } | 
|  | 419 |  | 
|  | 420 | propertiesValue = propertiesValue.Elem() | 
|  | 421 | if propertiesValue.Kind() != reflect.Struct { | 
|  | 422 | panic("properties must be a pointer to a struct") | 
|  | 423 | } | 
|  | 424 |  | 
|  | 425 | archProperties := &archProperties{} | 
|  | 426 | forEachInterface(reflect.ValueOf(archProperties), func(v reflect.Value) { | 
| Colin Cross | 3ab7d88 | 2015-05-19 13:03:01 -0700 | [diff] [blame] | 427 | newValue := proptools.CloneEmptyProperties(propertiesValue) | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 428 | v.Set(newValue) | 
|  | 429 | }) | 
|  | 430 |  | 
|  | 431 | base.archProperties = append(base.archProperties, archProperties) | 
|  | 432 | } | 
|  | 433 |  | 
|  | 434 | var allProperties []interface{} | 
|  | 435 | allProperties = append(allProperties, base.generalProperties...) | 
|  | 436 | for _, asp := range base.archProperties { | 
|  | 437 | allProperties = append(allProperties, asp) | 
|  | 438 | } | 
|  | 439 |  | 
|  | 440 | return m, allProperties | 
|  | 441 | } | 
|  | 442 |  | 
| Colin Cross | ec19363 | 2015-07-06 17:49:43 -0700 | [diff] [blame] | 443 | var dashToUnderscoreReplacer = strings.NewReplacer("-", "_") | 
|  | 444 |  | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 445 | // Rewrite the module's properties structs to contain arch-specific values. | 
| Colin Cross | d3ba039 | 2015-05-07 14:11:29 -0700 | [diff] [blame] | 446 | func (a *AndroidModuleBase) setArchProperties(ctx blueprint.EarlyMutatorContext) { | 
|  | 447 | arch := a.commonProperties.CompileArch | 
|  | 448 | hod := a.commonProperties.CompileHostOrDevice | 
|  | 449 |  | 
| Colin Cross | 2fe6687 | 2015-03-30 17:20:39 -0700 | [diff] [blame] | 450 | if arch.ArchType == Common { | 
|  | 451 | return | 
|  | 452 | } | 
|  | 453 |  | 
| Colin Cross | 7f64b6d | 2015-07-09 13:57:48 -0700 | [diff] [blame] | 454 | callback := func(srcPropertyName, dstPropertyName string) { | 
|  | 455 | a.extendedProperties[dstPropertyName] = struct{}{} | 
|  | 456 | } | 
|  | 457 |  | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 458 | for i := range a.generalProperties { | 
| Colin Cross | 7f64b6d | 2015-07-09 13:57:48 -0700 | [diff] [blame] | 459 | generalPropsValue := []reflect.Value{reflect.ValueOf(a.generalProperties[i]).Elem()} | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 460 |  | 
|  | 461 | // Handle arch-specific properties in the form: | 
| Colin Cross | b05bff2 | 2015-04-30 15:08:04 -0700 | [diff] [blame] | 462 | // arch: { | 
|  | 463 | //     arm64: { | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 464 | //         key: value, | 
|  | 465 | //     }, | 
|  | 466 | // }, | 
|  | 467 | t := arch.ArchType | 
| Colin Cross | ec19363 | 2015-07-06 17:49:43 -0700 | [diff] [blame] | 468 | field := proptools.FieldNameForProperty(t.Name) | 
| Colin Cross | 7f64b6d | 2015-07-09 13:57:48 -0700 | [diff] [blame] | 469 | extendProperties(ctx, "arch_variant", "arch."+t.Name, generalPropsValue, | 
|  | 470 | reflect.ValueOf(a.archProperties[i].Arch).FieldByName(field).Elem().Elem(), callback) | 
| Colin Cross | ec19363 | 2015-07-06 17:49:43 -0700 | [diff] [blame] | 471 |  | 
|  | 472 | // Handle arch-variant-specific properties in the form: | 
|  | 473 | // arch: { | 
|  | 474 | //     variant: { | 
|  | 475 | //         key: value, | 
|  | 476 | //     }, | 
|  | 477 | // }, | 
|  | 478 | v := dashToUnderscoreReplacer.Replace(arch.ArchVariant) | 
|  | 479 | if v != "" { | 
|  | 480 | field := proptools.FieldNameForProperty(v) | 
| Colin Cross | 7f64b6d | 2015-07-09 13:57:48 -0700 | [diff] [blame] | 481 | extendProperties(ctx, "arch_variant", "arch."+v, generalPropsValue, | 
|  | 482 | reflect.ValueOf(a.archProperties[i].Arch).FieldByName(field).Elem().Elem(), callback) | 
| Colin Cross | ec19363 | 2015-07-06 17:49:43 -0700 | [diff] [blame] | 483 | } | 
|  | 484 |  | 
|  | 485 | // Handle cpu-variant-specific properties in the form: | 
|  | 486 | // arch: { | 
|  | 487 | //     variant: { | 
|  | 488 | //         key: value, | 
|  | 489 | //     }, | 
|  | 490 | // }, | 
|  | 491 | c := dashToUnderscoreReplacer.Replace(arch.CpuVariant) | 
|  | 492 | if c != "" { | 
|  | 493 | field := proptools.FieldNameForProperty(c) | 
| Colin Cross | 7f64b6d | 2015-07-09 13:57:48 -0700 | [diff] [blame] | 494 | extendProperties(ctx, "arch_variant", "arch."+c, generalPropsValue, | 
|  | 495 | reflect.ValueOf(a.archProperties[i].Arch).FieldByName(field).Elem().Elem(), callback) | 
| Colin Cross | ec19363 | 2015-07-06 17:49:43 -0700 | [diff] [blame] | 496 | } | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 497 |  | 
|  | 498 | // Handle multilib-specific properties in the form: | 
| Colin Cross | b05bff2 | 2015-04-30 15:08:04 -0700 | [diff] [blame] | 499 | // multilib: { | 
|  | 500 | //     lib32: { | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 501 | //         key: value, | 
|  | 502 | //     }, | 
|  | 503 | // }, | 
| Colin Cross | ec19363 | 2015-07-06 17:49:43 -0700 | [diff] [blame] | 504 | multilibField := proptools.FieldNameForProperty(t.Multilib) | 
| Colin Cross | 7f64b6d | 2015-07-09 13:57:48 -0700 | [diff] [blame] | 505 | extendProperties(ctx, "arch_variant", "multilib."+t.Multilib, generalPropsValue, | 
|  | 506 | reflect.ValueOf(a.archProperties[i].Multilib).FieldByName(multilibField).Elem().Elem(), callback) | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 507 |  | 
|  | 508 | // Handle host-or-device-specific properties in the form: | 
| Colin Cross | b05bff2 | 2015-04-30 15:08:04 -0700 | [diff] [blame] | 509 | // target: { | 
|  | 510 | //     host: { | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 511 | //         key: value, | 
|  | 512 | //     }, | 
|  | 513 | // }, | 
| Colin Cross | ec19363 | 2015-07-06 17:49:43 -0700 | [diff] [blame] | 514 | hodProperty := hod.Property() | 
|  | 515 | hodField := proptools.FieldNameForProperty(hodProperty) | 
| Colin Cross | 7f64b6d | 2015-07-09 13:57:48 -0700 | [diff] [blame] | 516 | extendProperties(ctx, "arch_variant", "target."+hodProperty, generalPropsValue, | 
|  | 517 | reflect.ValueOf(a.archProperties[i].Target).FieldByName(hodField).Elem().Elem(), callback) | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 518 |  | 
|  | 519 | // Handle host target properties in the form: | 
| Colin Cross | b05bff2 | 2015-04-30 15:08:04 -0700 | [diff] [blame] | 520 | // target: { | 
|  | 521 | //     linux: { | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 522 | //         key: value, | 
|  | 523 | //     }, | 
| Colin Cross | b05bff2 | 2015-04-30 15:08:04 -0700 | [diff] [blame] | 524 | //     not_windows: { | 
|  | 525 | //         key: value, | 
|  | 526 | //     }, | 
|  | 527 | //     linux_x86: { | 
|  | 528 | //         key: value, | 
|  | 529 | //     }, | 
|  | 530 | //     linux_arm: { | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 531 | //         key: value, | 
|  | 532 | //     }, | 
|  | 533 | // }, | 
|  | 534 | var osList = []struct { | 
|  | 535 | goos  string | 
|  | 536 | field string | 
|  | 537 | }{ | 
|  | 538 | {"darwin", "Darwin"}, | 
|  | 539 | {"linux", "Linux"}, | 
|  | 540 | {"windows", "Windows"}, | 
|  | 541 | } | 
|  | 542 |  | 
|  | 543 | if hod.Host() { | 
|  | 544 | for _, v := range osList { | 
|  | 545 | if v.goos == runtime.GOOS { | 
| Colin Cross | 7f64b6d | 2015-07-09 13:57:48 -0700 | [diff] [blame] | 546 | extendProperties(ctx, "arch_variant", "target."+v.goos, generalPropsValue, | 
|  | 547 | reflect.ValueOf(a.archProperties[i].Target).FieldByName(v.field).Elem().Elem(), callback) | 
| Colin Cross | b05bff2 | 2015-04-30 15:08:04 -0700 | [diff] [blame] | 548 | t := arch.ArchType | 
| Colin Cross | 7f64b6d | 2015-07-09 13:57:48 -0700 | [diff] [blame] | 549 | extendProperties(ctx, "arch_variant", "target."+v.goos+"_"+t.Name, generalPropsValue, | 
|  | 550 | reflect.ValueOf(a.archProperties[i].Target).FieldByName(v.field+"_"+t.Name).Elem().Elem(), callback) | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 551 | } | 
|  | 552 | } | 
| Colin Cross | 7f64b6d | 2015-07-09 13:57:48 -0700 | [diff] [blame] | 553 | extendProperties(ctx, "arch_variant", "target.not_windows", generalPropsValue, | 
|  | 554 | reflect.ValueOf(a.archProperties[i].Target).FieldByName("Not_windows").Elem().Elem(), callback) | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 555 | } | 
|  | 556 |  | 
| Colin Cross | f820941 | 2015-03-26 14:44:26 -0700 | [diff] [blame] | 557 | // Handle 64-bit device properties in the form: | 
|  | 558 | // target { | 
|  | 559 | //     android64 { | 
|  | 560 | //         key: value, | 
|  | 561 | //     }, | 
|  | 562 | //     android32 { | 
|  | 563 | //         key: value, | 
|  | 564 | //     }, | 
|  | 565 | // }, | 
|  | 566 | // WARNING: this is probably not what you want to use in your blueprints file, it selects | 
|  | 567 | // options for all targets on a device that supports 64-bit binaries, not just the targets | 
|  | 568 | // that are being compiled for 64-bit.  Its expected use case is binaries like linker and | 
|  | 569 | // debuggerd that need to know when they are a 32-bit process running on a 64-bit device | 
|  | 570 | if hod.Device() { | 
|  | 571 | if true /* && target_is_64_bit */ { | 
| Colin Cross | 7f64b6d | 2015-07-09 13:57:48 -0700 | [diff] [blame] | 572 | extendProperties(ctx, "arch_variant", "target.android64", generalPropsValue, | 
|  | 573 | reflect.ValueOf(a.archProperties[i].Target).FieldByName("Android64").Elem().Elem(), callback) | 
| Colin Cross | f820941 | 2015-03-26 14:44:26 -0700 | [diff] [blame] | 574 | } else { | 
| Colin Cross | 7f64b6d | 2015-07-09 13:57:48 -0700 | [diff] [blame] | 575 | extendProperties(ctx, "arch_variant", "target.android32", generalPropsValue, | 
|  | 576 | reflect.ValueOf(a.archProperties[i].Target).FieldByName("Android32").Elem().Elem(), callback) | 
| Colin Cross | f820941 | 2015-03-26 14:44:26 -0700 | [diff] [blame] | 577 | } | 
|  | 578 | } | 
| Colin Cross | b05bff2 | 2015-04-30 15:08:04 -0700 | [diff] [blame] | 579 |  | 
|  | 580 | // Handle device architecture properties in the form: | 
|  | 581 | // target { | 
|  | 582 | //     android_arm { | 
|  | 583 | //         key: value, | 
|  | 584 | //     }, | 
|  | 585 | //     android_x86 { | 
|  | 586 | //         key: value, | 
|  | 587 | //     }, | 
|  | 588 | // }, | 
|  | 589 | if hod.Device() { | 
|  | 590 | t := arch.ArchType | 
| Colin Cross | 7f64b6d | 2015-07-09 13:57:48 -0700 | [diff] [blame] | 591 | extendProperties(ctx, "arch_variant", "target.android_"+t.Name, generalPropsValue, | 
|  | 592 | reflect.ValueOf(a.archProperties[i].Target).FieldByName("Android_"+t.Name).Elem().Elem(), callback) | 
| Colin Cross | b05bff2 | 2015-04-30 15:08:04 -0700 | [diff] [blame] | 593 | } | 
|  | 594 |  | 
| Colin Cross | 3f40fa4 | 2015-01-30 17:27:36 -0800 | [diff] [blame] | 595 | if ctx.Failed() { | 
|  | 596 | return | 
|  | 597 | } | 
|  | 598 | } | 
|  | 599 | } | 
|  | 600 |  | 
|  | 601 | func forEachInterface(v reflect.Value, f func(reflect.Value)) { | 
|  | 602 | switch v.Kind() { | 
|  | 603 | case reflect.Interface: | 
|  | 604 | f(v) | 
|  | 605 | case reflect.Struct: | 
|  | 606 | for i := 0; i < v.NumField(); i++ { | 
|  | 607 | forEachInterface(v.Field(i), f) | 
|  | 608 | } | 
|  | 609 | case reflect.Ptr: | 
|  | 610 | forEachInterface(v.Elem(), f) | 
|  | 611 | default: | 
|  | 612 | panic(fmt.Errorf("Unsupported kind %s", v.Kind())) | 
|  | 613 | } | 
|  | 614 | } |