blob: 45afec146e6fa9183446cee8d137b5e5f6816560 [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 Cross6ff51382015-12-17 16:39:19 -080018 "fmt"
Colin Cross3f40fa42015-01-30 17:27:36 -080019 "path/filepath"
Colin Cross6ff51382015-12-17 16:39:19 -080020 "strings"
Colin Crossf6566ed2015-03-24 11:13:38 -070021
Dan Willemsen0effe062015-11-30 16:06:01 -080022 "android/soong"
Colin Cross8f101b42015-06-17 15:09:06 -070023 "android/soong/glob"
24
Colin Crossf6566ed2015-03-24 11:13:38 -070025 "github.com/google/blueprint"
Colin Cross3f40fa42015-01-30 17:27:36 -080026)
27
Colin Crossa120ec12016-08-19 16:07:38 -070028func init() {
29 RegisterTopDownMutator("customizer", customizerMutator).Parallel()
30 RegisterBottomUpMutator("defaults_deps", defaultsDepsMutator).Parallel()
31 RegisterTopDownMutator("defaults", defaultsMutator).Parallel()
32
33 RegisterBottomUpMutator("arch", ArchMutator).Parallel()
34}
35
Colin Cross3f40fa42015-01-30 17:27:36 -080036var (
37 DeviceSharedLibrary = "shared_library"
38 DeviceStaticLibrary = "static_library"
39 DeviceExecutable = "executable"
40 HostSharedLibrary = "host_shared_library"
41 HostStaticLibrary = "host_static_library"
42 HostExecutable = "host_executable"
43)
44
Dan Willemsen34cc69e2015-09-23 15:26:20 -070045type ModuleBuildParams struct {
46 Rule blueprint.Rule
47 Output WritablePath
48 Outputs WritablePaths
49 Input Path
50 Inputs Paths
51 Implicit Path
52 Implicits Paths
53 OrderOnly Paths
54 Default bool
55 Args map[string]string
56}
57
Colin Crossf6566ed2015-03-24 11:13:38 -070058type androidBaseContext interface {
Colin Crossa1ad8d12016-06-01 17:09:44 -070059 Target() Target
Colin Crossf6566ed2015-03-24 11:13:38 -070060 Arch() Arch
Colin Crossa1ad8d12016-06-01 17:09:44 -070061 Os() OsType
Colin Crossf6566ed2015-03-24 11:13:38 -070062 Host() bool
63 Device() bool
Colin Cross0af4b842015-04-30 16:36:18 -070064 Darwin() bool
Colin Crossf6566ed2015-03-24 11:13:38 -070065 Debug() bool
Colin Cross1e7d3702016-08-24 15:25:47 -070066 PrimaryArch() bool
Colin Cross1332b002015-04-07 17:11:30 -070067 AConfig() Config
Colin Cross9272ade2016-08-17 15:24:12 -070068 DeviceConfig() DeviceConfig
Colin Crossf6566ed2015-03-24 11:13:38 -070069}
70
Colin Cross635c3b02016-05-18 15:37:25 -070071type BaseContext interface {
Colin Crossf6566ed2015-03-24 11:13:38 -070072 blueprint.BaseModuleContext
73 androidBaseContext
74}
75
Colin Cross635c3b02016-05-18 15:37:25 -070076type ModuleContext interface {
Colin Cross3f40fa42015-01-30 17:27:36 -080077 blueprint.ModuleContext
Colin Crossf6566ed2015-03-24 11:13:38 -070078 androidBaseContext
Colin Cross3f40fa42015-01-30 17:27:36 -080079
Dan Willemsen34cc69e2015-09-23 15:26:20 -070080 // Similar to Build, but takes Paths instead of []string,
81 // and performs more verification.
82 ModuleBuild(pctx blueprint.PackageContext, params ModuleBuildParams)
Colin Cross8f101b42015-06-17 15:09:06 -070083
Dan Willemsen34cc69e2015-09-23 15:26:20 -070084 ExpandSources(srcFiles, excludes []string) Paths
85 Glob(outDir, globPattern string, excludes []string) Paths
86
Colin Crossa2344662016-03-24 13:14:12 -070087 InstallFile(installPath OutputPath, srcPath Path, deps ...Path) OutputPath
88 InstallFileName(installPath OutputPath, name string, srcPath Path, deps ...Path) OutputPath
Colin Cross3854a602016-01-11 12:49:11 -080089 InstallSymlink(installPath OutputPath, name string, srcPath OutputPath) OutputPath
Dan Willemsen34cc69e2015-09-23 15:26:20 -070090 CheckbuildFile(srcPath Path)
Dan Willemsen6553f5e2016-03-10 18:14:25 -080091
92 AddMissingDependencies(deps []string)
Colin Cross8d8f8e22016-08-03 11:57:50 -070093
94 Proprietary() bool
95 InstallInData() bool
Colin Cross3f40fa42015-01-30 17:27:36 -080096}
97
Colin Cross635c3b02016-05-18 15:37:25 -070098type Module interface {
Colin Cross3f40fa42015-01-30 17:27:36 -080099 blueprint.Module
100
Colin Cross635c3b02016-05-18 15:37:25 -0700101 GenerateAndroidBuildActions(ModuleContext)
Colin Cross3f40fa42015-01-30 17:27:36 -0800102
Colin Cross635c3b02016-05-18 15:37:25 -0700103 base() *ModuleBase
Dan Willemsen0effe062015-11-30 16:06:01 -0800104 Enabled() bool
Colin Crossa1ad8d12016-06-01 17:09:44 -0700105 Target() Target
Dan Willemsen782a2d12015-12-21 14:55:28 -0800106 InstallInData() bool
Colin Cross3f40fa42015-01-30 17:27:36 -0800107}
108
Colin Cross3f40fa42015-01-30 17:27:36 -0800109type commonProperties struct {
Colin Crossc77f9d12015-04-02 13:54:39 -0700110 Name string
111 Deps []string
112 Tags []string
Colin Cross3f40fa42015-01-30 17:27:36 -0800113
Dan Willemsen0effe062015-11-30 16:06:01 -0800114 // emit build rules for this module
115 Enabled *bool `android:"arch_variant"`
Colin Cross3f40fa42015-01-30 17:27:36 -0800116
Colin Cross7d5136f2015-05-11 13:39:40 -0700117 // control whether this module compiles for 32-bit, 64-bit, or both. Possible values
Colin Cross3f40fa42015-01-30 17:27:36 -0800118 // are "32" (compile for 32-bit only), "64" (compile for 64-bit only), "both" (compile for both
119 // architectures), or "first" (compile for 64-bit on a 64-bit platform, and 32-bit on a 32-bit
120 // platform
121 Compile_multilib string
122
Dan Willemsen782a2d12015-12-21 14:55:28 -0800123 // whether this is a proprietary vendor module, and should be installed into /vendor
124 Proprietary bool
125
Dan Willemsen0fda89f2016-06-01 15:25:32 -0700126 // *.logtags files, to combine together in order to generate the /system/etc/event-log-tags
127 // file
128 Logtags []string
129
Dan Willemsen2277bcb2016-07-25 20:27:39 -0700130 // init.rc files to be installed if this module is installed
131 Init_rc []string
132
Chris Wolfe998306e2016-08-15 14:47:23 -0400133 // names of other modules to install if this module is installed
134 Required []string
135
Colin Crossa1ad8d12016-06-01 17:09:44 -0700136 // Set by TargetMutator
137 CompileTarget Target `blueprint:"mutated"`
Colin Cross3f40fa42015-01-30 17:27:36 -0800138
139 // Set by InitAndroidModule
140 HostOrDeviceSupported HostOrDeviceSupported `blueprint:"mutated"`
141}
142
143type hostAndDeviceProperties struct {
Colin Crossa4190c12016-07-12 13:11:25 -0700144 Host_supported *bool
145 Device_supported *bool
Colin Cross3f40fa42015-01-30 17:27:36 -0800146}
147
Colin Crossc472d572015-03-17 15:06:21 -0700148type Multilib string
149
150const (
Dan Willemsen218f6562015-07-08 18:13:11 -0700151 MultilibBoth Multilib = "both"
152 MultilibFirst Multilib = "first"
153 MultilibCommon Multilib = "common"
154 MultilibDefault Multilib = ""
Colin Crossc472d572015-03-17 15:06:21 -0700155)
156
Colin Crossa1ad8d12016-06-01 17:09:44 -0700157type HostOrDeviceSupported int
158
159const (
160 _ HostOrDeviceSupported = iota
161 HostSupported
162 DeviceSupported
163 HostAndDeviceSupported
164 HostAndDeviceDefault
165)
166
Colin Cross635c3b02016-05-18 15:37:25 -0700167func InitAndroidModule(m Module,
Colin Cross3f40fa42015-01-30 17:27:36 -0800168 propertyStructs ...interface{}) (blueprint.Module, []interface{}) {
169
170 base := m.base()
171 base.module = m
Colin Cross5049f022015-03-18 13:28:46 -0700172
Colin Cross7f64b6d2015-07-09 13:57:48 -0700173 propertyStructs = append(propertyStructs, &base.commonProperties, &base.variableProperties)
Colin Cross5049f022015-03-18 13:28:46 -0700174
175 return m, propertyStructs
176}
177
Colin Cross635c3b02016-05-18 15:37:25 -0700178func InitAndroidArchModule(m Module, hod HostOrDeviceSupported, defaultMultilib Multilib,
Colin Cross5049f022015-03-18 13:28:46 -0700179 propertyStructs ...interface{}) (blueprint.Module, []interface{}) {
180
181 _, propertyStructs = InitAndroidModule(m, propertyStructs...)
182
183 base := m.base()
Colin Cross3f40fa42015-01-30 17:27:36 -0800184 base.commonProperties.HostOrDeviceSupported = hod
Colin Crosscfad1192015-11-02 16:43:11 -0800185 base.commonProperties.Compile_multilib = string(defaultMultilib)
Colin Cross3f40fa42015-01-30 17:27:36 -0800186
Dan Willemsen218f6562015-07-08 18:13:11 -0700187 switch hod {
188 case HostAndDeviceSupported:
Colin Cross3f40fa42015-01-30 17:27:36 -0800189 // Default to module to device supported, host not supported, can override in module
190 // properties
Colin Crossa4190c12016-07-12 13:11:25 -0700191 base.hostAndDeviceProperties.Device_supported = boolPtr(true)
Dan Willemsen218f6562015-07-08 18:13:11 -0700192 fallthrough
193 case HostAndDeviceDefault:
Colin Cross3f40fa42015-01-30 17:27:36 -0800194 propertyStructs = append(propertyStructs, &base.hostAndDeviceProperties)
195 }
196
Colin Crosscfad1192015-11-02 16:43:11 -0800197 return InitArchModule(m, propertyStructs...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800198}
199
Colin Crossa120ec12016-08-19 16:07:38 -0700200func AddCustomizer(m blueprint.Module, c PropertyCustomizer) {
201 base := m.(Module).base()
202 base.customizers = append(base.customizers, c)
203}
204
Colin Cross3f40fa42015-01-30 17:27:36 -0800205// A AndroidModuleBase object contains the properties that are common to all Android
206// modules. It should be included as an anonymous field in every module
207// struct definition. InitAndroidModule should then be called from the module's
208// factory function, and the return values from InitAndroidModule should be
209// returned from the factory function.
210//
211// The AndroidModuleBase type is responsible for implementing the
212// GenerateBuildActions method to support the blueprint.Module interface. This
213// method will then call the module's GenerateAndroidBuildActions method once
214// for each build variant that is to be built. GenerateAndroidBuildActions is
215// passed a AndroidModuleContext rather than the usual blueprint.ModuleContext.
216// AndroidModuleContext exposes extra functionality specific to the Android build
217// system including details about the particular build variant that is to be
218// generated.
219//
220// For example:
221//
222// import (
223// "android/soong/common"
Colin Cross70b40592015-03-23 12:57:34 -0700224// "github.com/google/blueprint"
Colin Cross3f40fa42015-01-30 17:27:36 -0800225// )
226//
227// type myModule struct {
228// common.AndroidModuleBase
229// properties struct {
230// MyProperty string
231// }
232// }
233//
234// func NewMyModule() (blueprint.Module, []interface{}) {
235// m := &myModule{}
236// return common.InitAndroidModule(m, &m.properties)
237// }
238//
239// func (m *myModule) GenerateAndroidBuildActions(ctx common.AndroidModuleContext) {
240// // Get the CPU architecture for the current build variant.
241// variantArch := ctx.Arch()
242//
243// // ...
244// }
Colin Cross635c3b02016-05-18 15:37:25 -0700245type ModuleBase struct {
Colin Cross3f40fa42015-01-30 17:27:36 -0800246 // Putting the curiously recurring thing pointing to the thing that contains
247 // the thing pattern to good use.
Colin Cross635c3b02016-05-18 15:37:25 -0700248 module Module
Colin Cross3f40fa42015-01-30 17:27:36 -0800249
250 commonProperties commonProperties
Colin Cross7f64b6d2015-07-09 13:57:48 -0700251 variableProperties variableProperties
Colin Cross3f40fa42015-01-30 17:27:36 -0800252 hostAndDeviceProperties hostAndDeviceProperties
253 generalProperties []interface{}
254 archProperties []*archProperties
Colin Crossa120ec12016-08-19 16:07:38 -0700255 customizableProperties []interface{}
Colin Cross3f40fa42015-01-30 17:27:36 -0800256
257 noAddressSanitizer bool
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700258 installFiles Paths
259 checkbuildFiles Paths
Colin Cross1f8c52b2015-06-16 16:38:17 -0700260
261 // Used by buildTargetSingleton to create checkbuild and per-directory build targets
262 // Only set on the final variant of each module
263 installTarget string
264 checkbuildTarget string
265 blueprintDir string
Colin Crossa120ec12016-08-19 16:07:38 -0700266
267 customizers []PropertyCustomizer
Colin Cross3f40fa42015-01-30 17:27:36 -0800268}
269
Colin Cross635c3b02016-05-18 15:37:25 -0700270func (a *ModuleBase) base() *ModuleBase {
Colin Cross3f40fa42015-01-30 17:27:36 -0800271 return a
272}
273
Colin Crossa1ad8d12016-06-01 17:09:44 -0700274func (a *ModuleBase) SetTarget(target Target) {
275 a.commonProperties.CompileTarget = target
Colin Crossd3ba0392015-05-07 14:11:29 -0700276}
277
Colin Crossa1ad8d12016-06-01 17:09:44 -0700278func (a *ModuleBase) Target() Target {
279 return a.commonProperties.CompileTarget
Dan Willemsen490fd492015-11-24 17:53:15 -0800280}
281
Colin Crossa1ad8d12016-06-01 17:09:44 -0700282func (a *ModuleBase) Os() OsType {
283 return a.Target().Os
Dan Willemsen490fd492015-11-24 17:53:15 -0800284}
285
Colin Cross635c3b02016-05-18 15:37:25 -0700286func (a *ModuleBase) Host() bool {
Colin Crossa1ad8d12016-06-01 17:09:44 -0700287 return a.Os().Class == Host || a.Os().Class == HostCross
Dan Willemsen97750522016-02-09 17:43:51 -0800288}
289
Colin Cross635c3b02016-05-18 15:37:25 -0700290func (a *ModuleBase) Arch() Arch {
Colin Crossa1ad8d12016-06-01 17:09:44 -0700291 return a.Target().Arch
Dan Willemsen97750522016-02-09 17:43:51 -0800292}
293
Colin Crossa1ad8d12016-06-01 17:09:44 -0700294func (a *ModuleBase) OsClassSupported() []OsClass {
295 switch a.commonProperties.HostOrDeviceSupported {
296 case HostSupported:
297 // TODO(ccross): explicitly mark host cross support
298 return []OsClass{Host, HostCross}
299 case DeviceSupported:
300 return []OsClass{Device}
301 case HostAndDeviceSupported:
302 var supported []OsClass
Colin Crossa4190c12016-07-12 13:11:25 -0700303 if Bool(a.hostAndDeviceProperties.Host_supported) {
Colin Crossa1ad8d12016-06-01 17:09:44 -0700304 supported = append(supported, Host, HostCross)
305 }
Colin Crossa4190c12016-07-12 13:11:25 -0700306 if Bool(a.hostAndDeviceProperties.Device_supported) {
Colin Crossa1ad8d12016-06-01 17:09:44 -0700307 supported = append(supported, Device)
308 }
309 return supported
310 default:
311 return nil
312 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800313}
314
Colin Cross635c3b02016-05-18 15:37:25 -0700315func (a *ModuleBase) DeviceSupported() bool {
Colin Cross3f40fa42015-01-30 17:27:36 -0800316 return a.commonProperties.HostOrDeviceSupported == DeviceSupported ||
317 a.commonProperties.HostOrDeviceSupported == HostAndDeviceSupported &&
Colin Crossa4190c12016-07-12 13:11:25 -0700318 Bool(a.hostAndDeviceProperties.Device_supported)
Colin Cross3f40fa42015-01-30 17:27:36 -0800319}
320
Colin Cross635c3b02016-05-18 15:37:25 -0700321func (a *ModuleBase) Enabled() bool {
Dan Willemsen0effe062015-11-30 16:06:01 -0800322 if a.commonProperties.Enabled == nil {
Colin Crossa1ad8d12016-06-01 17:09:44 -0700323 return a.Os().Class != HostCross
Dan Willemsen490fd492015-11-24 17:53:15 -0800324 }
Dan Willemsen0effe062015-11-30 16:06:01 -0800325 return *a.commonProperties.Enabled
Colin Cross3f40fa42015-01-30 17:27:36 -0800326}
327
Colin Cross635c3b02016-05-18 15:37:25 -0700328func (a *ModuleBase) computeInstallDeps(
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700329 ctx blueprint.ModuleContext) Paths {
Colin Cross3f40fa42015-01-30 17:27:36 -0800330
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700331 result := Paths{}
Colin Cross3f40fa42015-01-30 17:27:36 -0800332 ctx.VisitDepsDepthFirstIf(isFileInstaller,
333 func(m blueprint.Module) {
334 fileInstaller := m.(fileInstaller)
335 files := fileInstaller.filesToInstall()
336 result = append(result, files...)
337 })
338
339 return result
340}
341
Colin Cross635c3b02016-05-18 15:37:25 -0700342func (a *ModuleBase) filesToInstall() Paths {
Colin Cross3f40fa42015-01-30 17:27:36 -0800343 return a.installFiles
344}
345
Colin Cross635c3b02016-05-18 15:37:25 -0700346func (p *ModuleBase) NoAddressSanitizer() bool {
Colin Cross3f40fa42015-01-30 17:27:36 -0800347 return p.noAddressSanitizer
348}
349
Colin Cross635c3b02016-05-18 15:37:25 -0700350func (p *ModuleBase) InstallInData() bool {
Dan Willemsen782a2d12015-12-21 14:55:28 -0800351 return false
352}
353
Colin Cross635c3b02016-05-18 15:37:25 -0700354func (a *ModuleBase) generateModuleTarget(ctx blueprint.ModuleContext) {
355 if a != ctx.FinalModule().(Module).base() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800356 return
357 }
358
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700359 allInstalledFiles := Paths{}
360 allCheckbuildFiles := Paths{}
Colin Cross3f40fa42015-01-30 17:27:36 -0800361 ctx.VisitAllModuleVariants(func(module blueprint.Module) {
Colin Cross635c3b02016-05-18 15:37:25 -0700362 a := module.(Module).base()
Colin Crossc9404352015-03-26 16:10:12 -0700363 allInstalledFiles = append(allInstalledFiles, a.installFiles...)
364 allCheckbuildFiles = append(allCheckbuildFiles, a.checkbuildFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800365 })
366
Colin Cross9454bfa2015-03-17 13:24:18 -0700367 deps := []string{}
368
Colin Cross3f40fa42015-01-30 17:27:36 -0800369 if len(allInstalledFiles) > 0 {
Colin Cross9454bfa2015-03-17 13:24:18 -0700370 name := ctx.ModuleName() + "-install"
Colin Cross3f40fa42015-01-30 17:27:36 -0800371 ctx.Build(pctx, blueprint.BuildParams{
Colin Cross9454bfa2015-03-17 13:24:18 -0700372 Rule: blueprint.Phony,
373 Outputs: []string{name},
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700374 Implicits: allInstalledFiles.Strings(),
Colin Cross346aa132015-12-17 17:19:51 -0800375 Optional: ctx.Config().(Config).EmbeddedInMake(),
Colin Cross9454bfa2015-03-17 13:24:18 -0700376 })
377 deps = append(deps, name)
Colin Cross1f8c52b2015-06-16 16:38:17 -0700378 a.installTarget = name
Colin Cross9454bfa2015-03-17 13:24:18 -0700379 }
380
381 if len(allCheckbuildFiles) > 0 {
382 name := ctx.ModuleName() + "-checkbuild"
383 ctx.Build(pctx, blueprint.BuildParams{
384 Rule: blueprint.Phony,
385 Outputs: []string{name},
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700386 Implicits: allCheckbuildFiles.Strings(),
Colin Cross9454bfa2015-03-17 13:24:18 -0700387 Optional: true,
388 })
389 deps = append(deps, name)
Colin Cross1f8c52b2015-06-16 16:38:17 -0700390 a.checkbuildTarget = name
Colin Cross9454bfa2015-03-17 13:24:18 -0700391 }
392
393 if len(deps) > 0 {
Dan Willemsen5ba07e82015-12-11 13:51:06 -0800394 suffix := ""
395 if ctx.Config().(Config).EmbeddedInMake() {
396 suffix = "-soong"
397 }
398
Colin Cross9454bfa2015-03-17 13:24:18 -0700399 ctx.Build(pctx, blueprint.BuildParams{
400 Rule: blueprint.Phony,
Dan Willemsen5ba07e82015-12-11 13:51:06 -0800401 Outputs: []string{ctx.ModuleName() + suffix},
Colin Cross9454bfa2015-03-17 13:24:18 -0700402 Implicits: deps,
403 Optional: true,
Colin Cross3f40fa42015-01-30 17:27:36 -0800404 })
Colin Cross1f8c52b2015-06-16 16:38:17 -0700405
406 a.blueprintDir = ctx.ModuleDir()
Colin Cross3f40fa42015-01-30 17:27:36 -0800407 }
408}
409
Colin Cross635c3b02016-05-18 15:37:25 -0700410func (a *ModuleBase) androidBaseContextFactory(ctx blueprint.BaseModuleContext) androidBaseContextImpl {
Colin Cross6362e272015-10-29 15:25:03 -0700411 return androidBaseContextImpl{
Colin Cross8d8f8e22016-08-03 11:57:50 -0700412 target: a.commonProperties.CompileTarget,
413 config: ctx.Config().(Config),
Colin Cross3f40fa42015-01-30 17:27:36 -0800414 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800415}
416
Colin Cross635c3b02016-05-18 15:37:25 -0700417func (a *ModuleBase) GenerateBuildActions(ctx blueprint.ModuleContext) {
Colin Cross3f40fa42015-01-30 17:27:36 -0800418 androidCtx := &androidModuleContext{
Colin Cross8d8f8e22016-08-03 11:57:50 -0700419 module: a.module,
Colin Cross6362e272015-10-29 15:25:03 -0700420 ModuleContext: ctx,
421 androidBaseContextImpl: a.androidBaseContextFactory(ctx),
422 installDeps: a.computeInstallDeps(ctx),
423 installFiles: a.installFiles,
Colin Cross6ff51382015-12-17 16:39:19 -0800424 missingDeps: ctx.GetMissingDependencies(),
Colin Cross3f40fa42015-01-30 17:27:36 -0800425 }
426
Dan Willemsen0effe062015-11-30 16:06:01 -0800427 if !a.Enabled() {
Colin Cross3f40fa42015-01-30 17:27:36 -0800428 return
429 }
430
431 a.module.GenerateAndroidBuildActions(androidCtx)
432 if ctx.Failed() {
433 return
434 }
435
Colin Crossc9404352015-03-26 16:10:12 -0700436 a.installFiles = append(a.installFiles, androidCtx.installFiles...)
437 a.checkbuildFiles = append(a.checkbuildFiles, androidCtx.checkbuildFiles...)
438
Colin Cross3f40fa42015-01-30 17:27:36 -0800439 a.generateModuleTarget(ctx)
440 if ctx.Failed() {
441 return
442 }
443}
444
Colin Crossf6566ed2015-03-24 11:13:38 -0700445type androidBaseContextImpl struct {
Colin Cross8d8f8e22016-08-03 11:57:50 -0700446 target Target
447 debug bool
448 config Config
Colin Crossf6566ed2015-03-24 11:13:38 -0700449}
450
Colin Cross3f40fa42015-01-30 17:27:36 -0800451type androidModuleContext struct {
452 blueprint.ModuleContext
Colin Crossf6566ed2015-03-24 11:13:38 -0700453 androidBaseContextImpl
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700454 installDeps Paths
455 installFiles Paths
456 checkbuildFiles Paths
Colin Cross6ff51382015-12-17 16:39:19 -0800457 missingDeps []string
Colin Cross8d8f8e22016-08-03 11:57:50 -0700458 module Module
Colin Cross6ff51382015-12-17 16:39:19 -0800459}
460
461func (a *androidModuleContext) ninjaError(outputs []string, err error) {
462 a.ModuleContext.Build(pctx, blueprint.BuildParams{
463 Rule: ErrorRule,
464 Outputs: outputs,
465 Optional: true,
466 Args: map[string]string{
467 "error": err.Error(),
468 },
469 })
470 return
Colin Cross3f40fa42015-01-30 17:27:36 -0800471}
472
Dan Willemsen14e5c2a2015-11-30 13:59:34 -0800473func (a *androidModuleContext) Build(pctx blueprint.PackageContext, params blueprint.BuildParams) {
Colin Crosse2c48742016-04-27 13:47:35 -0700474 if a.missingDeps != nil && params.Rule != globRule {
Colin Cross6ff51382015-12-17 16:39:19 -0800475 a.ninjaError(params.Outputs, fmt.Errorf("module %s missing dependencies: %s\n",
476 a.ModuleName(), strings.Join(a.missingDeps, ", ")))
477 return
478 }
479
Colin Cross3f40fa42015-01-30 17:27:36 -0800480 params.Optional = true
481 a.ModuleContext.Build(pctx, params)
482}
483
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700484func (a *androidModuleContext) ModuleBuild(pctx blueprint.PackageContext, params ModuleBuildParams) {
485 bparams := blueprint.BuildParams{
486 Rule: params.Rule,
487 Outputs: params.Outputs.Strings(),
488 Inputs: params.Inputs.Strings(),
489 Implicits: params.Implicits.Strings(),
490 OrderOnly: params.OrderOnly.Strings(),
491 Args: params.Args,
492 Optional: !params.Default,
493 }
494
495 if params.Output != nil {
496 bparams.Outputs = append(bparams.Outputs, params.Output.String())
497 }
498 if params.Input != nil {
499 bparams.Inputs = append(bparams.Inputs, params.Input.String())
500 }
501 if params.Implicit != nil {
502 bparams.Implicits = append(bparams.Implicits, params.Implicit.String())
503 }
504
Colin Cross6ff51382015-12-17 16:39:19 -0800505 if a.missingDeps != nil {
506 a.ninjaError(bparams.Outputs, fmt.Errorf("module %s missing dependencies: %s\n",
507 a.ModuleName(), strings.Join(a.missingDeps, ", ")))
508 return
509 }
510
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700511 a.ModuleContext.Build(pctx, bparams)
512}
513
Colin Cross6ff51382015-12-17 16:39:19 -0800514func (a *androidModuleContext) GetMissingDependencies() []string {
515 return a.missingDeps
516}
517
Dan Willemsen6553f5e2016-03-10 18:14:25 -0800518func (a *androidModuleContext) AddMissingDependencies(deps []string) {
519 if deps != nil {
520 a.missingDeps = append(a.missingDeps, deps...)
521 }
522}
523
Colin Crossa1ad8d12016-06-01 17:09:44 -0700524func (a *androidBaseContextImpl) Target() Target {
525 return a.target
526}
527
Colin Crossf6566ed2015-03-24 11:13:38 -0700528func (a *androidBaseContextImpl) Arch() Arch {
Colin Crossa1ad8d12016-06-01 17:09:44 -0700529 return a.target.Arch
Colin Cross3f40fa42015-01-30 17:27:36 -0800530}
531
Colin Crossa1ad8d12016-06-01 17:09:44 -0700532func (a *androidBaseContextImpl) Os() OsType {
533 return a.target.Os
Dan Willemsen490fd492015-11-24 17:53:15 -0800534}
535
Colin Crossf6566ed2015-03-24 11:13:38 -0700536func (a *androidBaseContextImpl) Host() bool {
Colin Crossa1ad8d12016-06-01 17:09:44 -0700537 return a.target.Os.Class == Host || a.target.Os.Class == HostCross
Colin Crossf6566ed2015-03-24 11:13:38 -0700538}
539
540func (a *androidBaseContextImpl) Device() bool {
Colin Crossa1ad8d12016-06-01 17:09:44 -0700541 return a.target.Os.Class == Device
Colin Crossf6566ed2015-03-24 11:13:38 -0700542}
543
Colin Cross0af4b842015-04-30 16:36:18 -0700544func (a *androidBaseContextImpl) Darwin() bool {
Colin Crossa1ad8d12016-06-01 17:09:44 -0700545 return a.target.Os == Darwin
Colin Cross0af4b842015-04-30 16:36:18 -0700546}
547
Colin Crossf6566ed2015-03-24 11:13:38 -0700548func (a *androidBaseContextImpl) Debug() bool {
549 return a.debug
550}
551
Colin Cross1e7d3702016-08-24 15:25:47 -0700552func (a *androidBaseContextImpl) PrimaryArch() bool {
553 return a.target.Arch.ArchType == a.config.Targets[a.target.Os.Class][0].Arch.ArchType
554}
555
Colin Cross1332b002015-04-07 17:11:30 -0700556func (a *androidBaseContextImpl) AConfig() Config {
557 return a.config
558}
559
Colin Cross9272ade2016-08-17 15:24:12 -0700560func (a *androidBaseContextImpl) DeviceConfig() DeviceConfig {
561 return DeviceConfig{a.config.deviceConfig}
562}
563
Colin Cross8d8f8e22016-08-03 11:57:50 -0700564func (a *androidModuleContext) Proprietary() bool {
565 return a.module.base().commonProperties.Proprietary
Dan Willemsen782a2d12015-12-21 14:55:28 -0800566}
567
Colin Cross8d8f8e22016-08-03 11:57:50 -0700568func (a *androidModuleContext) InstallInData() bool {
569 return a.module.InstallInData()
Dan Willemsen782a2d12015-12-21 14:55:28 -0800570}
571
572func (a *androidModuleContext) InstallFileName(installPath OutputPath, name string, srcPath Path,
Colin Crossa2344662016-03-24 13:14:12 -0700573 deps ...Path) OutputPath {
Colin Cross35cec122015-04-02 14:37:16 -0700574
Dan Willemsen782a2d12015-12-21 14:55:28 -0800575 fullInstallPath := installPath.Join(a, name)
Colin Cross3f40fa42015-01-30 17:27:36 -0800576
Dan Willemsen7f730fd2016-01-14 11:22:23 -0800577 if a.Host() || !a.AConfig().SkipDeviceInstall() {
Dan Willemsen322acaf2016-01-12 23:07:05 -0800578 deps = append(deps, a.installDeps...)
Colin Cross35cec122015-04-02 14:37:16 -0700579
Dan Willemsen322acaf2016-01-12 23:07:05 -0800580 a.ModuleBuild(pctx, ModuleBuildParams{
581 Rule: Cp,
582 Output: fullInstallPath,
583 Input: srcPath,
584 OrderOnly: Paths(deps),
Dan Willemsen7f730fd2016-01-14 11:22:23 -0800585 Default: !a.AConfig().EmbeddedInMake(),
Dan Willemsen322acaf2016-01-12 23:07:05 -0800586 })
Colin Cross3f40fa42015-01-30 17:27:36 -0800587
Dan Willemsen322acaf2016-01-12 23:07:05 -0800588 a.installFiles = append(a.installFiles, fullInstallPath)
589 }
Colin Cross1f8c52b2015-06-16 16:38:17 -0700590 a.checkbuildFiles = append(a.checkbuildFiles, srcPath)
Colin Cross35cec122015-04-02 14:37:16 -0700591 return fullInstallPath
592}
593
Colin Crossa2344662016-03-24 13:14:12 -0700594func (a *androidModuleContext) InstallFile(installPath OutputPath, srcPath Path, deps ...Path) OutputPath {
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700595 return a.InstallFileName(installPath, filepath.Base(srcPath.String()), srcPath, deps...)
Colin Cross3f40fa42015-01-30 17:27:36 -0800596}
597
Colin Cross3854a602016-01-11 12:49:11 -0800598func (a *androidModuleContext) InstallSymlink(installPath OutputPath, name string, srcPath OutputPath) OutputPath {
599 fullInstallPath := installPath.Join(a, name)
600
Colin Cross12fc4972016-01-11 12:49:11 -0800601 if a.Host() || !a.AConfig().SkipDeviceInstall() {
602 a.ModuleBuild(pctx, ModuleBuildParams{
603 Rule: Symlink,
604 Output: fullInstallPath,
605 OrderOnly: Paths{srcPath},
606 Default: !a.AConfig().EmbeddedInMake(),
607 Args: map[string]string{
608 "fromPath": srcPath.String(),
609 },
610 })
Colin Cross3854a602016-01-11 12:49:11 -0800611
Colin Cross12fc4972016-01-11 12:49:11 -0800612 a.installFiles = append(a.installFiles, fullInstallPath)
613 a.checkbuildFiles = append(a.checkbuildFiles, srcPath)
614 }
Colin Cross3854a602016-01-11 12:49:11 -0800615 return fullInstallPath
616}
617
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700618func (a *androidModuleContext) CheckbuildFile(srcPath Path) {
Colin Cross3f40fa42015-01-30 17:27:36 -0800619 a.checkbuildFiles = append(a.checkbuildFiles, srcPath)
620}
621
Colin Cross3f40fa42015-01-30 17:27:36 -0800622type fileInstaller interface {
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700623 filesToInstall() Paths
Colin Cross3f40fa42015-01-30 17:27:36 -0800624}
625
626func isFileInstaller(m blueprint.Module) bool {
627 _, ok := m.(fileInstaller)
628 return ok
629}
630
631func isAndroidModule(m blueprint.Module) bool {
Colin Cross635c3b02016-05-18 15:37:25 -0700632 _, ok := m.(Module)
Colin Cross3f40fa42015-01-30 17:27:36 -0800633 return ok
634}
Colin Crossfce53272015-04-08 11:21:40 -0700635
Dan Willemsen2ef08f42015-06-30 18:15:24 -0700636func findStringInSlice(str string, slice []string) int {
637 for i, s := range slice {
638 if s == str {
639 return i
Colin Crossfce53272015-04-08 11:21:40 -0700640 }
641 }
Dan Willemsen2ef08f42015-06-30 18:15:24 -0700642 return -1
643}
644
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700645func (ctx *androidModuleContext) ExpandSources(srcFiles, excludes []string) Paths {
646 prefix := PathForModuleSrc(ctx).String()
Dan Willemsen2ef08f42015-06-30 18:15:24 -0700647 for i, e := range excludes {
648 j := findStringInSlice(e, srcFiles)
649 if j != -1 {
650 srcFiles = append(srcFiles[:j], srcFiles[j+1:]...)
651 }
652
653 excludes[i] = filepath.Join(prefix, e)
654 }
655
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700656 globbedSrcFiles := make(Paths, 0, len(srcFiles))
Colin Cross8f101b42015-06-17 15:09:06 -0700657 for _, s := range srcFiles {
Dan Willemsen2ef08f42015-06-30 18:15:24 -0700658 if glob.IsGlob(s) {
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700659 globbedSrcFiles = append(globbedSrcFiles, ctx.Glob("src_glob", filepath.Join(prefix, s), excludes)...)
Colin Cross8f101b42015-06-17 15:09:06 -0700660 } else {
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700661 globbedSrcFiles = append(globbedSrcFiles, PathForModuleSrc(ctx, s))
Colin Cross8f101b42015-06-17 15:09:06 -0700662 }
663 }
664
665 return globbedSrcFiles
666}
667
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700668func (ctx *androidModuleContext) Glob(outDir, globPattern string, excludes []string) Paths {
669 ret, err := Glob(ctx, PathForModuleOut(ctx, outDir).String(), globPattern, excludes)
Colin Cross8f101b42015-06-17 15:09:06 -0700670 if err != nil {
671 ctx.ModuleErrorf("glob: %s", err.Error())
672 }
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700673 return pathsForModuleSrcFromFullPath(ctx, ret)
Colin Crossfce53272015-04-08 11:21:40 -0700674}
Colin Cross1f8c52b2015-06-16 16:38:17 -0700675
Colin Cross463a90e2015-06-17 14:20:06 -0700676func init() {
677 soong.RegisterSingletonType("buildtarget", BuildTargetSingleton)
678}
679
Colin Cross1f8c52b2015-06-16 16:38:17 -0700680func BuildTargetSingleton() blueprint.Singleton {
681 return &buildTargetSingleton{}
682}
683
684type buildTargetSingleton struct{}
685
686func (c *buildTargetSingleton) GenerateBuildActions(ctx blueprint.SingletonContext) {
687 checkbuildDeps := []string{}
688
689 dirModules := make(map[string][]string)
690
691 ctx.VisitAllModules(func(module blueprint.Module) {
Colin Cross635c3b02016-05-18 15:37:25 -0700692 if a, ok := module.(Module); ok {
Colin Cross1f8c52b2015-06-16 16:38:17 -0700693 blueprintDir := a.base().blueprintDir
694 installTarget := a.base().installTarget
695 checkbuildTarget := a.base().checkbuildTarget
696
697 if checkbuildTarget != "" {
698 checkbuildDeps = append(checkbuildDeps, checkbuildTarget)
699 dirModules[blueprintDir] = append(dirModules[blueprintDir], checkbuildTarget)
700 }
701
702 if installTarget != "" {
703 dirModules[blueprintDir] = append(dirModules[blueprintDir], installTarget)
704 }
705 }
706 })
707
Dan Willemsen5ba07e82015-12-11 13:51:06 -0800708 suffix := ""
709 if ctx.Config().(Config).EmbeddedInMake() {
710 suffix = "-soong"
711 }
712
Colin Cross1f8c52b2015-06-16 16:38:17 -0700713 // Create a top-level checkbuild target that depends on all modules
714 ctx.Build(pctx, blueprint.BuildParams{
715 Rule: blueprint.Phony,
Dan Willemsen5ba07e82015-12-11 13:51:06 -0800716 Outputs: []string{"checkbuild" + suffix},
Colin Cross1f8c52b2015-06-16 16:38:17 -0700717 Implicits: checkbuildDeps,
Dan Willemsen218f6562015-07-08 18:13:11 -0700718 Optional: true,
Colin Cross1f8c52b2015-06-16 16:38:17 -0700719 })
720
721 // Create a mm/<directory> target that depends on all modules in a directory
722 dirs := sortedKeys(dirModules)
723 for _, dir := range dirs {
724 ctx.Build(pctx, blueprint.BuildParams{
725 Rule: blueprint.Phony,
726 Outputs: []string{filepath.Join("mm", dir)},
727 Implicits: dirModules[dir],
Dan Willemsen5ba07e82015-12-11 13:51:06 -0800728 // HACK: checkbuild should be an optional build, but force it
729 // enabled for now in standalone builds
Colin Cross1604ecf2015-12-17 16:33:43 -0800730 Optional: ctx.Config().(Config).EmbeddedInMake(),
Colin Cross1f8c52b2015-06-16 16:38:17 -0700731 })
732 }
733}
Colin Crossd779da42015-12-17 18:00:23 -0800734
735type AndroidModulesByName struct {
Colin Cross635c3b02016-05-18 15:37:25 -0700736 slice []Module
Colin Crossd779da42015-12-17 18:00:23 -0800737 ctx interface {
738 ModuleName(blueprint.Module) string
739 ModuleSubDir(blueprint.Module) string
740 }
741}
742
743func (s AndroidModulesByName) Len() int { return len(s.slice) }
744func (s AndroidModulesByName) Less(i, j int) bool {
745 mi, mj := s.slice[i], s.slice[j]
746 ni, nj := s.ctx.ModuleName(mi), s.ctx.ModuleName(mj)
747
748 if ni != nj {
749 return ni < nj
750 } else {
751 return s.ctx.ModuleSubDir(mi) < s.ctx.ModuleSubDir(mj)
752 }
753}
754func (s AndroidModulesByName) Swap(i, j int) { s.slice[i], s.slice[j] = s.slice[j], s.slice[i] }