Colin Cross | 5049f02 | 2015-03-18 13:28:46 -0700 | [diff] [blame] | 1 | // 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 | |
| 15 | package genrule |
| 16 | |
| 17 | import ( |
Dan Willemsen | 3f4539b | 2016-09-28 16:19:10 -0700 | [diff] [blame^] | 18 | "os" |
| 19 | |
Colin Cross | 70b4059 | 2015-03-23 12:57:34 -0700 | [diff] [blame] | 20 | "github.com/google/blueprint" |
Colin Cross | 5049f02 | 2015-03-18 13:28:46 -0700 | [diff] [blame] | 21 | |
Colin Cross | 463a90e | 2015-06-17 14:20:06 -0700 | [diff] [blame] | 22 | "android/soong" |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 23 | "android/soong/android" |
Colin Cross | 5049f02 | 2015-03-18 13:28:46 -0700 | [diff] [blame] | 24 | ) |
| 25 | |
Colin Cross | 463a90e | 2015-06-17 14:20:06 -0700 | [diff] [blame] | 26 | func init() { |
| 27 | soong.RegisterModuleType("gensrcs", GenSrcsFactory) |
| 28 | soong.RegisterModuleType("genrule", GenRuleFactory) |
Colin Cross | 6362e27 | 2015-10-29 15:25:03 -0700 | [diff] [blame] | 29 | |
Colin Cross | e8a67a7 | 2016-08-07 21:17:54 -0700 | [diff] [blame] | 30 | android.RegisterBottomUpMutator("genrule_deps", genruleDepsMutator).Parallel() |
Colin Cross | 463a90e | 2015-06-17 14:20:06 -0700 | [diff] [blame] | 31 | } |
| 32 | |
Colin Cross | 5049f02 | 2015-03-18 13:28:46 -0700 | [diff] [blame] | 33 | var ( |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 34 | pctx = android.NewPackageContext("android/soong/genrule") |
Colin Cross | 5049f02 | 2015-03-18 13:28:46 -0700 | [diff] [blame] | 35 | ) |
| 36 | |
| 37 | func init() { |
Dan Willemsen | 34cc69e | 2015-09-23 15:26:20 -0700 | [diff] [blame] | 38 | pctx.SourcePathVariable("srcDir", "") |
| 39 | pctx.HostBinToolVariable("hostBin", "") |
Colin Cross | 5049f02 | 2015-03-18 13:28:46 -0700 | [diff] [blame] | 40 | } |
| 41 | |
| 42 | type SourceFileGenerator interface { |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 43 | GeneratedSourceFiles() android.Paths |
| 44 | GeneratedHeaderDir() android.Path |
Colin Cross | 5049f02 | 2015-03-18 13:28:46 -0700 | [diff] [blame] | 45 | } |
| 46 | |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 47 | type HostToolProvider interface { |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 48 | HostToolPath() android.OptionalPath |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 49 | } |
Colin Cross | 5049f02 | 2015-03-18 13:28:46 -0700 | [diff] [blame] | 50 | |
Colin Cross | 7d5136f | 2015-05-11 13:39:40 -0700 | [diff] [blame] | 51 | type generatorProperties struct { |
| 52 | // command to run on one or more input files. Available variables for substitution: |
Dan Willemsen | f7f3d69 | 2016-04-20 14:54:32 -0700 | [diff] [blame] | 53 | // $tool: the path to the `tool` or `tool_file` |
Colin Cross | 7d5136f | 2015-05-11 13:39:40 -0700 | [diff] [blame] | 54 | // $in: one or more input files |
| 55 | // $out: a single output file |
| 56 | // $srcDir: the root directory of the source tree |
Dan Willemsen | 3f4539b | 2016-09-28 16:19:10 -0700 | [diff] [blame^] | 57 | // $genDir: the sandbox directory for this tool; contains $out |
Colin Cross | 7d5136f | 2015-05-11 13:39:40 -0700 | [diff] [blame] | 58 | // The host bin directory will be in the path |
| 59 | Cmd string |
| 60 | |
| 61 | // name of the module (if any) that produces the host executable. Leave empty for |
| 62 | // prebuilts or scripts that do not need a module to build them. |
| 63 | Tool string |
Dan Willemsen | f7f3d69 | 2016-04-20 14:54:32 -0700 | [diff] [blame] | 64 | |
| 65 | // Local file that is used as the tool |
| 66 | Tool_file string |
Colin Cross | 7d5136f | 2015-05-11 13:39:40 -0700 | [diff] [blame] | 67 | } |
| 68 | |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 69 | type generator struct { |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 70 | android.ModuleBase |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 71 | |
Colin Cross | 7d5136f | 2015-05-11 13:39:40 -0700 | [diff] [blame] | 72 | properties generatorProperties |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 73 | |
| 74 | tasks taskFunc |
| 75 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 76 | deps android.Paths |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 77 | rule blueprint.Rule |
| 78 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 79 | genPath android.Path |
Dan Willemsen | b40aab6 | 2016-04-20 14:21:14 -0700 | [diff] [blame] | 80 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 81 | outputFiles android.Paths |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 82 | } |
| 83 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 84 | type taskFunc func(ctx android.ModuleContext) []generateTask |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 85 | |
| 86 | type generateTask struct { |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 87 | in android.Paths |
| 88 | out android.ModuleGenPath |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 89 | } |
| 90 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 91 | func (g *generator) GeneratedSourceFiles() android.Paths { |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 92 | return g.outputFiles |
| 93 | } |
| 94 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 95 | func (g *generator) GeneratedHeaderDir() android.Path { |
Dan Willemsen | b40aab6 | 2016-04-20 14:21:14 -0700 | [diff] [blame] | 96 | return g.genPath |
| 97 | } |
| 98 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 99 | func genruleDepsMutator(ctx android.BottomUpMutatorContext) { |
Colin Cross | 6362e27 | 2015-10-29 15:25:03 -0700 | [diff] [blame] | 100 | if g, ok := ctx.Module().(*generator); ok { |
| 101 | if g.properties.Tool != "" { |
Dan Willemsen | 490fd49 | 2015-11-24 17:53:15 -0800 | [diff] [blame] | 102 | ctx.AddFarVariationDependencies([]blueprint.Variation{ |
Colin Cross | a1ad8d1 | 2016-06-01 17:09:44 -0700 | [diff] [blame] | 103 | {"arch", ctx.AConfig().BuildOsVariant}, |
Colin Cross | c99deeb | 2016-04-11 15:06:20 -0700 | [diff] [blame] | 104 | }, nil, g.properties.Tool) |
Colin Cross | 6362e27 | 2015-10-29 15:25:03 -0700 | [diff] [blame] | 105 | } |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 106 | } |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 107 | } |
| 108 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 109 | func (g *generator) GenerateAndroidBuildActions(ctx android.ModuleContext) { |
Dan Willemsen | f7f3d69 | 2016-04-20 14:54:32 -0700 | [diff] [blame] | 110 | if g.properties.Tool != "" && g.properties.Tool_file != "" { |
| 111 | ctx.ModuleErrorf("`tool` and `tool_file` may not be specified at the same time") |
| 112 | return |
| 113 | } |
| 114 | |
Dan Willemsen | 3f4539b | 2016-09-28 16:19:10 -0700 | [diff] [blame^] | 115 | g.genPath = android.PathForModuleGen(ctx, "") |
| 116 | |
| 117 | cmd := os.Expand(g.properties.Cmd, func(name string) string { |
| 118 | switch name { |
| 119 | case "$": |
| 120 | return "$$" |
| 121 | case "tool": |
| 122 | return "${tool}" |
| 123 | case "in": |
| 124 | return "${in}" |
| 125 | case "out": |
| 126 | return "${out}" |
| 127 | case "srcDir": |
| 128 | return "${srcDir}" |
| 129 | case "genDir": |
| 130 | return g.genPath.String() |
| 131 | default: |
| 132 | ctx.PropertyErrorf("cmd", "unknown variable '%s'", name) |
| 133 | } |
| 134 | return "" |
| 135 | }) |
| 136 | |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 137 | g.rule = ctx.Rule(pctx, "generator", blueprint.RuleParams{ |
Dan Willemsen | 3f4539b | 2016-09-28 16:19:10 -0700 | [diff] [blame^] | 138 | Command: "PATH=$$PATH:$hostBin " + cmd, |
Dan Willemsen | f7f3d69 | 2016-04-20 14:54:32 -0700 | [diff] [blame] | 139 | }, "tool") |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 140 | |
Dan Willemsen | f7f3d69 | 2016-04-20 14:54:32 -0700 | [diff] [blame] | 141 | var tool string |
| 142 | if g.properties.Tool_file != "" { |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 143 | toolpath := android.PathForModuleSrc(ctx, g.properties.Tool_file) |
Dan Willemsen | f7f3d69 | 2016-04-20 14:54:32 -0700 | [diff] [blame] | 144 | g.deps = append(g.deps, toolpath) |
| 145 | tool = toolpath.String() |
| 146 | } else if g.properties.Tool != "" { |
| 147 | ctx.VisitDirectDeps(func(module blueprint.Module) { |
| 148 | if t, ok := module.(HostToolProvider); ok { |
| 149 | p := t.HostToolPath() |
| 150 | if p.Valid() { |
| 151 | g.deps = append(g.deps, p.Path()) |
| 152 | tool = p.String() |
| 153 | } else { |
| 154 | ctx.ModuleErrorf("host tool %q missing output file", ctx.OtherModuleName(module)) |
| 155 | } |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 156 | } else { |
Dan Willemsen | f7f3d69 | 2016-04-20 14:54:32 -0700 | [diff] [blame] | 157 | ctx.ModuleErrorf("unknown dependency %q", ctx.OtherModuleName(module)) |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 158 | } |
Dan Willemsen | f7f3d69 | 2016-04-20 14:54:32 -0700 | [diff] [blame] | 159 | }) |
| 160 | } |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 161 | |
| 162 | for _, task := range g.tasks(ctx) { |
Dan Willemsen | f7f3d69 | 2016-04-20 14:54:32 -0700 | [diff] [blame] | 163 | g.generateSourceFile(ctx, task, tool) |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 164 | } |
| 165 | } |
| 166 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 167 | func (g *generator) generateSourceFile(ctx android.ModuleContext, task generateTask, tool string) { |
| 168 | ctx.ModuleBuild(pctx, android.ModuleBuildParams{ |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 169 | Rule: g.rule, |
Dan Willemsen | 34cc69e | 2015-09-23 15:26:20 -0700 | [diff] [blame] | 170 | Output: task.out, |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 171 | Inputs: task.in, |
| 172 | Implicits: g.deps, |
Dan Willemsen | f7f3d69 | 2016-04-20 14:54:32 -0700 | [diff] [blame] | 173 | Args: map[string]string{ |
| 174 | "tool": tool, |
| 175 | }, |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 176 | }) |
| 177 | |
| 178 | g.outputFiles = append(g.outputFiles, task.out) |
| 179 | } |
| 180 | |
| 181 | func generatorFactory(tasks taskFunc, props ...interface{}) (blueprint.Module, []interface{}) { |
| 182 | module := &generator{ |
| 183 | tasks: tasks, |
| 184 | } |
| 185 | |
| 186 | props = append(props, &module.properties) |
| 187 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 188 | return android.InitAndroidModule(module, props...) |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 189 | } |
| 190 | |
| 191 | func GenSrcsFactory() (blueprint.Module, []interface{}) { |
| 192 | properties := &genSrcsProperties{} |
| 193 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 194 | tasks := func(ctx android.ModuleContext) []generateTask { |
Dan Willemsen | 2ef08f4 | 2015-06-30 18:15:24 -0700 | [diff] [blame] | 195 | srcFiles := ctx.ExpandSources(properties.Srcs, nil) |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 196 | tasks := make([]generateTask, 0, len(srcFiles)) |
| 197 | for _, in := range srcFiles { |
Dan Willemsen | 34cc69e | 2015-09-23 15:26:20 -0700 | [diff] [blame] | 198 | tasks = append(tasks, generateTask{ |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 199 | in: android.Paths{in}, |
| 200 | out: android.GenPathWithExt(ctx, in, properties.Output_extension), |
Dan Willemsen | 34cc69e | 2015-09-23 15:26:20 -0700 | [diff] [blame] | 201 | }) |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 202 | } |
| 203 | return tasks |
| 204 | } |
| 205 | |
| 206 | return generatorFactory(tasks, properties) |
| 207 | } |
| 208 | |
| 209 | type genSrcsProperties struct { |
Colin Cross | 7d5136f | 2015-05-11 13:39:40 -0700 | [diff] [blame] | 210 | // list of input files |
Colin Cross | 5049f02 | 2015-03-18 13:28:46 -0700 | [diff] [blame] | 211 | Srcs []string |
| 212 | |
Colin Cross | 7d5136f | 2015-05-11 13:39:40 -0700 | [diff] [blame] | 213 | // extension that will be substituted for each output file |
Colin Cross | 5049f02 | 2015-03-18 13:28:46 -0700 | [diff] [blame] | 214 | Output_extension string |
| 215 | } |
| 216 | |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 217 | func GenRuleFactory() (blueprint.Module, []interface{}) { |
| 218 | properties := &genRuleProperties{} |
Colin Cross | 5049f02 | 2015-03-18 13:28:46 -0700 | [diff] [blame] | 219 | |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 220 | tasks := func(ctx android.ModuleContext) []generateTask { |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 221 | return []generateTask{ |
| 222 | { |
Dan Willemsen | 2ef08f4 | 2015-06-30 18:15:24 -0700 | [diff] [blame] | 223 | in: ctx.ExpandSources(properties.Srcs, nil), |
Colin Cross | 635c3b0 | 2016-05-18 15:37:25 -0700 | [diff] [blame] | 224 | out: android.PathForModuleGen(ctx, properties.Out), |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 225 | }, |
| 226 | } |
Colin Cross | 5049f02 | 2015-03-18 13:28:46 -0700 | [diff] [blame] | 227 | } |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 228 | |
| 229 | return generatorFactory(tasks, properties) |
Colin Cross | 5049f02 | 2015-03-18 13:28:46 -0700 | [diff] [blame] | 230 | } |
| 231 | |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 232 | type genRuleProperties struct { |
Colin Cross | 7d5136f | 2015-05-11 13:39:40 -0700 | [diff] [blame] | 233 | // list of input files |
Colin Cross | d350ecd | 2015-04-28 13:25:36 -0700 | [diff] [blame] | 234 | Srcs []string |
Colin Cross | 5049f02 | 2015-03-18 13:28:46 -0700 | [diff] [blame] | 235 | |
Colin Cross | 7d5136f | 2015-05-11 13:39:40 -0700 | [diff] [blame] | 236 | // name of the output file that will be generated |
Dan Willemsen | f7f3d69 | 2016-04-20 14:54:32 -0700 | [diff] [blame] | 237 | Out string |
Colin Cross | 5049f02 | 2015-03-18 13:28:46 -0700 | [diff] [blame] | 238 | } |