blob: ce39dd5110c5e4b1eeeb837e1ec42b601fbcd16e [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 Cross3f68a132017-10-23 17:10:29 -0700210
Colin Cross0875c522017-11-28 17:34:01 -0800211 PrimaryModule() Module
212 FinalModule() Module
213 VisitAllModuleVariants(visit func(Module))
Colin Cross3f68a132017-10-23 17:10:29 -0700214
215 GetMissingDependencies() []string
Jeff Gaston088e29e2017-11-29 16:47:17 -0800216 Namespace() blueprint.Namespace
Colin Cross3f40fa42015-01-30 17:27:36 -0800217}
218
Colin Cross635c3b02016-05-18 15:37:25 -0700219type Module interface {
Colin Cross3f40fa42015-01-30 17:27:36 -0800220 blueprint.Module
221
Jeff Gastonaf3cc2d2017-09-27 17:01:44 -0700222 // GenerateAndroidBuildActions is analogous to Blueprints' GenerateBuildActions,
223 // but GenerateAndroidBuildActions also has access to Android-specific information.
224 // For more information, see Module.GenerateBuildActions within Blueprint's module_ctx.go
Colin Cross635c3b02016-05-18 15:37:25 -0700225 GenerateAndroidBuildActions(ModuleContext)
Jeff Gastonaf3cc2d2017-09-27 17:01:44 -0700226
Colin Cross1e676be2016-10-12 14:38:15 -0700227 DepsMutator(BottomUpMutatorContext)
Colin Cross3f40fa42015-01-30 17:27:36 -0800228
Colin Cross635c3b02016-05-18 15:37:25 -0700229 base() *ModuleBase
Inseob Kimeec88e12020-01-22 11:11:29 +0900230 Disable()
Dan Willemsen0effe062015-11-30 16:06:01 -0800231 Enabled() bool
Colin Crossa1ad8d12016-06-01 17:09:44 -0700232 Target() Target
Dan Willemsen782a2d12015-12-21 14:55:28 -0800233 InstallInData() bool
Jaewoong Jung0949f312019-09-11 10:25:18 -0700234 InstallInTestcases() bool
Vishwath Mohan1dd88392017-03-29 22:00:18 -0700235 InstallInSanitizerDir() bool
Yifan Hong1b3348d2020-01-21 15:53:22 -0800236 InstallInRamdisk() bool
Jiyong Parkf9332f12018-02-01 00:54:12 +0900237 InstallInRecovery() bool
Colin Cross90ba5f42019-10-02 11:10:58 -0700238 InstallInRoot() bool
Colin Cross607d8582019-07-29 16:44:46 -0700239 InstallBypassMake() bool
Colin Cross6e359402020-02-10 15:29:54 -0800240 InstallForceOS() *OsType
Colin Crossa2f296f2016-11-29 15:16:18 -0800241 SkipInstall()
Ulya Trafimovichb28cc372020-01-13 15:18:16 +0000242 IsSkipInstall() bool
Jiyong Park374510b2018-03-19 18:23:01 +0900243 ExportedToMake() bool
Inseob Kim8471cda2019-11-15 09:59:12 +0900244 InitRc() Paths
245 VintfFragments() Paths
Bob Badoura75b0572020-02-18 20:21:55 -0800246 NoticeFiles() Paths
Colin Cross36242852017-06-23 15:06:31 -0700247
248 AddProperties(props ...interface{})
249 GetProperties() []interface{}
Colin Crosscec81712017-07-13 14:43:27 -0700250
Colin Crossae887032017-10-23 17:16:14 -0700251 BuildParamsForTests() []BuildParams
Colin Cross4c83e5c2019-02-25 14:54:28 -0800252 RuleParamsForTests() map[blueprint.Rule]blueprint.RuleParams
Jaewoong Jung38e4fb22018-12-12 09:01:34 -0800253 VariablesForTests() map[string]string
Paul Duffine2453c72019-05-31 14:00:04 +0100254
Colin Cross9a362232019-07-01 15:32:45 -0700255 // String returns a string that includes the module name and variants for printing during debugging.
256 String() string
257
Paul Duffine2453c72019-05-31 14:00:04 +0100258 // Get the qualified module id for this module.
259 qualifiedModuleId(ctx BaseModuleContext) qualifiedModuleName
260
261 // Get information about the properties that can contain visibility rules.
262 visibilityProperties() []visibilityProperty
Paul Duffin63c6e182019-07-24 14:24:38 +0100263
Jiyong Park6a8cf5f2019-12-30 16:31:09 +0900264 RequiredModuleNames() []string
265 HostRequiredModuleNames() []string
266 TargetRequiredModuleNames() []string
Colin Cross897266e2020-02-13 13:22:08 -0800267
268 filesToInstall() InstallPaths
Paul Duffine2453c72019-05-31 14:00:04 +0100269}
270
271// Qualified id for a module
272type qualifiedModuleName struct {
273 // The package (i.e. directory) in which the module is defined, without trailing /
274 pkg string
275
276 // The name of the module, empty string if package.
277 name string
278}
279
280func (q qualifiedModuleName) String() string {
281 if q.name == "" {
282 return "//" + q.pkg
283 }
284 return "//" + q.pkg + ":" + q.name
285}
286
Paul Duffine484f472019-06-20 16:38:08 +0100287func (q qualifiedModuleName) isRootPackage() bool {
288 return q.pkg == "" && q.name == ""
289}
290
Paul Duffine2453c72019-05-31 14:00:04 +0100291// Get the id for the package containing this module.
292func (q qualifiedModuleName) getContainingPackageId() qualifiedModuleName {
293 pkg := q.pkg
294 if q.name == "" {
Paul Duffine484f472019-06-20 16:38:08 +0100295 if pkg == "" {
296 panic(fmt.Errorf("Cannot get containing package id of root package"))
297 }
298
299 index := strings.LastIndex(pkg, "/")
300 if index == -1 {
301 pkg = ""
302 } else {
303 pkg = pkg[:index]
304 }
Paul Duffine2453c72019-05-31 14:00:04 +0100305 }
306 return newPackageId(pkg)
307}
308
309func newPackageId(pkg string) qualifiedModuleName {
310 // A qualified id for a package module has no name.
311 return qualifiedModuleName{pkg: pkg, name: ""}
Colin Cross3f40fa42015-01-30 17:27:36 -0800312}
313
Colin Crossfc754582016-05-17 16:34:16 -0700314type nameProperties struct {
315 // The name of the module. Must be unique across all modules.
Nan Zhang0007d812017-11-07 10:57:05 -0800316 Name *string
Colin Crossfc754582016-05-17 16:34:16 -0700317}
318
319type commonProperties struct {
Dan Willemsen0effe062015-11-30 16:06:01 -0800320 // emit build rules for this module
Paul Duffin54d9bb72020-02-12 10:20:56 +0000321 //
322 // Disabling a module should only be done for those modules that cannot be built
323 // in the current environment. Modules that can build in the current environment
324 // but are not usually required (e.g. superceded by a prebuilt) should not be
325 // disabled as that will prevent them from being built by the checkbuild target
326 // and so prevent early detection of changes that have broken those modules.
Dan Willemsen0effe062015-11-30 16:06:01 -0800327 Enabled *bool `android:"arch_variant"`
Colin Cross3f40fa42015-01-30 17:27:36 -0800328
Paul Duffin2e61fa62019-03-28 14:10:57 +0000329 // Controls the visibility of this module to other modules. Allowable values are one or more of
330 // these formats:
331 //
332 // ["//visibility:public"]: Anyone can use this module.
333 // ["//visibility:private"]: Only rules in the module's package (not its subpackages) can use
334 // this module.
Paul Duffin51084ff2020-05-05 19:19:22 +0100335 // ["//visibility:override"]: Discards any rules inherited from defaults or a creating module.
336 // Can only be used at the beginning of a list of visibility rules.
Paul Duffin2e61fa62019-03-28 14:10:57 +0000337 // ["//some/package:__pkg__", "//other/package:__pkg__"]: Only modules in some/package and
338 // other/package (defined in some/package/*.bp and other/package/*.bp) have access to
339 // this module. Note that sub-packages do not have access to the rule; for example,
340 // //some/package/foo:bar or //other/package/testing:bla wouldn't have access. __pkg__
341 // is a special module and must be used verbatim. It represents all of the modules in the
342 // package.
343 // ["//project:__subpackages__", "//other:__subpackages__"]: Only modules in packages project
344 // or other or in one of their sub-packages have access to this module. For example,
345 // //project:rule, //project/library:lib or //other/testing/internal:munge are allowed
346 // to depend on this rule (but not //independent:evil)
347 // ["//project"]: This is shorthand for ["//project:__pkg__"]
348 // [":__subpackages__"]: This is shorthand for ["//project:__subpackages__"] where
349 // //project is the module's package. e.g. using [":__subpackages__"] in
350 // packages/apps/Settings/Android.bp is equivalent to
351 // //packages/apps/Settings:__subpackages__.
352 // ["//visibility:legacy_public"]: The default visibility, behaves as //visibility:public
353 // for now. It is an error if it is used in a module.
Paul Duffine2453c72019-05-31 14:00:04 +0100354 //
355 // If a module does not specify the `visibility` property then it uses the
356 // `default_visibility` property of the `package` module in the module's package.
357 //
358 // If the `default_visibility` property is not set for the module's package then
Paul Duffine484f472019-06-20 16:38:08 +0100359 // it will use the `default_visibility` of its closest ancestor package for which
360 // a `default_visibility` property is specified.
361 //
362 // If no `default_visibility` property can be found then the module uses the
363 // global default of `//visibility:legacy_public`.
Paul Duffine2453c72019-05-31 14:00:04 +0100364 //
Paul Duffin95d53b52019-07-24 13:45:05 +0100365 // The `visibility` property has no effect on a defaults module although it does
366 // apply to any non-defaults module that uses it. To set the visibility of a
367 // defaults module, use the `defaults_visibility` property on the defaults module;
368 // not to be confused with the `default_visibility` property on the package module.
369 //
Paul Duffin2e61fa62019-03-28 14:10:57 +0000370 // See https://android.googlesource.com/platform/build/soong/+/master/README.md#visibility for
371 // more details.
372 Visibility []string
373
Colin Cross7d5136f2015-05-11 13:39:40 -0700374 // control whether this module compiles for 32-bit, 64-bit, or both. Possible values
Colin Cross3f40fa42015-01-30 17:27:36 -0800375 // are "32" (compile for 32-bit only), "64" (compile for 64-bit only), "both" (compile for both
376 // architectures), or "first" (compile for 64-bit on a 64-bit platform, and 32-bit on a 32-bit
377 // platform
Colin Cross7d716ba2017-11-01 10:38:29 -0700378 Compile_multilib *string `android:"arch_variant"`
Colin Cross69617d32016-09-06 10:39:07 -0700379
380 Target struct {
381 Host struct {
Colin Cross7d716ba2017-11-01 10:38:29 -0700382 Compile_multilib *string
Colin Cross69617d32016-09-06 10:39:07 -0700383 }
384 Android struct {
Colin Cross7d716ba2017-11-01 10:38:29 -0700385 Compile_multilib *string
Colin Cross69617d32016-09-06 10:39:07 -0700386 }
387 }
388
Paul Duffinca7f0ef2020-02-25 15:50:49 +0000389 // If set to true then the archMutator will create variants for each arch specific target
390 // (e.g. 32/64) that the module is required to produce. If set to false then it will only
391 // create a variant for the architecture and will list the additional arch specific targets
392 // that the variant needs to produce in the CompileMultiTargets property.
Colin Crossee0bc3b2018-10-02 22:01:37 -0700393 UseTargetVariants bool `blueprint:"mutated"`
394 Default_multilib string `blueprint:"mutated"`
Colin Cross3f40fa42015-01-30 17:27:36 -0800395
Dan Willemsen782a2d12015-12-21 14:55:28 -0800396 // whether this is a proprietary vendor module, and should be installed into /vendor
Colin Cross7d716ba2017-11-01 10:38:29 -0700397 Proprietary *bool
Dan Willemsen782a2d12015-12-21 14:55:28 -0800398
Colin Cross55708f32017-03-20 13:23:34 -0700399 // vendor who owns this module
Dan Willemsenefac4a82017-07-18 19:42:09 -0700400 Owner *string
Colin Cross55708f32017-03-20 13:23:34 -0700401
Jiyong Park2db76922017-11-08 16:03:48 +0900402 // whether this module is specific to an SoC (System-On-a-Chip). When set to true,
403 // it is installed into /vendor (or /system/vendor if vendor partition does not exist).
404 // Use `soc_specific` instead for better meaning.
Colin Cross7d716ba2017-11-01 10:38:29 -0700405 Vendor *bool
Dan Willemsenaa118f92017-04-06 12:49:58 -0700406
Jiyong Park2db76922017-11-08 16:03:48 +0900407 // whether this module is specific to an SoC (System-On-a-Chip). When set to true,
408 // it is installed into /vendor (or /system/vendor if vendor partition does not exist).
409 Soc_specific *bool
410
411 // whether this module is specific to a device, not only for SoC, but also for off-chip
412 // peripherals. When set to true, it is installed into /odm (or /vendor/odm if odm partition
413 // does not exist, or /system/vendor/odm if both odm and vendor partitions do not exist).
414 // This implies `soc_specific:true`.
415 Device_specific *bool
416
417 // whether this module is specific to a software configuration of a product (e.g. country,
Jaekyun Seok5cfbfbb2018-01-10 19:00:15 +0900418 // network operator, etc). When set to true, it is installed into /product (or
419 // /system/product if product partition does not exist).
Jiyong Park2db76922017-11-08 16:03:48 +0900420 Product_specific *bool
421
Justin Yund5f6c822019-06-25 16:47:17 +0900422 // whether this module extends system. When set to true, it is installed into /system_ext
423 // (or /system/system_ext if system_ext partition does not exist).
424 System_ext_specific *bool
425
Jiyong Parkf9332f12018-02-01 00:54:12 +0900426 // Whether this module is installed to recovery partition
427 Recovery *bool
428
Yifan Hong1b3348d2020-01-21 15:53:22 -0800429 // Whether this module is installed to ramdisk
430 Ramdisk *bool
431
dimitry1f33e402019-03-26 12:39:31 +0100432 // Whether this module is built for non-native architecures (also known as native bridge binary)
433 Native_bridge_supported *bool `android:"arch_variant"`
434
Dan Willemsen2277bcb2016-07-25 20:27:39 -0700435 // init.rc files to be installed if this module is installed
Colin Cross27b922f2019-03-04 22:35:41 -0800436 Init_rc []string `android:"path"`
Dan Willemsen2277bcb2016-07-25 20:27:39 -0700437
Steven Moreland57a23d22018-04-04 15:42:19 -0700438 // VINTF manifest fragments to be installed if this module is installed
Colin Cross27b922f2019-03-04 22:35:41 -0800439 Vintf_fragments []string `android:"path"`
Steven Moreland57a23d22018-04-04 15:42:19 -0700440
Chris Wolfe998306e2016-08-15 14:47:23 -0400441 // names of other modules to install if this module is installed
Colin Crossc602b7d2017-05-05 13:36:36 -0700442 Required []string `android:"arch_variant"`
Chris Wolfe998306e2016-08-15 14:47:23 -0400443
Sasha Smundakb6d23052019-04-01 18:37:36 -0700444 // names of other modules to install on host if this module is installed
445 Host_required []string `android:"arch_variant"`
446
447 // names of other modules to install on target if this module is installed
448 Target_required []string `android:"arch_variant"`
449
Colin Cross5aac3622017-08-31 15:07:09 -0700450 // relative path to a file to include in the list of notices for the device
Colin Cross27b922f2019-03-04 22:35:41 -0800451 Notice *string `android:"path"`
Colin Cross5aac3622017-08-31 15:07:09 -0700452
Dan Willemsen569edc52018-11-19 09:33:29 -0800453 Dist struct {
454 // copy the output of this module to the $DIST_DIR when `dist` is specified on the
455 // command line and any of these targets are also on the command line, or otherwise
456 // built
457 Targets []string `android:"arch_variant"`
458
459 // The name of the output artifact. This defaults to the basename of the output of
460 // the module.
461 Dest *string `android:"arch_variant"`
462
463 // The directory within the dist directory to store the artifact. Defaults to the
464 // top level directory ("").
465 Dir *string `android:"arch_variant"`
466
467 // A suffix to add to the artifact file name (before any extension).
468 Suffix *string `android:"arch_variant"`
469 } `android:"arch_variant"`
470
Paul Duffinca7f0ef2020-02-25 15:50:49 +0000471 // The OsType of artifacts that this module variant is responsible for creating.
472 //
473 // Set by osMutator
474 CompileOS OsType `blueprint:"mutated"`
475
476 // The Target of artifacts that this module variant is responsible for creating.
477 //
478 // Set by archMutator
479 CompileTarget Target `blueprint:"mutated"`
480
481 // The additional arch specific targets (e.g. 32/64 bit) that this module variant is
482 // responsible for creating.
483 //
484 // By default this is nil as, where necessary, separate variants are created for the
485 // different multilib types supported and that information is encapsulated in the
486 // CompileTarget so the module variant simply needs to create artifacts for that.
487 //
488 // However, if UseTargetVariants is set to false (e.g. by
489 // InitAndroidMultiTargetsArchModule) then no separate variants are created for the
490 // multilib targets. Instead a single variant is created for the architecture and
491 // this contains the multilib specific targets that this variant should create.
492 //
493 // Set by archMutator
Colin Crossee0bc3b2018-10-02 22:01:37 -0700494 CompileMultiTargets []Target `blueprint:"mutated"`
Paul Duffinca7f0ef2020-02-25 15:50:49 +0000495
496 // True if the module variant's CompileTarget is the primary target
497 //
498 // Set by archMutator
499 CompilePrimary bool `blueprint:"mutated"`
Colin Cross3f40fa42015-01-30 17:27:36 -0800500
501 // Set by InitAndroidModule
502 HostOrDeviceSupported HostOrDeviceSupported `blueprint:"mutated"`
Dan Willemsen0b24c742016-10-04 15:13:37 -0700503 ArchSpecific bool `blueprint:"mutated"`
Colin Crossce75d2c2016-10-06 16:12:58 -0700504
Paul Duffin1356d8c2020-02-25 19:26:33 +0000505 // If set to true then a CommonOS variant will be created which will have dependencies
506 // on all its OsType specific variants. Used by sdk/module_exports to create a snapshot
507 // that covers all os and architecture variants.
508 //
509 // The OsType specific variants can be retrieved by calling
510 // GetOsSpecificVariantsOfCommonOSVariant
511 //
512 // Set at module initialization time by calling InitCommonOSAndroidMultiTargetsArchModule
513 CreateCommonOSVariant bool `blueprint:"mutated"`
514
515 // If set to true then this variant is the CommonOS variant that has dependencies on its
516 // OsType specific variants.
517 //
518 // Set by osMutator.
519 CommonOSVariant bool `blueprint:"mutated"`
520
Colin Crossce75d2c2016-10-06 16:12:58 -0700521 SkipInstall bool `blueprint:"mutated"`
Jeff Gaston088e29e2017-11-29 16:47:17 -0800522
523 NamespaceExportedToMake bool `blueprint:"mutated"`
Colin Cross6c4f21f2019-06-06 15:41:36 -0700524
525 MissingDeps []string `blueprint:"mutated"`
Colin Cross9a362232019-07-01 15:32:45 -0700526
527 // Name and variant strings stored by mutators to enable Module.String()
528 DebugName string `blueprint:"mutated"`
529 DebugMutators []string `blueprint:"mutated"`
530 DebugVariations []string `blueprint:"mutated"`
Colin Cross7228ecd2019-11-18 16:00:16 -0800531
532 // set by ImageMutator
533 ImageVariation string `blueprint:"mutated"`
Colin Cross3f40fa42015-01-30 17:27:36 -0800534}
535
536type hostAndDeviceProperties struct {
Colin Cross4e81d702018-11-09 10:36:55 -0800537 // If set to true, build a variant of the module for the host. Defaults to false.
538 Host_supported *bool
539
540 // If set to true, build a variant of the module for the device. Defaults to true.
Colin Crossa4190c12016-07-12 13:11:25 -0700541 Device_supported *bool
Colin Cross3f40fa42015-01-30 17:27:36 -0800542}
543
Colin Crossc472d572015-03-17 15:06:21 -0700544type Multilib string
545
546const (
Colin Cross6b4a32d2017-12-05 13:42:45 -0800547 MultilibBoth Multilib = "both"
548 MultilibFirst Multilib = "first"
549 MultilibCommon Multilib = "common"
550 MultilibCommonFirst Multilib = "common_first"
551 MultilibDefault Multilib = ""
Colin Crossc472d572015-03-17 15:06:21 -0700552)
553
Colin Crossa1ad8d12016-06-01 17:09:44 -0700554type HostOrDeviceSupported int
555
556const (
557 _ HostOrDeviceSupported = iota
Dan Albert0981b5c2018-08-02 13:46:35 -0700558
559 // Host and HostCross are built by default. Device is not supported.
Colin Crossa1ad8d12016-06-01 17:09:44 -0700560 HostSupported
Dan Albert0981b5c2018-08-02 13:46:35 -0700561
562 // Host is built by default. HostCross and Device are not supported.
Dan Albertc6345fb2016-10-20 01:36:11 -0700563 HostSupportedNoCross
Dan Albert0981b5c2018-08-02 13:46:35 -0700564
565 // Device is built by default. Host and HostCross are not supported.
Colin Crossa1ad8d12016-06-01 17:09:44 -0700566 DeviceSupported
Dan Albert0981b5c2018-08-02 13:46:35 -0700567
568 // Device is built by default. Host and HostCross are supported.
Colin Crossa1ad8d12016-06-01 17:09:44 -0700569 HostAndDeviceSupported
Dan Albert0981b5c2018-08-02 13:46:35 -0700570
571 // Host, HostCross, and Device are built by default.
Colin Crossa1ad8d12016-06-01 17:09:44 -0700572 HostAndDeviceDefault
Dan Albert0981b5c2018-08-02 13:46:35 -0700573
574 // Nothing is supported. This is not exposed to the user, but used to mark a
575 // host only module as unsupported when the module type is not supported on
576 // the host OS. E.g. benchmarks are supported on Linux but not Darwin.
Dan Willemsen0b24c742016-10-04 15:13:37 -0700577 NeitherHostNorDeviceSupported
Colin Crossa1ad8d12016-06-01 17:09:44 -0700578)
579
Jiyong Park2db76922017-11-08 16:03:48 +0900580type moduleKind int
581
582const (
583 platformModule moduleKind = iota
584 deviceSpecificModule
585 socSpecificModule
586 productSpecificModule
Justin Yund5f6c822019-06-25 16:47:17 +0900587 systemExtSpecificModule
Jiyong Park2db76922017-11-08 16:03:48 +0900588)
589
590func (k moduleKind) String() string {
591 switch k {
592 case platformModule:
593 return "platform"
594 case deviceSpecificModule:
595 return "device-specific"
596 case socSpecificModule:
597 return "soc-specific"
598 case productSpecificModule:
599 return "product-specific"
Justin Yund5f6c822019-06-25 16:47:17 +0900600 case systemExtSpecificModule:
601 return "systemext-specific"
Jiyong Park2db76922017-11-08 16:03:48 +0900602 default:
603 panic(fmt.Errorf("unknown module kind %d", k))
604 }
605}
606
Colin Cross9d34f352019-11-22 16:03:51 -0800607func initAndroidModuleBase(m Module) {
608 m.base().module = m
609}
610
Colin Cross36242852017-06-23 15:06:31 -0700611func InitAndroidModule(m Module) {
Colin Cross9d34f352019-11-22 16:03:51 -0800612 initAndroidModuleBase(m)
Colin Cross3f40fa42015-01-30 17:27:36 -0800613 base := m.base()
Colin Cross5049f022015-03-18 13:28:46 -0700614
Colin Cross36242852017-06-23 15:06:31 -0700615 m.AddProperties(
Colin Crossfc754582016-05-17 16:34:16 -0700616 &base.nameProperties,
Colin Cross18c46802019-09-24 22:19:02 -0700617 &base.commonProperties)
618
Colin Crosseabaedd2020-02-06 17:01:55 -0800619 initProductVariableModule(m)
Colin Cross18c46802019-09-24 22:19:02 -0700620
Colin Crossa3a97412019-03-18 12:24:29 -0700621 base.generalProperties = m.GetProperties()
Pirama Arumuga Nainar955dc492018-04-17 14:58:42 -0700622 base.customizableProperties = m.GetProperties()
Paul Duffin63c6e182019-07-24 14:24:38 +0100623
624 // The default_visibility property needs to be checked and parsed by the visibility module during
Paul Duffin5ec73ec2020-05-01 17:52:01 +0100625 // its checking and parsing phases so make it the primary visibility property.
626 setPrimaryVisibilityProperty(m, "visibility", &base.commonProperties.Visibility)
Colin Cross5049f022015-03-18 13:28:46 -0700627}
628
Colin Cross36242852017-06-23 15:06:31 -0700629func InitAndroidArchModule(m Module, hod HostOrDeviceSupported, defaultMultilib Multilib) {
630 InitAndroidModule(m)
Colin Cross5049f022015-03-18 13:28:46 -0700631
632 base := m.base()
Colin Cross3f40fa42015-01-30 17:27:36 -0800633 base.commonProperties.HostOrDeviceSupported = hod
Colin Cross69617d32016-09-06 10:39:07 -0700634 base.commonProperties.Default_multilib = string(defaultMultilib)
Dan Willemsen0b24c742016-10-04 15:13:37 -0700635 base.commonProperties.ArchSpecific = true
Colin Crossee0bc3b2018-10-02 22:01:37 -0700636 base.commonProperties.UseTargetVariants = true
Colin Cross3f40fa42015-01-30 17:27:36 -0800637
Dan Willemsen218f6562015-07-08 18:13:11 -0700638 switch hod {
Nan Zhang1a0f09b2017-07-05 10:35:11 -0700639 case HostAndDeviceSupported, HostAndDeviceDefault:
Colin Cross36242852017-06-23 15:06:31 -0700640 m.AddProperties(&base.hostAndDeviceProperties)
Colin Cross3f40fa42015-01-30 17:27:36 -0800641 }
642
Colin Cross36242852017-06-23 15:06:31 -0700643 InitArchModule(m)
Colin Cross3f40fa42015-01-30 17:27:36 -0800644}
645
Colin Crossee0bc3b2018-10-02 22:01:37 -0700646func InitAndroidMultiTargetsArchModule(m Module, hod HostOrDeviceSupported, defaultMultilib Multilib) {
647 InitAndroidArchModule(m, hod, defaultMultilib)
648 m.base().commonProperties.UseTargetVariants = false
649}
650
Paul Duffin1356d8c2020-02-25 19:26:33 +0000651// As InitAndroidMultiTargetsArchModule except it creates an additional CommonOS variant that
652// has dependencies on all the OsType specific variants.
653func InitCommonOSAndroidMultiTargetsArchModule(m Module, hod HostOrDeviceSupported, defaultMultilib Multilib) {
654 InitAndroidArchModule(m, hod, defaultMultilib)
655 m.base().commonProperties.UseTargetVariants = false
656 m.base().commonProperties.CreateCommonOSVariant = true
657}
658
Nan Zhangb9eeb1d2017-02-02 10:46:07 -0800659// A ModuleBase object contains the properties that are common to all Android
Colin Cross3f40fa42015-01-30 17:27:36 -0800660// modules. It should be included as an anonymous field in every module
661// struct definition. InitAndroidModule should then be called from the module's
662// factory function, and the return values from InitAndroidModule should be
663// returned from the factory function.
664//
Nan Zhangb9eeb1d2017-02-02 10:46:07 -0800665// The ModuleBase type is responsible for implementing the GenerateBuildActions
666// method to support the blueprint.Module interface. This method will then call
667// the module's GenerateAndroidBuildActions method once for each build variant
Colin Cross25de6c32019-06-06 14:29:25 -0700668// that is to be built. GenerateAndroidBuildActions is passed a ModuleContext
669// rather than the usual blueprint.ModuleContext.
670// ModuleContext exposes extra functionality specific to the Android build
Colin Cross3f40fa42015-01-30 17:27:36 -0800671// system including details about the particular build variant that is to be
672// generated.
673//
674// For example:
675//
676// import (
Nan Zhangb9eeb1d2017-02-02 10:46:07 -0800677// "android/soong/android"
Colin Cross3f40fa42015-01-30 17:27:36 -0800678// )
679//
680// type myModule struct {
Nan Zhangb9eeb1d2017-02-02 10:46:07 -0800681// android.ModuleBase
Colin Cross3f40fa42015-01-30 17:27:36 -0800682// properties struct {
683// MyProperty string
684// }
685// }
686//
Colin Cross36242852017-06-23 15:06:31 -0700687// func NewMyModule() android.Module) {
Colin Cross3f40fa42015-01-30 17:27:36 -0800688// m := &myModule{}
Colin Cross36242852017-06-23 15:06:31 -0700689// m.AddProperties(&m.properties)
690// android.InitAndroidModule(m)
691// return m
Colin Cross3f40fa42015-01-30 17:27:36 -0800692// }
693//
Nan Zhangb9eeb1d2017-02-02 10:46:07 -0800694// func (m *myModule) GenerateAndroidBuildActions(ctx android.ModuleContext) {
Colin Cross3f40fa42015-01-30 17:27:36 -0800695// // Get the CPU architecture for the current build variant.
696// variantArch := ctx.Arch()
697//
698// // ...
699// }
Colin Cross635c3b02016-05-18 15:37:25 -0700700type ModuleBase struct {
Colin Cross3f40fa42015-01-30 17:27:36 -0800701 // Putting the curiously recurring thing pointing to the thing that contains
702 // the thing pattern to good use.
Colin Cross36242852017-06-23 15:06:31 -0700703 // TODO: remove this
Colin Cross635c3b02016-05-18 15:37:25 -0700704 module Module
Colin Cross3f40fa42015-01-30 17:27:36 -0800705
Colin Crossfc754582016-05-17 16:34:16 -0700706 nameProperties nameProperties
Colin Cross3f40fa42015-01-30 17:27:36 -0800707 commonProperties commonProperties
Colin Cross18c46802019-09-24 22:19:02 -0700708 variableProperties interface{}
Colin Cross3f40fa42015-01-30 17:27:36 -0800709 hostAndDeviceProperties hostAndDeviceProperties
710 generalProperties []interface{}
Colin Crossc17727d2018-10-24 12:42:09 -0700711 archProperties [][]interface{}
Colin Crossa120ec12016-08-19 16:07:38 -0700712 customizableProperties []interface{}
Colin Cross3f40fa42015-01-30 17:27:36 -0800713
Paul Duffin63c6e182019-07-24 14:24:38 +0100714 // Information about all the properties on the module that contains visibility rules that need
715 // checking.
716 visibilityPropertyInfo []visibilityProperty
717
718 // The primary visibility property, may be nil, that controls access to the module.
719 primaryVisibilityProperty visibilityProperty
720
Colin Cross3f40fa42015-01-30 17:27:36 -0800721 noAddressSanitizer bool
Colin Cross897266e2020-02-13 13:22:08 -0800722 installFiles InstallPaths
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700723 checkbuildFiles Paths
Bob Badoura75b0572020-02-18 20:21:55 -0800724 noticeFiles Paths
Colin Cross1f8c52b2015-06-16 16:38:17 -0700725
726 // Used by buildTargetSingleton to create checkbuild and per-directory build targets
727 // Only set on the final variant of each module
Colin Cross0875c522017-11-28 17:34:01 -0800728 installTarget WritablePath
729 checkbuildTarget WritablePath
Colin Cross1f8c52b2015-06-16 16:38:17 -0700730 blueprintDir string
Colin Crossa120ec12016-08-19 16:07:38 -0700731
Colin Cross178a5092016-09-13 13:42:32 -0700732 hooks hooks
Colin Cross36242852017-06-23 15:06:31 -0700733
734 registerProps []interface{}
Colin Crosscec81712017-07-13 14:43:27 -0700735
736 // For tests
Colin Crossae887032017-10-23 17:16:14 -0700737 buildParams []BuildParams
Colin Cross4c83e5c2019-02-25 14:54:28 -0800738 ruleParams map[blueprint.Rule]blueprint.RuleParams
Jaewoong Jung38e4fb22018-12-12 09:01:34 -0800739 variables map[string]string
Colin Crossa9d8bee2018-10-02 13:59:46 -0700740
Inseob Kim8471cda2019-11-15 09:59:12 +0900741 initRcPaths Paths
742 vintfFragmentsPaths Paths
743
Colin Crossa9d8bee2018-10-02 13:59:46 -0700744 prefer32 func(ctx BaseModuleContext, base *ModuleBase, class OsClass) bool
Colin Cross36242852017-06-23 15:06:31 -0700745}
746
Colin Cross4157e882019-06-06 16:57:04 -0700747func (m *ModuleBase) DepsMutator(BottomUpMutatorContext) {}
Colin Cross5f692ec2019-02-01 16:53:07 -0800748
Colin Cross4157e882019-06-06 16:57:04 -0700749func (m *ModuleBase) AddProperties(props ...interface{}) {
750 m.registerProps = append(m.registerProps, props...)
Colin Cross36242852017-06-23 15:06:31 -0700751}
752
Colin Cross4157e882019-06-06 16:57:04 -0700753func (m *ModuleBase) GetProperties() []interface{} {
754 return m.registerProps
Colin Cross3f40fa42015-01-30 17:27:36 -0800755}
756
Colin Cross4157e882019-06-06 16:57:04 -0700757func (m *ModuleBase) BuildParamsForTests() []BuildParams {
758 return m.buildParams
Colin Crosscec81712017-07-13 14:43:27 -0700759}
760
Colin Cross4157e882019-06-06 16:57:04 -0700761func (m *ModuleBase) RuleParamsForTests() map[blueprint.Rule]blueprint.RuleParams {
762 return m.ruleParams
Colin Cross4c83e5c2019-02-25 14:54:28 -0800763}
764
Colin Cross4157e882019-06-06 16:57:04 -0700765func (m *ModuleBase) VariablesForTests() map[string]string {
766 return m.variables
Jaewoong Jung38e4fb22018-12-12 09:01:34 -0800767}
768
Colin Cross4157e882019-06-06 16:57:04 -0700769func (m *ModuleBase) Prefer32(prefer32 func(ctx BaseModuleContext, base *ModuleBase, class OsClass) bool) {
770 m.prefer32 = prefer32
Colin Crossa9d8bee2018-10-02 13:59:46 -0700771}
772
Colin Crossce75d2c2016-10-06 16:12:58 -0700773// Name returns the name of the module. It may be overridden by individual module types, for
774// example prebuilts will prepend prebuilt_ to the name.
Colin Cross4157e882019-06-06 16:57:04 -0700775func (m *ModuleBase) Name() string {
776 return String(m.nameProperties.Name)
Colin Crossfc754582016-05-17 16:34:16 -0700777}
778
Colin Cross9a362232019-07-01 15:32:45 -0700779// String returns a string that includes the module name and variants for printing during debugging.
780func (m *ModuleBase) String() string {
781 sb := strings.Builder{}
782 sb.WriteString(m.commonProperties.DebugName)
783 sb.WriteString("{")
784 for i := range m.commonProperties.DebugMutators {
785 if i != 0 {
786 sb.WriteString(",")
787 }
788 sb.WriteString(m.commonProperties.DebugMutators[i])
789 sb.WriteString(":")
790 sb.WriteString(m.commonProperties.DebugVariations[i])
791 }
792 sb.WriteString("}")
793 return sb.String()
794}
795
Colin Crossce75d2c2016-10-06 16:12:58 -0700796// BaseModuleName returns the name of the module as specified in the blueprints file.
Colin Cross4157e882019-06-06 16:57:04 -0700797func (m *ModuleBase) BaseModuleName() string {
798 return String(m.nameProperties.Name)
Colin Crossce75d2c2016-10-06 16:12:58 -0700799}
800
Colin Cross4157e882019-06-06 16:57:04 -0700801func (m *ModuleBase) base() *ModuleBase {
802 return m
Colin Cross3f40fa42015-01-30 17:27:36 -0800803}
804
Paul Duffine2453c72019-05-31 14:00:04 +0100805func (m *ModuleBase) qualifiedModuleId(ctx BaseModuleContext) qualifiedModuleName {
806 return qualifiedModuleName{pkg: ctx.ModuleDir(), name: ctx.ModuleName()}
807}
808
809func (m *ModuleBase) visibilityProperties() []visibilityProperty {
Paul Duffin63c6e182019-07-24 14:24:38 +0100810 return m.visibilityPropertyInfo
811}
812
Colin Cross4157e882019-06-06 16:57:04 -0700813func (m *ModuleBase) Target() Target {
814 return m.commonProperties.CompileTarget
Dan Willemsen490fd492015-11-24 17:53:15 -0800815}
816
Colin Cross4157e882019-06-06 16:57:04 -0700817func (m *ModuleBase) TargetPrimary() bool {
818 return m.commonProperties.CompilePrimary
Colin Cross8b74d172016-09-13 09:59:14 -0700819}
820
Colin Cross4157e882019-06-06 16:57:04 -0700821func (m *ModuleBase) MultiTargets() []Target {
822 return m.commonProperties.CompileMultiTargets
Colin Crossee0bc3b2018-10-02 22:01:37 -0700823}
824
Colin Cross4157e882019-06-06 16:57:04 -0700825func (m *ModuleBase) Os() OsType {
826 return m.Target().Os
Dan Willemsen490fd492015-11-24 17:53:15 -0800827}
828
Colin Cross4157e882019-06-06 16:57:04 -0700829func (m *ModuleBase) Host() bool {
830 return m.Os().Class == Host || m.Os().Class == HostCross
Dan Willemsen97750522016-02-09 17:43:51 -0800831}
832
Yo Chiangbba545e2020-06-09 16:15:37 +0800833func (m *ModuleBase) Device() bool {
834 return m.Os().Class == Device
835}
836
Colin Cross4157e882019-06-06 16:57:04 -0700837func (m *ModuleBase) Arch() Arch {
838 return m.Target().Arch
Dan Willemsen97750522016-02-09 17:43:51 -0800839}
840
Colin Cross4157e882019-06-06 16:57:04 -0700841func (m *ModuleBase) ArchSpecific() bool {
842 return m.commonProperties.ArchSpecific
Dan Willemsen0b24c742016-10-04 15:13:37 -0700843}
844
Paul Duffin1356d8c2020-02-25 19:26:33 +0000845// True if the current variant is a CommonOS variant, false otherwise.
846func (m *ModuleBase) IsCommonOSVariant() bool {
847 return m.commonProperties.CommonOSVariant
848}
849
Colin Cross4157e882019-06-06 16:57:04 -0700850func (m *ModuleBase) OsClassSupported() []OsClass {
851 switch m.commonProperties.HostOrDeviceSupported {
Colin Crossa1ad8d12016-06-01 17:09:44 -0700852 case HostSupported:
Colin Crossa1ad8d12016-06-01 17:09:44 -0700853 return []OsClass{Host, HostCross}
Dan Albertc6345fb2016-10-20 01:36:11 -0700854 case HostSupportedNoCross:
855 return []OsClass{Host}
Colin Crossa1ad8d12016-06-01 17:09:44 -0700856 case DeviceSupported:
857 return []OsClass{Device}
Dan Albert0981b5c2018-08-02 13:46:35 -0700858 case HostAndDeviceSupported, HostAndDeviceDefault:
Colin Crossa1ad8d12016-06-01 17:09:44 -0700859 var supported []OsClass
Colin Cross4157e882019-06-06 16:57:04 -0700860 if Bool(m.hostAndDeviceProperties.Host_supported) ||
861 (m.commonProperties.HostOrDeviceSupported == HostAndDeviceDefault &&
862 m.hostAndDeviceProperties.Host_supported == nil) {
Colin Crossa1ad8d12016-06-01 17:09:44 -0700863 supported = append(supported, Host, HostCross)
864 }
Colin Cross4157e882019-06-06 16:57:04 -0700865 if m.hostAndDeviceProperties.Device_supported == nil ||
866 *m.hostAndDeviceProperties.Device_supported {
Colin Crossa1ad8d12016-06-01 17:09:44 -0700867 supported = append(supported, Device)
868 }
869 return supported
870 default:
871 return nil
872 }
Colin Cross3f40fa42015-01-30 17:27:36 -0800873}
874
Colin Cross4157e882019-06-06 16:57:04 -0700875func (m *ModuleBase) DeviceSupported() bool {
876 return m.commonProperties.HostOrDeviceSupported == DeviceSupported ||
877 m.commonProperties.HostOrDeviceSupported == HostAndDeviceSupported &&
878 (m.hostAndDeviceProperties.Device_supported == nil ||
879 *m.hostAndDeviceProperties.Device_supported)
Colin Cross3f40fa42015-01-30 17:27:36 -0800880}
881
Paul Duffine44358f2019-11-26 18:04:12 +0000882func (m *ModuleBase) HostSupported() bool {
883 return m.commonProperties.HostOrDeviceSupported == HostSupported ||
884 m.commonProperties.HostOrDeviceSupported == HostAndDeviceSupported &&
885 (m.hostAndDeviceProperties.Host_supported != nil &&
886 *m.hostAndDeviceProperties.Host_supported)
887}
888
Colin Cross4157e882019-06-06 16:57:04 -0700889func (m *ModuleBase) Platform() bool {
Justin Yund5f6c822019-06-25 16:47:17 +0900890 return !m.DeviceSpecific() && !m.SocSpecific() && !m.ProductSpecific() && !m.SystemExtSpecific()
Jiyong Parkc678ad32018-04-10 13:07:10 +0900891}
892
Colin Cross4157e882019-06-06 16:57:04 -0700893func (m *ModuleBase) DeviceSpecific() bool {
894 return Bool(m.commonProperties.Device_specific)
Jiyong Parkc678ad32018-04-10 13:07:10 +0900895}
896
Colin Cross4157e882019-06-06 16:57:04 -0700897func (m *ModuleBase) SocSpecific() bool {
898 return Bool(m.commonProperties.Vendor) || Bool(m.commonProperties.Proprietary) || Bool(m.commonProperties.Soc_specific)
Jiyong Parkc678ad32018-04-10 13:07:10 +0900899}
900
Colin Cross4157e882019-06-06 16:57:04 -0700901func (m *ModuleBase) ProductSpecific() bool {
902 return Bool(m.commonProperties.Product_specific)
Jiyong Parkc678ad32018-04-10 13:07:10 +0900903}
904
Justin Yund5f6c822019-06-25 16:47:17 +0900905func (m *ModuleBase) SystemExtSpecific() bool {
906 return Bool(m.commonProperties.System_ext_specific)
Dario Frenifd05a742018-05-29 13:28:54 +0100907}
908
Colin Crossc2d24052020-05-13 11:05:02 -0700909// RequiresStableAPIs returns true if the module will be installed to a partition that may
910// be updated separately from the system image.
911func (m *ModuleBase) RequiresStableAPIs(ctx BaseModuleContext) bool {
912 return m.SocSpecific() || m.DeviceSpecific() ||
913 (m.ProductSpecific() && ctx.Config().EnforceProductPartitionInterface())
914}
915
Bill Peckhamfff3f8a2020-03-20 18:33:20 -0700916func (m *ModuleBase) PartitionTag(config DeviceConfig) string {
917 partition := "system"
918 if m.SocSpecific() {
919 // A SoC-specific module could be on the vendor partition at
920 // "vendor" or the system partition at "system/vendor".
921 if config.VendorPath() == "vendor" {
922 partition = "vendor"
923 }
924 } else if m.DeviceSpecific() {
925 // A device-specific module could be on the odm partition at
926 // "odm", the vendor partition at "vendor/odm", or the system
927 // partition at "system/vendor/odm".
928 if config.OdmPath() == "odm" {
929 partition = "odm"
Ramy Medhat944839a2020-03-31 22:14:52 -0400930 } else if strings.HasPrefix(config.OdmPath(), "vendor/") {
Bill Peckhamfff3f8a2020-03-20 18:33:20 -0700931 partition = "vendor"
932 }
933 } else if m.ProductSpecific() {
934 // A product-specific module could be on the product partition
935 // at "product" or the system partition at "system/product".
936 if config.ProductPath() == "product" {
937 partition = "product"
938 }
939 } else if m.SystemExtSpecific() {
940 // A system_ext-specific module could be on the system_ext
941 // partition at "system_ext" or the system partition at
942 // "system/system_ext".
943 if config.SystemExtPath() == "system_ext" {
944 partition = "system_ext"
945 }
946 }
947 return partition
948}
949
Colin Cross4157e882019-06-06 16:57:04 -0700950func (m *ModuleBase) Enabled() bool {
951 if m.commonProperties.Enabled == nil {
952 return !m.Os().DefaultDisabled
Dan Willemsen490fd492015-11-24 17:53:15 -0800953 }
Colin Cross4157e882019-06-06 16:57:04 -0700954 return *m.commonProperties.Enabled
Colin Cross3f40fa42015-01-30 17:27:36 -0800955}
956
Inseob Kimeec88e12020-01-22 11:11:29 +0900957func (m *ModuleBase) Disable() {
958 m.commonProperties.Enabled = proptools.BoolPtr(false)
959}
960
Colin Cross4157e882019-06-06 16:57:04 -0700961func (m *ModuleBase) SkipInstall() {
962 m.commonProperties.SkipInstall = true
Colin Crossce75d2c2016-10-06 16:12:58 -0700963}
964
Ulya Trafimovichb28cc372020-01-13 15:18:16 +0000965func (m *ModuleBase) IsSkipInstall() bool {
966 return m.commonProperties.SkipInstall == true
967}
968
Colin Cross4157e882019-06-06 16:57:04 -0700969func (m *ModuleBase) ExportedToMake() bool {
970 return m.commonProperties.NamespaceExportedToMake
Jiyong Park374510b2018-03-19 18:23:01 +0900971}
972
Colin Cross897266e2020-02-13 13:22:08 -0800973func (m *ModuleBase) computeInstallDeps(ctx blueprint.ModuleContext) InstallPaths {
Colin Cross3f40fa42015-01-30 17:27:36 -0800974
Colin Cross897266e2020-02-13 13:22:08 -0800975 var result InstallPaths
Colin Cross6b753602018-06-21 13:03:07 -0700976 // TODO(ccross): we need to use WalkDeps and have some way to know which dependencies require installation
Colin Cross897266e2020-02-13 13:22:08 -0800977 ctx.VisitDepsDepthFirst(func(m blueprint.Module) {
978 if a, ok := m.(Module); ok {
979 result = append(result, a.filesToInstall()...)
980 }
981 })
Colin Cross3f40fa42015-01-30 17:27:36 -0800982
983 return result
984}
985
Colin Cross897266e2020-02-13 13:22:08 -0800986func (m *ModuleBase) filesToInstall() InstallPaths {
Colin Cross4157e882019-06-06 16:57:04 -0700987 return m.installFiles
Colin Cross3f40fa42015-01-30 17:27:36 -0800988}
989
Colin Cross4157e882019-06-06 16:57:04 -0700990func (m *ModuleBase) NoAddressSanitizer() bool {
991 return m.noAddressSanitizer
Colin Cross3f40fa42015-01-30 17:27:36 -0800992}
993
Colin Cross4157e882019-06-06 16:57:04 -0700994func (m *ModuleBase) InstallInData() bool {
Dan Willemsen782a2d12015-12-21 14:55:28 -0800995 return false
996}
997
Jaewoong Jung0949f312019-09-11 10:25:18 -0700998func (m *ModuleBase) InstallInTestcases() bool {
999 return false
1000}
1001
Colin Cross4157e882019-06-06 16:57:04 -07001002func (m *ModuleBase) InstallInSanitizerDir() bool {
Vishwath Mohan1dd88392017-03-29 22:00:18 -07001003 return false
1004}
1005
Yifan Hong1b3348d2020-01-21 15:53:22 -08001006func (m *ModuleBase) InstallInRamdisk() bool {
1007 return Bool(m.commonProperties.Ramdisk)
1008}
1009
Colin Cross4157e882019-06-06 16:57:04 -07001010func (m *ModuleBase) InstallInRecovery() bool {
1011 return Bool(m.commonProperties.Recovery)
Jiyong Parkf9332f12018-02-01 00:54:12 +09001012}
1013
Colin Cross90ba5f42019-10-02 11:10:58 -07001014func (m *ModuleBase) InstallInRoot() bool {
1015 return false
1016}
1017
Colin Cross607d8582019-07-29 16:44:46 -07001018func (m *ModuleBase) InstallBypassMake() bool {
1019 return false
1020}
1021
Colin Cross6e359402020-02-10 15:29:54 -08001022func (m *ModuleBase) InstallForceOS() *OsType {
1023 return nil
1024}
1025
Colin Cross4157e882019-06-06 16:57:04 -07001026func (m *ModuleBase) Owner() string {
1027 return String(m.commonProperties.Owner)
Sundong Ahn4fd04bb2018-08-31 18:01:37 +09001028}
1029
Bob Badoura75b0572020-02-18 20:21:55 -08001030func (m *ModuleBase) NoticeFiles() Paths {
1031 return m.noticeFiles
Jiyong Park52818fc2019-03-18 12:01:38 +09001032}
1033
Colin Cross7228ecd2019-11-18 16:00:16 -08001034func (m *ModuleBase) setImageVariation(variant string) {
1035 m.commonProperties.ImageVariation = variant
1036}
1037
1038func (m *ModuleBase) ImageVariation() blueprint.Variation {
1039 return blueprint.Variation{
1040 Mutator: "image",
1041 Variation: m.base().commonProperties.ImageVariation,
1042 }
1043}
1044
Paul Duffin9b76c0b2020-03-12 10:24:35 +00001045func (m *ModuleBase) getVariationByMutatorName(mutator string) string {
1046 for i, v := range m.commonProperties.DebugMutators {
1047 if v == mutator {
1048 return m.commonProperties.DebugVariations[i]
1049 }
1050 }
1051
1052 return ""
1053}
1054
Yifan Hong1b3348d2020-01-21 15:53:22 -08001055func (m *ModuleBase) InRamdisk() bool {
1056 return m.base().commonProperties.ImageVariation == RamdiskVariation
1057}
1058
Colin Cross7228ecd2019-11-18 16:00:16 -08001059func (m *ModuleBase) InRecovery() bool {
1060 return m.base().commonProperties.ImageVariation == RecoveryVariation
1061}
1062
Jiyong Park6a8cf5f2019-12-30 16:31:09 +09001063func (m *ModuleBase) RequiredModuleNames() []string {
1064 return m.base().commonProperties.Required
1065}
1066
1067func (m *ModuleBase) HostRequiredModuleNames() []string {
1068 return m.base().commonProperties.Host_required
1069}
1070
1071func (m *ModuleBase) TargetRequiredModuleNames() []string {
1072 return m.base().commonProperties.Target_required
1073}
1074
Inseob Kim8471cda2019-11-15 09:59:12 +09001075func (m *ModuleBase) InitRc() Paths {
1076 return append(Paths{}, m.initRcPaths...)
1077}
1078
1079func (m *ModuleBase) VintfFragments() Paths {
1080 return append(Paths{}, m.vintfFragmentsPaths...)
1081}
1082
Colin Cross4157e882019-06-06 16:57:04 -07001083func (m *ModuleBase) generateModuleTarget(ctx ModuleContext) {
Colin Cross897266e2020-02-13 13:22:08 -08001084 var allInstalledFiles InstallPaths
1085 var allCheckbuildFiles Paths
Colin Cross0875c522017-11-28 17:34:01 -08001086 ctx.VisitAllModuleVariants(func(module Module) {
1087 a := module.base()
Colin Crossc9404352015-03-26 16:10:12 -07001088 allInstalledFiles = append(allInstalledFiles, a.installFiles...)
1089 allCheckbuildFiles = append(allCheckbuildFiles, a.checkbuildFiles...)
Colin Cross3f40fa42015-01-30 17:27:36 -08001090 })
1091
Colin Cross0875c522017-11-28 17:34:01 -08001092 var deps Paths
Colin Cross9454bfa2015-03-17 13:24:18 -07001093
Jeff Gaston088e29e2017-11-29 16:47:17 -08001094 namespacePrefix := ctx.Namespace().(*Namespace).id
1095 if namespacePrefix != "" {
1096 namespacePrefix = namespacePrefix + "-"
1097 }
1098
Colin Cross3f40fa42015-01-30 17:27:36 -08001099 if len(allInstalledFiles) > 0 {
Jeff Gaston088e29e2017-11-29 16:47:17 -08001100 name := PathForPhony(ctx, namespacePrefix+ctx.ModuleName()+"-install")
Colin Cross0875c522017-11-28 17:34:01 -08001101 ctx.Build(pctx, BuildParams{
Colin Cross9454bfa2015-03-17 13:24:18 -07001102 Rule: blueprint.Phony,
Colin Cross0875c522017-11-28 17:34:01 -08001103 Output: name,
Colin Cross897266e2020-02-13 13:22:08 -08001104 Implicits: allInstalledFiles.Paths(),
Colin Crossaabf6792017-11-29 00:27:14 -08001105 Default: !ctx.Config().EmbeddedInMake(),
Colin Cross9454bfa2015-03-17 13:24:18 -07001106 })
1107 deps = append(deps, name)
Colin Cross4157e882019-06-06 16:57:04 -07001108 m.installTarget = name
Colin Cross9454bfa2015-03-17 13:24:18 -07001109 }
1110
1111 if len(allCheckbuildFiles) > 0 {
Jeff Gaston088e29e2017-11-29 16:47:17 -08001112 name := PathForPhony(ctx, namespacePrefix+ctx.ModuleName()+"-checkbuild")
Colin Cross0875c522017-11-28 17:34:01 -08001113 ctx.Build(pctx, BuildParams{
Colin Cross9454bfa2015-03-17 13:24:18 -07001114 Rule: blueprint.Phony,
Colin Cross0875c522017-11-28 17:34:01 -08001115 Output: name,
1116 Implicits: allCheckbuildFiles,
Colin Cross9454bfa2015-03-17 13:24:18 -07001117 })
1118 deps = append(deps, name)
Colin Cross4157e882019-06-06 16:57:04 -07001119 m.checkbuildTarget = name
Colin Cross9454bfa2015-03-17 13:24:18 -07001120 }
1121
1122 if len(deps) > 0 {
Dan Willemsen5ba07e82015-12-11 13:51:06 -08001123 suffix := ""
Colin Crossaabf6792017-11-29 00:27:14 -08001124 if ctx.Config().EmbeddedInMake() {
Dan Willemsen5ba07e82015-12-11 13:51:06 -08001125 suffix = "-soong"
1126 }
1127
Jeff Gaston088e29e2017-11-29 16:47:17 -08001128 name := PathForPhony(ctx, namespacePrefix+ctx.ModuleName()+suffix)
Colin Cross0875c522017-11-28 17:34:01 -08001129 ctx.Build(pctx, BuildParams{
Colin Cross9454bfa2015-03-17 13:24:18 -07001130 Rule: blueprint.Phony,
Jeff Gaston088e29e2017-11-29 16:47:17 -08001131 Outputs: []WritablePath{name},
Colin Cross9454bfa2015-03-17 13:24:18 -07001132 Implicits: deps,
Colin Cross3f40fa42015-01-30 17:27:36 -08001133 })
Colin Cross1f8c52b2015-06-16 16:38:17 -07001134
Colin Cross4157e882019-06-06 16:57:04 -07001135 m.blueprintDir = ctx.ModuleDir()
Colin Cross3f40fa42015-01-30 17:27:36 -08001136 }
1137}
1138
Colin Crossc34d2322020-01-03 15:23:27 -08001139func determineModuleKind(m *ModuleBase, ctx blueprint.EarlyModuleContext) moduleKind {
Colin Cross4157e882019-06-06 16:57:04 -07001140 var socSpecific = Bool(m.commonProperties.Vendor) || Bool(m.commonProperties.Proprietary) || Bool(m.commonProperties.Soc_specific)
1141 var deviceSpecific = Bool(m.commonProperties.Device_specific)
1142 var productSpecific = Bool(m.commonProperties.Product_specific)
Justin Yund5f6c822019-06-25 16:47:17 +09001143 var systemExtSpecific = Bool(m.commonProperties.System_ext_specific)
Jiyong Park2db76922017-11-08 16:03:48 +09001144
Dario Frenifd05a742018-05-29 13:28:54 +01001145 msg := "conflicting value set here"
1146 if socSpecific && deviceSpecific {
1147 ctx.PropertyErrorf("device_specific", "a module cannot be specific to SoC and device at the same time.")
Colin Cross4157e882019-06-06 16:57:04 -07001148 if Bool(m.commonProperties.Vendor) {
Jiyong Park2db76922017-11-08 16:03:48 +09001149 ctx.PropertyErrorf("vendor", msg)
1150 }
Colin Cross4157e882019-06-06 16:57:04 -07001151 if Bool(m.commonProperties.Proprietary) {
Jiyong Park2db76922017-11-08 16:03:48 +09001152 ctx.PropertyErrorf("proprietary", msg)
1153 }
Colin Cross4157e882019-06-06 16:57:04 -07001154 if Bool(m.commonProperties.Soc_specific) {
Jiyong Park2db76922017-11-08 16:03:48 +09001155 ctx.PropertyErrorf("soc_specific", msg)
1156 }
1157 }
1158
Justin Yund5f6c822019-06-25 16:47:17 +09001159 if productSpecific && systemExtSpecific {
1160 ctx.PropertyErrorf("product_specific", "a module cannot be specific to product and system_ext at the same time.")
1161 ctx.PropertyErrorf("system_ext_specific", msg)
Dario Frenifd05a742018-05-29 13:28:54 +01001162 }
1163
Justin Yund5f6c822019-06-25 16:47:17 +09001164 if (socSpecific || deviceSpecific) && (productSpecific || systemExtSpecific) {
Dario Frenifd05a742018-05-29 13:28:54 +01001165 if productSpecific {
1166 ctx.PropertyErrorf("product_specific", "a module cannot be specific to SoC or device and product at the same time.")
1167 } else {
Justin Yund5f6c822019-06-25 16:47:17 +09001168 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 +01001169 }
1170 if deviceSpecific {
1171 ctx.PropertyErrorf("device_specific", msg)
1172 } else {
Colin Cross4157e882019-06-06 16:57:04 -07001173 if Bool(m.commonProperties.Vendor) {
Dario Frenifd05a742018-05-29 13:28:54 +01001174 ctx.PropertyErrorf("vendor", msg)
1175 }
Colin Cross4157e882019-06-06 16:57:04 -07001176 if Bool(m.commonProperties.Proprietary) {
Dario Frenifd05a742018-05-29 13:28:54 +01001177 ctx.PropertyErrorf("proprietary", msg)
1178 }
Colin Cross4157e882019-06-06 16:57:04 -07001179 if Bool(m.commonProperties.Soc_specific) {
Dario Frenifd05a742018-05-29 13:28:54 +01001180 ctx.PropertyErrorf("soc_specific", msg)
1181 }
1182 }
1183 }
1184
Jiyong Park2db76922017-11-08 16:03:48 +09001185 if productSpecific {
1186 return productSpecificModule
Justin Yund5f6c822019-06-25 16:47:17 +09001187 } else if systemExtSpecific {
1188 return systemExtSpecificModule
Jiyong Park2db76922017-11-08 16:03:48 +09001189 } else if deviceSpecific {
1190 return deviceSpecificModule
1191 } else if socSpecific {
1192 return socSpecificModule
1193 } else {
1194 return platformModule
1195 }
1196}
1197
Colin Crossc34d2322020-01-03 15:23:27 -08001198func (m *ModuleBase) earlyModuleContextFactory(ctx blueprint.EarlyModuleContext) earlyModuleContext {
Colin Cross1184b642019-12-30 18:43:07 -08001199 return earlyModuleContext{
Colin Crossc34d2322020-01-03 15:23:27 -08001200 EarlyModuleContext: ctx,
1201 kind: determineModuleKind(m, ctx),
1202 config: ctx.Config().(Config),
Colin Cross3f40fa42015-01-30 17:27:36 -08001203 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001204}
1205
Colin Cross1184b642019-12-30 18:43:07 -08001206func (m *ModuleBase) baseModuleContextFactory(ctx blueprint.BaseModuleContext) baseModuleContext {
1207 return baseModuleContext{
1208 bp: ctx,
1209 earlyModuleContext: m.earlyModuleContextFactory(ctx),
1210 os: m.commonProperties.CompileOS,
1211 target: m.commonProperties.CompileTarget,
1212 targetPrimary: m.commonProperties.CompilePrimary,
1213 multiTargets: m.commonProperties.CompileMultiTargets,
1214 }
1215}
1216
Colin Cross4157e882019-06-06 16:57:04 -07001217func (m *ModuleBase) GenerateBuildActions(blueprintCtx blueprint.ModuleContext) {
Colin Cross25de6c32019-06-06 14:29:25 -07001218 ctx := &moduleContext{
Colin Cross0ea8ba82019-06-06 14:33:29 -07001219 module: m.module,
Colin Crossdc35e212019-06-06 16:13:11 -07001220 bp: blueprintCtx,
Colin Cross0ea8ba82019-06-06 14:33:29 -07001221 baseModuleContext: m.baseModuleContextFactory(blueprintCtx),
1222 installDeps: m.computeInstallDeps(blueprintCtx),
1223 installFiles: m.installFiles,
Colin Cross0ea8ba82019-06-06 14:33:29 -07001224 variables: make(map[string]string),
Colin Cross3f40fa42015-01-30 17:27:36 -08001225 }
1226
Colin Cross6c4f21f2019-06-06 15:41:36 -07001227 // Temporarily continue to call blueprintCtx.GetMissingDependencies() to maintain the previous behavior of never
1228 // reporting missing dependency errors in Blueprint when AllowMissingDependencies == true.
1229 // TODO: This will be removed once defaults modules handle missing dependency errors
1230 blueprintCtx.GetMissingDependencies()
1231
Colin Crossdc35e212019-06-06 16:13:11 -07001232 // For the final GenerateAndroidBuildActions pass, require that all visited dependencies Soong modules and
Paul Duffin1356d8c2020-02-25 19:26:33 +00001233 // are enabled. Unless the module is a CommonOS variant which may have dependencies on disabled variants
1234 // (because the dependencies are added before the modules are disabled). The
1235 // GetOsSpecificVariantsOfCommonOSVariant(...) method will ensure that the disabled variants are
1236 // ignored.
1237 ctx.baseModuleContext.strictVisitDeps = !m.IsCommonOSVariant()
Colin Crossdc35e212019-06-06 16:13:11 -07001238
Colin Cross4c83e5c2019-02-25 14:54:28 -08001239 if ctx.config.captureBuild {
1240 ctx.ruleParams = make(map[blueprint.Rule]blueprint.RuleParams)
1241 }
1242
Colin Cross67a5c132017-05-09 13:45:28 -07001243 desc := "//" + ctx.ModuleDir() + ":" + ctx.ModuleName() + " "
1244 var suffix []string
Colin Cross0875c522017-11-28 17:34:01 -08001245 if ctx.Os().Class != Device && ctx.Os().Class != Generic {
1246 suffix = append(suffix, ctx.Os().String())
Colin Cross67a5c132017-05-09 13:45:28 -07001247 }
Colin Cross0875c522017-11-28 17:34:01 -08001248 if !ctx.PrimaryArch() {
1249 suffix = append(suffix, ctx.Arch().ArchType.String())
Colin Cross67a5c132017-05-09 13:45:28 -07001250 }
Dan Willemsenb13a9482020-02-14 11:25:54 -08001251 if apex, ok := m.module.(ApexModule); ok && !apex.IsForPlatform() {
1252 suffix = append(suffix, apex.ApexName())
1253 }
Colin Cross67a5c132017-05-09 13:45:28 -07001254
1255 ctx.Variable(pctx, "moduleDesc", desc)
1256
1257 s := ""
1258 if len(suffix) > 0 {
1259 s = " [" + strings.Join(suffix, " ") + "]"
1260 }
1261 ctx.Variable(pctx, "moduleDescSuffix", s)
1262
Dan Willemsen569edc52018-11-19 09:33:29 -08001263 // Some common property checks for properties that will be used later in androidmk.go
Colin Cross4157e882019-06-06 16:57:04 -07001264 if m.commonProperties.Dist.Dest != nil {
1265 _, err := validateSafePath(*m.commonProperties.Dist.Dest)
Dan Willemsen569edc52018-11-19 09:33:29 -08001266 if err != nil {
1267 ctx.PropertyErrorf("dist.dest", "%s", err.Error())
1268 }
1269 }
Colin Cross4157e882019-06-06 16:57:04 -07001270 if m.commonProperties.Dist.Dir != nil {
1271 _, err := validateSafePath(*m.commonProperties.Dist.Dir)
Dan Willemsen569edc52018-11-19 09:33:29 -08001272 if err != nil {
1273 ctx.PropertyErrorf("dist.dir", "%s", err.Error())
1274 }
1275 }
Colin Cross4157e882019-06-06 16:57:04 -07001276 if m.commonProperties.Dist.Suffix != nil {
1277 if strings.Contains(*m.commonProperties.Dist.Suffix, "/") {
Dan Willemsen569edc52018-11-19 09:33:29 -08001278 ctx.PropertyErrorf("dist.suffix", "Suffix may not contain a '/' character.")
1279 }
1280 }
1281
Colin Cross4157e882019-06-06 16:57:04 -07001282 if m.Enabled() {
Jooyung Hand48f3c32019-08-23 11:18:57 +09001283 // ensure all direct android.Module deps are enabled
1284 ctx.VisitDirectDepsBlueprint(func(bm blueprint.Module) {
1285 if _, ok := bm.(Module); ok {
1286 ctx.validateAndroidModule(bm, ctx.baseModuleContext.strictVisitDeps)
1287 }
1288 })
1289
Bob Badoura75b0572020-02-18 20:21:55 -08001290 m.noticeFiles = make([]Path, 0)
1291 optPath := OptionalPath{}
1292 notice := proptools.StringDefault(m.commonProperties.Notice, "")
Colin Cross4157e882019-06-06 16:57:04 -07001293 if module := SrcIsModule(notice); module != "" {
Bob Badoura75b0572020-02-18 20:21:55 -08001294 optPath = ctx.ExpandOptionalSource(&notice, "notice")
1295 } else if notice != "" {
Jiyong Park52818fc2019-03-18 12:01:38 +09001296 noticePath := filepath.Join(ctx.ModuleDir(), notice)
Bob Badoura75b0572020-02-18 20:21:55 -08001297 optPath = ExistentPathForSource(ctx, noticePath)
1298 }
1299 if optPath.Valid() {
1300 m.noticeFiles = append(m.noticeFiles, optPath.Path())
1301 } else {
1302 for _, notice = range []string{"LICENSE", "LICENCE", "NOTICE"} {
1303 noticePath := filepath.Join(ctx.ModuleDir(), notice)
1304 optPath = ExistentPathForSource(ctx, noticePath)
1305 if optPath.Valid() {
1306 m.noticeFiles = append(m.noticeFiles, optPath.Path())
1307 }
1308 }
Jaewoong Jung62707f72018-11-16 13:26:43 -08001309 }
Jaewoong Jung5b425e22019-06-17 17:40:56 -07001310
1311 m.module.GenerateAndroidBuildActions(ctx)
1312 if ctx.Failed() {
1313 return
1314 }
1315
1316 m.installFiles = append(m.installFiles, ctx.installFiles...)
1317 m.checkbuildFiles = append(m.checkbuildFiles, ctx.checkbuildFiles...)
Inseob Kim8471cda2019-11-15 09:59:12 +09001318 m.initRcPaths = PathsForModuleSrc(ctx, m.commonProperties.Init_rc)
1319 m.vintfFragmentsPaths = PathsForModuleSrc(ctx, m.commonProperties.Vintf_fragments)
Colin Crossdc35e212019-06-06 16:13:11 -07001320 } else if ctx.Config().AllowMissingDependencies() {
1321 // If the module is not enabled it will not create any build rules, nothing will call
1322 // ctx.GetMissingDependencies(), and blueprint will consider the missing dependencies to be unhandled
1323 // and report them as an error even when AllowMissingDependencies = true. Call
1324 // ctx.GetMissingDependencies() here to tell blueprint not to handle them.
1325 ctx.GetMissingDependencies()
Colin Cross3f40fa42015-01-30 17:27:36 -08001326 }
1327
Colin Cross4157e882019-06-06 16:57:04 -07001328 if m == ctx.FinalModule().(Module).base() {
1329 m.generateModuleTarget(ctx)
Colin Cross9b1d13d2016-09-19 15:18:11 -07001330 if ctx.Failed() {
1331 return
1332 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001333 }
Colin Crosscec81712017-07-13 14:43:27 -07001334
Colin Cross4157e882019-06-06 16:57:04 -07001335 m.buildParams = ctx.buildParams
1336 m.ruleParams = ctx.ruleParams
1337 m.variables = ctx.variables
Colin Cross3f40fa42015-01-30 17:27:36 -08001338}
1339
Colin Cross1184b642019-12-30 18:43:07 -08001340type earlyModuleContext struct {
Colin Crossc34d2322020-01-03 15:23:27 -08001341 blueprint.EarlyModuleContext
Colin Cross1184b642019-12-30 18:43:07 -08001342
1343 kind moduleKind
1344 config Config
1345}
1346
1347func (e *earlyModuleContext) Glob(globPattern string, excludes []string) Paths {
1348 ret, err := e.GlobWithDeps(globPattern, excludes)
1349 if err != nil {
1350 e.ModuleErrorf("glob: %s", err.Error())
1351 }
1352 return pathsForModuleSrcFromFullPath(e, ret, true)
1353}
1354
1355func (e *earlyModuleContext) GlobFiles(globPattern string, excludes []string) Paths {
1356 ret, err := e.GlobWithDeps(globPattern, excludes)
1357 if err != nil {
1358 e.ModuleErrorf("glob: %s", err.Error())
1359 }
1360 return pathsForModuleSrcFromFullPath(e, ret, false)
1361}
1362
Colin Cross988414c2020-01-11 01:11:46 +00001363func (b *earlyModuleContext) IsSymlink(path Path) bool {
1364 fileInfo, err := b.config.fs.Lstat(path.String())
1365 if err != nil {
1366 b.ModuleErrorf("os.Lstat(%q) failed: %s", path.String(), err)
1367 }
1368 return fileInfo.Mode()&os.ModeSymlink == os.ModeSymlink
1369}
1370
1371func (b *earlyModuleContext) Readlink(path Path) string {
1372 dest, err := b.config.fs.Readlink(path.String())
1373 if err != nil {
1374 b.ModuleErrorf("os.Readlink(%q) failed: %s", path.String(), err)
1375 }
1376 return dest
1377}
1378
Colin Cross1184b642019-12-30 18:43:07 -08001379func (e *earlyModuleContext) Module() Module {
Colin Crossc34d2322020-01-03 15:23:27 -08001380 module, _ := e.EarlyModuleContext.Module().(Module)
Colin Cross1184b642019-12-30 18:43:07 -08001381 return module
1382}
1383
1384func (e *earlyModuleContext) Config() Config {
Colin Crossc34d2322020-01-03 15:23:27 -08001385 return e.EarlyModuleContext.Config().(Config)
Colin Cross1184b642019-12-30 18:43:07 -08001386}
1387
1388func (e *earlyModuleContext) AConfig() Config {
1389 return e.config
1390}
1391
1392func (e *earlyModuleContext) DeviceConfig() DeviceConfig {
1393 return DeviceConfig{e.config.deviceConfig}
1394}
1395
1396func (e *earlyModuleContext) Platform() bool {
1397 return e.kind == platformModule
1398}
1399
1400func (e *earlyModuleContext) DeviceSpecific() bool {
1401 return e.kind == deviceSpecificModule
1402}
1403
1404func (e *earlyModuleContext) SocSpecific() bool {
1405 return e.kind == socSpecificModule
1406}
1407
1408func (e *earlyModuleContext) ProductSpecific() bool {
1409 return e.kind == productSpecificModule
1410}
1411
1412func (e *earlyModuleContext) SystemExtSpecific() bool {
1413 return e.kind == systemExtSpecificModule
1414}
1415
1416type baseModuleContext struct {
1417 bp blueprint.BaseModuleContext
1418 earlyModuleContext
Colin Crossfb0c16e2019-11-20 17:12:35 -08001419 os OsType
Colin Cross8b74d172016-09-13 09:59:14 -07001420 target Target
Colin Crossee0bc3b2018-10-02 22:01:37 -07001421 multiTargets []Target
Colin Cross8b74d172016-09-13 09:59:14 -07001422 targetPrimary bool
1423 debug bool
Colin Crossdc35e212019-06-06 16:13:11 -07001424
1425 walkPath []Module
Paul Duffinc5192442020-03-31 11:31:36 +01001426 tagPath []blueprint.DependencyTag
Colin Crossdc35e212019-06-06 16:13:11 -07001427
1428 strictVisitDeps bool // If true, enforce that all dependencies are enabled
Colin Crossf6566ed2015-03-24 11:13:38 -07001429}
1430
Paul Duffinca7f0ef2020-02-25 15:50:49 +00001431func (b *baseModuleContext) OtherModuleName(m blueprint.Module) string {
1432 return b.bp.OtherModuleName(m)
1433}
1434func (b *baseModuleContext) OtherModuleDir(m blueprint.Module) string { return b.bp.OtherModuleDir(m) }
Colin Cross1184b642019-12-30 18:43:07 -08001435func (b *baseModuleContext) OtherModuleErrorf(m blueprint.Module, fmt string, args ...interface{}) {
Jooyung Hancd87c692020-02-26 02:05:18 +09001436 b.bp.OtherModuleErrorf(m, fmt, args...)
Colin Cross1184b642019-12-30 18:43:07 -08001437}
1438func (b *baseModuleContext) OtherModuleDependencyTag(m blueprint.Module) blueprint.DependencyTag {
1439 return b.bp.OtherModuleDependencyTag(m)
1440}
Paul Duffinca7f0ef2020-02-25 15:50:49 +00001441func (b *baseModuleContext) OtherModuleExists(name string) bool { return b.bp.OtherModuleExists(name) }
Martin Stjernholm009a9dc2020-03-05 17:34:13 +00001442func (b *baseModuleContext) OtherModuleDependencyVariantExists(variations []blueprint.Variation, name string) bool {
1443 return b.bp.OtherModuleDependencyVariantExists(variations, name)
1444}
1445func (b *baseModuleContext) OtherModuleReverseDependencyVariantExists(name string) bool {
1446 return b.bp.OtherModuleReverseDependencyVariantExists(name)
1447}
Paul Duffinca7f0ef2020-02-25 15:50:49 +00001448func (b *baseModuleContext) OtherModuleType(m blueprint.Module) string {
1449 return b.bp.OtherModuleType(m)
1450}
Colin Cross1184b642019-12-30 18:43:07 -08001451
1452func (b *baseModuleContext) GetDirectDepWithTag(name string, tag blueprint.DependencyTag) blueprint.Module {
1453 return b.bp.GetDirectDepWithTag(name, tag)
1454}
1455
Paul Duffinf88d8e02020-05-07 20:21:34 +01001456func (b *baseModuleContext) blueprintBaseModuleContext() blueprint.BaseModuleContext {
1457 return b.bp
1458}
1459
Colin Cross25de6c32019-06-06 14:29:25 -07001460type moduleContext struct {
Colin Crossdc35e212019-06-06 16:13:11 -07001461 bp blueprint.ModuleContext
Colin Cross0ea8ba82019-06-06 14:33:29 -07001462 baseModuleContext
Colin Cross897266e2020-02-13 13:22:08 -08001463 installDeps InstallPaths
1464 installFiles InstallPaths
Dan Willemsen34cc69e2015-09-23 15:26:20 -07001465 checkbuildFiles Paths
Colin Cross8d8f8e22016-08-03 11:57:50 -07001466 module Module
Colin Crosscec81712017-07-13 14:43:27 -07001467
1468 // For tests
Colin Crossae887032017-10-23 17:16:14 -07001469 buildParams []BuildParams
Colin Cross4c83e5c2019-02-25 14:54:28 -08001470 ruleParams map[blueprint.Rule]blueprint.RuleParams
Jaewoong Jung38e4fb22018-12-12 09:01:34 -08001471 variables map[string]string
Colin Cross6ff51382015-12-17 16:39:19 -08001472}
1473
Colin Crossb88b3c52019-06-10 15:15:17 -07001474func (m *moduleContext) ninjaError(params BuildParams, err error) (PackageContext, BuildParams) {
1475 return pctx, BuildParams{
Colin Cross4b69c492019-06-07 13:06:06 -07001476 Rule: ErrorRule,
1477 Description: params.Description,
1478 Output: params.Output,
1479 Outputs: params.Outputs,
1480 ImplicitOutput: params.ImplicitOutput,
1481 ImplicitOutputs: params.ImplicitOutputs,
Colin Cross6ff51382015-12-17 16:39:19 -08001482 Args: map[string]string{
1483 "error": err.Error(),
1484 },
Colin Crossb88b3c52019-06-10 15:15:17 -07001485 }
Colin Cross3f40fa42015-01-30 17:27:36 -08001486}
1487
Colin Cross25de6c32019-06-06 14:29:25 -07001488func (m *moduleContext) ModuleBuild(pctx PackageContext, params ModuleBuildParams) {
1489 m.Build(pctx, BuildParams(params))
Colin Cross3f40fa42015-01-30 17:27:36 -08001490}
1491
Colin Cross0875c522017-11-28 17:34:01 -08001492func convertBuildParams(params BuildParams) blueprint.BuildParams {
Dan Willemsen34cc69e2015-09-23 15:26:20 -07001493 bparams := blueprint.BuildParams{
Dan Willemsen9f3c5742016-11-03 14:28:31 -07001494 Rule: params.Rule,
Colin Cross0875c522017-11-28 17:34:01 -08001495 Description: params.Description,
Colin Cross33bfb0a2016-11-21 17:23:08 -08001496 Deps: params.Deps,
Dan Willemsen9f3c5742016-11-03 14:28:31 -07001497 Outputs: params.Outputs.Strings(),
1498 ImplicitOutputs: params.ImplicitOutputs.Strings(),
1499 Inputs: params.Inputs.Strings(),
1500 Implicits: params.Implicits.Strings(),
1501 OrderOnly: params.OrderOnly.Strings(),
1502 Args: params.Args,
1503 Optional: !params.Default,
Dan Willemsen34cc69e2015-09-23 15:26:20 -07001504 }
1505
Colin Cross33bfb0a2016-11-21 17:23:08 -08001506 if params.Depfile != nil {
1507 bparams.Depfile = params.Depfile.String()
1508 }
Dan Willemsen34cc69e2015-09-23 15:26:20 -07001509 if params.Output != nil {
1510 bparams.Outputs = append(bparams.Outputs, params.Output.String())
1511 }
Dan Willemsen9f3c5742016-11-03 14:28:31 -07001512 if params.ImplicitOutput != nil {
1513 bparams.ImplicitOutputs = append(bparams.ImplicitOutputs, params.ImplicitOutput.String())
1514 }
Dan Willemsen34cc69e2015-09-23 15:26:20 -07001515 if params.Input != nil {
1516 bparams.Inputs = append(bparams.Inputs, params.Input.String())
1517 }
1518 if params.Implicit != nil {
1519 bparams.Implicits = append(bparams.Implicits, params.Implicit.String())
1520 }
1521
Colin Cross0b9f31f2019-02-28 11:00:01 -08001522 bparams.Outputs = proptools.NinjaEscapeList(bparams.Outputs)
1523 bparams.ImplicitOutputs = proptools.NinjaEscapeList(bparams.ImplicitOutputs)
1524 bparams.Inputs = proptools.NinjaEscapeList(bparams.Inputs)
1525 bparams.Implicits = proptools.NinjaEscapeList(bparams.Implicits)
1526 bparams.OrderOnly = proptools.NinjaEscapeList(bparams.OrderOnly)
1527 bparams.Depfile = proptools.NinjaEscapeList([]string{bparams.Depfile})[0]
Colin Crossfe4bc362018-09-12 10:02:13 -07001528
Colin Cross0875c522017-11-28 17:34:01 -08001529 return bparams
1530}
1531
Colin Cross25de6c32019-06-06 14:29:25 -07001532func (m *moduleContext) Variable(pctx PackageContext, name, value string) {
1533 if m.config.captureBuild {
1534 m.variables[name] = value
Jaewoong Jung38e4fb22018-12-12 09:01:34 -08001535 }
1536
Colin Crossdc35e212019-06-06 16:13:11 -07001537 m.bp.Variable(pctx.PackageContext, name, value)
Colin Cross0875c522017-11-28 17:34:01 -08001538}
1539
Colin Cross25de6c32019-06-06 14:29:25 -07001540func (m *moduleContext) Rule(pctx PackageContext, name string, params blueprint.RuleParams,
Colin Cross0875c522017-11-28 17:34:01 -08001541 argNames ...string) blueprint.Rule {
1542
Ramy Medhat944839a2020-03-31 22:14:52 -04001543 if m.config.UseRemoteBuild() {
1544 if params.Pool == nil {
1545 // When USE_GOMA=true or USE_RBE=true are set and the rule is not supported by goma/RBE, restrict
1546 // jobs to the local parallelism value
1547 params.Pool = localPool
1548 } else if params.Pool == remotePool {
1549 // remotePool is a fake pool used to identify rule that are supported for remoting. If the rule's
1550 // pool is the remotePool, replace with nil so that ninja runs it at NINJA_REMOTE_NUM_JOBS
1551 // parallelism.
1552 params.Pool = nil
1553 }
Colin Cross2e2dbc22019-09-25 13:31:46 -07001554 }
1555
Colin Crossdc35e212019-06-06 16:13:11 -07001556 rule := m.bp.Rule(pctx.PackageContext, name, params, argNames...)
Colin Cross4c83e5c2019-02-25 14:54:28 -08001557
Colin Cross25de6c32019-06-06 14:29:25 -07001558 if m.config.captureBuild {
1559 m.ruleParams[rule] = params
Colin Cross4c83e5c2019-02-25 14:54:28 -08001560 }
1561
1562 return rule
Colin Cross0875c522017-11-28 17:34:01 -08001563}
1564
Colin Cross25de6c32019-06-06 14:29:25 -07001565func (m *moduleContext) Build(pctx PackageContext, params BuildParams) {
Colin Crossb88b3c52019-06-10 15:15:17 -07001566 if params.Description != "" {
1567 params.Description = "${moduleDesc}" + params.Description + "${moduleDescSuffix}"
1568 }
1569
1570 if missingDeps := m.GetMissingDependencies(); len(missingDeps) > 0 {
1571 pctx, params = m.ninjaError(params, fmt.Errorf("module %s missing dependencies: %s\n",
1572 m.ModuleName(), strings.Join(missingDeps, ", ")))
1573 }
1574
Colin Cross25de6c32019-06-06 14:29:25 -07001575 if m.config.captureBuild {
1576 m.buildParams = append(m.buildParams, params)
Colin Cross0875c522017-11-28 17:34:01 -08001577 }
1578
Colin Crossdc35e212019-06-06 16:13:11 -07001579 m.bp.Build(pctx.PackageContext, convertBuildParams(params))
Dan Willemsen34cc69e2015-09-23 15:26:20 -07001580}
Colin Cross25de6c32019-06-06 14:29:25 -07001581func (m *moduleContext) GetMissingDependencies() []string {
Colin Cross6c4f21f2019-06-06 15:41:36 -07001582 var missingDeps []string
1583 missingDeps = append(missingDeps, m.Module().base().commonProperties.MissingDeps...)
Colin Crossdc35e212019-06-06 16:13:11 -07001584 missingDeps = append(missingDeps, m.bp.GetMissingDependencies()...)
Colin Cross6c4f21f2019-06-06 15:41:36 -07001585 missingDeps = FirstUniqueStrings(missingDeps)
1586 return missingDeps
Colin Cross6ff51382015-12-17 16:39:19 -08001587}
1588
Colin Crossdc35e212019-06-06 16:13:11 -07001589func (b *baseModuleContext) AddMissingDependencies(deps []string) {
Dan Willemsen6553f5e2016-03-10 18:14:25 -08001590 if deps != nil {
Colin Crossdc35e212019-06-06 16:13:11 -07001591 missingDeps := &b.Module().base().commonProperties.MissingDeps
Colin Cross6c4f21f2019-06-06 15:41:36 -07001592 *missingDeps = append(*missingDeps, deps...)
1593 *missingDeps = FirstUniqueStrings(*missingDeps)
Dan Willemsen6553f5e2016-03-10 18:14:25 -08001594 }
1595}
1596
Colin Crossdc35e212019-06-06 16:13:11 -07001597func (b *baseModuleContext) validateAndroidModule(module blueprint.Module, strict bool) Module {
Colin Crossd11fcda2017-10-23 17:59:01 -07001598 aModule, _ := module.(Module)
Colin Crossdc35e212019-06-06 16:13:11 -07001599
1600 if !strict {
1601 return aModule
1602 }
1603
Colin Cross380c69a2019-06-10 17:49:58 +00001604 if aModule == nil {
Colin Crossdc35e212019-06-06 16:13:11 -07001605 b.ModuleErrorf("module %q not an android module", b.OtherModuleName(module))
Colin Cross380c69a2019-06-10 17:49:58 +00001606 return nil
1607 }
1608
1609 if !aModule.Enabled() {
Colin Crossdc35e212019-06-06 16:13:11 -07001610 if b.Config().AllowMissingDependencies() {
1611 b.AddMissingDependencies([]string{b.OtherModuleName(aModule)})
Colin Cross380c69a2019-06-10 17:49:58 +00001612 } else {
Colin Crossdc35e212019-06-06 16:13:11 -07001613 b.ModuleErrorf("depends on disabled module %q", b.OtherModuleName(aModule))
Colin Cross380c69a2019-06-10 17:49:58 +00001614 }
1615 return nil
1616 }
Colin Crossd11fcda2017-10-23 17:59:01 -07001617 return aModule
1618}
1619
Colin Crossdc35e212019-06-06 16:13:11 -07001620func (b *baseModuleContext) getDirectDepInternal(name string, tag blueprint.DependencyTag) (blueprint.Module, blueprint.DependencyTag) {
Jiyong Parkf2976302019-04-17 21:47:37 +09001621 type dep struct {
1622 mod blueprint.Module
1623 tag blueprint.DependencyTag
1624 }
1625 var deps []dep
Colin Crossdc35e212019-06-06 16:13:11 -07001626 b.VisitDirectDepsBlueprint(func(module blueprint.Module) {
Colin Cross25de6c32019-06-06 14:29:25 -07001627 if aModule, _ := module.(Module); aModule != nil && aModule.base().BaseModuleName() == name {
Colin Cross1184b642019-12-30 18:43:07 -08001628 returnedTag := b.bp.OtherModuleDependencyTag(aModule)
Jiyong Parkf2976302019-04-17 21:47:37 +09001629 if tag == nil || returnedTag == tag {
1630 deps = append(deps, dep{aModule, returnedTag})
1631 }
1632 }
1633 })
1634 if len(deps) == 1 {
1635 return deps[0].mod, deps[0].tag
1636 } else if len(deps) >= 2 {
1637 panic(fmt.Errorf("Multiple dependencies having same BaseModuleName() %q found from %q",
Colin Crossdc35e212019-06-06 16:13:11 -07001638 name, b.ModuleName()))
Jiyong Parkf2976302019-04-17 21:47:37 +09001639 } else {
1640 return nil, nil
1641 }
1642}
1643
Colin Crossdc35e212019-06-06 16:13:11 -07001644func (b *baseModuleContext) GetDirectDepsWithTag(tag blueprint.DependencyTag) []Module {
Colin Cross0ef08162019-05-01 15:50:51 -07001645 var deps []Module
Colin Crossdc35e212019-06-06 16:13:11 -07001646 b.VisitDirectDepsBlueprint(func(module blueprint.Module) {
Colin Cross25de6c32019-06-06 14:29:25 -07001647 if aModule, _ := module.(Module); aModule != nil {
Colin Cross1184b642019-12-30 18:43:07 -08001648 if b.bp.OtherModuleDependencyTag(aModule) == tag {
Colin Cross0ef08162019-05-01 15:50:51 -07001649 deps = append(deps, aModule)
1650 }
1651 }
1652 })
1653 return deps
1654}
1655
Colin Cross25de6c32019-06-06 14:29:25 -07001656func (m *moduleContext) GetDirectDepWithTag(name string, tag blueprint.DependencyTag) blueprint.Module {
1657 module, _ := m.getDirectDepInternal(name, tag)
1658 return module
Jiyong Parkf2976302019-04-17 21:47:37 +09001659}
1660
Colin Crossdc35e212019-06-06 16:13:11 -07001661func (b *baseModuleContext) GetDirectDep(name string) (blueprint.Module, blueprint.DependencyTag) {
1662 return b.getDirectDepInternal(name, nil)
Jiyong Parkf2976302019-04-17 21:47:37 +09001663}
1664
Colin Crossdc35e212019-06-06 16:13:11 -07001665func (b *baseModuleContext) VisitDirectDepsBlueprint(visit func(blueprint.Module)) {
Colin Cross1184b642019-12-30 18:43:07 -08001666 b.bp.VisitDirectDeps(visit)
Colin Cross35143d02017-11-16 00:11:20 -08001667}
1668
Colin Crossdc35e212019-06-06 16:13:11 -07001669func (b *baseModuleContext) VisitDirectDeps(visit func(Module)) {
Colin Cross1184b642019-12-30 18:43:07 -08001670 b.bp.VisitDirectDeps(func(module blueprint.Module) {
Colin Crossdc35e212019-06-06 16:13:11 -07001671 if aModule := b.validateAndroidModule(module, b.strictVisitDeps); aModule != nil {
Colin Crossd11fcda2017-10-23 17:59:01 -07001672 visit(aModule)
1673 }
1674 })
1675}
1676
Colin Crossdc35e212019-06-06 16:13:11 -07001677func (b *baseModuleContext) VisitDirectDepsWithTag(tag blueprint.DependencyTag, visit func(Module)) {
Colin Cross1184b642019-12-30 18:43:07 -08001678 b.bp.VisitDirectDeps(func(module blueprint.Module) {
Colin Crossdc35e212019-06-06 16:13:11 -07001679 if aModule := b.validateAndroidModule(module, b.strictVisitDeps); aModule != nil {
Colin Cross1184b642019-12-30 18:43:07 -08001680 if b.bp.OtherModuleDependencyTag(aModule) == tag {
Colin Crossee6143c2017-12-30 17:54:27 -08001681 visit(aModule)
1682 }
1683 }
1684 })
1685}
1686
Colin Crossdc35e212019-06-06 16:13:11 -07001687func (b *baseModuleContext) VisitDirectDepsIf(pred func(Module) bool, visit func(Module)) {
Colin Cross1184b642019-12-30 18:43:07 -08001688 b.bp.VisitDirectDepsIf(
Colin Crossd11fcda2017-10-23 17:59:01 -07001689 // pred
1690 func(module blueprint.Module) bool {
Colin Crossdc35e212019-06-06 16:13:11 -07001691 if aModule := b.validateAndroidModule(module, b.strictVisitDeps); aModule != nil {
Colin Crossd11fcda2017-10-23 17:59:01 -07001692 return pred(aModule)
1693 } else {
1694 return false
1695 }
1696 },
1697 // visit
1698 func(module blueprint.Module) {
1699 visit(module.(Module))
1700 })
1701}
1702
Colin Crossdc35e212019-06-06 16:13:11 -07001703func (b *baseModuleContext) VisitDepsDepthFirst(visit func(Module)) {
Colin Cross1184b642019-12-30 18:43:07 -08001704 b.bp.VisitDepsDepthFirst(func(module blueprint.Module) {
Colin Crossdc35e212019-06-06 16:13:11 -07001705 if aModule := b.validateAndroidModule(module, b.strictVisitDeps); aModule != nil {
Colin Crossd11fcda2017-10-23 17:59:01 -07001706 visit(aModule)
1707 }
1708 })
1709}
1710
Colin Crossdc35e212019-06-06 16:13:11 -07001711func (b *baseModuleContext) VisitDepsDepthFirstIf(pred func(Module) bool, visit func(Module)) {
Colin Cross1184b642019-12-30 18:43:07 -08001712 b.bp.VisitDepsDepthFirstIf(
Colin Crossd11fcda2017-10-23 17:59:01 -07001713 // pred
1714 func(module blueprint.Module) bool {
Colin Crossdc35e212019-06-06 16:13:11 -07001715 if aModule := b.validateAndroidModule(module, b.strictVisitDeps); aModule != nil {
Colin Crossd11fcda2017-10-23 17:59:01 -07001716 return pred(aModule)
1717 } else {
1718 return false
1719 }
1720 },
1721 // visit
1722 func(module blueprint.Module) {
1723 visit(module.(Module))
1724 })
1725}
1726
Colin Crossdc35e212019-06-06 16:13:11 -07001727func (b *baseModuleContext) WalkDepsBlueprint(visit func(blueprint.Module, blueprint.Module) bool) {
Colin Cross1184b642019-12-30 18:43:07 -08001728 b.bp.WalkDeps(visit)
Alex Light778127a2019-02-27 14:19:50 -08001729}
1730
Colin Crossdc35e212019-06-06 16:13:11 -07001731func (b *baseModuleContext) WalkDeps(visit func(Module, Module) bool) {
1732 b.walkPath = []Module{b.Module()}
Paul Duffinc5192442020-03-31 11:31:36 +01001733 b.tagPath = []blueprint.DependencyTag{}
Colin Cross1184b642019-12-30 18:43:07 -08001734 b.bp.WalkDeps(func(child, parent blueprint.Module) bool {
Colin Crossdc35e212019-06-06 16:13:11 -07001735 childAndroidModule, _ := child.(Module)
1736 parentAndroidModule, _ := parent.(Module)
Colin Crossd11fcda2017-10-23 17:59:01 -07001737 if childAndroidModule != nil && parentAndroidModule != nil {
Colin Crossdc35e212019-06-06 16:13:11 -07001738 // record walkPath before visit
1739 for b.walkPath[len(b.walkPath)-1] != parentAndroidModule {
1740 b.walkPath = b.walkPath[0 : len(b.walkPath)-1]
Paul Duffinc5192442020-03-31 11:31:36 +01001741 b.tagPath = b.tagPath[0 : len(b.tagPath)-1]
Colin Crossdc35e212019-06-06 16:13:11 -07001742 }
1743 b.walkPath = append(b.walkPath, childAndroidModule)
Paul Duffinc5192442020-03-31 11:31:36 +01001744 b.tagPath = append(b.tagPath, b.OtherModuleDependencyTag(childAndroidModule))
Colin Crossd11fcda2017-10-23 17:59:01 -07001745 return visit(childAndroidModule, parentAndroidModule)
1746 } else {
1747 return false
1748 }
1749 })
1750}
1751
Colin Crossdc35e212019-06-06 16:13:11 -07001752func (b *baseModuleContext) GetWalkPath() []Module {
1753 return b.walkPath
1754}
1755
Paul Duffinc5192442020-03-31 11:31:36 +01001756func (b *baseModuleContext) GetTagPath() []blueprint.DependencyTag {
1757 return b.tagPath
1758}
1759
Jiyong Park1c7e9622020-05-07 16:12:13 +09001760// A regexp for removing boilerplate from BaseDependencyTag from the string representation of
1761// a dependency tag.
1762var tagCleaner = regexp.MustCompile(`\QBaseDependencyTag:blueprint.BaseDependencyTag{}\E(, )?`)
1763
1764// PrettyPrintTag returns string representation of the tag, but prefers
1765// custom String() method if available.
1766func PrettyPrintTag(tag blueprint.DependencyTag) string {
1767 // Use tag's custom String() method if available.
1768 if stringer, ok := tag.(fmt.Stringer); ok {
1769 return stringer.String()
1770 }
1771
1772 // Otherwise, get a default string representation of the tag's struct.
1773 tagString := fmt.Sprintf("%#v", tag)
1774
1775 // Remove the boilerplate from BaseDependencyTag as it adds no value.
1776 tagString = tagCleaner.ReplaceAllString(tagString, "")
1777 return tagString
1778}
1779
1780func (b *baseModuleContext) GetPathString(skipFirst bool) string {
1781 sb := strings.Builder{}
1782 tagPath := b.GetTagPath()
1783 walkPath := b.GetWalkPath()
1784 if !skipFirst {
1785 sb.WriteString(walkPath[0].String())
1786 }
1787 for i, m := range walkPath[1:] {
1788 sb.WriteString("\n")
1789 sb.WriteString(fmt.Sprintf(" via tag %s\n", PrettyPrintTag(tagPath[i])))
1790 sb.WriteString(fmt.Sprintf(" -> %s", m.String()))
1791 }
1792 return sb.String()
1793}
1794
Colin Cross25de6c32019-06-06 14:29:25 -07001795func (m *moduleContext) VisitAllModuleVariants(visit func(Module)) {
Colin Crossdc35e212019-06-06 16:13:11 -07001796 m.bp.VisitAllModuleVariants(func(module blueprint.Module) {
Colin Cross0875c522017-11-28 17:34:01 -08001797 visit(module.(Module))
1798 })
1799}
1800
Colin Cross25de6c32019-06-06 14:29:25 -07001801func (m *moduleContext) PrimaryModule() Module {
Colin Crossdc35e212019-06-06 16:13:11 -07001802 return m.bp.PrimaryModule().(Module)
Colin Cross0875c522017-11-28 17:34:01 -08001803}
1804
Colin Cross25de6c32019-06-06 14:29:25 -07001805func (m *moduleContext) FinalModule() Module {
Colin Crossdc35e212019-06-06 16:13:11 -07001806 return m.bp.FinalModule().(Module)
1807}
1808
1809func (m *moduleContext) ModuleSubDir() string {
1810 return m.bp.ModuleSubDir()
Colin Cross0875c522017-11-28 17:34:01 -08001811}
1812
Colin Cross0ea8ba82019-06-06 14:33:29 -07001813func (b *baseModuleContext) Target() Target {
Colin Cross25de6c32019-06-06 14:29:25 -07001814 return b.target
Colin Crossa1ad8d12016-06-01 17:09:44 -07001815}
1816
Colin Cross0ea8ba82019-06-06 14:33:29 -07001817func (b *baseModuleContext) TargetPrimary() bool {
Colin Cross25de6c32019-06-06 14:29:25 -07001818 return b.targetPrimary
Colin Cross8b74d172016-09-13 09:59:14 -07001819}
1820
Colin Cross0ea8ba82019-06-06 14:33:29 -07001821func (b *baseModuleContext) MultiTargets() []Target {
Colin Cross25de6c32019-06-06 14:29:25 -07001822 return b.multiTargets
Colin Crossee0bc3b2018-10-02 22:01:37 -07001823}
1824
Colin Cross0ea8ba82019-06-06 14:33:29 -07001825func (b *baseModuleContext) Arch() Arch {
Colin Cross25de6c32019-06-06 14:29:25 -07001826 return b.target.Arch
Colin Cross3f40fa42015-01-30 17:27:36 -08001827}
1828
Colin Cross0ea8ba82019-06-06 14:33:29 -07001829func (b *baseModuleContext) Os() OsType {
Colin Crossfb0c16e2019-11-20 17:12:35 -08001830 return b.os
Dan Willemsen490fd492015-11-24 17:53:15 -08001831}
1832
Colin Cross0ea8ba82019-06-06 14:33:29 -07001833func (b *baseModuleContext) Host() bool {
Colin Crossfb0c16e2019-11-20 17:12:35 -08001834 return b.os.Class == Host || b.os.Class == HostCross
Colin Crossf6566ed2015-03-24 11:13:38 -07001835}
1836
Colin Cross0ea8ba82019-06-06 14:33:29 -07001837func (b *baseModuleContext) Device() bool {
Colin Crossfb0c16e2019-11-20 17:12:35 -08001838 return b.os.Class == Device
Colin Crossf6566ed2015-03-24 11:13:38 -07001839}
1840
Colin Cross0ea8ba82019-06-06 14:33:29 -07001841func (b *baseModuleContext) Darwin() bool {
Colin Crossfb0c16e2019-11-20 17:12:35 -08001842 return b.os == Darwin
Colin Cross0af4b842015-04-30 16:36:18 -07001843}
1844
Colin Cross0ea8ba82019-06-06 14:33:29 -07001845func (b *baseModuleContext) Fuchsia() bool {
Colin Crossfb0c16e2019-11-20 17:12:35 -08001846 return b.os == Fuchsia
Doug Horn21b94272019-01-16 12:06:11 -08001847}
1848
Colin Cross0ea8ba82019-06-06 14:33:29 -07001849func (b *baseModuleContext) Windows() bool {
Colin Crossfb0c16e2019-11-20 17:12:35 -08001850 return b.os == Windows
Colin Cross3edeee12017-04-04 12:59:48 -07001851}
1852
Colin Cross0ea8ba82019-06-06 14:33:29 -07001853func (b *baseModuleContext) Debug() bool {
Colin Cross25de6c32019-06-06 14:29:25 -07001854 return b.debug
Colin Crossf6566ed2015-03-24 11:13:38 -07001855}
1856
Colin Cross0ea8ba82019-06-06 14:33:29 -07001857func (b *baseModuleContext) PrimaryArch() bool {
Colin Cross25de6c32019-06-06 14:29:25 -07001858 if len(b.config.Targets[b.target.Os]) <= 1 {
Colin Cross67a5c132017-05-09 13:45:28 -07001859 return true
1860 }
Colin Cross25de6c32019-06-06 14:29:25 -07001861 return b.target.Arch.ArchType == b.config.Targets[b.target.Os][0].Arch.ArchType
Colin Cross1e7d3702016-08-24 15:25:47 -07001862}
1863
Jiyong Park5baac542018-08-28 09:55:37 +09001864// Makes this module a platform module, i.e. not specific to soc, device,
Justin Yund5f6c822019-06-25 16:47:17 +09001865// product, or system_ext.
Colin Cross4157e882019-06-06 16:57:04 -07001866func (m *ModuleBase) MakeAsPlatform() {
1867 m.commonProperties.Vendor = boolPtr(false)
1868 m.commonProperties.Proprietary = boolPtr(false)
1869 m.commonProperties.Soc_specific = boolPtr(false)
1870 m.commonProperties.Product_specific = boolPtr(false)
Justin Yund5f6c822019-06-25 16:47:17 +09001871 m.commonProperties.System_ext_specific = boolPtr(false)
Jiyong Park5baac542018-08-28 09:55:37 +09001872}
1873
Colin Cross4157e882019-06-06 16:57:04 -07001874func (m *ModuleBase) EnableNativeBridgeSupportByDefault() {
1875 m.commonProperties.Native_bridge_supported = boolPtr(true)
dimitry03dc3f62019-05-09 14:07:34 +02001876}
1877
Sundong Ahnd95aa2d2019-10-08 19:34:03 +09001878func (m *ModuleBase) MakeAsSystemExt() {
Jooyung Han91df2082019-11-20 01:49:42 +09001879 m.commonProperties.Vendor = boolPtr(false)
1880 m.commonProperties.Proprietary = boolPtr(false)
1881 m.commonProperties.Soc_specific = boolPtr(false)
1882 m.commonProperties.Product_specific = boolPtr(false)
1883 m.commonProperties.System_ext_specific = boolPtr(true)
Sundong Ahnd95aa2d2019-10-08 19:34:03 +09001884}
1885
Jooyung Han344d5432019-08-23 11:17:39 +09001886// IsNativeBridgeSupported returns true if "native_bridge_supported" is explicitly set as "true"
1887func (m *ModuleBase) IsNativeBridgeSupported() bool {
1888 return proptools.Bool(m.commonProperties.Native_bridge_supported)
1889}
1890
Colin Cross25de6c32019-06-06 14:29:25 -07001891func (m *moduleContext) InstallInData() bool {
1892 return m.module.InstallInData()
Dan Willemsen782a2d12015-12-21 14:55:28 -08001893}
1894
Jaewoong Jung0949f312019-09-11 10:25:18 -07001895func (m *moduleContext) InstallInTestcases() bool {
1896 return m.module.InstallInTestcases()
1897}
1898
Colin Cross25de6c32019-06-06 14:29:25 -07001899func (m *moduleContext) InstallInSanitizerDir() bool {
1900 return m.module.InstallInSanitizerDir()
Vishwath Mohan1dd88392017-03-29 22:00:18 -07001901}
1902
Yifan Hong1b3348d2020-01-21 15:53:22 -08001903func (m *moduleContext) InstallInRamdisk() bool {
1904 return m.module.InstallInRamdisk()
1905}
1906
Colin Cross25de6c32019-06-06 14:29:25 -07001907func (m *moduleContext) InstallInRecovery() bool {
1908 return m.module.InstallInRecovery()
Jiyong Parkf9332f12018-02-01 00:54:12 +09001909}
1910
Colin Cross90ba5f42019-10-02 11:10:58 -07001911func (m *moduleContext) InstallInRoot() bool {
1912 return m.module.InstallInRoot()
1913}
1914
Colin Cross607d8582019-07-29 16:44:46 -07001915func (m *moduleContext) InstallBypassMake() bool {
1916 return m.module.InstallBypassMake()
1917}
1918
Colin Cross6e359402020-02-10 15:29:54 -08001919func (m *moduleContext) InstallForceOS() *OsType {
1920 return m.module.InstallForceOS()
1921}
1922
Colin Cross70dda7e2019-10-01 22:05:35 -07001923func (m *moduleContext) skipInstall(fullInstallPath InstallPath) bool {
Colin Cross25de6c32019-06-06 14:29:25 -07001924 if m.module.base().commonProperties.SkipInstall {
Colin Cross893d8162017-04-26 17:34:03 -07001925 return true
1926 }
1927
Colin Cross3607f212018-05-07 15:28:05 -07001928 // We'll need a solution for choosing which of modules with the same name in different
1929 // namespaces to install. For now, reuse the list of namespaces exported to Make as the
1930 // list of namespaces to install in a Soong-only build.
Colin Cross25de6c32019-06-06 14:29:25 -07001931 if !m.module.base().commonProperties.NamespaceExportedToMake {
Colin Cross3607f212018-05-07 15:28:05 -07001932 return true
1933 }
1934
Colin Cross25de6c32019-06-06 14:29:25 -07001935 if m.Device() {
Colin Cross607d8582019-07-29 16:44:46 -07001936 if m.Config().EmbeddedInMake() && !m.InstallBypassMake() {
Colin Cross893d8162017-04-26 17:34:03 -07001937 return true
1938 }
1939
Colin Cross25de6c32019-06-06 14:29:25 -07001940 if m.Config().SkipMegaDeviceInstall(fullInstallPath.String()) {
Colin Cross893d8162017-04-26 17:34:03 -07001941 return true
1942 }
1943 }
1944
1945 return false
1946}
1947
Colin Cross70dda7e2019-10-01 22:05:35 -07001948func (m *moduleContext) InstallFile(installPath InstallPath, name string, srcPath Path,
1949 deps ...Path) InstallPath {
Colin Cross25de6c32019-06-06 14:29:25 -07001950 return m.installFile(installPath, name, srcPath, Cp, deps)
Colin Cross5c517922017-08-31 12:29:17 -07001951}
1952
Colin Cross70dda7e2019-10-01 22:05:35 -07001953func (m *moduleContext) InstallExecutable(installPath InstallPath, name string, srcPath Path,
1954 deps ...Path) InstallPath {
Colin Cross25de6c32019-06-06 14:29:25 -07001955 return m.installFile(installPath, name, srcPath, CpExecutable, deps)
Colin Cross5c517922017-08-31 12:29:17 -07001956}
1957
Colin Cross70dda7e2019-10-01 22:05:35 -07001958func (m *moduleContext) installFile(installPath InstallPath, name string, srcPath Path,
1959 rule blueprint.Rule, deps []Path) InstallPath {
Colin Cross35cec122015-04-02 14:37:16 -07001960
Colin Cross25de6c32019-06-06 14:29:25 -07001961 fullInstallPath := installPath.Join(m, name)
1962 m.module.base().hooks.runInstallHooks(m, fullInstallPath, false)
Colin Cross3f40fa42015-01-30 17:27:36 -08001963
Colin Cross25de6c32019-06-06 14:29:25 -07001964 if !m.skipInstall(fullInstallPath) {
Colin Crossce75d2c2016-10-06 16:12:58 -07001965
Colin Cross897266e2020-02-13 13:22:08 -08001966 deps = append(deps, m.installDeps.Paths()...)
Colin Cross35cec122015-04-02 14:37:16 -07001967
Colin Cross89562dc2016-10-03 17:47:19 -07001968 var implicitDeps, orderOnlyDeps Paths
1969
Colin Cross25de6c32019-06-06 14:29:25 -07001970 if m.Host() {
Colin Cross89562dc2016-10-03 17:47:19 -07001971 // Installed host modules might be used during the build, depend directly on their
1972 // dependencies so their timestamp is updated whenever their dependency is updated
1973 implicitDeps = deps
1974 } else {
1975 orderOnlyDeps = deps
1976 }
1977
Colin Cross25de6c32019-06-06 14:29:25 -07001978 m.Build(pctx, BuildParams{
Colin Cross5c517922017-08-31 12:29:17 -07001979 Rule: rule,
Colin Cross67a5c132017-05-09 13:45:28 -07001980 Description: "install " + fullInstallPath.Base(),
1981 Output: fullInstallPath,
1982 Input: srcPath,
1983 Implicits: implicitDeps,
1984 OrderOnly: orderOnlyDeps,
Colin Cross25de6c32019-06-06 14:29:25 -07001985 Default: !m.Config().EmbeddedInMake(),
Dan Willemsen322acaf2016-01-12 23:07:05 -08001986 })
Colin Cross3f40fa42015-01-30 17:27:36 -08001987
Colin Cross25de6c32019-06-06 14:29:25 -07001988 m.installFiles = append(m.installFiles, fullInstallPath)
Dan Willemsen322acaf2016-01-12 23:07:05 -08001989 }
Colin Cross25de6c32019-06-06 14:29:25 -07001990 m.checkbuildFiles = append(m.checkbuildFiles, srcPath)
Colin Cross35cec122015-04-02 14:37:16 -07001991 return fullInstallPath
1992}
1993
Colin Cross70dda7e2019-10-01 22:05:35 -07001994func (m *moduleContext) InstallSymlink(installPath InstallPath, name string, srcPath InstallPath) InstallPath {
Colin Cross25de6c32019-06-06 14:29:25 -07001995 fullInstallPath := installPath.Join(m, name)
1996 m.module.base().hooks.runInstallHooks(m, fullInstallPath, true)
Colin Cross3854a602016-01-11 12:49:11 -08001997
Colin Cross25de6c32019-06-06 14:29:25 -07001998 if !m.skipInstall(fullInstallPath) {
Colin Crossce75d2c2016-10-06 16:12:58 -07001999
Alex Lightfb4353d2019-01-17 13:57:45 -08002000 relPath, err := filepath.Rel(path.Dir(fullInstallPath.String()), srcPath.String())
2001 if err != nil {
2002 panic(fmt.Sprintf("Unable to generate symlink between %q and %q: %s", fullInstallPath.Base(), srcPath.Base(), err))
2003 }
Colin Cross25de6c32019-06-06 14:29:25 -07002004 m.Build(pctx, BuildParams{
Colin Cross67a5c132017-05-09 13:45:28 -07002005 Rule: Symlink,
2006 Description: "install symlink " + fullInstallPath.Base(),
2007 Output: fullInstallPath,
Dan Willemsen40efa1c2020-01-14 15:19:52 -08002008 Input: srcPath,
Colin Cross25de6c32019-06-06 14:29:25 -07002009 Default: !m.Config().EmbeddedInMake(),
Colin Cross12fc4972016-01-11 12:49:11 -08002010 Args: map[string]string{
Alex Lightfb4353d2019-01-17 13:57:45 -08002011 "fromPath": relPath,
Colin Cross12fc4972016-01-11 12:49:11 -08002012 },
2013 })
Colin Cross3854a602016-01-11 12:49:11 -08002014
Colin Cross25de6c32019-06-06 14:29:25 -07002015 m.installFiles = append(m.installFiles, fullInstallPath)
2016 m.checkbuildFiles = append(m.checkbuildFiles, srcPath)
Colin Cross12fc4972016-01-11 12:49:11 -08002017 }
Colin Cross3854a602016-01-11 12:49:11 -08002018 return fullInstallPath
2019}
2020
Jiyong Parkf1194352019-02-25 11:05:47 +09002021// installPath/name -> absPath where absPath might be a path that is available only at runtime
2022// (e.g. /apex/...)
Colin Cross70dda7e2019-10-01 22:05:35 -07002023func (m *moduleContext) InstallAbsoluteSymlink(installPath InstallPath, name string, absPath string) InstallPath {
Colin Cross25de6c32019-06-06 14:29:25 -07002024 fullInstallPath := installPath.Join(m, name)
2025 m.module.base().hooks.runInstallHooks(m, fullInstallPath, true)
Jiyong Parkf1194352019-02-25 11:05:47 +09002026
Colin Cross25de6c32019-06-06 14:29:25 -07002027 if !m.skipInstall(fullInstallPath) {
2028 m.Build(pctx, BuildParams{
Jiyong Parkf1194352019-02-25 11:05:47 +09002029 Rule: Symlink,
2030 Description: "install symlink " + fullInstallPath.Base() + " -> " + absPath,
2031 Output: fullInstallPath,
Colin Cross25de6c32019-06-06 14:29:25 -07002032 Default: !m.Config().EmbeddedInMake(),
Jiyong Parkf1194352019-02-25 11:05:47 +09002033 Args: map[string]string{
2034 "fromPath": absPath,
2035 },
2036 })
2037
Colin Cross25de6c32019-06-06 14:29:25 -07002038 m.installFiles = append(m.installFiles, fullInstallPath)
Jiyong Parkf1194352019-02-25 11:05:47 +09002039 }
2040 return fullInstallPath
2041}
2042
Colin Cross25de6c32019-06-06 14:29:25 -07002043func (m *moduleContext) CheckbuildFile(srcPath Path) {
2044 m.checkbuildFiles = append(m.checkbuildFiles, srcPath)
Colin Cross3f40fa42015-01-30 17:27:36 -08002045}
2046
Dan Willemsen2ef08f42015-06-30 18:15:24 -07002047func findStringInSlice(str string, slice []string) int {
2048 for i, s := range slice {
2049 if s == str {
2050 return i
Colin Crossfce53272015-04-08 11:21:40 -07002051 }
2052 }
Dan Willemsen2ef08f42015-06-30 18:15:24 -07002053 return -1
2054}
2055
Colin Cross41955e82019-05-29 14:40:35 -07002056// SrcIsModule decodes module references in the format ":name" into the module name, or empty string if the input
2057// was not a module reference.
2058func SrcIsModule(s string) (module string) {
Colin Cross068e0fe2016-12-13 15:23:47 -08002059 if len(s) > 1 && s[0] == ':' {
2060 return s[1:]
2061 }
2062 return ""
2063}
2064
Colin Cross41955e82019-05-29 14:40:35 -07002065// SrcIsModule decodes module references in the format ":name{.tag}" into the module name and tag, ":name" into the
2066// module name and an empty string for the tag, or empty strings if the input was not a module reference.
2067func SrcIsModuleWithTag(s string) (module, tag string) {
2068 if len(s) > 1 && s[0] == ':' {
2069 module = s[1:]
2070 if tagStart := strings.IndexByte(module, '{'); tagStart > 0 {
2071 if module[len(module)-1] == '}' {
2072 tag = module[tagStart+1 : len(module)-1]
2073 module = module[:tagStart]
2074 return module, tag
2075 }
2076 }
2077 return module, ""
2078 }
2079 return "", ""
Colin Cross068e0fe2016-12-13 15:23:47 -08002080}
2081
Colin Cross41955e82019-05-29 14:40:35 -07002082type sourceOrOutputDependencyTag struct {
2083 blueprint.BaseDependencyTag
2084 tag string
2085}
2086
2087func sourceOrOutputDepTag(tag string) blueprint.DependencyTag {
2088 return sourceOrOutputDependencyTag{tag: tag}
2089}
2090
2091var SourceDepTag = sourceOrOutputDepTag("")
Colin Cross068e0fe2016-12-13 15:23:47 -08002092
Colin Cross366938f2017-12-11 16:29:02 -08002093// Adds necessary dependencies to satisfy filegroup or generated sources modules listed in srcFiles
2094// using ":module" syntax, if any.
Colin Cross27b922f2019-03-04 22:35:41 -08002095//
2096// Deprecated: tag the property with `android:"path"` instead.
Colin Cross068e0fe2016-12-13 15:23:47 -08002097func ExtractSourcesDeps(ctx BottomUpMutatorContext, srcFiles []string) {
Nan Zhang2439eb72017-04-10 11:27:50 -07002098 set := make(map[string]bool)
2099
Colin Cross068e0fe2016-12-13 15:23:47 -08002100 for _, s := range srcFiles {
Colin Cross41955e82019-05-29 14:40:35 -07002101 if m, t := SrcIsModuleWithTag(s); m != "" {
2102 if _, found := set[s]; found {
2103 ctx.ModuleErrorf("found source dependency duplicate: %q!", s)
Nan Zhang2439eb72017-04-10 11:27:50 -07002104 } else {
Colin Cross41955e82019-05-29 14:40:35 -07002105 set[s] = true
2106 ctx.AddDependency(ctx.Module(), sourceOrOutputDepTag(t), m)
Nan Zhang2439eb72017-04-10 11:27:50 -07002107 }
Colin Cross068e0fe2016-12-13 15:23:47 -08002108 }
2109 }
Colin Cross068e0fe2016-12-13 15:23:47 -08002110}
2111
Colin Cross366938f2017-12-11 16:29:02 -08002112// Adds necessary dependencies to satisfy filegroup or generated sources modules specified in s
2113// using ":module" syntax, if any.
Colin Cross27b922f2019-03-04 22:35:41 -08002114//
2115// Deprecated: tag the property with `android:"path"` instead.
Colin Cross366938f2017-12-11 16:29:02 -08002116func ExtractSourceDeps(ctx BottomUpMutatorContext, s *string) {
2117 if s != nil {
Colin Cross41955e82019-05-29 14:40:35 -07002118 if m, t := SrcIsModuleWithTag(*s); m != "" {
2119 ctx.AddDependency(ctx.Module(), sourceOrOutputDepTag(t), m)
Colin Cross366938f2017-12-11 16:29:02 -08002120 }
2121 }
2122}
2123
Colin Cross41955e82019-05-29 14:40:35 -07002124// A module that implements SourceFileProducer can be referenced from any property that is tagged with `android:"path"`
2125// 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 -08002126type SourceFileProducer interface {
2127 Srcs() Paths
2128}
2129
Colin Cross41955e82019-05-29 14:40:35 -07002130// A module that implements OutputFileProducer can be referenced from any property that is tagged with `android:"path"`
Roland Levillain97c1f342019-11-22 14:20:54 +00002131// 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 -07002132// listed in the property.
2133type OutputFileProducer interface {
2134 OutputFiles(tag string) (Paths, error)
2135}
2136
Colin Cross5e708052019-08-06 13:59:50 -07002137// OutputFilesForModule returns the paths from an OutputFileProducer with the given tag. On error, including if the
2138// module produced zero paths, it reports errors to the ctx and returns nil.
2139func OutputFilesForModule(ctx PathContext, module blueprint.Module, tag string) Paths {
2140 paths, err := outputFilesForModule(ctx, module, tag)
2141 if err != nil {
2142 reportPathError(ctx, err)
2143 return nil
2144 }
2145 return paths
2146}
2147
2148// OutputFileForModule returns the path from an OutputFileProducer with the given tag. On error, including if the
2149// module produced zero or multiple paths, it reports errors to the ctx and returns nil.
2150func OutputFileForModule(ctx PathContext, module blueprint.Module, tag string) Path {
2151 paths, err := outputFilesForModule(ctx, module, tag)
2152 if err != nil {
2153 reportPathError(ctx, err)
2154 return nil
2155 }
2156 if len(paths) > 1 {
2157 reportPathErrorf(ctx, "got multiple output files from module %q, expected exactly one",
2158 pathContextName(ctx, module))
2159 return nil
2160 }
2161 return paths[0]
2162}
2163
2164func outputFilesForModule(ctx PathContext, module blueprint.Module, tag string) (Paths, error) {
2165 if outputFileProducer, ok := module.(OutputFileProducer); ok {
2166 paths, err := outputFileProducer.OutputFiles(tag)
2167 if err != nil {
2168 return nil, fmt.Errorf("failed to get output file from module %q: %s",
2169 pathContextName(ctx, module), err.Error())
2170 }
2171 if len(paths) == 0 {
2172 return nil, fmt.Errorf("failed to get output files from module %q", pathContextName(ctx, module))
2173 }
2174 return paths, nil
2175 } else {
2176 return nil, fmt.Errorf("module %q is not an OutputFileProducer", pathContextName(ctx, module))
2177 }
2178}
2179
Colin Crossfe17f6f2019-03-28 19:30:56 -07002180type HostToolProvider interface {
2181 HostToolPath() OptionalPath
2182}
2183
Colin Cross27b922f2019-03-04 22:35:41 -08002184// Returns a list of paths expanded from globs and modules referenced using ":module" syntax. The property must
2185// be tagged with `android:"path" to support automatic source module dependency resolution.
Colin Cross8a497952019-03-05 22:25:09 -08002186//
2187// Deprecated: use PathsForModuleSrc or PathsForModuleSrcExcludes instead.
Colin Cross25de6c32019-06-06 14:29:25 -07002188func (m *moduleContext) ExpandSources(srcFiles, excludes []string) Paths {
2189 return PathsForModuleSrcExcludes(m, srcFiles, excludes)
Colin Cross8f101b42015-06-17 15:09:06 -07002190}
2191
Colin Cross2fafa3e2019-03-05 12:39:51 -08002192// Returns a single path expanded from globs and modules referenced using ":module" syntax. The property must
2193// be tagged with `android:"path" to support automatic source module dependency resolution.
Colin Cross8a497952019-03-05 22:25:09 -08002194//
2195// Deprecated: use PathForModuleSrc instead.
Colin Cross25de6c32019-06-06 14:29:25 -07002196func (m *moduleContext) ExpandSource(srcFile, prop string) Path {
2197 return PathForModuleSrc(m, srcFile)
Colin Cross2fafa3e2019-03-05 12:39:51 -08002198}
2199
2200// Returns an optional single path expanded from globs and modules referenced using ":module" syntax if
2201// the srcFile is non-nil. The property must be tagged with `android:"path" to support automatic source module
2202// dependency resolution.
Colin Cross25de6c32019-06-06 14:29:25 -07002203func (m *moduleContext) ExpandOptionalSource(srcFile *string, prop string) OptionalPath {
Colin Cross2fafa3e2019-03-05 12:39:51 -08002204 if srcFile != nil {
Colin Cross25de6c32019-06-06 14:29:25 -07002205 return OptionalPathForPath(PathForModuleSrc(m, *srcFile))
Colin Cross2fafa3e2019-03-05 12:39:51 -08002206 }
2207 return OptionalPath{}
2208}
2209
Colin Cross25de6c32019-06-06 14:29:25 -07002210func (m *moduleContext) RequiredModuleNames() []string {
Jiyong Park6a8cf5f2019-12-30 16:31:09 +09002211 return m.module.RequiredModuleNames()
Nan Zhang6d34b302017-02-04 17:47:46 -08002212}
2213
Colin Cross25de6c32019-06-06 14:29:25 -07002214func (m *moduleContext) HostRequiredModuleNames() []string {
Jiyong Park6a8cf5f2019-12-30 16:31:09 +09002215 return m.module.HostRequiredModuleNames()
Sasha Smundakb6d23052019-04-01 18:37:36 -07002216}
2217
Colin Cross25de6c32019-06-06 14:29:25 -07002218func (m *moduleContext) TargetRequiredModuleNames() []string {
Jiyong Park6a8cf5f2019-12-30 16:31:09 +09002219 return m.module.TargetRequiredModuleNames()
Sasha Smundakb6d23052019-04-01 18:37:36 -07002220}
2221
Colin Cross463a90e2015-06-17 14:20:06 -07002222func init() {
Colin Cross798bfce2016-10-12 14:28:16 -07002223 RegisterSingletonType("buildtarget", BuildTargetSingleton)
Colin Cross463a90e2015-06-17 14:20:06 -07002224}
2225
Colin Cross0875c522017-11-28 17:34:01 -08002226func BuildTargetSingleton() Singleton {
Colin Cross1f8c52b2015-06-16 16:38:17 -07002227 return &buildTargetSingleton{}
2228}
2229
Colin Cross87d8b562017-04-25 10:01:55 -07002230func parentDir(dir string) string {
2231 dir, _ = filepath.Split(dir)
2232 return filepath.Clean(dir)
2233}
2234
Colin Cross1f8c52b2015-06-16 16:38:17 -07002235type buildTargetSingleton struct{}
2236
Colin Cross0875c522017-11-28 17:34:01 -08002237func (c *buildTargetSingleton) GenerateBuildActions(ctx SingletonContext) {
2238 var checkbuildDeps Paths
Colin Cross1f8c52b2015-06-16 16:38:17 -07002239
Colin Cross0875c522017-11-28 17:34:01 -08002240 mmTarget := func(dir string) WritablePath {
2241 return PathForPhony(ctx,
2242 "MODULES-IN-"+strings.Replace(filepath.Clean(dir), "/", "-", -1))
Colin Cross87d8b562017-04-25 10:01:55 -07002243 }
2244
Colin Cross0875c522017-11-28 17:34:01 -08002245 modulesInDir := make(map[string]Paths)
Colin Cross1f8c52b2015-06-16 16:38:17 -07002246
Colin Cross0875c522017-11-28 17:34:01 -08002247 ctx.VisitAllModules(func(module Module) {
2248 blueprintDir := module.base().blueprintDir
2249 installTarget := module.base().installTarget
2250 checkbuildTarget := module.base().checkbuildTarget
Colin Cross1f8c52b2015-06-16 16:38:17 -07002251
Colin Cross0875c522017-11-28 17:34:01 -08002252 if checkbuildTarget != nil {
2253 checkbuildDeps = append(checkbuildDeps, checkbuildTarget)
2254 modulesInDir[blueprintDir] = append(modulesInDir[blueprintDir], checkbuildTarget)
2255 }
Colin Cross1f8c52b2015-06-16 16:38:17 -07002256
Colin Cross0875c522017-11-28 17:34:01 -08002257 if installTarget != nil {
2258 modulesInDir[blueprintDir] = append(modulesInDir[blueprintDir], installTarget)
Colin Cross1f8c52b2015-06-16 16:38:17 -07002259 }
2260 })
2261
Dan Willemsen5ba07e82015-12-11 13:51:06 -08002262 suffix := ""
Colin Crossaabf6792017-11-29 00:27:14 -08002263 if ctx.Config().EmbeddedInMake() {
Dan Willemsen5ba07e82015-12-11 13:51:06 -08002264 suffix = "-soong"
2265 }
2266
Colin Cross1f8c52b2015-06-16 16:38:17 -07002267 // Create a top-level checkbuild target that depends on all modules
Colin Cross0875c522017-11-28 17:34:01 -08002268 ctx.Build(pctx, BuildParams{
Colin Cross1f8c52b2015-06-16 16:38:17 -07002269 Rule: blueprint.Phony,
Colin Cross0875c522017-11-28 17:34:01 -08002270 Output: PathForPhony(ctx, "checkbuild"+suffix),
Colin Cross1f8c52b2015-06-16 16:38:17 -07002271 Implicits: checkbuildDeps,
Colin Cross1f8c52b2015-06-16 16:38:17 -07002272 })
2273
Dan Willemsend2e95fb2017-09-20 14:30:50 -07002274 // Make will generate the MODULES-IN-* targets
Colin Crossaabf6792017-11-29 00:27:14 -08002275 if ctx.Config().EmbeddedInMake() {
Dan Willemsend2e95fb2017-09-20 14:30:50 -07002276 return
2277 }
2278
Colin Cross87d8b562017-04-25 10:01:55 -07002279 // Ensure ancestor directories are in modulesInDir
Inseob Kim1a365c62019-06-08 15:47:51 +09002280 dirs := SortedStringKeys(modulesInDir)
Colin Cross87d8b562017-04-25 10:01:55 -07002281 for _, dir := range dirs {
2282 dir := parentDir(dir)
2283 for dir != "." && dir != "/" {
2284 if _, exists := modulesInDir[dir]; exists {
2285 break
2286 }
2287 modulesInDir[dir] = nil
2288 dir = parentDir(dir)
2289 }
2290 }
2291
2292 // Make directories build their direct subdirectories
Colin Cross87d8b562017-04-25 10:01:55 -07002293 for _, dir := range dirs {
2294 p := parentDir(dir)
2295 if p != "." && p != "/" {
2296 modulesInDir[p] = append(modulesInDir[p], mmTarget(dir))
2297 }
2298 }
2299
Dan Willemsend2e95fb2017-09-20 14:30:50 -07002300 // Create a MODULES-IN-<directory> target that depends on all modules in a directory, and
2301 // depends on the MODULES-IN-* targets of all of its subdirectories that contain Android.bp
2302 // files.
Colin Cross1f8c52b2015-06-16 16:38:17 -07002303 for _, dir := range dirs {
Colin Cross0875c522017-11-28 17:34:01 -08002304 ctx.Build(pctx, BuildParams{
Colin Cross1f8c52b2015-06-16 16:38:17 -07002305 Rule: blueprint.Phony,
Colin Cross0875c522017-11-28 17:34:01 -08002306 Output: mmTarget(dir),
Colin Cross87d8b562017-04-25 10:01:55 -07002307 Implicits: modulesInDir[dir],
Dan Willemsen5ba07e82015-12-11 13:51:06 -08002308 // HACK: checkbuild should be an optional build, but force it
2309 // enabled for now in standalone builds
Colin Crossaabf6792017-11-29 00:27:14 -08002310 Default: !ctx.Config().EmbeddedInMake(),
Colin Cross1f8c52b2015-06-16 16:38:17 -07002311 })
2312 }
Dan Willemsen61d88b82017-09-20 17:29:08 -07002313
2314 // Create (host|host-cross|target)-<OS> phony rules to build a reduced checkbuild.
2315 osDeps := map[OsType]Paths{}
Colin Cross0875c522017-11-28 17:34:01 -08002316 ctx.VisitAllModules(func(module Module) {
2317 if module.Enabled() {
2318 os := module.Target().Os
2319 osDeps[os] = append(osDeps[os], module.base().checkbuildFiles...)
Dan Willemsen61d88b82017-09-20 17:29:08 -07002320 }
2321 })
2322
Colin Cross0875c522017-11-28 17:34:01 -08002323 osClass := make(map[string]Paths)
Dan Willemsen61d88b82017-09-20 17:29:08 -07002324 for os, deps := range osDeps {
2325 var className string
2326
2327 switch os.Class {
2328 case Host:
2329 className = "host"
2330 case HostCross:
2331 className = "host-cross"
2332 case Device:
2333 className = "target"
2334 default:
2335 continue
2336 }
2337
Colin Cross0875c522017-11-28 17:34:01 -08002338 name := PathForPhony(ctx, className+"-"+os.Name)
Dan Willemsen61d88b82017-09-20 17:29:08 -07002339 osClass[className] = append(osClass[className], name)
2340
Colin Cross0875c522017-11-28 17:34:01 -08002341 ctx.Build(pctx, BuildParams{
Dan Willemsen61d88b82017-09-20 17:29:08 -07002342 Rule: blueprint.Phony,
Colin Cross0875c522017-11-28 17:34:01 -08002343 Output: name,
2344 Implicits: deps,
Dan Willemsen61d88b82017-09-20 17:29:08 -07002345 })
2346 }
2347
2348 // Wrap those into host|host-cross|target phony rules
Inseob Kim1a365c62019-06-08 15:47:51 +09002349 for _, class := range SortedStringKeys(osClass) {
Colin Cross0875c522017-11-28 17:34:01 -08002350 ctx.Build(pctx, BuildParams{
Dan Willemsen61d88b82017-09-20 17:29:08 -07002351 Rule: blueprint.Phony,
Colin Cross0875c522017-11-28 17:34:01 -08002352 Output: PathForPhony(ctx, class),
Dan Willemsen61d88b82017-09-20 17:29:08 -07002353 Implicits: osClass[class],
Dan Willemsen61d88b82017-09-20 17:29:08 -07002354 })
2355 }
Colin Cross1f8c52b2015-06-16 16:38:17 -07002356}
Colin Crossd779da42015-12-17 18:00:23 -08002357
Brandon Lee5d45c6f2018-08-15 15:35:38 -07002358// Collect information for opening IDE project files in java/jdeps.go.
2359type IDEInfo interface {
2360 IDEInfo(ideInfo *IdeInfo)
2361 BaseModuleName() string
2362}
2363
2364// Extract the base module name from the Import name.
2365// Often the Import name has a prefix "prebuilt_".
2366// Remove the prefix explicitly if needed
2367// until we find a better solution to get the Import name.
2368type IDECustomizedModuleName interface {
2369 IDECustomizedModuleName() string
2370}
2371
2372type IdeInfo struct {
2373 Deps []string `json:"dependencies,omitempty"`
2374 Srcs []string `json:"srcs,omitempty"`
2375 Aidl_include_dirs []string `json:"aidl_include_dirs,omitempty"`
2376 Jarjar_rules []string `json:"jarjar_rules,omitempty"`
2377 Jars []string `json:"jars,omitempty"`
2378 Classes []string `json:"class,omitempty"`
2379 Installed_paths []string `json:"installed,omitempty"`
patricktu18c82ff2019-05-10 15:48:50 +08002380 SrcJars []string `json:"srcjars,omitempty"`
bralee1fbf4402020-05-21 10:11:59 +08002381 Paths []string `json:"path,omitempty"`
Brandon Lee5d45c6f2018-08-15 15:35:38 -07002382}
Paul Duffinf88d8e02020-05-07 20:21:34 +01002383
2384func CheckBlueprintSyntax(ctx BaseModuleContext, filename string, contents string) []error {
2385 bpctx := ctx.blueprintBaseModuleContext()
2386 return blueprint.CheckBlueprintSyntax(bpctx.ModuleFactories(), filename, contents)
2387}