blob: a488c0d7766760b42b7d65a97c8d7afd45873b47 [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 Cross988414c2020-01-11 01:11:46 +000019 "os"
Alex Lightfb4353d2019-01-17 13:57:45 -080020 "path"
Colin Cross3f40fa42015-01-30 17:27:36 -080021 "path/filepath"
Jiyong Park1c7e9622020-05-07 16:12:13 +090022 "regexp"
Colin Cross6ff51382015-12-17 16:39:19 -080023 "strings"
Colin Crossaabf6792017-11-29 00:27:14 -080024 "text/scanner"
Colin Crossf6566ed2015-03-24 11:13:38 -070025
26 "github.com/google/blueprint"
Colin Crossfe4bc362018-09-12 10:02:13 -070027 "github.com/google/blueprint/proptools"
Colin Cross3f40fa42015-01-30 17:27:36 -080028)
29
30var (
31 DeviceSharedLibrary = "shared_library"
32 DeviceStaticLibrary = "static_library"
33 DeviceExecutable = "executable"
34 HostSharedLibrary = "host_shared_library"
35 HostStaticLibrary = "host_static_library"
36 HostExecutable = "host_executable"
37)
38
Colin Crossae887032017-10-23 17:16:14 -070039type BuildParams struct {
Dan Willemsen9f3c5742016-11-03 14:28:31 -070040 Rule blueprint.Rule
Colin Cross33bfb0a2016-11-21 17:23:08 -080041 Deps blueprint.Deps
42 Depfile WritablePath
Colin Cross67a5c132017-05-09 13:45:28 -070043 Description string
Dan Willemsen9f3c5742016-11-03 14:28:31 -070044 Output WritablePath
45 Outputs WritablePaths
46 ImplicitOutput WritablePath
47 ImplicitOutputs WritablePaths
48 Input Path
49 Inputs Paths
50 Implicit Path
51 Implicits Paths
52 OrderOnly Paths
53 Default bool
54 Args map[string]string
Dan Willemsen34cc69e2015-09-23 15:26:20 -070055}
56
Colin Crossae887032017-10-23 17:16:14 -070057type ModuleBuildParams BuildParams
58
Colin Cross1184b642019-12-30 18:43:07 -080059// EarlyModuleContext provides methods that can be called early, as soon as the properties have
60// been parsed into the module and before any mutators have run.
61type EarlyModuleContext interface {
62 Module() Module
63 ModuleName() string
64 ModuleDir() string
65 ModuleType() string
Colin Cross9d34f352019-11-22 16:03:51 -080066 BlueprintsFile() string
Colin Cross1184b642019-12-30 18:43:07 -080067
68 ContainsProperty(name string) bool
69 Errorf(pos scanner.Position, fmt string, args ...interface{})
70 ModuleErrorf(fmt string, args ...interface{})
71 PropertyErrorf(property, fmt string, args ...interface{})
72 Failed() bool
73
74 AddNinjaFileDeps(deps ...string)
75
76 DeviceSpecific() bool
77 SocSpecific() bool
78 ProductSpecific() bool
79 SystemExtSpecific() bool
80 Platform() bool
81
82 Config() Config
83 DeviceConfig() DeviceConfig
84
85 // Deprecated: use Config()
86 AConfig() Config
87
88 // GlobWithDeps returns a list of files that match the specified pattern but do not match any
89 // of the patterns in excludes. It also adds efficient dependencies to rerun the primary
90 // builder whenever a file matching the pattern as added or removed, without rerunning if a
91 // file that does not match the pattern is added to a searched directory.
92 GlobWithDeps(pattern string, excludes []string) ([]string, error)
93
94 Glob(globPattern string, excludes []string) Paths
95 GlobFiles(globPattern string, excludes []string) Paths
Colin Cross988414c2020-01-11 01:11:46 +000096 IsSymlink(path Path) bool
97 Readlink(path Path) string
Colin Cross1184b642019-12-30 18:43:07 -080098}
99
Colin Cross0ea8ba82019-06-06 14:33:29 -0700100// BaseModuleContext is the same as blueprint.BaseModuleContext except that Config() returns
Colin Crossdc35e212019-06-06 16:13:11 -0700101// a Config instead of an interface{}, and some methods have been wrapped to use an android.Module
102// instead of a blueprint.Module, plus some extra methods that return Android-specific information
Colin Cross0ea8ba82019-06-06 14:33:29 -0700103// about the current module.
104type BaseModuleContext interface {
Colin Cross1184b642019-12-30 18:43:07 -0800105 EarlyModuleContext
Colin Cross0ea8ba82019-06-06 14:33:29 -0700106
Paul Duffinf88d8e02020-05-07 20:21:34 +0100107 blueprintBaseModuleContext() blueprint.BaseModuleContext
108
Colin Crossdc35e212019-06-06 16:13:11 -0700109 OtherModuleName(m blueprint.Module) string
110 OtherModuleDir(m blueprint.Module) string
111 OtherModuleErrorf(m blueprint.Module, fmt string, args ...interface{})
112 OtherModuleDependencyTag(m blueprint.Module) blueprint.DependencyTag
113 OtherModuleExists(name string) bool
Martin Stjernholm009a9dc2020-03-05 17:34:13 +0000114 OtherModuleDependencyVariantExists(variations []blueprint.Variation, name string) bool
115 OtherModuleReverseDependencyVariantExists(name string) bool
Jiyong Park9e6c2422019-08-09 20:39:45 +0900116 OtherModuleType(m blueprint.Module) string
Colin Crossdc35e212019-06-06 16:13:11 -0700117
118 GetDirectDepsWithTag(tag blueprint.DependencyTag) []Module
119 GetDirectDepWithTag(name string, tag blueprint.DependencyTag) blueprint.Module
120 GetDirectDep(name string) (blueprint.Module, blueprint.DependencyTag)
121
122 VisitDirectDepsBlueprint(visit func(blueprint.Module))
123 VisitDirectDeps(visit func(Module))
124 VisitDirectDepsWithTag(tag blueprint.DependencyTag, visit func(Module))
125 VisitDirectDepsIf(pred func(Module) bool, visit func(Module))
126 // Deprecated: use WalkDeps instead to support multiple dependency tags on the same module
127 VisitDepsDepthFirst(visit func(Module))
128 // Deprecated: use WalkDeps instead to support multiple dependency tags on the same module
129 VisitDepsDepthFirstIf(pred func(Module) bool, visit func(Module))
130 WalkDeps(visit func(Module, Module) bool)
131 WalkDepsBlueprint(visit func(blueprint.Module, blueprint.Module) bool)
132 // GetWalkPath is supposed to be called in visit function passed in WalkDeps()
133 // and returns a top-down dependency path from a start module to current child module.
134 GetWalkPath() []Module
135
Paul Duffinc5192442020-03-31 11:31:36 +0100136 // GetTagPath is supposed to be called in visit function passed in WalkDeps()
137 // and returns a top-down dependency tags path from a start module to current child module.
138 // It has one less entry than GetWalkPath() as it contains the dependency tags that
139 // exist between each adjacent pair of modules in the GetWalkPath().
140 // GetTagPath()[i] is the tag between GetWalkPath()[i] and GetWalkPath()[i+1]
141 GetTagPath() []blueprint.DependencyTag
142
Jiyong Park1c7e9622020-05-07 16:12:13 +0900143 // GetPathString is supposed to be called in visit function passed in WalkDeps()
144 // and returns a multi-line string showing the modules and dependency tags
145 // among them along the top-down dependency path from a start module to current child module.
146 // skipFirst when set to true, the output doesn't include the start module,
147 // which is already printed when this function is used along with ModuleErrorf().
148 GetPathString(skipFirst bool) string
149
Colin Crossdc35e212019-06-06 16:13:11 -0700150 AddMissingDependencies(missingDeps []string)
151
Colin Crossa1ad8d12016-06-01 17:09:44 -0700152 Target() Target
Colin Cross8b74d172016-09-13 09:59:14 -0700153 TargetPrimary() bool
Paul Duffinca7f0ef2020-02-25 15:50:49 +0000154
155 // The additional arch specific targets (e.g. 32/64 bit) that this module variant is
156 // responsible for creating.
Colin Crossee0bc3b2018-10-02 22:01:37 -0700157 MultiTargets() []Target
Colin Crossf6566ed2015-03-24 11:13:38 -0700158 Arch() Arch
Colin Crossa1ad8d12016-06-01 17:09:44 -0700159 Os() OsType
Colin Crossf6566ed2015-03-24 11:13:38 -0700160 Host() bool
161 Device() bool
Colin Cross0af4b842015-04-30 16:36:18 -0700162 Darwin() bool
Doug Horn21b94272019-01-16 12:06:11 -0800163 Fuchsia() bool
Colin Cross3edeee12017-04-04 12:59:48 -0700164 Windows() bool
Colin Crossf6566ed2015-03-24 11:13:38 -0700165 Debug() bool
Colin Cross1e7d3702016-08-24 15:25:47 -0700166 PrimaryArch() bool
Colin Crossf6566ed2015-03-24 11:13:38 -0700167}
168
Colin Cross1184b642019-12-30 18:43:07 -0800169// Deprecated: use EarlyModuleContext instead
Colin Cross635c3b02016-05-18 15:37:25 -0700170type BaseContext interface {
Colin Cross1184b642019-12-30 18:43:07 -0800171 EarlyModuleContext
Colin Crossaabf6792017-11-29 00:27:14 -0800172}
173
Colin Cross635c3b02016-05-18 15:37:25 -0700174type ModuleContext interface {
Colin Crossaabf6792017-11-29 00:27:14 -0800175 BaseModuleContext
Colin Cross3f40fa42015-01-30 17:27:36 -0800176
Colin Crossae887032017-10-23 17:16:14 -0700177 // Deprecated: use ModuleContext.Build instead.
Colin Cross0875c522017-11-28 17:34:01 -0800178 ModuleBuild(pctx PackageContext, params ModuleBuildParams)
Colin Cross8f101b42015-06-17 15:09:06 -0700179
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700180 ExpandSources(srcFiles, excludes []string) Paths
Colin Cross366938f2017-12-11 16:29:02 -0800181 ExpandSource(srcFile, prop string) Path
Colin Cross2383f3b2018-02-06 14:40:13 -0800182 ExpandOptionalSource(srcFile *string, prop string) OptionalPath
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700183
Colin Cross70dda7e2019-10-01 22:05:35 -0700184 InstallExecutable(installPath InstallPath, name string, srcPath Path, deps ...Path) InstallPath
185 InstallFile(installPath InstallPath, name string, srcPath Path, deps ...Path) InstallPath
186 InstallSymlink(installPath InstallPath, name string, srcPath InstallPath) InstallPath
187 InstallAbsoluteSymlink(installPath InstallPath, name string, absPath string) InstallPath
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700188 CheckbuildFile(srcPath Path)
Dan Willemsen6553f5e2016-03-10 18:14:25 -0800189
Colin Cross8d8f8e22016-08-03 11:57:50 -0700190 InstallInData() bool
Jaewoong Jung0949f312019-09-11 10:25:18 -0700191 InstallInTestcases() bool
Vishwath Mohan1dd88392017-03-29 22:00:18 -0700192 InstallInSanitizerDir() bool
Yifan Hong1b3348d2020-01-21 15:53:22 -0800193 InstallInRamdisk() bool
Jiyong Parkf9332f12018-02-01 00:54:12 +0900194 InstallInRecovery() bool
Colin Cross90ba5f42019-10-02 11:10:58 -0700195 InstallInRoot() bool
Colin Cross607d8582019-07-29 16:44:46 -0700196 InstallBypassMake() bool
Colin Cross6e359402020-02-10 15:29:54 -0800197 InstallForceOS() *OsType
Nan Zhang6d34b302017-02-04 17:47:46 -0800198
199 RequiredModuleNames() []string
Sasha Smundakb6d23052019-04-01 18:37:36 -0700200 HostRequiredModuleNames() []string
201 TargetRequiredModuleNames() []string
Colin Cross3f68a132017-10-23 17:10:29 -0700202
Colin Cross3f68a132017-10-23 17:10:29 -0700203 ModuleSubDir() string
204
Colin Cross0875c522017-11-28 17:34:01 -0800205 Variable(pctx PackageContext, name, value string)
206 Rule(pctx PackageContext, name string, params blueprint.RuleParams, argNames ...string) blueprint.Rule
Colin Crossae887032017-10-23 17:16:14 -0700207 // Similar to blueprint.ModuleContext.Build, but takes Paths instead of []string,
208 // and performs more verification.
Colin Cross0875c522017-11-28 17:34:01 -0800209 Build(pctx PackageContext, params BuildParams)
Colin Crossc3d87d32020-06-04 13:25:17 -0700210 // Phony creates a Make-style phony rule, a rule with no commands that can depend on other
211 // phony rules or real files. Phony can be called on the same name multiple times to add
212 // additional dependencies.
213 Phony(phony string, deps ...Path)
Colin Cross3f68a132017-10-23 17:10:29 -0700214
Colin Cross0875c522017-11-28 17:34:01 -0800215 PrimaryModule() Module
216 FinalModule() Module
217 VisitAllModuleVariants(visit func(Module))
Colin Cross3f68a132017-10-23 17:10:29 -0700218
219 GetMissingDependencies() []string
Jeff Gaston088e29e2017-11-29 16:47:17 -0800220 Namespace() blueprint.Namespace
Colin Cross3f40fa42015-01-30 17:27:36 -0800221}
222
Colin Cross635c3b02016-05-18 15:37:25 -0700223type Module interface {
Colin Cross3f40fa42015-01-30 17:27:36 -0800224 blueprint.Module
225
Jeff Gastonaf3cc2d2017-09-27 17:01:44 -0700226 // GenerateAndroidBuildActions is analogous to Blueprints' GenerateBuildActions,
227 // but GenerateAndroidBuildActions also has access to Android-specific information.
228 // For more information, see Module.GenerateBuildActions within Blueprint's module_ctx.go
Colin Cross635c3b02016-05-18 15:37:25 -0700229 GenerateAndroidBuildActions(ModuleContext)
Jeff Gastonaf3cc2d2017-09-27 17:01:44 -0700230
Colin Cross1e676be2016-10-12 14:38:15 -0700231 DepsMutator(BottomUpMutatorContext)
Colin Cross3f40fa42015-01-30 17:27:36 -0800232
Colin Cross635c3b02016-05-18 15:37:25 -0700233 base() *ModuleBase
Inseob Kimeec88e12020-01-22 11:11:29 +0900234 Disable()
Dan Willemsen0effe062015-11-30 16:06:01 -0800235 Enabled() bool
Colin Crossa1ad8d12016-06-01 17:09:44 -0700236 Target() Target
Dan Willemsen782a2d12015-12-21 14:55:28 -0800237 InstallInData() bool
Jaewoong Jung0949f312019-09-11 10:25:18 -0700238 InstallInTestcases() bool
Vishwath Mohan1dd88392017-03-29 22:00:18 -0700239 InstallInSanitizerDir() bool
Yifan Hong1b3348d2020-01-21 15:53:22 -0800240 InstallInRamdisk() bool
Jiyong Parkf9332f12018-02-01 00:54:12 +0900241 InstallInRecovery() bool
Colin Cross90ba5f42019-10-02 11:10:58 -0700242 InstallInRoot() bool
Colin Cross607d8582019-07-29 16:44:46 -0700243 InstallBypassMake() bool
Colin Cross6e359402020-02-10 15:29:54 -0800244 InstallForceOS() *OsType
Colin Crossa2f296f2016-11-29 15:16:18 -0800245 SkipInstall()
Ulya Trafimovichb28cc372020-01-13 15:18:16 +0000246 IsSkipInstall() bool
Jiyong Park374510b2018-03-19 18:23:01 +0900247 ExportedToMake() bool
Inseob Kim8471cda2019-11-15 09:59:12 +0900248 InitRc() Paths
249 VintfFragments() Paths
Bob Badoura75b0572020-02-18 20:21:55 -0800250 NoticeFiles() Paths
Colin Cross36242852017-06-23 15:06:31 -0700251
252 AddProperties(props ...interface{})
253 GetProperties() []interface{}
Colin Crosscec81712017-07-13 14:43:27 -0700254
Colin Crossae887032017-10-23 17:16:14 -0700255 BuildParamsForTests() []BuildParams
Colin Cross4c83e5c2019-02-25 14:54:28 -0800256 RuleParamsForTests() map[blueprint.Rule]blueprint.RuleParams
Jaewoong Jung38e4fb22018-12-12 09:01:34 -0800257 VariablesForTests() map[string]string
Paul Duffine2453c72019-05-31 14:00:04 +0100258
Colin Cross9a362232019-07-01 15:32:45 -0700259 // String returns a string that includes the module name and variants for printing during debugging.
260 String() string
261
Paul Duffine2453c72019-05-31 14:00:04 +0100262 // Get the qualified module id for this module.
263 qualifiedModuleId(ctx BaseModuleContext) qualifiedModuleName
264
265 // Get information about the properties that can contain visibility rules.
266 visibilityProperties() []visibilityProperty
Paul Duffin63c6e182019-07-24 14:24:38 +0100267
Jiyong Park6a8cf5f2019-12-30 16:31:09 +0900268 RequiredModuleNames() []string
269 HostRequiredModuleNames() []string
270 TargetRequiredModuleNames() []string
Colin Cross897266e2020-02-13 13:22:08 -0800271
272 filesToInstall() InstallPaths
Paul Duffine2453c72019-05-31 14:00:04 +0100273}
274
275// Qualified id for a module
276type qualifiedModuleName struct {
277 // The package (i.e. directory) in which the module is defined, without trailing /
278 pkg string
279
280 // The name of the module, empty string if package.
281 name string
282}
283
284func (q qualifiedModuleName) String() string {
285 if q.name == "" {
286 return "//" + q.pkg
287 }
288 return "//" + q.pkg + ":" + q.name
289}
290
Paul Duffine484f472019-06-20 16:38:08 +0100291func (q qualifiedModuleName) isRootPackage() bool {
292 return q.pkg == "" && q.name == ""
293}
294
Paul Duffine2453c72019-05-31 14:00:04 +0100295// Get the id for the package containing this module.
296func (q qualifiedModuleName) getContainingPackageId() qualifiedModuleName {
297 pkg := q.pkg
298 if q.name == "" {
Paul Duffine484f472019-06-20 16:38:08 +0100299 if pkg == "" {
300 panic(fmt.Errorf("Cannot get containing package id of root package"))
301 }
302
303 index := strings.LastIndex(pkg, "/")
304 if index == -1 {
305 pkg = ""
306 } else {
307 pkg = pkg[:index]
308 }
Paul Duffine2453c72019-05-31 14:00:04 +0100309 }
310 return newPackageId(pkg)
311}
312
313func newPackageId(pkg string) qualifiedModuleName {
314 // A qualified id for a package module has no name.
315 return qualifiedModuleName{pkg: pkg, name: ""}
Colin Cross3f40fa42015-01-30 17:27:36 -0800316}
317
Colin Crossfc754582016-05-17 16:34:16 -0700318type nameProperties struct {
319 // The name of the module. Must be unique across all modules.
Nan Zhang0007d812017-11-07 10:57:05 -0800320 Name *string
Colin Crossfc754582016-05-17 16:34:16 -0700321}
322
323type commonProperties struct {
Dan Willemsen0effe062015-11-30 16:06:01 -0800324 // emit build rules for this module
Paul Duffin54d9bb72020-02-12 10:20:56 +0000325 //
326 // Disabling a module should only be done for those modules that cannot be built
327 // in the current environment. Modules that can build in the current environment
328 // but are not usually required (e.g. superceded by a prebuilt) should not be
329 // disabled as that will prevent them from being built by the checkbuild target
330 // and so prevent early detection of changes that have broken those modules.
Dan Willemsen0effe062015-11-30 16:06:01 -0800331 Enabled *bool `android:"arch_variant"`
Colin Cross3f40fa42015-01-30 17:27:36 -0800332
Paul Duffin2e61fa62019-03-28 14:10:57 +0000333 // Controls the visibility of this module to other modules. Allowable values are one or more of
334 // these formats:
335 //
336 // ["//visibility:public"]: Anyone can use this module.
337 // ["//visibility:private"]: Only rules in the module's package (not its subpackages) can use
338 // this module.
Paul Duffin51084ff2020-05-05 19:19:22 +0100339 // ["//visibility:override"]: Discards any rules inherited from defaults or a creating module.
340 // Can only be used at the beginning of a list of visibility rules.
Paul Duffin2e61fa62019-03-28 14:10:57 +0000341 // ["//some/package:__pkg__", "//other/package:__pkg__"]: Only modules in some/package and
342 // other/package (defined in some/package/*.bp and other/package/*.bp) have access to
343 // this module. Note that sub-packages do not have access to the rule; for example,
344 // //some/package/foo:bar or //other/package/testing:bla wouldn't have access. __pkg__
345 // is a special module and must be used verbatim. It represents all of the modules in the
346 // package.
347 // ["//project:__subpackages__", "//other:__subpackages__"]: Only modules in packages project
348 // or other or in one of their sub-packages have access to this module. For example,
349 // //project:rule, //project/library:lib or //other/testing/internal:munge are allowed
350 // to depend on this rule (but not //independent:evil)
351 // ["//project"]: This is shorthand for ["//project:__pkg__"]
352 // [":__subpackages__"]: This is shorthand for ["//project:__subpackages__"] where
353 // //project is the module's package. e.g. using [":__subpackages__"] in
354 // packages/apps/Settings/Android.bp is equivalent to
355 // //packages/apps/Settings:__subpackages__.
356 // ["//visibility:legacy_public"]: The default visibility, behaves as //visibility:public
357 // for now. It is an error if it is used in a module.
Paul Duffine2453c72019-05-31 14:00:04 +0100358 //
359 // If a module does not specify the `visibility` property then it uses the
360 // `default_visibility` property of the `package` module in the module's package.
361 //
362 // If the `default_visibility` property is not set for the module's package then
Paul Duffine484f472019-06-20 16:38:08 +0100363 // it will use the `default_visibility` of its closest ancestor package for which
364 // a `default_visibility` property is specified.
365 //
366 // If no `default_visibility` property can be found then the module uses the
367 // global default of `//visibility:legacy_public`.
Paul Duffine2453c72019-05-31 14:00:04 +0100368 //
Paul Duffin95d53b52019-07-24 13:45:05 +0100369 // The `visibility` property has no effect on a defaults module although it does
370 // apply to any non-defaults module that uses it. To set the visibility of a
371 // defaults module, use the `defaults_visibility` property on the defaults module;
372 // not to be confused with the `default_visibility` property on the package module.
373 //
Paul Duffin2e61fa62019-03-28 14:10:57 +0000374 // See https://android.googlesource.com/platform/build/soong/+/master/README.md#visibility for
375 // more details.
376 Visibility []string
377
Colin Cross7d5136f2015-05-11 13:39:40 -0700378 // control whether this module compiles for 32-bit, 64-bit, or both. Possible values
Colin Cross3f40fa42015-01-30 17:27:36 -0800379 // are "32" (compile for 32-bit only), "64" (compile for 64-bit only), "both" (compile for both
380 // architectures), or "first" (compile for 64-bit on a 64-bit platform, and 32-bit on a 32-bit
381 // platform
Colin Cross7d716ba2017-11-01 10:38:29 -0700382 Compile_multilib *string `android:"arch_variant"`
Colin Cross69617d32016-09-06 10:39:07 -0700383
384 Target struct {
385 Host struct {
Colin Cross7d716ba2017-11-01 10:38:29 -0700386 Compile_multilib *string
Colin Cross69617d32016-09-06 10:39:07 -0700387 }
388 Android struct {
Colin Cross7d716ba2017-11-01 10:38:29 -0700389 Compile_multilib *string
Colin Cross69617d32016-09-06 10:39:07 -0700390 }
391 }
392
Paul Duffinca7f0ef2020-02-25 15:50:49 +0000393 // If set to true then the archMutator will create variants for each arch specific target
394 // (e.g. 32/64) that the module is required to produce. If set to false then it will only
395 // create a variant for the architecture and will list the additional arch specific targets
396 // that the variant needs to produce in the CompileMultiTargets property.
Colin Crossee0bc3b2018-10-02 22:01:37 -0700397 UseTargetVariants bool `blueprint:"mutated"`
398 Default_multilib string `blueprint:"mutated"`
Colin Cross3f40fa42015-01-30 17:27:36 -0800399
Dan Willemsen782a2d12015-12-21 14:55:28 -0800400 // whether this is a proprietary vendor module, and should be installed into /vendor
Colin Cross7d716ba2017-11-01 10:38:29 -0700401 Proprietary *bool
Dan Willemsen782a2d12015-12-21 14:55:28 -0800402
Colin Cross55708f32017-03-20 13:23:34 -0700403 // vendor who owns this module
Dan Willemsenefac4a82017-07-18 19:42:09 -0700404 Owner *string
Colin Cross55708f32017-03-20 13:23:34 -0700405
Jiyong Park2db76922017-11-08 16:03:48 +0900406 // whether this module is specific to an SoC (System-On-a-Chip). When set to true,
407 // it is installed into /vendor (or /system/vendor if vendor partition does not exist).
408 // Use `soc_specific` instead for better meaning.
Colin Cross7d716ba2017-11-01 10:38:29 -0700409 Vendor *bool
Dan Willemsenaa118f92017-04-06 12:49:58 -0700410
Jiyong Park2db76922017-11-08 16:03:48 +0900411 // whether this module is specific to an SoC (System-On-a-Chip). When set to true,
412 // it is installed into /vendor (or /system/vendor if vendor partition does not exist).
413 Soc_specific *bool
414
415 // whether this module is specific to a device, not only for SoC, but also for off-chip
416 // peripherals. When set to true, it is installed into /odm (or /vendor/odm if odm partition
417 // does not exist, or /system/vendor/odm if both odm and vendor partitions do not exist).
418 // This implies `soc_specific:true`.
419 Device_specific *bool
420
421 // whether this module is specific to a software configuration of a product (e.g. country,
Jaekyun Seok5cfbfbb2018-01-10 19:00:15 +0900422 // network operator, etc). When set to true, it is installed into /product (or
423 // /system/product if product partition does not exist).
Jiyong Park2db76922017-11-08 16:03:48 +0900424 Product_specific *bool
425
Justin Yund5f6c822019-06-25 16:47:17 +0900426 // whether this module extends system. When set to true, it is installed into /system_ext
427 // (or /system/system_ext if system_ext partition does not exist).
428 System_ext_specific *bool
429
Jiyong Parkf9332f12018-02-01 00:54:12 +0900430 // Whether this module is installed to recovery partition
431 Recovery *bool
432
Yifan Hong1b3348d2020-01-21 15:53:22 -0800433 // Whether this module is installed to ramdisk
434 Ramdisk *bool
435
dimitry1f33e402019-03-26 12:39:31 +0100436 // Whether this module is built for non-native architecures (also known as native bridge binary)
437 Native_bridge_supported *bool `android:"arch_variant"`
438
Dan Willemsen2277bcb2016-07-25 20:27:39 -0700439 // init.rc files to be installed if this module is installed
Colin Cross27b922f2019-03-04 22:35:41 -0800440 Init_rc []string `android:"path"`
Dan Willemsen2277bcb2016-07-25 20:27:39 -0700441
Steven Moreland57a23d22018-04-04 15:42:19 -0700442 // VINTF manifest fragments to be installed if this module is installed
Colin Cross27b922f2019-03-04 22:35:41 -0800443 Vintf_fragments []string `android:"path"`
Steven Moreland57a23d22018-04-04 15:42:19 -0700444
Chris Wolfe998306e2016-08-15 14:47:23 -0400445 // names of other modules to install if this module is installed
Colin Crossc602b7d2017-05-05 13:36:36 -0700446 Required []string `android:"arch_variant"`
Chris Wolfe998306e2016-08-15 14:47:23 -0400447
Sasha Smundakb6d23052019-04-01 18:37:36 -0700448 // names of other modules to install on host if this module is installed
449 Host_required []string `android:"arch_variant"`
450
451 // names of other modules to install on target if this module is installed
452 Target_required []string `android:"arch_variant"`
453
Colin Cross5aac3622017-08-31 15:07:09 -0700454 // relative path to a file to include in the list of notices for the device
Colin Cross27b922f2019-03-04 22:35:41 -0800455 Notice *string `android:"path"`
Colin Cross5aac3622017-08-31 15:07:09 -0700456
Dan Willemsen569edc52018-11-19 09:33:29 -0800457 Dist struct {
458 // copy the output of this module to the $DIST_DIR when `dist` is specified on the
459 // command line and any of these targets are also on the command line, or otherwise
460 // built
461 Targets []string `android:"arch_variant"`
462
463 // The name of the output artifact. This defaults to the basename of the output of
464 // the module.
465 Dest *string `android:"arch_variant"`
466
467 // The directory within the dist directory to store the artifact. Defaults to the
468 // top level directory ("").
469 Dir *string `android:"arch_variant"`
470
471 // A suffix to add to the artifact file name (before any extension).
472 Suffix *string `android:"arch_variant"`
473 } `android:"arch_variant"`
474
Paul Duffinca7f0ef2020-02-25 15:50:49 +0000475 // The OsType of artifacts that this module variant is responsible for creating.
476 //
477 // Set by osMutator
478 CompileOS OsType `blueprint:"mutated"`
479
480 // The Target of artifacts that this module variant is responsible for creating.
481 //
482 // Set by archMutator
483 CompileTarget Target `blueprint:"mutated"`
484
485 // The additional arch specific targets (e.g. 32/64 bit) that this module variant is
486 // responsible for creating.
487 //
488 // By default this is nil as, where necessary, separate variants are created for the
489 // different multilib types supported and that information is encapsulated in the
490 // CompileTarget so the module variant simply needs to create artifacts for that.
491 //
492 // However, if UseTargetVariants is set to false (e.g. by
493 // InitAndroidMultiTargetsArchModule) then no separate variants are created for the
494 // multilib targets. Instead a single variant is created for the architecture and
495 // this contains the multilib specific targets that this variant should create.
496 //
497 // Set by archMutator
Colin Crossee0bc3b2018-10-02 22:01:37 -0700498 CompileMultiTargets []Target `blueprint:"mutated"`
Paul Duffinca7f0ef2020-02-25 15:50:49 +0000499
500 // True if the module variant's CompileTarget is the primary target
501 //
502 // Set by archMutator
503 CompilePrimary bool `blueprint:"mutated"`
Colin Cross3f40fa42015-01-30 17:27:36 -0800504
505 // Set by InitAndroidModule
506 HostOrDeviceSupported HostOrDeviceSupported `blueprint:"mutated"`
Dan Willemsen0b24c742016-10-04 15:13:37 -0700507 ArchSpecific bool `blueprint:"mutated"`
Colin Crossce75d2c2016-10-06 16:12:58 -0700508
Paul Duffin1356d8c2020-02-25 19:26:33 +0000509 // If set to true then a CommonOS variant will be created which will have dependencies
510 // on all its OsType specific variants. Used by sdk/module_exports to create a snapshot
511 // that covers all os and architecture variants.
512 //
513 // The OsType specific variants can be retrieved by calling
514 // GetOsSpecificVariantsOfCommonOSVariant
515 //
516 // Set at module initialization time by calling InitCommonOSAndroidMultiTargetsArchModule
517 CreateCommonOSVariant bool `blueprint:"mutated"`
518
519 // If set to true then this variant is the CommonOS variant that has dependencies on its
520 // OsType specific variants.
521 //
522 // Set by osMutator.
523 CommonOSVariant bool `blueprint:"mutated"`
524
Colin Crossce75d2c2016-10-06 16:12:58 -0700525 SkipInstall bool `blueprint:"mutated"`
Jeff Gaston088e29e2017-11-29 16:47:17 -0800526
527 NamespaceExportedToMake bool `blueprint:"mutated"`
Colin Cross6c4f21f2019-06-06 15:41:36 -0700528
529 MissingDeps []string `blueprint:"mutated"`
Colin Cross9a362232019-07-01 15:32:45 -0700530
531 // Name and variant strings stored by mutators to enable Module.String()
532 DebugName string `blueprint:"mutated"`
533 DebugMutators []string `blueprint:"mutated"`
534 DebugVariations []string `blueprint:"mutated"`
Colin Cross7228ecd2019-11-18 16:00:16 -0800535
536 // set by ImageMutator
537 ImageVariation string `blueprint:"mutated"`
Colin Cross3f40fa42015-01-30 17:27:36 -0800538}
539
540type hostAndDeviceProperties struct {
Colin Cross4e81d702018-11-09 10:36:55 -0800541 // If set to true, build a variant of the module for the host. Defaults to false.
542 Host_supported *bool
543
544 // If set to true, build a variant of the module for the device. Defaults to true.
Colin Crossa4190c12016-07-12 13:11:25 -0700545 Device_supported *bool
Colin Cross3f40fa42015-01-30 17:27:36 -0800546}
547
Colin Crossc472d572015-03-17 15:06:21 -0700548type Multilib string
549
550const (
Colin Cross6b4a32d2017-12-05 13:42:45 -0800551 MultilibBoth Multilib = "both"
552 MultilibFirst Multilib = "first"
553 MultilibCommon Multilib = "common"
554 MultilibCommonFirst Multilib = "common_first"
555 MultilibDefault Multilib = ""
Colin Crossc472d572015-03-17 15:06:21 -0700556)
557
Colin Crossa1ad8d12016-06-01 17:09:44 -0700558type HostOrDeviceSupported int
559
560const (
561 _ HostOrDeviceSupported = iota
Dan Albert0981b5c2018-08-02 13:46:35 -0700562
563 // Host and HostCross are built by default. Device is not supported.
Colin Crossa1ad8d12016-06-01 17:09:44 -0700564 HostSupported
Dan Albert0981b5c2018-08-02 13:46:35 -0700565
566 // Host is built by default. HostCross and Device are not supported.
Dan Albertc6345fb2016-10-20 01:36:11 -0700567 HostSupportedNoCross
Dan Albert0981b5c2018-08-02 13:46:35 -0700568
569 // Device is built by default. Host and HostCross are not supported.
Colin Crossa1ad8d12016-06-01 17:09:44 -0700570 DeviceSupported
Dan Albert0981b5c2018-08-02 13:46:35 -0700571
572 // Device is built by default. Host and HostCross are supported.
Colin Crossa1ad8d12016-06-01 17:09:44 -0700573 HostAndDeviceSupported
Dan Albert0981b5c2018-08-02 13:46:35 -0700574
575 // Host, HostCross, and Device are built by default.
Colin Crossa1ad8d12016-06-01 17:09:44 -0700576 HostAndDeviceDefault
Dan Albert0981b5c2018-08-02 13:46:35 -0700577
578 // Nothing is supported. This is not exposed to the user, but used to mark a
579 // host only module as unsupported when the module type is not supported on
580 // the host OS. E.g. benchmarks are supported on Linux but not Darwin.
Dan Willemsen0b24c742016-10-04 15:13:37 -0700581 NeitherHostNorDeviceSupported
Colin Crossa1ad8d12016-06-01 17:09:44 -0700582)
583
Jiyong Park2db76922017-11-08 16:03:48 +0900584type moduleKind int
585
586const (
587 platformModule moduleKind = iota
588 deviceSpecificModule
589 socSpecificModule
590 productSpecificModule
Justin Yund5f6c822019-06-25 16:47:17 +0900591 systemExtSpecificModule
Jiyong Park2db76922017-11-08 16:03:48 +0900592)
593
594func (k moduleKind) String() string {
595 switch k {
596 case platformModule:
597 return "platform"
598 case deviceSpecificModule:
599 return "device-specific"
600 case socSpecificModule:
601 return "soc-specific"
602 case productSpecificModule:
603 return "product-specific"
Justin Yund5f6c822019-06-25 16:47:17 +0900604 case systemExtSpecificModule:
605 return "systemext-specific"
Jiyong Park2db76922017-11-08 16:03:48 +0900606 default:
607 panic(fmt.Errorf("unknown module kind %d", k))
608 }
609}
610
Colin Cross9d34f352019-11-22 16:03:51 -0800611func initAndroidModuleBase(m Module) {
612 m.base().module = m
613}
614
Colin Cross36242852017-06-23 15:06:31 -0700615func InitAndroidModule(m Module) {
Colin Cross9d34f352019-11-22 16:03:51 -0800616 initAndroidModuleBase(m)
Colin Cross3f40fa42015-01-30 17:27:36 -0800617 base := m.base()
Colin Cross5049f022015-03-18 13:28:46 -0700618
Colin Cross36242852017-06-23 15:06:31 -0700619 m.AddProperties(
Colin Crossfc754582016-05-17 16:34:16 -0700620 &base.nameProperties,
Colin Cross18c46802019-09-24 22:19:02 -0700621 &base.commonProperties)
622
Colin Crosseabaedd2020-02-06 17:01:55 -0800623 initProductVariableModule(m)
Colin Cross18c46802019-09-24 22:19:02 -0700624
Colin Crossa3a97412019-03-18 12:24:29 -0700625 base.generalProperties = m.GetProperties()
Pirama Arumuga Nainar955dc492018-04-17 14:58:42 -0700626 base.customizableProperties = m.GetProperties()
Paul Duffin63c6e182019-07-24 14:24:38 +0100627
628 // The default_visibility property needs to be checked and parsed by the visibility module during
Paul Duffin5ec73ec2020-05-01 17:52:01 +0100629 // its checking and parsing phases so make it the primary visibility property.
630 setPrimaryVisibilityProperty(m, "visibility", &base.commonProperties.Visibility)
Colin Cross5049f022015-03-18 13:28:46 -0700631}
632
Colin Cross36242852017-06-23 15:06:31 -0700633func InitAndroidArchModule(m Module, hod HostOrDeviceSupported, defaultMultilib Multilib) {
634 InitAndroidModule(m)
Colin Cross5049f022015-03-18 13:28:46 -0700635
636 base := m.base()
Colin Cross3f40fa42015-01-30 17:27:36 -0800637 base.commonProperties.HostOrDeviceSupported = hod
Colin Cross69617d32016-09-06 10:39:07 -0700638 base.commonProperties.Default_multilib = string(defaultMultilib)
Dan Willemsen0b24c742016-10-04 15:13:37 -0700639 base.commonProperties.ArchSpecific = true
Colin Crossee0bc3b2018-10-02 22:01:37 -0700640 base.commonProperties.UseTargetVariants = true
Colin Cross3f40fa42015-01-30 17:27:36 -0800641
Dan Willemsen218f6562015-07-08 18:13:11 -0700642 switch hod {
Nan Zhang1a0f09b2017-07-05 10:35:11 -0700643 case HostAndDeviceSupported, HostAndDeviceDefault:
Colin Cross36242852017-06-23 15:06:31 -0700644 m.AddProperties(&base.hostAndDeviceProperties)
Colin Cross3f40fa42015-01-30 17:27:36 -0800645 }
646
Colin Cross36242852017-06-23 15:06:31 -0700647 InitArchModule(m)
Colin Cross3f40fa42015-01-30 17:27:36 -0800648}
649
Colin Crossee0bc3b2018-10-02 22:01:37 -0700650func InitAndroidMultiTargetsArchModule(m Module, hod HostOrDeviceSupported, defaultMultilib Multilib) {
651 InitAndroidArchModule(m, hod, defaultMultilib)
652 m.base().commonProperties.UseTargetVariants = false
653}
654
Paul Duffin1356d8c2020-02-25 19:26:33 +0000655// As InitAndroidMultiTargetsArchModule except it creates an additional CommonOS variant that
656// has dependencies on all the OsType specific variants.
657func InitCommonOSAndroidMultiTargetsArchModule(m Module, hod HostOrDeviceSupported, defaultMultilib Multilib) {
658 InitAndroidArchModule(m, hod, defaultMultilib)
659 m.base().commonProperties.UseTargetVariants = false
660 m.base().commonProperties.CreateCommonOSVariant = true
661}
662
Nan Zhangb9eeb1d2017-02-02 10:46:07 -0800663// A ModuleBase object contains the properties that are common to all Android
Colin Cross3f40fa42015-01-30 17:27:36 -0800664// modules. It should be included as an anonymous field in every module
665// struct definition. InitAndroidModule should then be called from the module's
666// factory function, and the return values from InitAndroidModule should be
667// returned from the factory function.
668//
Nan Zhangb9eeb1d2017-02-02 10:46:07 -0800669// The ModuleBase type is responsible for implementing the GenerateBuildActions
670// method to support the blueprint.Module interface. This method will then call
671// the module's GenerateAndroidBuildActions method once for each build variant
Colin Cross25de6c32019-06-06 14:29:25 -0700672// that is to be built. GenerateAndroidBuildActions is passed a ModuleContext
673// rather than the usual blueprint.ModuleContext.
674// ModuleContext exposes extra functionality specific to the Android build
Colin Cross3f40fa42015-01-30 17:27:36 -0800675// system including details about the particular build variant that is to be
676// generated.
677//
678// For example:
679//
680// import (
Nan Zhangb9eeb1d2017-02-02 10:46:07 -0800681// "android/soong/android"
Colin Cross3f40fa42015-01-30 17:27:36 -0800682// )
683//
684// type myModule struct {
Nan Zhangb9eeb1d2017-02-02 10:46:07 -0800685// android.ModuleBase
Colin Cross3f40fa42015-01-30 17:27:36 -0800686// properties struct {
687// MyProperty string
688// }
689// }
690//
Colin Cross36242852017-06-23 15:06:31 -0700691// func NewMyModule() android.Module) {
Colin Cross3f40fa42015-01-30 17:27:36 -0800692// m := &myModule{}
Colin Cross36242852017-06-23 15:06:31 -0700693// m.AddProperties(&m.properties)
694// android.InitAndroidModule(m)
695// return m
Colin Cross3f40fa42015-01-30 17:27:36 -0800696// }
697//
Nan Zhangb9eeb1d2017-02-02 10:46:07 -0800698// func (m *myModule) GenerateAndroidBuildActions(ctx android.ModuleContext) {
Colin Cross3f40fa42015-01-30 17:27:36 -0800699// // Get the CPU architecture for the current build variant.
700// variantArch := ctx.Arch()
701//
702// // ...
703// }
Colin Cross635c3b02016-05-18 15:37:25 -0700704type ModuleBase struct {
Colin Cross3f40fa42015-01-30 17:27:36 -0800705 // Putting the curiously recurring thing pointing to the thing that contains
706 // the thing pattern to good use.
Colin Cross36242852017-06-23 15:06:31 -0700707 // TODO: remove this
Colin Cross635c3b02016-05-18 15:37:25 -0700708 module Module
Colin Cross3f40fa42015-01-30 17:27:36 -0800709
Colin Crossfc754582016-05-17 16:34:16 -0700710 nameProperties nameProperties
Colin Cross3f40fa42015-01-30 17:27:36 -0800711 commonProperties commonProperties
Colin Cross18c46802019-09-24 22:19:02 -0700712 variableProperties interface{}
Colin Cross3f40fa42015-01-30 17:27:36 -0800713 hostAndDeviceProperties hostAndDeviceProperties
714 generalProperties []interface{}
Colin Crossc17727d2018-10-24 12:42:09 -0700715 archProperties [][]interface{}
Colin Crossa120ec12016-08-19 16:07:38 -0700716 customizableProperties []interface{}
Colin Cross3f40fa42015-01-30 17:27:36 -0800717
Paul Duffin63c6e182019-07-24 14:24:38 +0100718 // Information about all the properties on the module that contains visibility rules that need
719 // checking.
720 visibilityPropertyInfo []visibilityProperty
721
722 // The primary visibility property, may be nil, that controls access to the module.
723 primaryVisibilityProperty visibilityProperty
724
Colin Cross3f40fa42015-01-30 17:27:36 -0800725 noAddressSanitizer bool
Colin Cross897266e2020-02-13 13:22:08 -0800726 installFiles InstallPaths
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700727 checkbuildFiles Paths
Bob Badoura75b0572020-02-18 20:21:55 -0800728 noticeFiles Paths
Colin Crossc3d87d32020-06-04 13:25:17 -0700729 phonies map[string]Paths
Colin Cross1f8c52b2015-06-16 16:38:17 -0700730
731 // Used by buildTargetSingleton to create checkbuild and per-directory build targets
732 // Only set on the final variant of each module
Colin Cross0875c522017-11-28 17:34:01 -0800733 installTarget WritablePath
734 checkbuildTarget WritablePath
Colin Cross1f8c52b2015-06-16 16:38:17 -0700735 blueprintDir string
Colin Crossa120ec12016-08-19 16:07:38 -0700736
Colin Cross178a5092016-09-13 13:42:32 -0700737 hooks hooks
Colin Cross36242852017-06-23 15:06:31 -0700738
739 registerProps []interface{}
Colin Crosscec81712017-07-13 14:43:27 -0700740
741 // For tests
Colin Crossae887032017-10-23 17:16:14 -0700742 buildParams []BuildParams
Colin Cross4c83e5c2019-02-25 14:54:28 -0800743 ruleParams map[blueprint.Rule]blueprint.RuleParams
Jaewoong Jung38e4fb22018-12-12 09:01:34 -0800744 variables map[string]string
Colin Crossa9d8bee2018-10-02 13:59:46 -0700745
Inseob Kim8471cda2019-11-15 09:59:12 +0900746 initRcPaths Paths
747 vintfFragmentsPaths Paths
748
Colin Crossa9d8bee2018-10-02 13:59:46 -0700749 prefer32 func(ctx BaseModuleContext, base *ModuleBase, class OsClass) bool
Colin Cross36242852017-06-23 15:06:31 -0700750}
751
Colin Cross4157e882019-06-06 16:57:04 -0700752func (m *ModuleBase) DepsMutator(BottomUpMutatorContext) {}
Colin Cross5f692ec2019-02-01 16:53:07 -0800753
Colin Cross4157e882019-06-06 16:57:04 -0700754func (m *ModuleBase) AddProperties(props ...interface{}) {
755 m.registerProps = append(m.registerProps, props...)
Colin Cross36242852017-06-23 15:06:31 -0700756}
757
Colin Cross4157e882019-06-06 16:57:04 -0700758func (m *ModuleBase) GetProperties() []interface{} {
759 return m.registerProps
Colin Cross3f40fa42015-01-30 17:27:36 -0800760}
761
Colin Cross4157e882019-06-06 16:57:04 -0700762func (m *ModuleBase) BuildParamsForTests() []BuildParams {
763 return m.buildParams
Colin Crosscec81712017-07-13 14:43:27 -0700764}
765
Colin Cross4157e882019-06-06 16:57:04 -0700766func (m *ModuleBase) RuleParamsForTests() map[blueprint.Rule]blueprint.RuleParams {
767 return m.ruleParams
Colin Cross4c83e5c2019-02-25 14:54:28 -0800768}
769
Colin Cross4157e882019-06-06 16:57:04 -0700770func (m *ModuleBase) VariablesForTests() map[string]string {
771 return m.variables
Jaewoong Jung38e4fb22018-12-12 09:01:34 -0800772}
773
Colin Cross4157e882019-06-06 16:57:04 -0700774func (m *ModuleBase) Prefer32(prefer32 func(ctx BaseModuleContext, base *ModuleBase, class OsClass) bool) {
775 m.prefer32 = prefer32
Colin Crossa9d8bee2018-10-02 13:59:46 -0700776}
777
Colin Crossce75d2c2016-10-06 16:12:58 -0700778// Name returns the name of the module. It may be overridden by individual module types, for
779// example prebuilts will prepend prebuilt_ to the name.
Colin Cross4157e882019-06-06 16:57:04 -0700780func (m *ModuleBase) Name() string {
781 return String(m.nameProperties.Name)
Colin Crossfc754582016-05-17 16:34:16 -0700782}
783
Colin Cross9a362232019-07-01 15:32:45 -0700784// String returns a string that includes the module name and variants for printing during debugging.
785func (m *ModuleBase) String() string {
786 sb := strings.Builder{}
787 sb.WriteString(m.commonProperties.DebugName)
788 sb.WriteString("{")
789 for i := range m.commonProperties.DebugMutators {
790 if i != 0 {
791 sb.WriteString(",")
792 }
793 sb.WriteString(m.commonProperties.DebugMutators[i])
794 sb.WriteString(":")
795 sb.WriteString(m.commonProperties.DebugVariations[i])
796 }
797 sb.WriteString("}")
798 return sb.String()
799}
800
Colin Crossce75d2c2016-10-06 16:12:58 -0700801// BaseModuleName returns the name of the module as specified in the blueprints file.
Colin Cross4157e882019-06-06 16:57:04 -0700802func (m *ModuleBase) BaseModuleName() string {
803 return String(m.nameProperties.Name)
Colin Crossce75d2c2016-10-06 16:12:58 -0700804}
805
Colin Cross4157e882019-06-06 16:57:04 -0700806func (m *ModuleBase) base() *ModuleBase {
807 return m
Colin Cross3f40fa42015-01-30 17:27:36 -0800808}
809
Paul Duffine2453c72019-05-31 14:00:04 +0100810func (m *ModuleBase) qualifiedModuleId(ctx BaseModuleContext) qualifiedModuleName {
811 return qualifiedModuleName{pkg: ctx.ModuleDir(), name: ctx.ModuleName()}
812}
813
814func (m *ModuleBase) visibilityProperties() []visibilityProperty {
Paul Duffin63c6e182019-07-24 14:24:38 +0100815 return m.visibilityPropertyInfo
816}
817
Colin Cross4157e882019-06-06 16:57:04 -0700818func (m *ModuleBase) Target() Target {
819 return m.commonProperties.CompileTarget
Dan Willemsen490fd492015-11-24 17:53:15 -0800820}
821
Colin Cross4157e882019-06-06 16:57:04 -0700822func (m *ModuleBase) TargetPrimary() bool {
823 return m.commonProperties.CompilePrimary
Colin Cross8b74d172016-09-13 09:59:14 -0700824}
825
Colin Cross4157e882019-06-06 16:57:04 -0700826func (m *ModuleBase) MultiTargets() []Target {
827 return m.commonProperties.CompileMultiTargets
Colin Crossee0bc3b2018-10-02 22:01:37 -0700828}
829
Colin Cross4157e882019-06-06 16:57:04 -0700830func (m *ModuleBase) Os() OsType {
831 return m.Target().Os
Dan Willemsen490fd492015-11-24 17:53:15 -0800832}
833
Colin Cross4157e882019-06-06 16:57:04 -0700834func (m *ModuleBase) Host() bool {
835 return m.Os().Class == Host || m.Os().Class == HostCross
Dan Willemsen97750522016-02-09 17:43:51 -0800836}
837
Colin Cross4157e882019-06-06 16:57:04 -0700838func (m *ModuleBase) Arch() Arch {
839 return m.Target().Arch
Dan Willemsen97750522016-02-09 17:43:51 -0800840}
841
Colin Cross4157e882019-06-06 16:57:04 -0700842func (m *ModuleBase) ArchSpecific() bool {
843 return m.commonProperties.ArchSpecific
Dan Willemsen0b24c742016-10-04 15:13:37 -0700844}
845
Paul Duffin1356d8c2020-02-25 19:26:33 +0000846// True if the current variant is a CommonOS variant, false otherwise.
847func (m *ModuleBase) IsCommonOSVariant() bool {
848 return m.commonProperties.CommonOSVariant
849}
850
Colin Cross4157e882019-06-06 16:57:04 -0700851func (m *ModuleBase) OsClassSupported() []OsClass {
852 switch m.commonProperties.HostOrDeviceSupported {
Colin Crossa1ad8d12016-06-01 17:09:44 -0700853 case HostSupported:
Colin Crossa1ad8d12016-06-01 17:09:44 -0700854 return []OsClass{Host, HostCross}
Dan Albertc6345fb2016-10-20 01:36:11 -0700855 case HostSupportedNoCross:
856 return []OsClass{Host}
Colin Crossa1ad8d12016-06-01 17:09:44 -0700857 case DeviceSupported:
858 return []OsClass{Device}
Dan Albert0981b5c2018-08-02 13:46:35 -0700859 case HostAndDeviceSupported, HostAndDeviceDefault:
Colin Crossa1ad8d12016-06-01 17:09:44 -0700860 var supported []OsClass
Colin Cross4157e882019-06-06 16:57:04 -0700861 if Bool(m.hostAndDeviceProperties.Host_supported) ||
862 (m.commonProperties.HostOrDeviceSupported == HostAndDeviceDefault &&
863 m.hostAndDeviceProperties.Host_supported == nil) {
Colin Crossa1ad8d12016-06-01 17:09:44 -0700864 supported = append(supported, Host, HostCross)
865 }
Colin Cross4157e882019-06-06 16:57:04 -0700866 if m.hostAndDeviceProperties.Device_supported == nil ||
867 *m.hostAndDeviceProperties.Device_supported {
Colin Crossa1ad8d12016-06-01 17:09:44 -0700868 supported = append(supported, Device)
869 }
870 return supported
871 default:
872 return nil
873 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800874}
875
Colin Cross4157e882019-06-06 16:57:04 -0700876func (m *ModuleBase) DeviceSupported() bool {
877 return m.commonProperties.HostOrDeviceSupported == DeviceSupported ||
878 m.commonProperties.HostOrDeviceSupported == HostAndDeviceSupported &&
879 (m.hostAndDeviceProperties.Device_supported == nil ||
880 *m.hostAndDeviceProperties.Device_supported)
Colin Cross3f40fa42015-01-30 17:27:36 -0800881}
882
Paul Duffine44358f2019-11-26 18:04:12 +0000883func (m *ModuleBase) HostSupported() bool {
884 return m.commonProperties.HostOrDeviceSupported == HostSupported ||
885 m.commonProperties.HostOrDeviceSupported == HostAndDeviceSupported &&
886 (m.hostAndDeviceProperties.Host_supported != nil &&
887 *m.hostAndDeviceProperties.Host_supported)
888}
889
Colin Cross4157e882019-06-06 16:57:04 -0700890func (m *ModuleBase) Platform() bool {
Justin Yund5f6c822019-06-25 16:47:17 +0900891 return !m.DeviceSpecific() && !m.SocSpecific() && !m.ProductSpecific() && !m.SystemExtSpecific()
Jiyong Parkc678ad32018-04-10 13:07:10 +0900892}
893
Colin Cross4157e882019-06-06 16:57:04 -0700894func (m *ModuleBase) DeviceSpecific() bool {
895 return Bool(m.commonProperties.Device_specific)
Jiyong Parkc678ad32018-04-10 13:07:10 +0900896}
897
Colin Cross4157e882019-06-06 16:57:04 -0700898func (m *ModuleBase) SocSpecific() bool {
899 return Bool(m.commonProperties.Vendor) || Bool(m.commonProperties.Proprietary) || Bool(m.commonProperties.Soc_specific)
Jiyong Parkc678ad32018-04-10 13:07:10 +0900900}
901
Colin Cross4157e882019-06-06 16:57:04 -0700902func (m *ModuleBase) ProductSpecific() bool {
903 return Bool(m.commonProperties.Product_specific)
Jiyong Parkc678ad32018-04-10 13:07:10 +0900904}
905
Justin Yund5f6c822019-06-25 16:47:17 +0900906func (m *ModuleBase) SystemExtSpecific() bool {
907 return Bool(m.commonProperties.System_ext_specific)
Dario Frenifd05a742018-05-29 13:28:54 +0100908}
909
Colin Crossc2d24052020-05-13 11:05:02 -0700910// RequiresStableAPIs returns true if the module will be installed to a partition that may
911// be updated separately from the system image.
912func (m *ModuleBase) RequiresStableAPIs(ctx BaseModuleContext) bool {
913 return m.SocSpecific() || m.DeviceSpecific() ||
914 (m.ProductSpecific() && ctx.Config().EnforceProductPartitionInterface())
915}
916
Bill Peckhamfff3f8a2020-03-20 18:33:20 -0700917func (m *ModuleBase) PartitionTag(config DeviceConfig) string {
918 partition := "system"
919 if m.SocSpecific() {
920 // A SoC-specific module could be on the vendor partition at
921 // "vendor" or the system partition at "system/vendor".
922 if config.VendorPath() == "vendor" {
923 partition = "vendor"
924 }
925 } else if m.DeviceSpecific() {
926 // A device-specific module could be on the odm partition at
927 // "odm", the vendor partition at "vendor/odm", or the system
928 // partition at "system/vendor/odm".
929 if config.OdmPath() == "odm" {
930 partition = "odm"
Ramy Medhat944839a2020-03-31 22:14:52 -0400931 } else if strings.HasPrefix(config.OdmPath(), "vendor/") {
Bill Peckhamfff3f8a2020-03-20 18:33:20 -0700932 partition = "vendor"
933 }
934 } else if m.ProductSpecific() {
935 // A product-specific module could be on the product partition
936 // at "product" or the system partition at "system/product".
937 if config.ProductPath() == "product" {
938 partition = "product"
939 }
940 } else if m.SystemExtSpecific() {
941 // A system_ext-specific module could be on the system_ext
942 // partition at "system_ext" or the system partition at
943 // "system/system_ext".
944 if config.SystemExtPath() == "system_ext" {
945 partition = "system_ext"
946 }
947 }
948 return partition
949}
950
Colin Cross4157e882019-06-06 16:57:04 -0700951func (m *ModuleBase) Enabled() bool {
952 if m.commonProperties.Enabled == nil {
953 return !m.Os().DefaultDisabled
Dan Willemsen490fd492015-11-24 17:53:15 -0800954 }
Colin Cross4157e882019-06-06 16:57:04 -0700955 return *m.commonProperties.Enabled
Colin Cross3f40fa42015-01-30 17:27:36 -0800956}
957
Inseob Kimeec88e12020-01-22 11:11:29 +0900958func (m *ModuleBase) Disable() {
959 m.commonProperties.Enabled = proptools.BoolPtr(false)
960}
961
Colin Cross4157e882019-06-06 16:57:04 -0700962func (m *ModuleBase) SkipInstall() {
963 m.commonProperties.SkipInstall = true
Colin Crossce75d2c2016-10-06 16:12:58 -0700964}
965
Ulya Trafimovichb28cc372020-01-13 15:18:16 +0000966func (m *ModuleBase) IsSkipInstall() bool {
967 return m.commonProperties.SkipInstall == true
968}
969
Colin Cross4157e882019-06-06 16:57:04 -0700970func (m *ModuleBase) ExportedToMake() bool {
971 return m.commonProperties.NamespaceExportedToMake
Jiyong Park374510b2018-03-19 18:23:01 +0900972}
973
Colin Cross897266e2020-02-13 13:22:08 -0800974func (m *ModuleBase) computeInstallDeps(ctx blueprint.ModuleContext) InstallPaths {
Colin Cross3f40fa42015-01-30 17:27:36 -0800975
Colin Cross897266e2020-02-13 13:22:08 -0800976 var result InstallPaths
Colin Cross6b753602018-06-21 13:03:07 -0700977 // TODO(ccross): we need to use WalkDeps and have some way to know which dependencies require installation
Colin Cross897266e2020-02-13 13:22:08 -0800978 ctx.VisitDepsDepthFirst(func(m blueprint.Module) {
979 if a, ok := m.(Module); ok {
980 result = append(result, a.filesToInstall()...)
981 }
982 })
Colin Cross3f40fa42015-01-30 17:27:36 -0800983
984 return result
985}
986
Colin Cross897266e2020-02-13 13:22:08 -0800987func (m *ModuleBase) filesToInstall() InstallPaths {
Colin Cross4157e882019-06-06 16:57:04 -0700988 return m.installFiles
Colin Cross3f40fa42015-01-30 17:27:36 -0800989}
990
Colin Cross4157e882019-06-06 16:57:04 -0700991func (m *ModuleBase) NoAddressSanitizer() bool {
992 return m.noAddressSanitizer
Colin Cross3f40fa42015-01-30 17:27:36 -0800993}
994
Colin Cross4157e882019-06-06 16:57:04 -0700995func (m *ModuleBase) InstallInData() bool {
Dan Willemsen782a2d12015-12-21 14:55:28 -0800996 return false
997}
998
Jaewoong Jung0949f312019-09-11 10:25:18 -0700999func (m *ModuleBase) InstallInTestcases() bool {
1000 return false
1001}
1002
Colin Cross4157e882019-06-06 16:57:04 -07001003func (m *ModuleBase) InstallInSanitizerDir() bool {
Vishwath Mohan1dd88392017-03-29 22:00:18 -07001004 return false
1005}
1006
Yifan Hong1b3348d2020-01-21 15:53:22 -08001007func (m *ModuleBase) InstallInRamdisk() bool {
1008 return Bool(m.commonProperties.Ramdisk)
1009}
1010
Colin Cross4157e882019-06-06 16:57:04 -07001011func (m *ModuleBase) InstallInRecovery() bool {
1012 return Bool(m.commonProperties.Recovery)
Jiyong Parkf9332f12018-02-01 00:54:12 +09001013}
1014
Colin Cross90ba5f42019-10-02 11:10:58 -07001015func (m *ModuleBase) InstallInRoot() bool {
1016 return false
1017}
1018
Colin Cross607d8582019-07-29 16:44:46 -07001019func (m *ModuleBase) InstallBypassMake() bool {
1020 return false
1021}
1022
Colin Cross6e359402020-02-10 15:29:54 -08001023func (m *ModuleBase) InstallForceOS() *OsType {
1024 return nil
1025}
1026
Colin Cross4157e882019-06-06 16:57:04 -07001027func (m *ModuleBase) Owner() string {
1028 return String(m.commonProperties.Owner)
Sundong Ahn4fd04bb2018-08-31 18:01:37 +09001029}
1030
Bob Badoura75b0572020-02-18 20:21:55 -08001031func (m *ModuleBase) NoticeFiles() Paths {
1032 return m.noticeFiles
Jiyong Park52818fc2019-03-18 12:01:38 +09001033}
1034
Colin Cross7228ecd2019-11-18 16:00:16 -08001035func (m *ModuleBase) setImageVariation(variant string) {
1036 m.commonProperties.ImageVariation = variant
1037}
1038
1039func (m *ModuleBase) ImageVariation() blueprint.Variation {
1040 return blueprint.Variation{
1041 Mutator: "image",
1042 Variation: m.base().commonProperties.ImageVariation,
1043 }
1044}
1045
Paul Duffin9b76c0b2020-03-12 10:24:35 +00001046func (m *ModuleBase) getVariationByMutatorName(mutator string) string {
1047 for i, v := range m.commonProperties.DebugMutators {
1048 if v == mutator {
1049 return m.commonProperties.DebugVariations[i]
1050 }
1051 }
1052
1053 return ""
1054}
1055
Yifan Hong1b3348d2020-01-21 15:53:22 -08001056func (m *ModuleBase) InRamdisk() bool {
1057 return m.base().commonProperties.ImageVariation == RamdiskVariation
1058}
1059
Colin Cross7228ecd2019-11-18 16:00:16 -08001060func (m *ModuleBase) InRecovery() bool {
1061 return m.base().commonProperties.ImageVariation == RecoveryVariation
1062}
1063
Jiyong Park6a8cf5f2019-12-30 16:31:09 +09001064func (m *ModuleBase) RequiredModuleNames() []string {
1065 return m.base().commonProperties.Required
1066}
1067
1068func (m *ModuleBase) HostRequiredModuleNames() []string {
1069 return m.base().commonProperties.Host_required
1070}
1071
1072func (m *ModuleBase) TargetRequiredModuleNames() []string {
1073 return m.base().commonProperties.Target_required
1074}
1075
Inseob Kim8471cda2019-11-15 09:59:12 +09001076func (m *ModuleBase) InitRc() Paths {
1077 return append(Paths{}, m.initRcPaths...)
1078}
1079
1080func (m *ModuleBase) VintfFragments() Paths {
1081 return append(Paths{}, m.vintfFragmentsPaths...)
1082}
1083
Colin Cross4157e882019-06-06 16:57:04 -07001084func (m *ModuleBase) generateModuleTarget(ctx ModuleContext) {
Colin Cross897266e2020-02-13 13:22:08 -08001085 var allInstalledFiles InstallPaths
1086 var allCheckbuildFiles Paths
Colin Cross0875c522017-11-28 17:34:01 -08001087 ctx.VisitAllModuleVariants(func(module Module) {
1088 a := module.base()
Colin Crossc9404352015-03-26 16:10:12 -07001089 allInstalledFiles = append(allInstalledFiles, a.installFiles...)
1090 allCheckbuildFiles = append(allCheckbuildFiles, a.checkbuildFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001091 })
1092
Colin Cross0875c522017-11-28 17:34:01 -08001093 var deps Paths
Colin Cross9454bfa2015-03-17 13:24:18 -07001094
Jeff Gaston088e29e2017-11-29 16:47:17 -08001095 namespacePrefix := ctx.Namespace().(*Namespace).id
1096 if namespacePrefix != "" {
1097 namespacePrefix = namespacePrefix + "-"
1098 }
1099
Colin Cross3f40fa42015-01-30 17:27:36 -08001100 if len(allInstalledFiles) > 0 {
Colin Crossc3d87d32020-06-04 13:25:17 -07001101 name := namespacePrefix + ctx.ModuleName() + "-install"
1102 ctx.Phony(name, allInstalledFiles.Paths()...)
1103 m.installTarget = PathForPhony(ctx, name)
1104 deps = append(deps, m.installTarget)
Colin Cross9454bfa2015-03-17 13:24:18 -07001105 }
1106
1107 if len(allCheckbuildFiles) > 0 {
Colin Crossc3d87d32020-06-04 13:25:17 -07001108 name := namespacePrefix + ctx.ModuleName() + "-checkbuild"
1109 ctx.Phony(name, allCheckbuildFiles...)
1110 m.checkbuildTarget = PathForPhony(ctx, name)
1111 deps = append(deps, m.checkbuildTarget)
Colin Cross9454bfa2015-03-17 13:24:18 -07001112 }
1113
1114 if len(deps) > 0 {
Dan Willemsen5ba07e82015-12-11 13:51:06 -08001115 suffix := ""
Colin Crossaabf6792017-11-29 00:27:14 -08001116 if ctx.Config().EmbeddedInMake() {
Dan Willemsen5ba07e82015-12-11 13:51:06 -08001117 suffix = "-soong"
1118 }
1119
Colin Crossc3d87d32020-06-04 13:25:17 -07001120 ctx.Phony(namespacePrefix+ctx.ModuleName()+suffix, deps...)
Colin Cross1f8c52b2015-06-16 16:38:17 -07001121
Colin Cross4157e882019-06-06 16:57:04 -07001122 m.blueprintDir = ctx.ModuleDir()
Colin Cross3f40fa42015-01-30 17:27:36 -08001123 }
1124}
1125
Colin Crossc34d2322020-01-03 15:23:27 -08001126func determineModuleKind(m *ModuleBase, ctx blueprint.EarlyModuleContext) moduleKind {
Colin Cross4157e882019-06-06 16:57:04 -07001127 var socSpecific = Bool(m.commonProperties.Vendor) || Bool(m.commonProperties.Proprietary) || Bool(m.commonProperties.Soc_specific)
1128 var deviceSpecific = Bool(m.commonProperties.Device_specific)
1129 var productSpecific = Bool(m.commonProperties.Product_specific)
Justin Yund5f6c822019-06-25 16:47:17 +09001130 var systemExtSpecific = Bool(m.commonProperties.System_ext_specific)
Jiyong Park2db76922017-11-08 16:03:48 +09001131
Dario Frenifd05a742018-05-29 13:28:54 +01001132 msg := "conflicting value set here"
1133 if socSpecific && deviceSpecific {
1134 ctx.PropertyErrorf("device_specific", "a module cannot be specific to SoC and device at the same time.")
Colin Cross4157e882019-06-06 16:57:04 -07001135 if Bool(m.commonProperties.Vendor) {
Jiyong Park2db76922017-11-08 16:03:48 +09001136 ctx.PropertyErrorf("vendor", msg)
1137 }
Colin Cross4157e882019-06-06 16:57:04 -07001138 if Bool(m.commonProperties.Proprietary) {
Jiyong Park2db76922017-11-08 16:03:48 +09001139 ctx.PropertyErrorf("proprietary", msg)
1140 }
Colin Cross4157e882019-06-06 16:57:04 -07001141 if Bool(m.commonProperties.Soc_specific) {
Jiyong Park2db76922017-11-08 16:03:48 +09001142 ctx.PropertyErrorf("soc_specific", msg)
1143 }
1144 }
1145
Justin Yund5f6c822019-06-25 16:47:17 +09001146 if productSpecific && systemExtSpecific {
1147 ctx.PropertyErrorf("product_specific", "a module cannot be specific to product and system_ext at the same time.")
1148 ctx.PropertyErrorf("system_ext_specific", msg)
Dario Frenifd05a742018-05-29 13:28:54 +01001149 }
1150
Justin Yund5f6c822019-06-25 16:47:17 +09001151 if (socSpecific || deviceSpecific) && (productSpecific || systemExtSpecific) {
Dario Frenifd05a742018-05-29 13:28:54 +01001152 if productSpecific {
1153 ctx.PropertyErrorf("product_specific", "a module cannot be specific to SoC or device and product at the same time.")
1154 } else {
Justin Yund5f6c822019-06-25 16:47:17 +09001155 ctx.PropertyErrorf("system_ext_specific", "a module cannot be specific to SoC or device and system_ext at the same time.")
Dario Frenifd05a742018-05-29 13:28:54 +01001156 }
1157 if deviceSpecific {
1158 ctx.PropertyErrorf("device_specific", msg)
1159 } else {
Colin Cross4157e882019-06-06 16:57:04 -07001160 if Bool(m.commonProperties.Vendor) {
Dario Frenifd05a742018-05-29 13:28:54 +01001161 ctx.PropertyErrorf("vendor", msg)
1162 }
Colin Cross4157e882019-06-06 16:57:04 -07001163 if Bool(m.commonProperties.Proprietary) {
Dario Frenifd05a742018-05-29 13:28:54 +01001164 ctx.PropertyErrorf("proprietary", msg)
1165 }
Colin Cross4157e882019-06-06 16:57:04 -07001166 if Bool(m.commonProperties.Soc_specific) {
Dario Frenifd05a742018-05-29 13:28:54 +01001167 ctx.PropertyErrorf("soc_specific", msg)
1168 }
1169 }
1170 }
1171
Jiyong Park2db76922017-11-08 16:03:48 +09001172 if productSpecific {
1173 return productSpecificModule
Justin Yund5f6c822019-06-25 16:47:17 +09001174 } else if systemExtSpecific {
1175 return systemExtSpecificModule
Jiyong Park2db76922017-11-08 16:03:48 +09001176 } else if deviceSpecific {
1177 return deviceSpecificModule
1178 } else if socSpecific {
1179 return socSpecificModule
1180 } else {
1181 return platformModule
1182 }
1183}
1184
Colin Crossc34d2322020-01-03 15:23:27 -08001185func (m *ModuleBase) earlyModuleContextFactory(ctx blueprint.EarlyModuleContext) earlyModuleContext {
Colin Cross1184b642019-12-30 18:43:07 -08001186 return earlyModuleContext{
Colin Crossc34d2322020-01-03 15:23:27 -08001187 EarlyModuleContext: ctx,
1188 kind: determineModuleKind(m, ctx),
1189 config: ctx.Config().(Config),
Colin Cross3f40fa42015-01-30 17:27:36 -08001190 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001191}
1192
Colin Cross1184b642019-12-30 18:43:07 -08001193func (m *ModuleBase) baseModuleContextFactory(ctx blueprint.BaseModuleContext) baseModuleContext {
1194 return baseModuleContext{
1195 bp: ctx,
1196 earlyModuleContext: m.earlyModuleContextFactory(ctx),
1197 os: m.commonProperties.CompileOS,
1198 target: m.commonProperties.CompileTarget,
1199 targetPrimary: m.commonProperties.CompilePrimary,
1200 multiTargets: m.commonProperties.CompileMultiTargets,
1201 }
1202}
1203
Colin Cross4157e882019-06-06 16:57:04 -07001204func (m *ModuleBase) GenerateBuildActions(blueprintCtx blueprint.ModuleContext) {
Colin Cross25de6c32019-06-06 14:29:25 -07001205 ctx := &moduleContext{
Colin Cross0ea8ba82019-06-06 14:33:29 -07001206 module: m.module,
Colin Crossdc35e212019-06-06 16:13:11 -07001207 bp: blueprintCtx,
Colin Cross0ea8ba82019-06-06 14:33:29 -07001208 baseModuleContext: m.baseModuleContextFactory(blueprintCtx),
1209 installDeps: m.computeInstallDeps(blueprintCtx),
1210 installFiles: m.installFiles,
Colin Cross0ea8ba82019-06-06 14:33:29 -07001211 variables: make(map[string]string),
Colin Cross3f40fa42015-01-30 17:27:36 -08001212 }
1213
Colin Cross6c4f21f2019-06-06 15:41:36 -07001214 // Temporarily continue to call blueprintCtx.GetMissingDependencies() to maintain the previous behavior of never
1215 // reporting missing dependency errors in Blueprint when AllowMissingDependencies == true.
1216 // TODO: This will be removed once defaults modules handle missing dependency errors
1217 blueprintCtx.GetMissingDependencies()
1218
Colin Crossdc35e212019-06-06 16:13:11 -07001219 // For the final GenerateAndroidBuildActions pass, require that all visited dependencies Soong modules and
Paul Duffin1356d8c2020-02-25 19:26:33 +00001220 // are enabled. Unless the module is a CommonOS variant which may have dependencies on disabled variants
1221 // (because the dependencies are added before the modules are disabled). The
1222 // GetOsSpecificVariantsOfCommonOSVariant(...) method will ensure that the disabled variants are
1223 // ignored.
1224 ctx.baseModuleContext.strictVisitDeps = !m.IsCommonOSVariant()
Colin Crossdc35e212019-06-06 16:13:11 -07001225
Colin Cross4c83e5c2019-02-25 14:54:28 -08001226 if ctx.config.captureBuild {
1227 ctx.ruleParams = make(map[blueprint.Rule]blueprint.RuleParams)
1228 }
1229
Colin Cross67a5c132017-05-09 13:45:28 -07001230 desc := "//" + ctx.ModuleDir() + ":" + ctx.ModuleName() + " "
1231 var suffix []string
Colin Cross0875c522017-11-28 17:34:01 -08001232 if ctx.Os().Class != Device && ctx.Os().Class != Generic {
1233 suffix = append(suffix, ctx.Os().String())
Colin Cross67a5c132017-05-09 13:45:28 -07001234 }
Colin Cross0875c522017-11-28 17:34:01 -08001235 if !ctx.PrimaryArch() {
1236 suffix = append(suffix, ctx.Arch().ArchType.String())
Colin Cross67a5c132017-05-09 13:45:28 -07001237 }
Dan Willemsenb13a9482020-02-14 11:25:54 -08001238 if apex, ok := m.module.(ApexModule); ok && !apex.IsForPlatform() {
1239 suffix = append(suffix, apex.ApexName())
1240 }
Colin Cross67a5c132017-05-09 13:45:28 -07001241
1242 ctx.Variable(pctx, "moduleDesc", desc)
1243
1244 s := ""
1245 if len(suffix) > 0 {
1246 s = " [" + strings.Join(suffix, " ") + "]"
1247 }
1248 ctx.Variable(pctx, "moduleDescSuffix", s)
1249
Dan Willemsen569edc52018-11-19 09:33:29 -08001250 // Some common property checks for properties that will be used later in androidmk.go
Colin Cross4157e882019-06-06 16:57:04 -07001251 if m.commonProperties.Dist.Dest != nil {
1252 _, err := validateSafePath(*m.commonProperties.Dist.Dest)
Dan Willemsen569edc52018-11-19 09:33:29 -08001253 if err != nil {
1254 ctx.PropertyErrorf("dist.dest", "%s", err.Error())
1255 }
1256 }
Colin Cross4157e882019-06-06 16:57:04 -07001257 if m.commonProperties.Dist.Dir != nil {
1258 _, err := validateSafePath(*m.commonProperties.Dist.Dir)
Dan Willemsen569edc52018-11-19 09:33:29 -08001259 if err != nil {
1260 ctx.PropertyErrorf("dist.dir", "%s", err.Error())
1261 }
1262 }
Colin Cross4157e882019-06-06 16:57:04 -07001263 if m.commonProperties.Dist.Suffix != nil {
1264 if strings.Contains(*m.commonProperties.Dist.Suffix, "/") {
Dan Willemsen569edc52018-11-19 09:33:29 -08001265 ctx.PropertyErrorf("dist.suffix", "Suffix may not contain a '/' character.")
1266 }
1267 }
1268
Colin Cross4157e882019-06-06 16:57:04 -07001269 if m.Enabled() {
Jooyung Hand48f3c32019-08-23 11:18:57 +09001270 // ensure all direct android.Module deps are enabled
1271 ctx.VisitDirectDepsBlueprint(func(bm blueprint.Module) {
1272 if _, ok := bm.(Module); ok {
1273 ctx.validateAndroidModule(bm, ctx.baseModuleContext.strictVisitDeps)
1274 }
1275 })
1276
Bob Badoura75b0572020-02-18 20:21:55 -08001277 m.noticeFiles = make([]Path, 0)
1278 optPath := OptionalPath{}
1279 notice := proptools.StringDefault(m.commonProperties.Notice, "")
Colin Cross4157e882019-06-06 16:57:04 -07001280 if module := SrcIsModule(notice); module != "" {
Bob Badoura75b0572020-02-18 20:21:55 -08001281 optPath = ctx.ExpandOptionalSource(&notice, "notice")
1282 } else if notice != "" {
Jiyong Park52818fc2019-03-18 12:01:38 +09001283 noticePath := filepath.Join(ctx.ModuleDir(), notice)
Bob Badoura75b0572020-02-18 20:21:55 -08001284 optPath = ExistentPathForSource(ctx, noticePath)
1285 }
1286 if optPath.Valid() {
1287 m.noticeFiles = append(m.noticeFiles, optPath.Path())
1288 } else {
1289 for _, notice = range []string{"LICENSE", "LICENCE", "NOTICE"} {
1290 noticePath := filepath.Join(ctx.ModuleDir(), notice)
1291 optPath = ExistentPathForSource(ctx, noticePath)
1292 if optPath.Valid() {
1293 m.noticeFiles = append(m.noticeFiles, optPath.Path())
1294 }
1295 }
Jaewoong Jung62707f72018-11-16 13:26:43 -08001296 }
Jaewoong Jung5b425e22019-06-17 17:40:56 -07001297
1298 m.module.GenerateAndroidBuildActions(ctx)
1299 if ctx.Failed() {
1300 return
1301 }
1302
1303 m.installFiles = append(m.installFiles, ctx.installFiles...)
1304 m.checkbuildFiles = append(m.checkbuildFiles, ctx.checkbuildFiles...)
Inseob Kim8471cda2019-11-15 09:59:12 +09001305 m.initRcPaths = PathsForModuleSrc(ctx, m.commonProperties.Init_rc)
1306 m.vintfFragmentsPaths = PathsForModuleSrc(ctx, m.commonProperties.Vintf_fragments)
Colin Crossc3d87d32020-06-04 13:25:17 -07001307 for k, v := range ctx.phonies {
1308 m.phonies[k] = append(m.phonies[k], v...)
1309 }
Colin Crossdc35e212019-06-06 16:13:11 -07001310 } else if ctx.Config().AllowMissingDependencies() {
1311 // If the module is not enabled it will not create any build rules, nothing will call
1312 // ctx.GetMissingDependencies(), and blueprint will consider the missing dependencies to be unhandled
1313 // and report them as an error even when AllowMissingDependencies = true. Call
1314 // ctx.GetMissingDependencies() here to tell blueprint not to handle them.
1315 ctx.GetMissingDependencies()
Colin Cross3f40fa42015-01-30 17:27:36 -08001316 }
1317
Colin Cross4157e882019-06-06 16:57:04 -07001318 if m == ctx.FinalModule().(Module).base() {
1319 m.generateModuleTarget(ctx)
Colin Cross9b1d13d2016-09-19 15:18:11 -07001320 if ctx.Failed() {
1321 return
1322 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001323 }
Colin Crosscec81712017-07-13 14:43:27 -07001324
Colin Cross4157e882019-06-06 16:57:04 -07001325 m.buildParams = ctx.buildParams
1326 m.ruleParams = ctx.ruleParams
1327 m.variables = ctx.variables
Colin Cross3f40fa42015-01-30 17:27:36 -08001328}
1329
Colin Cross1184b642019-12-30 18:43:07 -08001330type earlyModuleContext struct {
Colin Crossc34d2322020-01-03 15:23:27 -08001331 blueprint.EarlyModuleContext
Colin Cross1184b642019-12-30 18:43:07 -08001332
1333 kind moduleKind
1334 config Config
1335}
1336
1337func (e *earlyModuleContext) Glob(globPattern string, excludes []string) Paths {
1338 ret, err := e.GlobWithDeps(globPattern, excludes)
1339 if err != nil {
1340 e.ModuleErrorf("glob: %s", err.Error())
1341 }
1342 return pathsForModuleSrcFromFullPath(e, ret, true)
1343}
1344
1345func (e *earlyModuleContext) GlobFiles(globPattern string, excludes []string) Paths {
1346 ret, err := e.GlobWithDeps(globPattern, excludes)
1347 if err != nil {
1348 e.ModuleErrorf("glob: %s", err.Error())
1349 }
1350 return pathsForModuleSrcFromFullPath(e, ret, false)
1351}
1352
Colin Cross988414c2020-01-11 01:11:46 +00001353func (b *earlyModuleContext) IsSymlink(path Path) bool {
1354 fileInfo, err := b.config.fs.Lstat(path.String())
1355 if err != nil {
1356 b.ModuleErrorf("os.Lstat(%q) failed: %s", path.String(), err)
1357 }
1358 return fileInfo.Mode()&os.ModeSymlink == os.ModeSymlink
1359}
1360
1361func (b *earlyModuleContext) Readlink(path Path) string {
1362 dest, err := b.config.fs.Readlink(path.String())
1363 if err != nil {
1364 b.ModuleErrorf("os.Readlink(%q) failed: %s", path.String(), err)
1365 }
1366 return dest
1367}
1368
Colin Cross1184b642019-12-30 18:43:07 -08001369func (e *earlyModuleContext) Module() Module {
Colin Crossc34d2322020-01-03 15:23:27 -08001370 module, _ := e.EarlyModuleContext.Module().(Module)
Colin Cross1184b642019-12-30 18:43:07 -08001371 return module
1372}
1373
1374func (e *earlyModuleContext) Config() Config {
Colin Crossc34d2322020-01-03 15:23:27 -08001375 return e.EarlyModuleContext.Config().(Config)
Colin Cross1184b642019-12-30 18:43:07 -08001376}
1377
1378func (e *earlyModuleContext) AConfig() Config {
1379 return e.config
1380}
1381
1382func (e *earlyModuleContext) DeviceConfig() DeviceConfig {
1383 return DeviceConfig{e.config.deviceConfig}
1384}
1385
1386func (e *earlyModuleContext) Platform() bool {
1387 return e.kind == platformModule
1388}
1389
1390func (e *earlyModuleContext) DeviceSpecific() bool {
1391 return e.kind == deviceSpecificModule
1392}
1393
1394func (e *earlyModuleContext) SocSpecific() bool {
1395 return e.kind == socSpecificModule
1396}
1397
1398func (e *earlyModuleContext) ProductSpecific() bool {
1399 return e.kind == productSpecificModule
1400}
1401
1402func (e *earlyModuleContext) SystemExtSpecific() bool {
1403 return e.kind == systemExtSpecificModule
1404}
1405
1406type baseModuleContext struct {
1407 bp blueprint.BaseModuleContext
1408 earlyModuleContext
Colin Crossfb0c16e2019-11-20 17:12:35 -08001409 os OsType
Colin Cross8b74d172016-09-13 09:59:14 -07001410 target Target
Colin Crossee0bc3b2018-10-02 22:01:37 -07001411 multiTargets []Target
Colin Cross8b74d172016-09-13 09:59:14 -07001412 targetPrimary bool
1413 debug bool
Colin Crossdc35e212019-06-06 16:13:11 -07001414
1415 walkPath []Module
Paul Duffinc5192442020-03-31 11:31:36 +01001416 tagPath []blueprint.DependencyTag
Colin Crossdc35e212019-06-06 16:13:11 -07001417
1418 strictVisitDeps bool // If true, enforce that all dependencies are enabled
Colin Crossf6566ed2015-03-24 11:13:38 -07001419}
1420
Paul Duffinca7f0ef2020-02-25 15:50:49 +00001421func (b *baseModuleContext) OtherModuleName(m blueprint.Module) string {
1422 return b.bp.OtherModuleName(m)
1423}
1424func (b *baseModuleContext) OtherModuleDir(m blueprint.Module) string { return b.bp.OtherModuleDir(m) }
Colin Cross1184b642019-12-30 18:43:07 -08001425func (b *baseModuleContext) OtherModuleErrorf(m blueprint.Module, fmt string, args ...interface{}) {
Jooyung Hancd87c692020-02-26 02:05:18 +09001426 b.bp.OtherModuleErrorf(m, fmt, args...)
Colin Cross1184b642019-12-30 18:43:07 -08001427}
1428func (b *baseModuleContext) OtherModuleDependencyTag(m blueprint.Module) blueprint.DependencyTag {
1429 return b.bp.OtherModuleDependencyTag(m)
1430}
Paul Duffinca7f0ef2020-02-25 15:50:49 +00001431func (b *baseModuleContext) OtherModuleExists(name string) bool { return b.bp.OtherModuleExists(name) }
Martin Stjernholm009a9dc2020-03-05 17:34:13 +00001432func (b *baseModuleContext) OtherModuleDependencyVariantExists(variations []blueprint.Variation, name string) bool {
1433 return b.bp.OtherModuleDependencyVariantExists(variations, name)
1434}
1435func (b *baseModuleContext) OtherModuleReverseDependencyVariantExists(name string) bool {
1436 return b.bp.OtherModuleReverseDependencyVariantExists(name)
1437}
Paul Duffinca7f0ef2020-02-25 15:50:49 +00001438func (b *baseModuleContext) OtherModuleType(m blueprint.Module) string {
1439 return b.bp.OtherModuleType(m)
1440}
Colin Cross1184b642019-12-30 18:43:07 -08001441
1442func (b *baseModuleContext) GetDirectDepWithTag(name string, tag blueprint.DependencyTag) blueprint.Module {
1443 return b.bp.GetDirectDepWithTag(name, tag)
1444}
1445
Paul Duffinf88d8e02020-05-07 20:21:34 +01001446func (b *baseModuleContext) blueprintBaseModuleContext() blueprint.BaseModuleContext {
1447 return b.bp
1448}
1449
Colin Cross25de6c32019-06-06 14:29:25 -07001450type moduleContext struct {
Colin Crossdc35e212019-06-06 16:13:11 -07001451 bp blueprint.ModuleContext
Colin Cross0ea8ba82019-06-06 14:33:29 -07001452 baseModuleContext
Colin Cross897266e2020-02-13 13:22:08 -08001453 installDeps InstallPaths
1454 installFiles InstallPaths
Dan Willemsen34cc69e2015-09-23 15:26:20 -07001455 checkbuildFiles Paths
Colin Cross8d8f8e22016-08-03 11:57:50 -07001456 module Module
Colin Crossc3d87d32020-06-04 13:25:17 -07001457 phonies map[string]Paths
Colin Crosscec81712017-07-13 14:43:27 -07001458
1459 // For tests
Colin Crossae887032017-10-23 17:16:14 -07001460 buildParams []BuildParams
Colin Cross4c83e5c2019-02-25 14:54:28 -08001461 ruleParams map[blueprint.Rule]blueprint.RuleParams
Jaewoong Jung38e4fb22018-12-12 09:01:34 -08001462 variables map[string]string
Colin Cross6ff51382015-12-17 16:39:19 -08001463}
1464
Colin Crossb88b3c52019-06-10 15:15:17 -07001465func (m *moduleContext) ninjaError(params BuildParams, err error) (PackageContext, BuildParams) {
1466 return pctx, BuildParams{
Colin Cross4b69c492019-06-07 13:06:06 -07001467 Rule: ErrorRule,
1468 Description: params.Description,
1469 Output: params.Output,
1470 Outputs: params.Outputs,
1471 ImplicitOutput: params.ImplicitOutput,
1472 ImplicitOutputs: params.ImplicitOutputs,
Colin Cross6ff51382015-12-17 16:39:19 -08001473 Args: map[string]string{
1474 "error": err.Error(),
1475 },
Colin Crossb88b3c52019-06-10 15:15:17 -07001476 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001477}
1478
Colin Cross25de6c32019-06-06 14:29:25 -07001479func (m *moduleContext) ModuleBuild(pctx PackageContext, params ModuleBuildParams) {
1480 m.Build(pctx, BuildParams(params))
Colin Cross3f40fa42015-01-30 17:27:36 -08001481}
1482
Colin Cross0875c522017-11-28 17:34:01 -08001483func convertBuildParams(params BuildParams) blueprint.BuildParams {
Dan Willemsen34cc69e2015-09-23 15:26:20 -07001484 bparams := blueprint.BuildParams{
Dan Willemsen9f3c5742016-11-03 14:28:31 -07001485 Rule: params.Rule,
Colin Cross0875c522017-11-28 17:34:01 -08001486 Description: params.Description,
Colin Cross33bfb0a2016-11-21 17:23:08 -08001487 Deps: params.Deps,
Dan Willemsen9f3c5742016-11-03 14:28:31 -07001488 Outputs: params.Outputs.Strings(),
1489 ImplicitOutputs: params.ImplicitOutputs.Strings(),
1490 Inputs: params.Inputs.Strings(),
1491 Implicits: params.Implicits.Strings(),
1492 OrderOnly: params.OrderOnly.Strings(),
1493 Args: params.Args,
1494 Optional: !params.Default,
Dan Willemsen34cc69e2015-09-23 15:26:20 -07001495 }
1496
Colin Cross33bfb0a2016-11-21 17:23:08 -08001497 if params.Depfile != nil {
1498 bparams.Depfile = params.Depfile.String()
1499 }
Dan Willemsen34cc69e2015-09-23 15:26:20 -07001500 if params.Output != nil {
1501 bparams.Outputs = append(bparams.Outputs, params.Output.String())
1502 }
Dan Willemsen9f3c5742016-11-03 14:28:31 -07001503 if params.ImplicitOutput != nil {
1504 bparams.ImplicitOutputs = append(bparams.ImplicitOutputs, params.ImplicitOutput.String())
1505 }
Dan Willemsen34cc69e2015-09-23 15:26:20 -07001506 if params.Input != nil {
1507 bparams.Inputs = append(bparams.Inputs, params.Input.String())
1508 }
1509 if params.Implicit != nil {
1510 bparams.Implicits = append(bparams.Implicits, params.Implicit.String())
1511 }
1512
Colin Cross0b9f31f2019-02-28 11:00:01 -08001513 bparams.Outputs = proptools.NinjaEscapeList(bparams.Outputs)
1514 bparams.ImplicitOutputs = proptools.NinjaEscapeList(bparams.ImplicitOutputs)
1515 bparams.Inputs = proptools.NinjaEscapeList(bparams.Inputs)
1516 bparams.Implicits = proptools.NinjaEscapeList(bparams.Implicits)
1517 bparams.OrderOnly = proptools.NinjaEscapeList(bparams.OrderOnly)
1518 bparams.Depfile = proptools.NinjaEscapeList([]string{bparams.Depfile})[0]
Colin Crossfe4bc362018-09-12 10:02:13 -07001519
Colin Cross0875c522017-11-28 17:34:01 -08001520 return bparams
1521}
1522
Colin Cross25de6c32019-06-06 14:29:25 -07001523func (m *moduleContext) Variable(pctx PackageContext, name, value string) {
1524 if m.config.captureBuild {
1525 m.variables[name] = value
Jaewoong Jung38e4fb22018-12-12 09:01:34 -08001526 }
1527
Colin Crossdc35e212019-06-06 16:13:11 -07001528 m.bp.Variable(pctx.PackageContext, name, value)
Colin Cross0875c522017-11-28 17:34:01 -08001529}
1530
Colin Cross25de6c32019-06-06 14:29:25 -07001531func (m *moduleContext) Rule(pctx PackageContext, name string, params blueprint.RuleParams,
Colin Cross0875c522017-11-28 17:34:01 -08001532 argNames ...string) blueprint.Rule {
1533
Ramy Medhat944839a2020-03-31 22:14:52 -04001534 if m.config.UseRemoteBuild() {
1535 if params.Pool == nil {
1536 // When USE_GOMA=true or USE_RBE=true are set and the rule is not supported by goma/RBE, restrict
1537 // jobs to the local parallelism value
1538 params.Pool = localPool
1539 } else if params.Pool == remotePool {
1540 // remotePool is a fake pool used to identify rule that are supported for remoting. If the rule's
1541 // pool is the remotePool, replace with nil so that ninja runs it at NINJA_REMOTE_NUM_JOBS
1542 // parallelism.
1543 params.Pool = nil
1544 }
Colin Cross2e2dbc22019-09-25 13:31:46 -07001545 }
1546
Colin Crossdc35e212019-06-06 16:13:11 -07001547 rule := m.bp.Rule(pctx.PackageContext, name, params, argNames...)
Colin Cross4c83e5c2019-02-25 14:54:28 -08001548
Colin Cross25de6c32019-06-06 14:29:25 -07001549 if m.config.captureBuild {
1550 m.ruleParams[rule] = params
Colin Cross4c83e5c2019-02-25 14:54:28 -08001551 }
1552
1553 return rule
Colin Cross0875c522017-11-28 17:34:01 -08001554}
1555
Colin Cross25de6c32019-06-06 14:29:25 -07001556func (m *moduleContext) Build(pctx PackageContext, params BuildParams) {
Colin Crossb88b3c52019-06-10 15:15:17 -07001557 if params.Description != "" {
1558 params.Description = "${moduleDesc}" + params.Description + "${moduleDescSuffix}"
1559 }
1560
1561 if missingDeps := m.GetMissingDependencies(); len(missingDeps) > 0 {
1562 pctx, params = m.ninjaError(params, fmt.Errorf("module %s missing dependencies: %s\n",
1563 m.ModuleName(), strings.Join(missingDeps, ", ")))
1564 }
1565
Colin Cross25de6c32019-06-06 14:29:25 -07001566 if m.config.captureBuild {
1567 m.buildParams = append(m.buildParams, params)
Colin Cross0875c522017-11-28 17:34:01 -08001568 }
1569
Colin Crossdc35e212019-06-06 16:13:11 -07001570 m.bp.Build(pctx.PackageContext, convertBuildParams(params))
Dan Willemsen34cc69e2015-09-23 15:26:20 -07001571}
Colin Crossc3d87d32020-06-04 13:25:17 -07001572
1573func (m *moduleContext) Phony(name string, deps ...Path) {
1574 addPhony(m.config, name, deps...)
1575}
1576
Colin Cross25de6c32019-06-06 14:29:25 -07001577func (m *moduleContext) GetMissingDependencies() []string {
Colin Cross6c4f21f2019-06-06 15:41:36 -07001578 var missingDeps []string
1579 missingDeps = append(missingDeps, m.Module().base().commonProperties.MissingDeps...)
Colin Crossdc35e212019-06-06 16:13:11 -07001580 missingDeps = append(missingDeps, m.bp.GetMissingDependencies()...)
Colin Cross6c4f21f2019-06-06 15:41:36 -07001581 missingDeps = FirstUniqueStrings(missingDeps)
1582 return missingDeps
Colin Cross6ff51382015-12-17 16:39:19 -08001583}
1584
Colin Crossdc35e212019-06-06 16:13:11 -07001585func (b *baseModuleContext) AddMissingDependencies(deps []string) {
Dan Willemsen6553f5e2016-03-10 18:14:25 -08001586 if deps != nil {
Colin Crossdc35e212019-06-06 16:13:11 -07001587 missingDeps := &b.Module().base().commonProperties.MissingDeps
Colin Cross6c4f21f2019-06-06 15:41:36 -07001588 *missingDeps = append(*missingDeps, deps...)
1589 *missingDeps = FirstUniqueStrings(*missingDeps)
Dan Willemsen6553f5e2016-03-10 18:14:25 -08001590 }
1591}
1592
Colin Crossdc35e212019-06-06 16:13:11 -07001593func (b *baseModuleContext) validateAndroidModule(module blueprint.Module, strict bool) Module {
Colin Crossd11fcda2017-10-23 17:59:01 -07001594 aModule, _ := module.(Module)
Colin Crossdc35e212019-06-06 16:13:11 -07001595
1596 if !strict {
1597 return aModule
1598 }
1599
Colin Cross380c69a2019-06-10 17:49:58 +00001600 if aModule == nil {
Colin Crossdc35e212019-06-06 16:13:11 -07001601 b.ModuleErrorf("module %q not an android module", b.OtherModuleName(module))
Colin Cross380c69a2019-06-10 17:49:58 +00001602 return nil
1603 }
1604
1605 if !aModule.Enabled() {
Colin Crossdc35e212019-06-06 16:13:11 -07001606 if b.Config().AllowMissingDependencies() {
1607 b.AddMissingDependencies([]string{b.OtherModuleName(aModule)})
Colin Cross380c69a2019-06-10 17:49:58 +00001608 } else {
Colin Crossdc35e212019-06-06 16:13:11 -07001609 b.ModuleErrorf("depends on disabled module %q", b.OtherModuleName(aModule))
Colin Cross380c69a2019-06-10 17:49:58 +00001610 }
1611 return nil
1612 }
Colin Crossd11fcda2017-10-23 17:59:01 -07001613 return aModule
1614}
1615
Colin Crossdc35e212019-06-06 16:13:11 -07001616func (b *baseModuleContext) getDirectDepInternal(name string, tag blueprint.DependencyTag) (blueprint.Module, blueprint.DependencyTag) {
Jiyong Parkf2976302019-04-17 21:47:37 +09001617 type dep struct {
1618 mod blueprint.Module
1619 tag blueprint.DependencyTag
1620 }
1621 var deps []dep
Colin Crossdc35e212019-06-06 16:13:11 -07001622 b.VisitDirectDepsBlueprint(func(module blueprint.Module) {
Colin Cross25de6c32019-06-06 14:29:25 -07001623 if aModule, _ := module.(Module); aModule != nil && aModule.base().BaseModuleName() == name {
Colin Cross1184b642019-12-30 18:43:07 -08001624 returnedTag := b.bp.OtherModuleDependencyTag(aModule)
Jiyong Parkf2976302019-04-17 21:47:37 +09001625 if tag == nil || returnedTag == tag {
1626 deps = append(deps, dep{aModule, returnedTag})
1627 }
1628 }
1629 })
1630 if len(deps) == 1 {
1631 return deps[0].mod, deps[0].tag
1632 } else if len(deps) >= 2 {
1633 panic(fmt.Errorf("Multiple dependencies having same BaseModuleName() %q found from %q",
Colin Crossdc35e212019-06-06 16:13:11 -07001634 name, b.ModuleName()))
Jiyong Parkf2976302019-04-17 21:47:37 +09001635 } else {
1636 return nil, nil
1637 }
1638}
1639
Colin Crossdc35e212019-06-06 16:13:11 -07001640func (b *baseModuleContext) GetDirectDepsWithTag(tag blueprint.DependencyTag) []Module {
Colin Cross0ef08162019-05-01 15:50:51 -07001641 var deps []Module
Colin Crossdc35e212019-06-06 16:13:11 -07001642 b.VisitDirectDepsBlueprint(func(module blueprint.Module) {
Colin Cross25de6c32019-06-06 14:29:25 -07001643 if aModule, _ := module.(Module); aModule != nil {
Colin Cross1184b642019-12-30 18:43:07 -08001644 if b.bp.OtherModuleDependencyTag(aModule) == tag {
Colin Cross0ef08162019-05-01 15:50:51 -07001645 deps = append(deps, aModule)
1646 }
1647 }
1648 })
1649 return deps
1650}
1651
Colin Cross25de6c32019-06-06 14:29:25 -07001652func (m *moduleContext) GetDirectDepWithTag(name string, tag blueprint.DependencyTag) blueprint.Module {
1653 module, _ := m.getDirectDepInternal(name, tag)
1654 return module
Jiyong Parkf2976302019-04-17 21:47:37 +09001655}
1656
Colin Crossdc35e212019-06-06 16:13:11 -07001657func (b *baseModuleContext) GetDirectDep(name string) (blueprint.Module, blueprint.DependencyTag) {
1658 return b.getDirectDepInternal(name, nil)
Jiyong Parkf2976302019-04-17 21:47:37 +09001659}
1660
Colin Crossdc35e212019-06-06 16:13:11 -07001661func (b *baseModuleContext) VisitDirectDepsBlueprint(visit func(blueprint.Module)) {
Colin Cross1184b642019-12-30 18:43:07 -08001662 b.bp.VisitDirectDeps(visit)
Colin Cross35143d02017-11-16 00:11:20 -08001663}
1664
Colin Crossdc35e212019-06-06 16:13:11 -07001665func (b *baseModuleContext) VisitDirectDeps(visit func(Module)) {
Colin Cross1184b642019-12-30 18:43:07 -08001666 b.bp.VisitDirectDeps(func(module blueprint.Module) {
Colin Crossdc35e212019-06-06 16:13:11 -07001667 if aModule := b.validateAndroidModule(module, b.strictVisitDeps); aModule != nil {
Colin Crossd11fcda2017-10-23 17:59:01 -07001668 visit(aModule)
1669 }
1670 })
1671}
1672
Colin Crossdc35e212019-06-06 16:13:11 -07001673func (b *baseModuleContext) VisitDirectDepsWithTag(tag blueprint.DependencyTag, visit func(Module)) {
Colin Cross1184b642019-12-30 18:43:07 -08001674 b.bp.VisitDirectDeps(func(module blueprint.Module) {
Colin Crossdc35e212019-06-06 16:13:11 -07001675 if aModule := b.validateAndroidModule(module, b.strictVisitDeps); aModule != nil {
Colin Cross1184b642019-12-30 18:43:07 -08001676 if b.bp.OtherModuleDependencyTag(aModule) == tag {
Colin Crossee6143c2017-12-30 17:54:27 -08001677 visit(aModule)
1678 }
1679 }
1680 })
1681}
1682
Colin Crossdc35e212019-06-06 16:13:11 -07001683func (b *baseModuleContext) VisitDirectDepsIf(pred func(Module) bool, visit func(Module)) {
Colin Cross1184b642019-12-30 18:43:07 -08001684 b.bp.VisitDirectDepsIf(
Colin Crossd11fcda2017-10-23 17:59:01 -07001685 // pred
1686 func(module blueprint.Module) bool {
Colin Crossdc35e212019-06-06 16:13:11 -07001687 if aModule := b.validateAndroidModule(module, b.strictVisitDeps); aModule != nil {
Colin Crossd11fcda2017-10-23 17:59:01 -07001688 return pred(aModule)
1689 } else {
1690 return false
1691 }
1692 },
1693 // visit
1694 func(module blueprint.Module) {
1695 visit(module.(Module))
1696 })
1697}
1698
Colin Crossdc35e212019-06-06 16:13:11 -07001699func (b *baseModuleContext) VisitDepsDepthFirst(visit func(Module)) {
Colin Cross1184b642019-12-30 18:43:07 -08001700 b.bp.VisitDepsDepthFirst(func(module blueprint.Module) {
Colin Crossdc35e212019-06-06 16:13:11 -07001701 if aModule := b.validateAndroidModule(module, b.strictVisitDeps); aModule != nil {
Colin Crossd11fcda2017-10-23 17:59:01 -07001702 visit(aModule)
1703 }
1704 })
1705}
1706
Colin Crossdc35e212019-06-06 16:13:11 -07001707func (b *baseModuleContext) VisitDepsDepthFirstIf(pred func(Module) bool, visit func(Module)) {
Colin Cross1184b642019-12-30 18:43:07 -08001708 b.bp.VisitDepsDepthFirstIf(
Colin Crossd11fcda2017-10-23 17:59:01 -07001709 // pred
1710 func(module blueprint.Module) bool {
Colin Crossdc35e212019-06-06 16:13:11 -07001711 if aModule := b.validateAndroidModule(module, b.strictVisitDeps); aModule != nil {
Colin Crossd11fcda2017-10-23 17:59:01 -07001712 return pred(aModule)
1713 } else {
1714 return false
1715 }
1716 },
1717 // visit
1718 func(module blueprint.Module) {
1719 visit(module.(Module))
1720 })
1721}
1722
Colin Crossdc35e212019-06-06 16:13:11 -07001723func (b *baseModuleContext) WalkDepsBlueprint(visit func(blueprint.Module, blueprint.Module) bool) {
Colin Cross1184b642019-12-30 18:43:07 -08001724 b.bp.WalkDeps(visit)
Alex Light778127a2019-02-27 14:19:50 -08001725}
1726
Colin Crossdc35e212019-06-06 16:13:11 -07001727func (b *baseModuleContext) WalkDeps(visit func(Module, Module) bool) {
1728 b.walkPath = []Module{b.Module()}
Paul Duffinc5192442020-03-31 11:31:36 +01001729 b.tagPath = []blueprint.DependencyTag{}
Colin Cross1184b642019-12-30 18:43:07 -08001730 b.bp.WalkDeps(func(child, parent blueprint.Module) bool {
Colin Crossdc35e212019-06-06 16:13:11 -07001731 childAndroidModule, _ := child.(Module)
1732 parentAndroidModule, _ := parent.(Module)
Colin Crossd11fcda2017-10-23 17:59:01 -07001733 if childAndroidModule != nil && parentAndroidModule != nil {
Colin Crossdc35e212019-06-06 16:13:11 -07001734 // record walkPath before visit
1735 for b.walkPath[len(b.walkPath)-1] != parentAndroidModule {
1736 b.walkPath = b.walkPath[0 : len(b.walkPath)-1]
Paul Duffinc5192442020-03-31 11:31:36 +01001737 b.tagPath = b.tagPath[0 : len(b.tagPath)-1]
Colin Crossdc35e212019-06-06 16:13:11 -07001738 }
1739 b.walkPath = append(b.walkPath, childAndroidModule)
Paul Duffinc5192442020-03-31 11:31:36 +01001740 b.tagPath = append(b.tagPath, b.OtherModuleDependencyTag(childAndroidModule))
Colin Crossd11fcda2017-10-23 17:59:01 -07001741 return visit(childAndroidModule, parentAndroidModule)
1742 } else {
1743 return false
1744 }
1745 })
1746}
1747
Colin Crossdc35e212019-06-06 16:13:11 -07001748func (b *baseModuleContext) GetWalkPath() []Module {
1749 return b.walkPath
1750}
1751
Paul Duffinc5192442020-03-31 11:31:36 +01001752func (b *baseModuleContext) GetTagPath() []blueprint.DependencyTag {
1753 return b.tagPath
1754}
1755
Jiyong Park1c7e9622020-05-07 16:12:13 +09001756// A regexp for removing boilerplate from BaseDependencyTag from the string representation of
1757// a dependency tag.
1758var tagCleaner = regexp.MustCompile(`\QBaseDependencyTag:blueprint.BaseDependencyTag{}\E(, )?`)
1759
1760// PrettyPrintTag returns string representation of the tag, but prefers
1761// custom String() method if available.
1762func PrettyPrintTag(tag blueprint.DependencyTag) string {
1763 // Use tag's custom String() method if available.
1764 if stringer, ok := tag.(fmt.Stringer); ok {
1765 return stringer.String()
1766 }
1767
1768 // Otherwise, get a default string representation of the tag's struct.
1769 tagString := fmt.Sprintf("%#v", tag)
1770
1771 // Remove the boilerplate from BaseDependencyTag as it adds no value.
1772 tagString = tagCleaner.ReplaceAllString(tagString, "")
1773 return tagString
1774}
1775
1776func (b *baseModuleContext) GetPathString(skipFirst bool) string {
1777 sb := strings.Builder{}
1778 tagPath := b.GetTagPath()
1779 walkPath := b.GetWalkPath()
1780 if !skipFirst {
1781 sb.WriteString(walkPath[0].String())
1782 }
1783 for i, m := range walkPath[1:] {
1784 sb.WriteString("\n")
1785 sb.WriteString(fmt.Sprintf(" via tag %s\n", PrettyPrintTag(tagPath[i])))
1786 sb.WriteString(fmt.Sprintf(" -> %s", m.String()))
1787 }
1788 return sb.String()
1789}
1790
Colin Cross25de6c32019-06-06 14:29:25 -07001791func (m *moduleContext) VisitAllModuleVariants(visit func(Module)) {
Colin Crossdc35e212019-06-06 16:13:11 -07001792 m.bp.VisitAllModuleVariants(func(module blueprint.Module) {
Colin Cross0875c522017-11-28 17:34:01 -08001793 visit(module.(Module))
1794 })
1795}
1796
Colin Cross25de6c32019-06-06 14:29:25 -07001797func (m *moduleContext) PrimaryModule() Module {
Colin Crossdc35e212019-06-06 16:13:11 -07001798 return m.bp.PrimaryModule().(Module)
Colin Cross0875c522017-11-28 17:34:01 -08001799}
1800
Colin Cross25de6c32019-06-06 14:29:25 -07001801func (m *moduleContext) FinalModule() Module {
Colin Crossdc35e212019-06-06 16:13:11 -07001802 return m.bp.FinalModule().(Module)
1803}
1804
1805func (m *moduleContext) ModuleSubDir() string {
1806 return m.bp.ModuleSubDir()
Colin Cross0875c522017-11-28 17:34:01 -08001807}
1808
Colin Cross0ea8ba82019-06-06 14:33:29 -07001809func (b *baseModuleContext) Target() Target {
Colin Cross25de6c32019-06-06 14:29:25 -07001810 return b.target
Colin Crossa1ad8d12016-06-01 17:09:44 -07001811}
1812
Colin Cross0ea8ba82019-06-06 14:33:29 -07001813func (b *baseModuleContext) TargetPrimary() bool {
Colin Cross25de6c32019-06-06 14:29:25 -07001814 return b.targetPrimary
Colin Cross8b74d172016-09-13 09:59:14 -07001815}
1816
Colin Cross0ea8ba82019-06-06 14:33:29 -07001817func (b *baseModuleContext) MultiTargets() []Target {
Colin Cross25de6c32019-06-06 14:29:25 -07001818 return b.multiTargets
Colin Crossee0bc3b2018-10-02 22:01:37 -07001819}
1820
Colin Cross0ea8ba82019-06-06 14:33:29 -07001821func (b *baseModuleContext) Arch() Arch {
Colin Cross25de6c32019-06-06 14:29:25 -07001822 return b.target.Arch
Colin Cross3f40fa42015-01-30 17:27:36 -08001823}
1824
Colin Cross0ea8ba82019-06-06 14:33:29 -07001825func (b *baseModuleContext) Os() OsType {
Colin Crossfb0c16e2019-11-20 17:12:35 -08001826 return b.os
Dan Willemsen490fd492015-11-24 17:53:15 -08001827}
1828
Colin Cross0ea8ba82019-06-06 14:33:29 -07001829func (b *baseModuleContext) Host() bool {
Colin Crossfb0c16e2019-11-20 17:12:35 -08001830 return b.os.Class == Host || b.os.Class == HostCross
Colin Crossf6566ed2015-03-24 11:13:38 -07001831}
1832
Colin Cross0ea8ba82019-06-06 14:33:29 -07001833func (b *baseModuleContext) Device() bool {
Colin Crossfb0c16e2019-11-20 17:12:35 -08001834 return b.os.Class == Device
Colin Crossf6566ed2015-03-24 11:13:38 -07001835}
1836
Colin Cross0ea8ba82019-06-06 14:33:29 -07001837func (b *baseModuleContext) Darwin() bool {
Colin Crossfb0c16e2019-11-20 17:12:35 -08001838 return b.os == Darwin
Colin Cross0af4b842015-04-30 16:36:18 -07001839}
1840
Colin Cross0ea8ba82019-06-06 14:33:29 -07001841func (b *baseModuleContext) Fuchsia() bool {
Colin Crossfb0c16e2019-11-20 17:12:35 -08001842 return b.os == Fuchsia
Doug Horn21b94272019-01-16 12:06:11 -08001843}
1844
Colin Cross0ea8ba82019-06-06 14:33:29 -07001845func (b *baseModuleContext) Windows() bool {
Colin Crossfb0c16e2019-11-20 17:12:35 -08001846 return b.os == Windows
Colin Cross3edeee12017-04-04 12:59:48 -07001847}
1848
Colin Cross0ea8ba82019-06-06 14:33:29 -07001849func (b *baseModuleContext) Debug() bool {
Colin Cross25de6c32019-06-06 14:29:25 -07001850 return b.debug
Colin Crossf6566ed2015-03-24 11:13:38 -07001851}
1852
Colin Cross0ea8ba82019-06-06 14:33:29 -07001853func (b *baseModuleContext) PrimaryArch() bool {
Colin Cross25de6c32019-06-06 14:29:25 -07001854 if len(b.config.Targets[b.target.Os]) <= 1 {
Colin Cross67a5c132017-05-09 13:45:28 -07001855 return true
1856 }
Colin Cross25de6c32019-06-06 14:29:25 -07001857 return b.target.Arch.ArchType == b.config.Targets[b.target.Os][0].Arch.ArchType
Colin Cross1e7d3702016-08-24 15:25:47 -07001858}
1859
Jiyong Park5baac542018-08-28 09:55:37 +09001860// Makes this module a platform module, i.e. not specific to soc, device,
Justin Yund5f6c822019-06-25 16:47:17 +09001861// product, or system_ext.
Colin Cross4157e882019-06-06 16:57:04 -07001862func (m *ModuleBase) MakeAsPlatform() {
1863 m.commonProperties.Vendor = boolPtr(false)
1864 m.commonProperties.Proprietary = boolPtr(false)
1865 m.commonProperties.Soc_specific = boolPtr(false)
1866 m.commonProperties.Product_specific = boolPtr(false)
Justin Yund5f6c822019-06-25 16:47:17 +09001867 m.commonProperties.System_ext_specific = boolPtr(false)
Jiyong Park5baac542018-08-28 09:55:37 +09001868}
1869
Colin Cross4157e882019-06-06 16:57:04 -07001870func (m *ModuleBase) EnableNativeBridgeSupportByDefault() {
1871 m.commonProperties.Native_bridge_supported = boolPtr(true)
dimitry03dc3f62019-05-09 14:07:34 +02001872}
1873
Sundong Ahnd95aa2d2019-10-08 19:34:03 +09001874func (m *ModuleBase) MakeAsSystemExt() {
Jooyung Han91df2082019-11-20 01:49:42 +09001875 m.commonProperties.Vendor = boolPtr(false)
1876 m.commonProperties.Proprietary = boolPtr(false)
1877 m.commonProperties.Soc_specific = boolPtr(false)
1878 m.commonProperties.Product_specific = boolPtr(false)
1879 m.commonProperties.System_ext_specific = boolPtr(true)
Sundong Ahnd95aa2d2019-10-08 19:34:03 +09001880}
1881
Jooyung Han344d5432019-08-23 11:17:39 +09001882// IsNativeBridgeSupported returns true if "native_bridge_supported" is explicitly set as "true"
1883func (m *ModuleBase) IsNativeBridgeSupported() bool {
1884 return proptools.Bool(m.commonProperties.Native_bridge_supported)
1885}
1886
Colin Cross25de6c32019-06-06 14:29:25 -07001887func (m *moduleContext) InstallInData() bool {
1888 return m.module.InstallInData()
Dan Willemsen782a2d12015-12-21 14:55:28 -08001889}
1890
Jaewoong Jung0949f312019-09-11 10:25:18 -07001891func (m *moduleContext) InstallInTestcases() bool {
1892 return m.module.InstallInTestcases()
1893}
1894
Colin Cross25de6c32019-06-06 14:29:25 -07001895func (m *moduleContext) InstallInSanitizerDir() bool {
1896 return m.module.InstallInSanitizerDir()
Vishwath Mohan1dd88392017-03-29 22:00:18 -07001897}
1898
Yifan Hong1b3348d2020-01-21 15:53:22 -08001899func (m *moduleContext) InstallInRamdisk() bool {
1900 return m.module.InstallInRamdisk()
1901}
1902
Colin Cross25de6c32019-06-06 14:29:25 -07001903func (m *moduleContext) InstallInRecovery() bool {
1904 return m.module.InstallInRecovery()
Jiyong Parkf9332f12018-02-01 00:54:12 +09001905}
1906
Colin Cross90ba5f42019-10-02 11:10:58 -07001907func (m *moduleContext) InstallInRoot() bool {
1908 return m.module.InstallInRoot()
1909}
1910
Colin Cross607d8582019-07-29 16:44:46 -07001911func (m *moduleContext) InstallBypassMake() bool {
1912 return m.module.InstallBypassMake()
1913}
1914
Colin Cross6e359402020-02-10 15:29:54 -08001915func (m *moduleContext) InstallForceOS() *OsType {
1916 return m.module.InstallForceOS()
1917}
1918
Colin Cross70dda7e2019-10-01 22:05:35 -07001919func (m *moduleContext) skipInstall(fullInstallPath InstallPath) bool {
Colin Cross25de6c32019-06-06 14:29:25 -07001920 if m.module.base().commonProperties.SkipInstall {
Colin Cross893d8162017-04-26 17:34:03 -07001921 return true
1922 }
1923
Colin Cross3607f212018-05-07 15:28:05 -07001924 // We'll need a solution for choosing which of modules with the same name in different
1925 // namespaces to install. For now, reuse the list of namespaces exported to Make as the
1926 // list of namespaces to install in a Soong-only build.
Colin Cross25de6c32019-06-06 14:29:25 -07001927 if !m.module.base().commonProperties.NamespaceExportedToMake {
Colin Cross3607f212018-05-07 15:28:05 -07001928 return true
1929 }
1930
Colin Cross25de6c32019-06-06 14:29:25 -07001931 if m.Device() {
Colin Cross607d8582019-07-29 16:44:46 -07001932 if m.Config().EmbeddedInMake() && !m.InstallBypassMake() {
Colin Cross893d8162017-04-26 17:34:03 -07001933 return true
1934 }
1935
Colin Cross25de6c32019-06-06 14:29:25 -07001936 if m.Config().SkipMegaDeviceInstall(fullInstallPath.String()) {
Colin Cross893d8162017-04-26 17:34:03 -07001937 return true
1938 }
1939 }
1940
1941 return false
1942}
1943
Colin Cross70dda7e2019-10-01 22:05:35 -07001944func (m *moduleContext) InstallFile(installPath InstallPath, name string, srcPath Path,
1945 deps ...Path) InstallPath {
Colin Cross25de6c32019-06-06 14:29:25 -07001946 return m.installFile(installPath, name, srcPath, Cp, deps)
Colin Cross5c517922017-08-31 12:29:17 -07001947}
1948
Colin Cross70dda7e2019-10-01 22:05:35 -07001949func (m *moduleContext) InstallExecutable(installPath InstallPath, name string, srcPath Path,
1950 deps ...Path) InstallPath {
Colin Cross25de6c32019-06-06 14:29:25 -07001951 return m.installFile(installPath, name, srcPath, CpExecutable, deps)
Colin Cross5c517922017-08-31 12:29:17 -07001952}
1953
Colin Cross70dda7e2019-10-01 22:05:35 -07001954func (m *moduleContext) installFile(installPath InstallPath, name string, srcPath Path,
1955 rule blueprint.Rule, deps []Path) InstallPath {
Colin Cross35cec122015-04-02 14:37:16 -07001956
Colin Cross25de6c32019-06-06 14:29:25 -07001957 fullInstallPath := installPath.Join(m, name)
1958 m.module.base().hooks.runInstallHooks(m, fullInstallPath, false)
Colin Cross3f40fa42015-01-30 17:27:36 -08001959
Colin Cross25de6c32019-06-06 14:29:25 -07001960 if !m.skipInstall(fullInstallPath) {
Colin Crossce75d2c2016-10-06 16:12:58 -07001961
Colin Cross897266e2020-02-13 13:22:08 -08001962 deps = append(deps, m.installDeps.Paths()...)
Colin Cross35cec122015-04-02 14:37:16 -07001963
Colin Cross89562dc2016-10-03 17:47:19 -07001964 var implicitDeps, orderOnlyDeps Paths
1965
Colin Cross25de6c32019-06-06 14:29:25 -07001966 if m.Host() {
Colin Cross89562dc2016-10-03 17:47:19 -07001967 // Installed host modules might be used during the build, depend directly on their
1968 // dependencies so their timestamp is updated whenever their dependency is updated
1969 implicitDeps = deps
1970 } else {
1971 orderOnlyDeps = deps
1972 }
1973
Colin Cross25de6c32019-06-06 14:29:25 -07001974 m.Build(pctx, BuildParams{
Colin Cross5c517922017-08-31 12:29:17 -07001975 Rule: rule,
Colin Cross67a5c132017-05-09 13:45:28 -07001976 Description: "install " + fullInstallPath.Base(),
1977 Output: fullInstallPath,
1978 Input: srcPath,
1979 Implicits: implicitDeps,
1980 OrderOnly: orderOnlyDeps,
Colin Cross25de6c32019-06-06 14:29:25 -07001981 Default: !m.Config().EmbeddedInMake(),
Dan Willemsen322acaf2016-01-12 23:07:05 -08001982 })
Colin Cross3f40fa42015-01-30 17:27:36 -08001983
Colin Cross25de6c32019-06-06 14:29:25 -07001984 m.installFiles = append(m.installFiles, fullInstallPath)
Dan Willemsen322acaf2016-01-12 23:07:05 -08001985 }
Colin Cross25de6c32019-06-06 14:29:25 -07001986 m.checkbuildFiles = append(m.checkbuildFiles, srcPath)
Colin Cross35cec122015-04-02 14:37:16 -07001987 return fullInstallPath
1988}
1989
Colin Cross70dda7e2019-10-01 22:05:35 -07001990func (m *moduleContext) InstallSymlink(installPath InstallPath, name string, srcPath InstallPath) InstallPath {
Colin Cross25de6c32019-06-06 14:29:25 -07001991 fullInstallPath := installPath.Join(m, name)
1992 m.module.base().hooks.runInstallHooks(m, fullInstallPath, true)
Colin Cross3854a602016-01-11 12:49:11 -08001993
Colin Cross25de6c32019-06-06 14:29:25 -07001994 if !m.skipInstall(fullInstallPath) {
Colin Crossce75d2c2016-10-06 16:12:58 -07001995
Alex Lightfb4353d2019-01-17 13:57:45 -08001996 relPath, err := filepath.Rel(path.Dir(fullInstallPath.String()), srcPath.String())
1997 if err != nil {
1998 panic(fmt.Sprintf("Unable to generate symlink between %q and %q: %s", fullInstallPath.Base(), srcPath.Base(), err))
1999 }
Colin Cross25de6c32019-06-06 14:29:25 -07002000 m.Build(pctx, BuildParams{
Colin Cross67a5c132017-05-09 13:45:28 -07002001 Rule: Symlink,
2002 Description: "install symlink " + fullInstallPath.Base(),
2003 Output: fullInstallPath,
Dan Willemsen40efa1c2020-01-14 15:19:52 -08002004 Input: srcPath,
Colin Cross25de6c32019-06-06 14:29:25 -07002005 Default: !m.Config().EmbeddedInMake(),
Colin Cross12fc4972016-01-11 12:49:11 -08002006 Args: map[string]string{
Alex Lightfb4353d2019-01-17 13:57:45 -08002007 "fromPath": relPath,
Colin Cross12fc4972016-01-11 12:49:11 -08002008 },
2009 })
Colin Cross3854a602016-01-11 12:49:11 -08002010
Colin Cross25de6c32019-06-06 14:29:25 -07002011 m.installFiles = append(m.installFiles, fullInstallPath)
2012 m.checkbuildFiles = append(m.checkbuildFiles, srcPath)
Colin Cross12fc4972016-01-11 12:49:11 -08002013 }
Colin Cross3854a602016-01-11 12:49:11 -08002014 return fullInstallPath
2015}
2016
Jiyong Parkf1194352019-02-25 11:05:47 +09002017// installPath/name -> absPath where absPath might be a path that is available only at runtime
2018// (e.g. /apex/...)
Colin Cross70dda7e2019-10-01 22:05:35 -07002019func (m *moduleContext) InstallAbsoluteSymlink(installPath InstallPath, name string, absPath string) InstallPath {
Colin Cross25de6c32019-06-06 14:29:25 -07002020 fullInstallPath := installPath.Join(m, name)
2021 m.module.base().hooks.runInstallHooks(m, fullInstallPath, true)
Jiyong Parkf1194352019-02-25 11:05:47 +09002022
Colin Cross25de6c32019-06-06 14:29:25 -07002023 if !m.skipInstall(fullInstallPath) {
2024 m.Build(pctx, BuildParams{
Jiyong Parkf1194352019-02-25 11:05:47 +09002025 Rule: Symlink,
2026 Description: "install symlink " + fullInstallPath.Base() + " -> " + absPath,
2027 Output: fullInstallPath,
Colin Cross25de6c32019-06-06 14:29:25 -07002028 Default: !m.Config().EmbeddedInMake(),
Jiyong Parkf1194352019-02-25 11:05:47 +09002029 Args: map[string]string{
2030 "fromPath": absPath,
2031 },
2032 })
2033
Colin Cross25de6c32019-06-06 14:29:25 -07002034 m.installFiles = append(m.installFiles, fullInstallPath)
Jiyong Parkf1194352019-02-25 11:05:47 +09002035 }
2036 return fullInstallPath
2037}
2038
Colin Cross25de6c32019-06-06 14:29:25 -07002039func (m *moduleContext) CheckbuildFile(srcPath Path) {
2040 m.checkbuildFiles = append(m.checkbuildFiles, srcPath)
Colin Cross3f40fa42015-01-30 17:27:36 -08002041}
2042
Dan Willemsen2ef08f42015-06-30 18:15:24 -07002043func findStringInSlice(str string, slice []string) int {
2044 for i, s := range slice {
2045 if s == str {
2046 return i
Colin Crossfce53272015-04-08 11:21:40 -07002047 }
2048 }
Dan Willemsen2ef08f42015-06-30 18:15:24 -07002049 return -1
2050}
2051
Colin Cross41955e82019-05-29 14:40:35 -07002052// SrcIsModule decodes module references in the format ":name" into the module name, or empty string if the input
2053// was not a module reference.
2054func SrcIsModule(s string) (module string) {
Colin Cross068e0fe2016-12-13 15:23:47 -08002055 if len(s) > 1 && s[0] == ':' {
2056 return s[1:]
2057 }
2058 return ""
2059}
2060
Colin Cross41955e82019-05-29 14:40:35 -07002061// SrcIsModule decodes module references in the format ":name{.tag}" into the module name and tag, ":name" into the
2062// module name and an empty string for the tag, or empty strings if the input was not a module reference.
2063func SrcIsModuleWithTag(s string) (module, tag string) {
2064 if len(s) > 1 && s[0] == ':' {
2065 module = s[1:]
2066 if tagStart := strings.IndexByte(module, '{'); tagStart > 0 {
2067 if module[len(module)-1] == '}' {
2068 tag = module[tagStart+1 : len(module)-1]
2069 module = module[:tagStart]
2070 return module, tag
2071 }
2072 }
2073 return module, ""
2074 }
2075 return "", ""
Colin Cross068e0fe2016-12-13 15:23:47 -08002076}
2077
Colin Cross41955e82019-05-29 14:40:35 -07002078type sourceOrOutputDependencyTag struct {
2079 blueprint.BaseDependencyTag
2080 tag string
2081}
2082
2083func sourceOrOutputDepTag(tag string) blueprint.DependencyTag {
2084 return sourceOrOutputDependencyTag{tag: tag}
2085}
2086
2087var SourceDepTag = sourceOrOutputDepTag("")
Colin Cross068e0fe2016-12-13 15:23:47 -08002088
Colin Cross366938f2017-12-11 16:29:02 -08002089// Adds necessary dependencies to satisfy filegroup or generated sources modules listed in srcFiles
2090// using ":module" syntax, if any.
Colin Cross27b922f2019-03-04 22:35:41 -08002091//
2092// Deprecated: tag the property with `android:"path"` instead.
Colin Cross068e0fe2016-12-13 15:23:47 -08002093func ExtractSourcesDeps(ctx BottomUpMutatorContext, srcFiles []string) {
Nan Zhang2439eb72017-04-10 11:27:50 -07002094 set := make(map[string]bool)
2095
Colin Cross068e0fe2016-12-13 15:23:47 -08002096 for _, s := range srcFiles {
Colin Cross41955e82019-05-29 14:40:35 -07002097 if m, t := SrcIsModuleWithTag(s); m != "" {
2098 if _, found := set[s]; found {
2099 ctx.ModuleErrorf("found source dependency duplicate: %q!", s)
Nan Zhang2439eb72017-04-10 11:27:50 -07002100 } else {
Colin Cross41955e82019-05-29 14:40:35 -07002101 set[s] = true
2102 ctx.AddDependency(ctx.Module(), sourceOrOutputDepTag(t), m)
Nan Zhang2439eb72017-04-10 11:27:50 -07002103 }
Colin Cross068e0fe2016-12-13 15:23:47 -08002104 }
2105 }
Colin Cross068e0fe2016-12-13 15:23:47 -08002106}
2107
Colin Cross366938f2017-12-11 16:29:02 -08002108// Adds necessary dependencies to satisfy filegroup or generated sources modules specified in s
2109// using ":module" syntax, if any.
Colin Cross27b922f2019-03-04 22:35:41 -08002110//
2111// Deprecated: tag the property with `android:"path"` instead.
Colin Cross366938f2017-12-11 16:29:02 -08002112func ExtractSourceDeps(ctx BottomUpMutatorContext, s *string) {
2113 if s != nil {
Colin Cross41955e82019-05-29 14:40:35 -07002114 if m, t := SrcIsModuleWithTag(*s); m != "" {
2115 ctx.AddDependency(ctx.Module(), sourceOrOutputDepTag(t), m)
Colin Cross366938f2017-12-11 16:29:02 -08002116 }
2117 }
2118}
2119
Colin Cross41955e82019-05-29 14:40:35 -07002120// A module that implements SourceFileProducer can be referenced from any property that is tagged with `android:"path"`
2121// using the ":module" syntax and provides a list of paths to be used as if they were listed in the property.
Colin Cross068e0fe2016-12-13 15:23:47 -08002122type SourceFileProducer interface {
2123 Srcs() Paths
2124}
2125
Colin Cross41955e82019-05-29 14:40:35 -07002126// A module that implements OutputFileProducer can be referenced from any property that is tagged with `android:"path"`
Roland Levillain97c1f342019-11-22 14:20:54 +00002127// using the ":module" syntax or ":module{.tag}" syntax and provides a list of output files to be used as if they were
Colin Cross41955e82019-05-29 14:40:35 -07002128// listed in the property.
2129type OutputFileProducer interface {
2130 OutputFiles(tag string) (Paths, error)
2131}
2132
Colin Cross5e708052019-08-06 13:59:50 -07002133// OutputFilesForModule returns the paths from an OutputFileProducer with the given tag. On error, including if the
2134// module produced zero paths, it reports errors to the ctx and returns nil.
2135func OutputFilesForModule(ctx PathContext, module blueprint.Module, tag string) Paths {
2136 paths, err := outputFilesForModule(ctx, module, tag)
2137 if err != nil {
2138 reportPathError(ctx, err)
2139 return nil
2140 }
2141 return paths
2142}
2143
2144// OutputFileForModule returns the path from an OutputFileProducer with the given tag. On error, including if the
2145// module produced zero or multiple paths, it reports errors to the ctx and returns nil.
2146func OutputFileForModule(ctx PathContext, module blueprint.Module, tag string) Path {
2147 paths, err := outputFilesForModule(ctx, module, tag)
2148 if err != nil {
2149 reportPathError(ctx, err)
2150 return nil
2151 }
2152 if len(paths) > 1 {
2153 reportPathErrorf(ctx, "got multiple output files from module %q, expected exactly one",
2154 pathContextName(ctx, module))
2155 return nil
2156 }
2157 return paths[0]
2158}
2159
2160func outputFilesForModule(ctx PathContext, module blueprint.Module, tag string) (Paths, error) {
2161 if outputFileProducer, ok := module.(OutputFileProducer); ok {
2162 paths, err := outputFileProducer.OutputFiles(tag)
2163 if err != nil {
2164 return nil, fmt.Errorf("failed to get output file from module %q: %s",
2165 pathContextName(ctx, module), err.Error())
2166 }
2167 if len(paths) == 0 {
2168 return nil, fmt.Errorf("failed to get output files from module %q", pathContextName(ctx, module))
2169 }
2170 return paths, nil
2171 } else {
2172 return nil, fmt.Errorf("module %q is not an OutputFileProducer", pathContextName(ctx, module))
2173 }
2174}
2175
Colin Crossfe17f6f2019-03-28 19:30:56 -07002176type HostToolProvider interface {
2177 HostToolPath() OptionalPath
2178}
2179
Colin Cross27b922f2019-03-04 22:35:41 -08002180// Returns a list of paths expanded from globs and modules referenced using ":module" syntax. The property must
2181// be tagged with `android:"path" to support automatic source module dependency resolution.
Colin Cross8a497952019-03-05 22:25:09 -08002182//
2183// Deprecated: use PathsForModuleSrc or PathsForModuleSrcExcludes instead.
Colin Cross25de6c32019-06-06 14:29:25 -07002184func (m *moduleContext) ExpandSources(srcFiles, excludes []string) Paths {
2185 return PathsForModuleSrcExcludes(m, srcFiles, excludes)
Colin Cross8f101b42015-06-17 15:09:06 -07002186}
2187
Colin Cross2fafa3e2019-03-05 12:39:51 -08002188// Returns a single path expanded from globs and modules referenced using ":module" syntax. The property must
2189// be tagged with `android:"path" to support automatic source module dependency resolution.
Colin Cross8a497952019-03-05 22:25:09 -08002190//
2191// Deprecated: use PathForModuleSrc instead.
Colin Cross25de6c32019-06-06 14:29:25 -07002192func (m *moduleContext) ExpandSource(srcFile, prop string) Path {
2193 return PathForModuleSrc(m, srcFile)
Colin Cross2fafa3e2019-03-05 12:39:51 -08002194}
2195
2196// Returns an optional single path expanded from globs and modules referenced using ":module" syntax if
2197// the srcFile is non-nil. The property must be tagged with `android:"path" to support automatic source module
2198// dependency resolution.
Colin Cross25de6c32019-06-06 14:29:25 -07002199func (m *moduleContext) ExpandOptionalSource(srcFile *string, prop string) OptionalPath {
Colin Cross2fafa3e2019-03-05 12:39:51 -08002200 if srcFile != nil {
Colin Cross25de6c32019-06-06 14:29:25 -07002201 return OptionalPathForPath(PathForModuleSrc(m, *srcFile))
Colin Cross2fafa3e2019-03-05 12:39:51 -08002202 }
2203 return OptionalPath{}
2204}
2205
Colin Cross25de6c32019-06-06 14:29:25 -07002206func (m *moduleContext) RequiredModuleNames() []string {
Jiyong Park6a8cf5f2019-12-30 16:31:09 +09002207 return m.module.RequiredModuleNames()
Nan Zhang6d34b302017-02-04 17:47:46 -08002208}
2209
Colin Cross25de6c32019-06-06 14:29:25 -07002210func (m *moduleContext) HostRequiredModuleNames() []string {
Jiyong Park6a8cf5f2019-12-30 16:31:09 +09002211 return m.module.HostRequiredModuleNames()
Sasha Smundakb6d23052019-04-01 18:37:36 -07002212}
2213
Colin Cross25de6c32019-06-06 14:29:25 -07002214func (m *moduleContext) TargetRequiredModuleNames() []string {
Jiyong Park6a8cf5f2019-12-30 16:31:09 +09002215 return m.module.TargetRequiredModuleNames()
Sasha Smundakb6d23052019-04-01 18:37:36 -07002216}
2217
Colin Cross463a90e2015-06-17 14:20:06 -07002218func init() {
Colin Cross798bfce2016-10-12 14:28:16 -07002219 RegisterSingletonType("buildtarget", BuildTargetSingleton)
Colin Cross463a90e2015-06-17 14:20:06 -07002220}
2221
Colin Cross0875c522017-11-28 17:34:01 -08002222func BuildTargetSingleton() Singleton {
Colin Cross1f8c52b2015-06-16 16:38:17 -07002223 return &buildTargetSingleton{}
2224}
2225
Colin Cross87d8b562017-04-25 10:01:55 -07002226func parentDir(dir string) string {
2227 dir, _ = filepath.Split(dir)
2228 return filepath.Clean(dir)
2229}
2230
Colin Cross1f8c52b2015-06-16 16:38:17 -07002231type buildTargetSingleton struct{}
2232
Colin Cross0875c522017-11-28 17:34:01 -08002233func (c *buildTargetSingleton) GenerateBuildActions(ctx SingletonContext) {
2234 var checkbuildDeps Paths
Colin Cross1f8c52b2015-06-16 16:38:17 -07002235
Colin Crossc3d87d32020-06-04 13:25:17 -07002236 mmTarget := func(dir string) string {
2237 return "MODULES-IN-" + strings.Replace(filepath.Clean(dir), "/", "-", -1)
Colin Cross87d8b562017-04-25 10:01:55 -07002238 }
2239
Colin Cross0875c522017-11-28 17:34:01 -08002240 modulesInDir := make(map[string]Paths)
Colin Cross1f8c52b2015-06-16 16:38:17 -07002241
Colin Cross0875c522017-11-28 17:34:01 -08002242 ctx.VisitAllModules(func(module Module) {
2243 blueprintDir := module.base().blueprintDir
2244 installTarget := module.base().installTarget
2245 checkbuildTarget := module.base().checkbuildTarget
Colin Cross1f8c52b2015-06-16 16:38:17 -07002246
Colin Cross0875c522017-11-28 17:34:01 -08002247 if checkbuildTarget != nil {
2248 checkbuildDeps = append(checkbuildDeps, checkbuildTarget)
2249 modulesInDir[blueprintDir] = append(modulesInDir[blueprintDir], checkbuildTarget)
2250 }
Colin Cross1f8c52b2015-06-16 16:38:17 -07002251
Colin Cross0875c522017-11-28 17:34:01 -08002252 if installTarget != nil {
2253 modulesInDir[blueprintDir] = append(modulesInDir[blueprintDir], installTarget)
Colin Cross1f8c52b2015-06-16 16:38:17 -07002254 }
2255 })
2256
Dan Willemsen5ba07e82015-12-11 13:51:06 -08002257 suffix := ""
Colin Crossaabf6792017-11-29 00:27:14 -08002258 if ctx.Config().EmbeddedInMake() {
Dan Willemsen5ba07e82015-12-11 13:51:06 -08002259 suffix = "-soong"
2260 }
2261
Colin Cross1f8c52b2015-06-16 16:38:17 -07002262 // Create a top-level checkbuild target that depends on all modules
Colin Crossc3d87d32020-06-04 13:25:17 -07002263 ctx.Phony("checkbuild"+suffix, checkbuildDeps...)
Colin Cross1f8c52b2015-06-16 16:38:17 -07002264
Dan Willemsend2e95fb2017-09-20 14:30:50 -07002265 // Make will generate the MODULES-IN-* targets
Colin Crossaabf6792017-11-29 00:27:14 -08002266 if ctx.Config().EmbeddedInMake() {
Dan Willemsend2e95fb2017-09-20 14:30:50 -07002267 return
2268 }
2269
Colin Cross87d8b562017-04-25 10:01:55 -07002270 // Ensure ancestor directories are in modulesInDir
Inseob Kim1a365c62019-06-08 15:47:51 +09002271 dirs := SortedStringKeys(modulesInDir)
Colin Cross87d8b562017-04-25 10:01:55 -07002272 for _, dir := range dirs {
2273 dir := parentDir(dir)
2274 for dir != "." && dir != "/" {
2275 if _, exists := modulesInDir[dir]; exists {
2276 break
2277 }
2278 modulesInDir[dir] = nil
2279 dir = parentDir(dir)
2280 }
2281 }
2282
2283 // Make directories build their direct subdirectories
Colin Cross87d8b562017-04-25 10:01:55 -07002284 for _, dir := range dirs {
2285 p := parentDir(dir)
2286 if p != "." && p != "/" {
Colin Crossc3d87d32020-06-04 13:25:17 -07002287 modulesInDir[p] = append(modulesInDir[p], PathForPhony(ctx, mmTarget(dir)))
Colin Cross87d8b562017-04-25 10:01:55 -07002288 }
2289 }
2290
Dan Willemsend2e95fb2017-09-20 14:30:50 -07002291 // Create a MODULES-IN-<directory> target that depends on all modules in a directory, and
2292 // depends on the MODULES-IN-* targets of all of its subdirectories that contain Android.bp
2293 // files.
Colin Cross1f8c52b2015-06-16 16:38:17 -07002294 for _, dir := range dirs {
Colin Crossc3d87d32020-06-04 13:25:17 -07002295 ctx.Phony(mmTarget(dir), modulesInDir[dir]...)
Colin Cross1f8c52b2015-06-16 16:38:17 -07002296 }
Dan Willemsen61d88b82017-09-20 17:29:08 -07002297
2298 // Create (host|host-cross|target)-<OS> phony rules to build a reduced checkbuild.
2299 osDeps := map[OsType]Paths{}
Colin Cross0875c522017-11-28 17:34:01 -08002300 ctx.VisitAllModules(func(module Module) {
2301 if module.Enabled() {
2302 os := module.Target().Os
2303 osDeps[os] = append(osDeps[os], module.base().checkbuildFiles...)
Dan Willemsen61d88b82017-09-20 17:29:08 -07002304 }
2305 })
2306
Colin Cross0875c522017-11-28 17:34:01 -08002307 osClass := make(map[string]Paths)
Dan Willemsen61d88b82017-09-20 17:29:08 -07002308 for os, deps := range osDeps {
2309 var className string
2310
2311 switch os.Class {
2312 case Host:
2313 className = "host"
2314 case HostCross:
2315 className = "host-cross"
2316 case Device:
2317 className = "target"
2318 default:
2319 continue
2320 }
2321
Colin Crossc3d87d32020-06-04 13:25:17 -07002322 name := className + "-" + os.Name
2323 osClass[className] = append(osClass[className], PathForPhony(ctx, name))
Dan Willemsen61d88b82017-09-20 17:29:08 -07002324
Colin Crossc3d87d32020-06-04 13:25:17 -07002325 ctx.Phony(name, deps...)
Dan Willemsen61d88b82017-09-20 17:29:08 -07002326 }
2327
2328 // Wrap those into host|host-cross|target phony rules
Inseob Kim1a365c62019-06-08 15:47:51 +09002329 for _, class := range SortedStringKeys(osClass) {
Colin Crossc3d87d32020-06-04 13:25:17 -07002330 ctx.Phony(class, osClass[class]...)
Dan Willemsen61d88b82017-09-20 17:29:08 -07002331 }
Colin Cross1f8c52b2015-06-16 16:38:17 -07002332}
Colin Crossd779da42015-12-17 18:00:23 -08002333
Brandon Lee5d45c6f2018-08-15 15:35:38 -07002334// Collect information for opening IDE project files in java/jdeps.go.
2335type IDEInfo interface {
2336 IDEInfo(ideInfo *IdeInfo)
2337 BaseModuleName() string
2338}
2339
2340// Extract the base module name from the Import name.
2341// Often the Import name has a prefix "prebuilt_".
2342// Remove the prefix explicitly if needed
2343// until we find a better solution to get the Import name.
2344type IDECustomizedModuleName interface {
2345 IDECustomizedModuleName() string
2346}
2347
2348type IdeInfo struct {
2349 Deps []string `json:"dependencies,omitempty"`
2350 Srcs []string `json:"srcs,omitempty"`
2351 Aidl_include_dirs []string `json:"aidl_include_dirs,omitempty"`
2352 Jarjar_rules []string `json:"jarjar_rules,omitempty"`
2353 Jars []string `json:"jars,omitempty"`
2354 Classes []string `json:"class,omitempty"`
2355 Installed_paths []string `json:"installed,omitempty"`
patricktu18c82ff2019-05-10 15:48:50 +08002356 SrcJars []string `json:"srcjars,omitempty"`
bralee1fbf4402020-05-21 10:11:59 +08002357 Paths []string `json:"path,omitempty"`
Brandon Lee5d45c6f2018-08-15 15:35:38 -07002358}
Paul Duffinf88d8e02020-05-07 20:21:34 +01002359
2360func CheckBlueprintSyntax(ctx BaseModuleContext, filename string, contents string) []error {
2361 bpctx := ctx.blueprintBaseModuleContext()
2362 return blueprint.CheckBlueprintSyntax(bpctx.ModuleFactories(), filename, contents)
2363}