| // Copyright 2015 Google Inc. All rights reserved. |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| package genrule |
| |
| import ( |
| "fmt" |
| "strings" |
| |
| "github.com/google/blueprint" |
| "github.com/google/blueprint/bootstrap" |
| "github.com/google/blueprint/proptools" |
| |
| "android/soong/android" |
| "android/soong/shared" |
| "path/filepath" |
| ) |
| |
| func init() { |
| android.RegisterModuleType("gensrcs", GenSrcsFactory) |
| android.RegisterModuleType("genrule", GenRuleFactory) |
| } |
| |
| var ( |
| pctx = android.NewPackageContext("android/soong/genrule") |
| ) |
| |
| func init() { |
| pctx.HostBinToolVariable("sboxCmd", "sbox") |
| } |
| |
| type SourceFileGenerator interface { |
| GeneratedSourceFiles() android.Paths |
| GeneratedHeaderDirs() android.Paths |
| GeneratedDeps() android.Paths |
| } |
| |
| type HostToolProvider interface { |
| HostToolPath() android.OptionalPath |
| } |
| |
| type hostToolDependencyTag struct { |
| blueprint.BaseDependencyTag |
| } |
| |
| var hostToolDepTag hostToolDependencyTag |
| |
| type generatorProperties struct { |
| // The command to run on one or more input files. Cmd supports substitution of a few variables |
| // (the actual substitution is implemented in GenerateAndroidBuildActions below) |
| // |
| // Available variables for substitution: |
| // |
| // $(location): the path to the first entry in tools or tool_files |
| // $(location <label>): the path to the tool or tool_file with name <label> |
| // $(in): one or more input files |
| // $(out): a single output file |
| // $(depfile): a file to which dependencies will be written, if the depfile property is set to true |
| // $(genDir): the sandbox directory for this tool; contains $(out) |
| // $$: a literal $ |
| // |
| // All files used must be declared as inputs (to ensure proper up-to-date checks). |
| // Use "$(in)" directly in Cmd to ensure that all inputs used are declared. |
| Cmd *string |
| |
| // Enable reading a file containing dependencies in gcc format after the command completes |
| Depfile *bool |
| |
| // name of the modules (if any) that produces the host executable. Leave empty for |
| // prebuilts or scripts that do not need a module to build them. |
| Tools []string |
| |
| // Local file that is used as the tool |
| Tool_files []string |
| |
| // List of directories to export generated headers from |
| Export_include_dirs []string |
| |
| // list of input files |
| Srcs []string |
| } |
| |
| type Module struct { |
| android.ModuleBase |
| |
| // For other packages to make their own genrules with extra |
| // properties |
| Extra interface{} |
| |
| properties generatorProperties |
| |
| taskGenerator taskFunc |
| |
| deps android.Paths |
| rule blueprint.Rule |
| |
| exportedIncludeDirs android.Paths |
| |
| outputFiles android.Paths |
| outputDeps android.Paths |
| } |
| |
| type taskFunc func(ctx android.ModuleContext, rawCommand string, srcFiles android.Paths) generateTask |
| |
| type generateTask struct { |
| in android.Paths |
| out android.WritablePaths |
| sandboxOuts []string |
| cmd string |
| } |
| |
| func (g *Module) GeneratedSourceFiles() android.Paths { |
| return g.outputFiles |
| } |
| |
| func (g *Module) Srcs() android.Paths { |
| return append(android.Paths{}, g.outputFiles...) |
| } |
| |
| func (g *Module) GeneratedHeaderDirs() android.Paths { |
| return g.exportedIncludeDirs |
| } |
| |
| func (g *Module) GeneratedDeps() android.Paths { |
| return g.outputDeps |
| } |
| |
| func (g *Module) DepsMutator(ctx android.BottomUpMutatorContext) { |
| android.ExtractSourcesDeps(ctx, g.properties.Srcs) |
| android.ExtractSourcesDeps(ctx, g.properties.Tool_files) |
| if g, ok := ctx.Module().(*Module); ok { |
| if len(g.properties.Tools) > 0 { |
| ctx.AddFarVariationDependencies([]blueprint.Variation{ |
| {Mutator: "arch", Variation: ctx.Config().BuildOsVariant}, |
| }, hostToolDepTag, g.properties.Tools...) |
| } |
| } |
| } |
| |
| func (g *Module) GenerateAndroidBuildActions(ctx android.ModuleContext) { |
| if len(g.properties.Export_include_dirs) > 0 { |
| for _, dir := range g.properties.Export_include_dirs { |
| g.exportedIncludeDirs = append(g.exportedIncludeDirs, |
| android.PathForModuleGen(ctx, ctx.ModuleDir(), dir)) |
| } |
| } else { |
| g.exportedIncludeDirs = append(g.exportedIncludeDirs, android.PathForModuleGen(ctx, "")) |
| } |
| |
| tools := map[string]android.Path{} |
| |
| if len(g.properties.Tools) > 0 { |
| ctx.VisitDirectDepsBlueprint(func(module blueprint.Module) { |
| switch ctx.OtherModuleDependencyTag(module) { |
| case hostToolDepTag: |
| tool := ctx.OtherModuleName(module) |
| var path android.OptionalPath |
| |
| if t, ok := module.(HostToolProvider); ok { |
| if !t.(android.Module).Enabled() { |
| if ctx.Config().AllowMissingDependencies() { |
| ctx.AddMissingDependencies([]string{tool}) |
| } else { |
| ctx.ModuleErrorf("depends on disabled module %q", tool) |
| } |
| break |
| } |
| path = t.HostToolPath() |
| } else if t, ok := module.(bootstrap.GoBinaryTool); ok { |
| if s, err := filepath.Rel(android.PathForOutput(ctx).String(), t.InstallPath()); err == nil { |
| path = android.OptionalPathForPath(android.PathForOutput(ctx, s)) |
| } else { |
| ctx.ModuleErrorf("cannot find path for %q: %v", tool, err) |
| break |
| } |
| } else { |
| ctx.ModuleErrorf("%q is not a host tool provider", tool) |
| break |
| } |
| |
| if path.Valid() { |
| g.deps = append(g.deps, path.Path()) |
| if _, exists := tools[tool]; !exists { |
| tools[tool] = path.Path() |
| } else { |
| ctx.ModuleErrorf("multiple tools for %q, %q and %q", tool, tools[tool], path.Path().String()) |
| } |
| } else { |
| ctx.ModuleErrorf("host tool %q missing output file", tool) |
| } |
| } |
| }) |
| } |
| |
| if ctx.Failed() { |
| return |
| } |
| |
| toolFiles := ctx.ExpandSources(g.properties.Tool_files, nil) |
| for _, tool := range toolFiles { |
| g.deps = append(g.deps, tool) |
| if _, exists := tools[tool.Rel()]; !exists { |
| tools[tool.Rel()] = tool |
| } else { |
| ctx.ModuleErrorf("multiple tools for %q, %q and %q", tool, tools[tool.Rel()], tool.Rel()) |
| } |
| } |
| |
| referencedDepfile := false |
| |
| srcFiles := ctx.ExpandSources(g.properties.Srcs, nil) |
| task := g.taskGenerator(ctx, String(g.properties.Cmd), srcFiles) |
| |
| rawCommand, err := android.Expand(task.cmd, func(name string) (string, error) { |
| // report the error directly without returning an error to android.Expand to catch multiple errors in a |
| // single run |
| reportError := func(fmt string, args ...interface{}) (string, error) { |
| ctx.PropertyErrorf("cmd", fmt, args...) |
| return "SOONG_ERROR", nil |
| } |
| |
| switch name { |
| case "location": |
| if len(g.properties.Tools) == 0 && len(toolFiles) == 0 { |
| return reportError("at least one `tools` or `tool_files` is required if $(location) is used") |
| } else if len(g.properties.Tools) > 0 { |
| return tools[g.properties.Tools[0]].String(), nil |
| } else { |
| return tools[toolFiles[0].Rel()].String(), nil |
| } |
| case "in": |
| return "${in}", nil |
| case "out": |
| return "__SBOX_OUT_FILES__", nil |
| case "depfile": |
| referencedDepfile = true |
| if !Bool(g.properties.Depfile) { |
| return reportError("$(depfile) used without depfile property") |
| } |
| return "__SBOX_DEPFILE__", nil |
| case "genDir": |
| return "__SBOX_OUT_DIR__", nil |
| default: |
| if strings.HasPrefix(name, "location ") { |
| label := strings.TrimSpace(strings.TrimPrefix(name, "location ")) |
| if tool, ok := tools[label]; ok { |
| return tool.String(), nil |
| } else { |
| return reportError("unknown location label %q", label) |
| } |
| } |
| return reportError("unknown variable '$(%s)'", name) |
| } |
| }) |
| |
| if err != nil { |
| ctx.PropertyErrorf("cmd", "%s", err.Error()) |
| return |
| } |
| |
| if Bool(g.properties.Depfile) && !referencedDepfile { |
| ctx.PropertyErrorf("cmd", "specified depfile=true but did not include a reference to '${depfile}' in cmd") |
| } |
| |
| // tell the sbox command which directory to use as its sandbox root |
| buildDir := android.PathForOutput(ctx).String() |
| sandboxPath := shared.TempDirForOutDir(buildDir) |
| |
| // recall that Sprintf replaces percent sign expressions, whereas dollar signs expressions remain as written, |
| // to be replaced later by ninja_strings.go |
| depfilePlaceholder := "" |
| if Bool(g.properties.Depfile) { |
| depfilePlaceholder = "$depfileArgs" |
| } |
| |
| genDir := android.PathForModuleGen(ctx) |
| // Escape the command for the shell |
| rawCommand = "'" + strings.Replace(rawCommand, "'", `'\''`, -1) + "'" |
| sandboxCommand := fmt.Sprintf("$sboxCmd --sandbox-path %s --output-root %s -c %s %s $allouts", |
| sandboxPath, genDir, rawCommand, depfilePlaceholder) |
| |
| ruleParams := blueprint.RuleParams{ |
| Command: sandboxCommand, |
| CommandDeps: []string{"$sboxCmd"}, |
| } |
| args := []string{"allouts"} |
| if Bool(g.properties.Depfile) { |
| ruleParams.Deps = blueprint.DepsGCC |
| args = append(args, "depfileArgs") |
| } |
| g.rule = ctx.Rule(pctx, "generator", ruleParams, args...) |
| |
| g.generateSourceFile(ctx, task) |
| |
| } |
| |
| func (g *Module) generateSourceFile(ctx android.ModuleContext, task generateTask) { |
| desc := "generate" |
| if len(task.out) == 0 { |
| ctx.ModuleErrorf("must have at least one output file") |
| return |
| } |
| if len(task.out) == 1 { |
| desc += " " + task.out[0].Base() |
| } |
| |
| var depFile android.ModuleGenPath |
| if Bool(g.properties.Depfile) { |
| depFile = android.PathForModuleGen(ctx, task.out[0].Rel()+".d") |
| } |
| |
| params := android.BuildParams{ |
| Rule: g.rule, |
| Description: "generate", |
| Output: task.out[0], |
| ImplicitOutputs: task.out[1:], |
| Inputs: task.in, |
| Implicits: g.deps, |
| Args: map[string]string{ |
| "allouts": strings.Join(task.sandboxOuts, " "), |
| }, |
| } |
| if Bool(g.properties.Depfile) { |
| params.Depfile = android.PathForModuleGen(ctx, task.out[0].Rel()+".d") |
| params.Args["depfileArgs"] = "--depfile-out " + depFile.String() |
| } |
| |
| ctx.Build(pctx, params) |
| |
| for _, outputFile := range task.out { |
| g.outputFiles = append(g.outputFiles, outputFile) |
| } |
| g.outputDeps = append(g.outputDeps, task.out[0]) |
| } |
| |
| func generatorFactory(taskGenerator taskFunc, props ...interface{}) *Module { |
| module := &Module{ |
| taskGenerator: taskGenerator, |
| } |
| |
| module.AddProperties(props...) |
| module.AddProperties(&module.properties) |
| |
| return module |
| } |
| |
| // replace "out" with "__SBOX_OUT_DIR__/<the value of ${out}>" |
| func pathToSandboxOut(path android.Path, genDir android.Path) string { |
| relOut, err := filepath.Rel(genDir.String(), path.String()) |
| if err != nil { |
| panic(fmt.Sprintf("Could not make ${out} relative: %v", err)) |
| } |
| return filepath.Join("__SBOX_OUT_DIR__", relOut) |
| |
| } |
| |
| func NewGenSrcs() *Module { |
| properties := &genSrcsProperties{} |
| |
| taskGenerator := func(ctx android.ModuleContext, rawCommand string, srcFiles android.Paths) generateTask { |
| commands := []string{} |
| outFiles := android.WritablePaths{} |
| genDir := android.PathForModuleGen(ctx) |
| sandboxOuts := []string{} |
| for _, in := range srcFiles { |
| outFile := android.GenPathWithExt(ctx, "", in, String(properties.Output_extension)) |
| outFiles = append(outFiles, outFile) |
| |
| sandboxOutfile := pathToSandboxOut(outFile, genDir) |
| sandboxOuts = append(sandboxOuts, sandboxOutfile) |
| |
| command, err := android.Expand(rawCommand, func(name string) (string, error) { |
| switch name { |
| case "in": |
| return in.String(), nil |
| case "out": |
| return sandboxOutfile, nil |
| default: |
| return "$(" + name + ")", nil |
| } |
| }) |
| if err != nil { |
| ctx.PropertyErrorf("cmd", err.Error()) |
| } |
| |
| // escape the command in case for example it contains '#', an odd number of '"', etc |
| command = fmt.Sprintf("bash -c %v", proptools.ShellEscape([]string{command})[0]) |
| commands = append(commands, command) |
| } |
| fullCommand := strings.Join(commands, " && ") |
| |
| return generateTask{ |
| in: srcFiles, |
| out: outFiles, |
| sandboxOuts: sandboxOuts, |
| cmd: fullCommand, |
| } |
| } |
| |
| return generatorFactory(taskGenerator, properties) |
| } |
| |
| func GenSrcsFactory() android.Module { |
| m := NewGenSrcs() |
| android.InitAndroidModule(m) |
| return m |
| } |
| |
| type genSrcsProperties struct { |
| // extension that will be substituted for each output file |
| Output_extension *string |
| } |
| |
| func NewGenRule() *Module { |
| properties := &genRuleProperties{} |
| |
| taskGenerator := func(ctx android.ModuleContext, rawCommand string, srcFiles android.Paths) generateTask { |
| outs := make(android.WritablePaths, len(properties.Out)) |
| sandboxOuts := make([]string, len(properties.Out)) |
| genDir := android.PathForModuleGen(ctx) |
| for i, out := range properties.Out { |
| outs[i] = android.PathForModuleGen(ctx, out) |
| sandboxOuts[i] = pathToSandboxOut(outs[i], genDir) |
| } |
| return generateTask{ |
| in: srcFiles, |
| out: outs, |
| sandboxOuts: sandboxOuts, |
| cmd: rawCommand, |
| } |
| } |
| |
| return generatorFactory(taskGenerator, properties) |
| } |
| |
| func GenRuleFactory() android.Module { |
| m := NewGenRule() |
| android.InitAndroidModule(m) |
| return m |
| } |
| |
| type genRuleProperties struct { |
| // names of the output files that will be generated |
| Out []string |
| } |
| |
| var Bool = proptools.Bool |
| var String = proptools.String |