| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 1 | // 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 |  | 
|  | 15 | package android | 
|  | 16 |  | 
|  | 17 | import ( | 
|  | 18 | "github.com/google/blueprint" | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 19 | ) | 
|  | 20 |  | 
|  | 21 | // SingletonContext | 
|  | 22 | type SingletonContext interface { | 
| Colin Cross | aabf679 | 2017-11-29 00:27:14 -0800 | [diff] [blame] | 23 | Config() Config | 
| Colin Cross | 65494b9 | 2019-02-07 14:25:51 -0800 | [diff] [blame] | 24 | DeviceConfig() DeviceConfig | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 25 |  | 
|  | 26 | ModuleName(module blueprint.Module) string | 
|  | 27 | ModuleDir(module blueprint.Module) string | 
|  | 28 | ModuleSubDir(module blueprint.Module) string | 
|  | 29 | ModuleType(module blueprint.Module) string | 
|  | 30 | BlueprintFile(module blueprint.Module) string | 
|  | 31 |  | 
| Bob Badour | eef4c1c | 2022-05-16 12:20:04 -0700 | [diff] [blame] | 32 | // ModuleVariantsFromName returns the list of module variants named `name` in the same namespace as `referer` enforcing visibility rules. | 
|  | 33 | // Allows generating build actions for `referer` based on the metadata for `name` deferred until the singleton context. | 
|  | 34 | ModuleVariantsFromName(referer Module, name string) []Module | 
|  | 35 |  | 
| Colin Cross | d27e7b8 | 2020-07-02 11:38:17 -0700 | [diff] [blame] | 36 | // ModuleProvider returns the value, if any, for the provider for a module.  If the value for the | 
|  | 37 | // provider was not set it returns the zero value of the type of the provider, which means the | 
|  | 38 | // return value can always be type-asserted to the type of the provider.  The return value should | 
|  | 39 | // always be considered read-only.  It panics if called before the appropriate mutator or | 
|  | 40 | // GenerateBuildActions pass for the provider on the module. | 
|  | 41 | ModuleProvider(module blueprint.Module, provider blueprint.ProviderKey) interface{} | 
|  | 42 |  | 
|  | 43 | // ModuleHasProvider returns true if the provider for the given module has been set. | 
|  | 44 | ModuleHasProvider(module blueprint.Module, provider blueprint.ProviderKey) bool | 
|  | 45 |  | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 46 | ModuleErrorf(module blueprint.Module, format string, args ...interface{}) | 
|  | 47 | Errorf(format string, args ...interface{}) | 
|  | 48 | Failed() bool | 
|  | 49 |  | 
|  | 50 | Variable(pctx PackageContext, name, value string) | 
| Colin Cross | 5901439 | 2017-12-12 11:05:06 -0800 | [diff] [blame] | 51 | Rule(pctx PackageContext, name string, params blueprint.RuleParams, argNames ...string) blueprint.Rule | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 52 | Build(pctx PackageContext, params BuildParams) | 
| Colin Cross | c3d87d3 | 2020-06-04 13:25:17 -0700 | [diff] [blame] | 53 |  | 
|  | 54 | // Phony creates a Make-style phony rule, a rule with no commands that can depend on other | 
|  | 55 | // phony rules or real files.  Phony can be called on the same name multiple times to add | 
|  | 56 | // additional dependencies. | 
|  | 57 | Phony(name string, deps ...Path) | 
|  | 58 |  | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 59 | RequireNinjaVersion(major, minor, micro int) | 
|  | 60 |  | 
| Lukacs T. Berki | 9f6c24a | 2021-08-26 15:07:24 +0200 | [diff] [blame] | 61 | // SetOutDir sets the value of the top-level "builddir" Ninja variable | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 62 | // that controls where Ninja stores its build log files.  This value can be | 
|  | 63 | // set at most one time for a single build, later calls are ignored. | 
| Lukacs T. Berki | 9f6c24a | 2021-08-26 15:07:24 +0200 | [diff] [blame] | 64 | SetOutDir(pctx PackageContext, value string) | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 65 |  | 
|  | 66 | // Eval takes a string with embedded ninja variables, and returns a string | 
|  | 67 | // with all of the variables recursively expanded. Any variables references | 
|  | 68 | // are expanded in the scope of the PackageContext. | 
|  | 69 | Eval(pctx PackageContext, ninjaStr string) (string, error) | 
|  | 70 |  | 
| Colin Cross | 2465c3d | 2018-09-28 10:19:18 -0700 | [diff] [blame] | 71 | VisitAllModulesBlueprint(visit func(blueprint.Module)) | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 72 | VisitAllModules(visit func(Module)) | 
|  | 73 | VisitAllModulesIf(pred func(Module) bool, visit func(Module)) | 
| Mitch Phillips | 3a7a31b | 2019-10-16 15:00:12 -0700 | [diff] [blame] | 74 |  | 
|  | 75 | VisitDirectDeps(module Module, visit func(Module)) | 
|  | 76 | VisitDirectDepsIf(module Module, pred func(Module) bool, visit func(Module)) | 
|  | 77 |  | 
| Colin Cross | 6b75360 | 2018-06-21 13:03:07 -0700 | [diff] [blame] | 78 | // Deprecated: use WalkDeps instead to support multiple dependency tags on the same module | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 79 | VisitDepsDepthFirst(module Module, visit func(Module)) | 
| Colin Cross | 6b75360 | 2018-06-21 13:03:07 -0700 | [diff] [blame] | 80 | // Deprecated: use WalkDeps instead to support multiple dependency tags on the same module | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 81 | VisitDepsDepthFirstIf(module Module, pred func(Module) bool, | 
|  | 82 | visit func(Module)) | 
|  | 83 |  | 
|  | 84 | VisitAllModuleVariants(module Module, visit func(Module)) | 
|  | 85 |  | 
|  | 86 | PrimaryModule(module Module) Module | 
|  | 87 | FinalModule(module Module) Module | 
|  | 88 |  | 
|  | 89 | AddNinjaFileDeps(deps ...string) | 
|  | 90 |  | 
|  | 91 | // GlobWithDeps returns a list of files that match the specified pattern but do not match any | 
|  | 92 | // of the patterns in excludes.  It also adds efficient dependencies to rerun the primary | 
|  | 93 | // builder whenever a file matching the pattern as added or removed, without rerunning if a | 
|  | 94 | // file that does not match the pattern is added to a searched directory. | 
|  | 95 | GlobWithDeps(pattern string, excludes []string) ([]string, error) | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 96 | } | 
|  | 97 |  | 
|  | 98 | type singletonAdaptor struct { | 
|  | 99 | Singleton | 
| Colin Cross | 4c83e5c | 2019-02-25 14:54:28 -0800 | [diff] [blame] | 100 |  | 
|  | 101 | buildParams []BuildParams | 
|  | 102 | ruleParams  map[blueprint.Rule]blueprint.RuleParams | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 103 | } | 
|  | 104 |  | 
| Colin Cross | 4c83e5c | 2019-02-25 14:54:28 -0800 | [diff] [blame] | 105 | var _ testBuildProvider = (*singletonAdaptor)(nil) | 
|  | 106 |  | 
|  | 107 | func (s *singletonAdaptor) GenerateBuildActions(ctx blueprint.SingletonContext) { | 
|  | 108 | sctx := &singletonContextAdaptor{SingletonContext: ctx} | 
|  | 109 | if sctx.Config().captureBuild { | 
|  | 110 | sctx.ruleParams = make(map[blueprint.Rule]blueprint.RuleParams) | 
|  | 111 | } | 
|  | 112 |  | 
|  | 113 | s.Singleton.GenerateBuildActions(sctx) | 
|  | 114 |  | 
|  | 115 | s.buildParams = sctx.buildParams | 
|  | 116 | s.ruleParams = sctx.ruleParams | 
|  | 117 | } | 
|  | 118 |  | 
|  | 119 | func (s *singletonAdaptor) BuildParamsForTests() []BuildParams { | 
|  | 120 | return s.buildParams | 
|  | 121 | } | 
|  | 122 |  | 
|  | 123 | func (s *singletonAdaptor) RuleParamsForTests() map[blueprint.Rule]blueprint.RuleParams { | 
|  | 124 | return s.ruleParams | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 125 | } | 
|  | 126 |  | 
|  | 127 | type Singleton interface { | 
|  | 128 | GenerateBuildActions(SingletonContext) | 
|  | 129 | } | 
|  | 130 |  | 
|  | 131 | type singletonContextAdaptor struct { | 
|  | 132 | blueprint.SingletonContext | 
| Colin Cross | 4c83e5c | 2019-02-25 14:54:28 -0800 | [diff] [blame] | 133 |  | 
|  | 134 | buildParams []BuildParams | 
|  | 135 | ruleParams  map[blueprint.Rule]blueprint.RuleParams | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 136 | } | 
|  | 137 |  | 
| Colin Cross | 4c83e5c | 2019-02-25 14:54:28 -0800 | [diff] [blame] | 138 | func (s *singletonContextAdaptor) Config() Config { | 
| Colin Cross | aabf679 | 2017-11-29 00:27:14 -0800 | [diff] [blame] | 139 | return s.SingletonContext.Config().(Config) | 
|  | 140 | } | 
|  | 141 |  | 
| Colin Cross | 4c83e5c | 2019-02-25 14:54:28 -0800 | [diff] [blame] | 142 | func (s *singletonContextAdaptor) DeviceConfig() DeviceConfig { | 
| Colin Cross | 65494b9 | 2019-02-07 14:25:51 -0800 | [diff] [blame] | 143 | return DeviceConfig{s.Config().deviceConfig} | 
|  | 144 | } | 
|  | 145 |  | 
| Colin Cross | 4c83e5c | 2019-02-25 14:54:28 -0800 | [diff] [blame] | 146 | func (s *singletonContextAdaptor) Variable(pctx PackageContext, name, value string) { | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 147 | s.SingletonContext.Variable(pctx.PackageContext, name, value) | 
|  | 148 | } | 
|  | 149 |  | 
| Colin Cross | 4c83e5c | 2019-02-25 14:54:28 -0800 | [diff] [blame] | 150 | func (s *singletonContextAdaptor) Rule(pctx PackageContext, name string, params blueprint.RuleParams, argNames ...string) blueprint.Rule { | 
| Ramy Medhat | 944839a | 2020-03-31 22:14:52 -0400 | [diff] [blame] | 151 | if s.Config().UseRemoteBuild() { | 
|  | 152 | if params.Pool == nil { | 
|  | 153 | // When USE_GOMA=true or USE_RBE=true are set and the rule is not supported by goma/RBE, restrict | 
|  | 154 | // jobs to the local parallelism value | 
|  | 155 | params.Pool = localPool | 
|  | 156 | } else if params.Pool == remotePool { | 
|  | 157 | // remotePool is a fake pool used to identify rule that are supported for remoting. If the rule's | 
|  | 158 | // pool is the remotePool, replace with nil so that ninja runs it at NINJA_REMOTE_NUM_JOBS | 
|  | 159 | // parallelism. | 
|  | 160 | params.Pool = nil | 
|  | 161 | } | 
| Colin Cross | 2e2dbc2 | 2019-09-25 13:31:46 -0700 | [diff] [blame] | 162 | } | 
| Colin Cross | 4c83e5c | 2019-02-25 14:54:28 -0800 | [diff] [blame] | 163 | rule := s.SingletonContext.Rule(pctx.PackageContext, name, params, argNames...) | 
|  | 164 | if s.Config().captureBuild { | 
|  | 165 | s.ruleParams[rule] = params | 
|  | 166 | } | 
|  | 167 | return rule | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 168 | } | 
|  | 169 |  | 
| Colin Cross | 4c83e5c | 2019-02-25 14:54:28 -0800 | [diff] [blame] | 170 | func (s *singletonContextAdaptor) Build(pctx PackageContext, params BuildParams) { | 
|  | 171 | if s.Config().captureBuild { | 
|  | 172 | s.buildParams = append(s.buildParams, params) | 
|  | 173 | } | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 174 | bparams := convertBuildParams(params) | 
| Jingwen Chen | ce679d2 | 2020-09-23 04:30:02 +0000 | [diff] [blame] | 175 | err := validateBuildParams(bparams) | 
|  | 176 | if err != nil { | 
|  | 177 | s.Errorf("%s: build parameter validation failed: %s", s.Name(), err.Error()) | 
|  | 178 | } | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 179 | s.SingletonContext.Build(pctx.PackageContext, bparams) | 
|  | 180 |  | 
|  | 181 | } | 
|  | 182 |  | 
| Colin Cross | c3d87d3 | 2020-06-04 13:25:17 -0700 | [diff] [blame] | 183 | func (s *singletonContextAdaptor) Phony(name string, deps ...Path) { | 
|  | 184 | addPhony(s.Config(), name, deps...) | 
|  | 185 | } | 
|  | 186 |  | 
| Lukacs T. Berki | 9f6c24a | 2021-08-26 15:07:24 +0200 | [diff] [blame] | 187 | func (s *singletonContextAdaptor) SetOutDir(pctx PackageContext, value string) { | 
|  | 188 | s.SingletonContext.SetOutDir(pctx.PackageContext, value) | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 189 | } | 
|  | 190 |  | 
| Colin Cross | 4c83e5c | 2019-02-25 14:54:28 -0800 | [diff] [blame] | 191 | func (s *singletonContextAdaptor) Eval(pctx PackageContext, ninjaStr string) (string, error) { | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 192 | 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. | 
|  | 198 | func 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. | 
|  | 209 | func 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 Cross | 4c83e5c | 2019-02-25 14:54:28 -0800 | [diff] [blame] | 219 | func (s *singletonContextAdaptor) VisitAllModulesBlueprint(visit func(blueprint.Module)) { | 
| Colin Cross | 2465c3d | 2018-09-28 10:19:18 -0700 | [diff] [blame] | 220 | s.SingletonContext.VisitAllModules(visit) | 
|  | 221 | } | 
|  | 222 |  | 
| Colin Cross | 4c83e5c | 2019-02-25 14:54:28 -0800 | [diff] [blame] | 223 | func (s *singletonContextAdaptor) VisitAllModules(visit func(Module)) { | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 224 | s.SingletonContext.VisitAllModules(visitAdaptor(visit)) | 
|  | 225 | } | 
|  | 226 |  | 
| Colin Cross | 4c83e5c | 2019-02-25 14:54:28 -0800 | [diff] [blame] | 227 | func (s *singletonContextAdaptor) VisitAllModulesIf(pred func(Module) bool, visit func(Module)) { | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 228 | s.SingletonContext.VisitAllModulesIf(predAdaptor(pred), visitAdaptor(visit)) | 
|  | 229 | } | 
|  | 230 |  | 
| Mitch Phillips | 3a7a31b | 2019-10-16 15:00:12 -0700 | [diff] [blame] | 231 | func (s *singletonContextAdaptor) VisitDirectDeps(module Module, visit func(Module)) { | 
|  | 232 | s.SingletonContext.VisitDirectDeps(module, visitAdaptor(visit)) | 
|  | 233 | } | 
|  | 234 |  | 
|  | 235 | func (s *singletonContextAdaptor) VisitDirectDepsIf(module Module, pred func(Module) bool, visit func(Module)) { | 
|  | 236 | s.SingletonContext.VisitDirectDepsIf(module, predAdaptor(pred), visitAdaptor(visit)) | 
|  | 237 | } | 
|  | 238 |  | 
| Colin Cross | 4c83e5c | 2019-02-25 14:54:28 -0800 | [diff] [blame] | 239 | func (s *singletonContextAdaptor) VisitDepsDepthFirst(module Module, visit func(Module)) { | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 240 | s.SingletonContext.VisitDepsDepthFirst(module, visitAdaptor(visit)) | 
|  | 241 | } | 
|  | 242 |  | 
| Colin Cross | 4c83e5c | 2019-02-25 14:54:28 -0800 | [diff] [blame] | 243 | func (s *singletonContextAdaptor) VisitDepsDepthFirstIf(module Module, pred func(Module) bool, visit func(Module)) { | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 244 | s.SingletonContext.VisitDepsDepthFirstIf(module, predAdaptor(pred), visitAdaptor(visit)) | 
|  | 245 | } | 
|  | 246 |  | 
| Colin Cross | 4c83e5c | 2019-02-25 14:54:28 -0800 | [diff] [blame] | 247 | func (s *singletonContextAdaptor) VisitAllModuleVariants(module Module, visit func(Module)) { | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 248 | s.SingletonContext.VisitAllModuleVariants(module, visitAdaptor(visit)) | 
|  | 249 | } | 
|  | 250 |  | 
| Colin Cross | 4c83e5c | 2019-02-25 14:54:28 -0800 | [diff] [blame] | 251 | func (s *singletonContextAdaptor) PrimaryModule(module Module) Module { | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 252 | return s.SingletonContext.PrimaryModule(module).(Module) | 
|  | 253 | } | 
|  | 254 |  | 
| Colin Cross | 4c83e5c | 2019-02-25 14:54:28 -0800 | [diff] [blame] | 255 | func (s *singletonContextAdaptor) FinalModule(module Module) Module { | 
| Colin Cross | 0875c52 | 2017-11-28 17:34:01 -0800 | [diff] [blame] | 256 | return s.SingletonContext.FinalModule(module).(Module) | 
|  | 257 | } | 
| Bob Badour | eef4c1c | 2022-05-16 12:20:04 -0700 | [diff] [blame] | 258 |  | 
|  | 259 | func (s *singletonContextAdaptor) ModuleVariantsFromName(referer Module, name string) []Module { | 
|  | 260 | // get qualified module name for visibility enforcement | 
|  | 261 | qualified := createQualifiedModuleName(s.ModuleName(referer), s.ModuleDir(referer)) | 
|  | 262 |  | 
|  | 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. | 
|  | 272 | if depQualified.pkg != qualified.pkg { | 
|  | 273 | rule := effectiveVisibilityRules(s.Config(), depQualified) | 
|  | 274 | if !rule.matches(qualified) { | 
| Bob Badour | a5ea247 | 2022-05-20 16:37:26 -0700 | [diff] [blame] | 275 | 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 Badour | eef4c1c | 2022-05-16 12:20:04 -0700 | [diff] [blame] | 277 | continue | 
|  | 278 | } | 
|  | 279 | } | 
|  | 280 | result = append(result, module) | 
|  | 281 | } | 
|  | 282 | } | 
|  | 283 | return result | 
|  | 284 | } |