blob: 913bf6a56b35ebe72f8330d04fe5684af0c55be7 [file] [log] [blame]
Colin Cross0875c522017-11-28 17:34:01 -08001// Copyright 2017 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
15package android
16
17import (
18 "github.com/google/blueprint"
Colin Cross0875c522017-11-28 17:34:01 -080019)
20
21// SingletonContext
22type SingletonContext interface {
Colin Cross3c0a83d2023-12-12 14:13:26 -080023 blueprintSingletonContext() blueprint.SingletonContext
24
Colin Crossaabf6792017-11-29 00:27:14 -080025 Config() Config
Colin Cross65494b92019-02-07 14:25:51 -080026 DeviceConfig() DeviceConfig
Colin Cross0875c522017-11-28 17:34:01 -080027
28 ModuleName(module blueprint.Module) string
29 ModuleDir(module blueprint.Module) string
30 ModuleSubDir(module blueprint.Module) string
31 ModuleType(module blueprint.Module) string
32 BlueprintFile(module blueprint.Module) string
33
Bob Badoureef4c1c2022-05-16 12:20:04 -070034 // ModuleVariantsFromName returns the list of module variants named `name` in the same namespace as `referer` enforcing visibility rules.
35 // Allows generating build actions for `referer` based on the metadata for `name` deferred until the singleton context.
36 ModuleVariantsFromName(referer Module, name string) []Module
37
Yu Liu663e4502024-08-12 18:23:59 +000038 otherModuleProvider(module blueprint.Module, provider blueprint.AnyProviderKey) (any, bool)
Colin Crossd27e7b82020-07-02 11:38:17 -070039
Colin Cross0875c522017-11-28 17:34:01 -080040 ModuleErrorf(module blueprint.Module, format string, args ...interface{})
41 Errorf(format string, args ...interface{})
42 Failed() bool
43
44 Variable(pctx PackageContext, name, value string)
Colin Cross59014392017-12-12 11:05:06 -080045 Rule(pctx PackageContext, name string, params blueprint.RuleParams, argNames ...string) blueprint.Rule
Colin Cross0875c522017-11-28 17:34:01 -080046 Build(pctx PackageContext, params BuildParams)
Colin Crossc3d87d32020-06-04 13:25:17 -070047
48 // Phony creates a Make-style phony rule, a rule with no commands that can depend on other
49 // phony rules or real files. Phony can be called on the same name multiple times to add
50 // additional dependencies.
51 Phony(name string, deps ...Path)
52
Colin Cross0875c522017-11-28 17:34:01 -080053 RequireNinjaVersion(major, minor, micro int)
54
Lukacs T. Berki9f6c24a2021-08-26 15:07:24 +020055 // SetOutDir sets the value of the top-level "builddir" Ninja variable
Colin Cross0875c522017-11-28 17:34:01 -080056 // that controls where Ninja stores its build log files. This value can be
57 // set at most one time for a single build, later calls are ignored.
Lukacs T. Berki9f6c24a2021-08-26 15:07:24 +020058 SetOutDir(pctx PackageContext, value string)
Colin Cross0875c522017-11-28 17:34:01 -080059
60 // Eval takes a string with embedded ninja variables, and returns a string
61 // with all of the variables recursively expanded. Any variables references
62 // are expanded in the scope of the PackageContext.
63 Eval(pctx PackageContext, ninjaStr string) (string, error)
64
Colin Cross2465c3d2018-09-28 10:19:18 -070065 VisitAllModulesBlueprint(visit func(blueprint.Module))
Colin Cross0875c522017-11-28 17:34:01 -080066 VisitAllModules(visit func(Module))
67 VisitAllModulesIf(pred func(Module) bool, visit func(Module))
Mitch Phillips3a7a31b2019-10-16 15:00:12 -070068
69 VisitDirectDeps(module Module, visit func(Module))
70 VisitDirectDepsIf(module Module, pred func(Module) bool, visit func(Module))
71
Colin Cross6b753602018-06-21 13:03:07 -070072 // Deprecated: use WalkDeps instead to support multiple dependency tags on the same module
Colin Cross0875c522017-11-28 17:34:01 -080073 VisitDepsDepthFirst(module Module, visit func(Module))
Colin Cross6b753602018-06-21 13:03:07 -070074 // Deprecated: use WalkDeps instead to support multiple dependency tags on the same module
Colin Cross0875c522017-11-28 17:34:01 -080075 VisitDepsDepthFirstIf(module Module, pred func(Module) bool,
76 visit func(Module))
77
78 VisitAllModuleVariants(module Module, visit func(Module))
79
80 PrimaryModule(module Module) Module
81 FinalModule(module Module) Module
82
83 AddNinjaFileDeps(deps ...string)
84
85 // GlobWithDeps returns a list of files that match the specified pattern but do not match any
86 // of the patterns in excludes. It also adds efficient dependencies to rerun the primary
87 // builder whenever a file matching the pattern as added or removed, without rerunning if a
88 // file that does not match the pattern is added to a searched directory.
89 GlobWithDeps(pattern string, excludes []string) ([]string, error)
Cole Faust02987bd2024-03-21 17:58:43 -070090
91 // OtherModulePropertyErrorf reports an error on the line number of the given property of the given module
92 OtherModulePropertyErrorf(module Module, property string, format string, args ...interface{})
Cole Faust4e2bf9f2024-09-11 13:26:20 -070093
94 // HasMutatorFinished returns true if the given mutator has finished running.
95 // It will panic if given an invalid mutator name.
96 HasMutatorFinished(mutatorName string) bool
Colin Cross0875c522017-11-28 17:34:01 -080097}
98
99type singletonAdaptor struct {
100 Singleton
Colin Cross4c83e5c2019-02-25 14:54:28 -0800101
102 buildParams []BuildParams
103 ruleParams map[blueprint.Rule]blueprint.RuleParams
Colin Cross0875c522017-11-28 17:34:01 -0800104}
105
Colin Cross4c83e5c2019-02-25 14:54:28 -0800106var _ testBuildProvider = (*singletonAdaptor)(nil)
107
108func (s *singletonAdaptor) GenerateBuildActions(ctx blueprint.SingletonContext) {
109 sctx := &singletonContextAdaptor{SingletonContext: ctx}
110 if sctx.Config().captureBuild {
111 sctx.ruleParams = make(map[blueprint.Rule]blueprint.RuleParams)
112 }
113
114 s.Singleton.GenerateBuildActions(sctx)
115
116 s.buildParams = sctx.buildParams
117 s.ruleParams = sctx.ruleParams
118}
119
120func (s *singletonAdaptor) BuildParamsForTests() []BuildParams {
121 return s.buildParams
122}
123
124func (s *singletonAdaptor) RuleParamsForTests() map[blueprint.Rule]blueprint.RuleParams {
125 return s.ruleParams
Colin Cross0875c522017-11-28 17:34:01 -0800126}
127
128type Singleton interface {
129 GenerateBuildActions(SingletonContext)
130}
131
132type singletonContextAdaptor struct {
133 blueprint.SingletonContext
Colin Cross4c83e5c2019-02-25 14:54:28 -0800134
135 buildParams []BuildParams
136 ruleParams map[blueprint.Rule]blueprint.RuleParams
Colin Cross0875c522017-11-28 17:34:01 -0800137}
138
Colin Cross3c0a83d2023-12-12 14:13:26 -0800139func (s *singletonContextAdaptor) blueprintSingletonContext() blueprint.SingletonContext {
140 return s.SingletonContext
141}
142
Colin Cross4c83e5c2019-02-25 14:54:28 -0800143func (s *singletonContextAdaptor) Config() Config {
Colin Crossaabf6792017-11-29 00:27:14 -0800144 return s.SingletonContext.Config().(Config)
145}
146
Colin Cross4c83e5c2019-02-25 14:54:28 -0800147func (s *singletonContextAdaptor) DeviceConfig() DeviceConfig {
Colin Cross65494b92019-02-07 14:25:51 -0800148 return DeviceConfig{s.Config().deviceConfig}
149}
150
Colin Cross4c83e5c2019-02-25 14:54:28 -0800151func (s *singletonContextAdaptor) Variable(pctx PackageContext, name, value string) {
Colin Cross0875c522017-11-28 17:34:01 -0800152 s.SingletonContext.Variable(pctx.PackageContext, name, value)
153}
154
Colin Cross4c83e5c2019-02-25 14:54:28 -0800155func (s *singletonContextAdaptor) Rule(pctx PackageContext, name string, params blueprint.RuleParams, argNames ...string) blueprint.Rule {
Ramy Medhat944839a2020-03-31 22:14:52 -0400156 if s.Config().UseRemoteBuild() {
157 if params.Pool == nil {
158 // When USE_GOMA=true or USE_RBE=true are set and the rule is not supported by goma/RBE, restrict
159 // jobs to the local parallelism value
160 params.Pool = localPool
161 } else if params.Pool == remotePool {
162 // remotePool is a fake pool used to identify rule that are supported for remoting. If the rule's
163 // pool is the remotePool, replace with nil so that ninja runs it at NINJA_REMOTE_NUM_JOBS
164 // parallelism.
165 params.Pool = nil
166 }
Colin Cross2e2dbc22019-09-25 13:31:46 -0700167 }
Colin Cross4c83e5c2019-02-25 14:54:28 -0800168 rule := s.SingletonContext.Rule(pctx.PackageContext, name, params, argNames...)
169 if s.Config().captureBuild {
170 s.ruleParams[rule] = params
171 }
172 return rule
Colin Cross0875c522017-11-28 17:34:01 -0800173}
174
Colin Cross4c83e5c2019-02-25 14:54:28 -0800175func (s *singletonContextAdaptor) Build(pctx PackageContext, params BuildParams) {
176 if s.Config().captureBuild {
177 s.buildParams = append(s.buildParams, params)
178 }
Colin Cross0875c522017-11-28 17:34:01 -0800179 bparams := convertBuildParams(params)
180 s.SingletonContext.Build(pctx.PackageContext, bparams)
Colin Cross0875c522017-11-28 17:34:01 -0800181}
182
Colin Crossc3d87d32020-06-04 13:25:17 -0700183func (s *singletonContextAdaptor) Phony(name string, deps ...Path) {
Yu Liu54513622024-08-19 20:00:32 +0000184 addSingletonPhony(s.Config(), name, deps...)
Colin Crossc3d87d32020-06-04 13:25:17 -0700185}
186
Lukacs T. Berki9f6c24a2021-08-26 15:07:24 +0200187func (s *singletonContextAdaptor) SetOutDir(pctx PackageContext, value string) {
188 s.SingletonContext.SetOutDir(pctx.PackageContext, value)
Colin Cross0875c522017-11-28 17:34:01 -0800189}
190
Colin Cross4c83e5c2019-02-25 14:54:28 -0800191func (s *singletonContextAdaptor) Eval(pctx PackageContext, ninjaStr string) (string, error) {
Colin Cross0875c522017-11-28 17:34:01 -0800192 return s.SingletonContext.Eval(pctx.PackageContext, ninjaStr)
193}
194
195// visitAdaptor wraps a visit function that takes an android.Module parameter into
196// a function that takes an blueprint.Module parameter and only calls the visit function if the
197// blueprint.Module is an android.Module.
198func visitAdaptor(visit func(Module)) func(blueprint.Module) {
199 return func(module blueprint.Module) {
200 if aModule, ok := module.(Module); ok {
201 visit(aModule)
202 }
203 }
204}
205
206// predAdaptor wraps a pred function that takes an android.Module parameter
207// into a function that takes an blueprint.Module parameter and only calls the visit function if the
208// blueprint.Module is an android.Module, otherwise returns false.
209func predAdaptor(pred func(Module) bool) func(blueprint.Module) bool {
210 return func(module blueprint.Module) bool {
211 if aModule, ok := module.(Module); ok {
212 return pred(aModule)
213 } else {
214 return false
215 }
216 }
217}
218
Colin Cross4c83e5c2019-02-25 14:54:28 -0800219func (s *singletonContextAdaptor) VisitAllModulesBlueprint(visit func(blueprint.Module)) {
Colin Cross2465c3d2018-09-28 10:19:18 -0700220 s.SingletonContext.VisitAllModules(visit)
221}
222
Colin Cross4c83e5c2019-02-25 14:54:28 -0800223func (s *singletonContextAdaptor) VisitAllModules(visit func(Module)) {
Colin Cross0875c522017-11-28 17:34:01 -0800224 s.SingletonContext.VisitAllModules(visitAdaptor(visit))
225}
226
Colin Cross4c83e5c2019-02-25 14:54:28 -0800227func (s *singletonContextAdaptor) VisitAllModulesIf(pred func(Module) bool, visit func(Module)) {
Colin Cross0875c522017-11-28 17:34:01 -0800228 s.SingletonContext.VisitAllModulesIf(predAdaptor(pred), visitAdaptor(visit))
229}
230
Mitch Phillips3a7a31b2019-10-16 15:00:12 -0700231func (s *singletonContextAdaptor) VisitDirectDeps(module Module, visit func(Module)) {
232 s.SingletonContext.VisitDirectDeps(module, visitAdaptor(visit))
233}
234
235func (s *singletonContextAdaptor) VisitDirectDepsIf(module Module, pred func(Module) bool, visit func(Module)) {
236 s.SingletonContext.VisitDirectDepsIf(module, predAdaptor(pred), visitAdaptor(visit))
237}
238
Colin Cross4c83e5c2019-02-25 14:54:28 -0800239func (s *singletonContextAdaptor) VisitDepsDepthFirst(module Module, visit func(Module)) {
Colin Cross0875c522017-11-28 17:34:01 -0800240 s.SingletonContext.VisitDepsDepthFirst(module, visitAdaptor(visit))
241}
242
Colin Cross4c83e5c2019-02-25 14:54:28 -0800243func (s *singletonContextAdaptor) VisitDepsDepthFirstIf(module Module, pred func(Module) bool, visit func(Module)) {
Colin Cross0875c522017-11-28 17:34:01 -0800244 s.SingletonContext.VisitDepsDepthFirstIf(module, predAdaptor(pred), visitAdaptor(visit))
245}
246
Colin Cross4c83e5c2019-02-25 14:54:28 -0800247func (s *singletonContextAdaptor) VisitAllModuleVariants(module Module, visit func(Module)) {
Colin Cross0875c522017-11-28 17:34:01 -0800248 s.SingletonContext.VisitAllModuleVariants(module, visitAdaptor(visit))
249}
250
Colin Cross4c83e5c2019-02-25 14:54:28 -0800251func (s *singletonContextAdaptor) PrimaryModule(module Module) Module {
Colin Cross0875c522017-11-28 17:34:01 -0800252 return s.SingletonContext.PrimaryModule(module).(Module)
253}
254
Colin Cross4c83e5c2019-02-25 14:54:28 -0800255func (s *singletonContextAdaptor) FinalModule(module Module) Module {
Colin Cross0875c522017-11-28 17:34:01 -0800256 return s.SingletonContext.FinalModule(module).(Module)
257}
Bob Badoureef4c1c2022-05-16 12:20:04 -0700258
259func (s *singletonContextAdaptor) ModuleVariantsFromName(referer Module, name string) []Module {
Cole Faust894bb3b2024-02-07 11:28:26 -0800260 // get module reference for visibility enforcement
Cole Faust9a24d902024-03-18 15:38:12 -0700261 qualified := createVisibilityModuleReference(s.ModuleName(referer), s.ModuleDir(referer), referer)
Bob Badoureef4c1c2022-05-16 12:20:04 -0700262
263 modules := s.SingletonContext.ModuleVariantsFromName(referer, name)
264 result := make([]Module, 0, len(modules))
265 for _, m := range modules {
266 if module, ok := m.(Module); ok {
267 // enforce visibility
268 depName := s.ModuleName(module)
269 depDir := s.ModuleDir(module)
270 depQualified := qualifiedModuleName{depDir, depName}
271 // Targets are always visible to other targets in their own package.
Cole Faust894bb3b2024-02-07 11:28:26 -0800272 if depQualified.pkg != qualified.name.pkg {
Bob Badoureef4c1c2022-05-16 12:20:04 -0700273 rule := effectiveVisibilityRules(s.Config(), depQualified)
274 if !rule.matches(qualified) {
Bob Badoura5ea2472022-05-20 16:37:26 -0700275 s.ModuleErrorf(referer, "module %q references %q which is not visible to this module\nYou may need to add %q to its visibility",
276 referer.Name(), depQualified, "//"+s.ModuleDir(referer))
Bob Badoureef4c1c2022-05-16 12:20:04 -0700277 continue
278 }
279 }
280 result = append(result, module)
281 }
282 }
283 return result
284}
Colin Cross3c0a83d2023-12-12 14:13:26 -0800285
Yu Liu663e4502024-08-12 18:23:59 +0000286func (s *singletonContextAdaptor) otherModuleProvider(module blueprint.Module, provider blueprint.AnyProviderKey) (any, bool) {
Colin Cross3c0a83d2023-12-12 14:13:26 -0800287 return s.SingletonContext.ModuleProvider(module, provider)
288}
Cole Faust02987bd2024-03-21 17:58:43 -0700289
290func (s *singletonContextAdaptor) OtherModulePropertyErrorf(module Module, property string, format string, args ...interface{}) {
Cole Fausta963b942024-04-11 17:43:00 -0700291 s.blueprintSingletonContext().OtherModulePropertyErrorf(module, property, format, args...)
Cole Faust02987bd2024-03-21 17:58:43 -0700292}
Cole Faust4e2bf9f2024-09-11 13:26:20 -0700293
294func (s *singletonContextAdaptor) HasMutatorFinished(mutatorName string) bool {
295 return s.blueprintSingletonContext().HasMutatorFinished(mutatorName)
296}