35143d0466
genrules lost the ability to depend on Go tools after I05e945f38915d49cd3c0ab72a86576949bc7eff2 which converted VisitDirectDeps from blueprint Modules to android Modules. Add VisitDirectDepsBlueprint to visit all modules including blueprint Modules, and use it in genrule. Also add a check for disabled modules that was being handled by VisitDirectDeps. Test: m checkbuild Change-Id: I65724283166c63596d071e598c08fed87ef32896
434 lines
12 KiB
Go
434 lines
12 KiB
Go
// 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
|
|
}
|
|
|
|
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
|
|
}
|
|
|
|
type taskFunc func(ctx android.ModuleContext, rawCommand string, srcFiles android.Paths) generateTask
|
|
|
|
type generateTask struct {
|
|
in android.Paths
|
|
out android.WritablePaths
|
|
cmd string
|
|
}
|
|
|
|
func (g *Module) GeneratedSourceFiles() android.Paths {
|
|
return g.outputFiles
|
|
}
|
|
|
|
func (g *Module) Srcs() android.Paths {
|
|
return g.outputFiles
|
|
}
|
|
|
|
func (g *Module) GeneratedHeaderDirs() android.Paths {
|
|
return g.exportedIncludeDirs
|
|
}
|
|
|
|
func (g *Module) DepsMutator(ctx android.BottomUpMutatorContext) {
|
|
android.ExtractSourcesDeps(ctx, g.properties.Srcs)
|
|
if g, ok := ctx.Module().(*Module); ok {
|
|
if len(g.properties.Tools) > 0 {
|
|
ctx.AddFarVariationDependencies([]blueprint.Variation{
|
|
{"arch", ctx.AConfig().BuildOsVariant},
|
|
}, hostToolDepTag, g.properties.Tools...)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (g *Module) GenerateAndroidBuildActions(ctx android.ModuleContext) {
|
|
if len(g.properties.Tools) == 0 && len(g.properties.Tool_files) == 0 {
|
|
ctx.ModuleErrorf("at least one `tools` or `tool_files` is required")
|
|
return
|
|
}
|
|
|
|
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 android.SourceDepTag:
|
|
// Nothing to do
|
|
case hostToolDepTag:
|
|
tool := ctx.OtherModuleName(module)
|
|
var path android.OptionalPath
|
|
|
|
if t, ok := module.(HostToolProvider); ok {
|
|
if !t.(android.Module).Enabled() {
|
|
if ctx.AConfig().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)
|
|
}
|
|
default:
|
|
ctx.ModuleErrorf("unknown dependency on %q", ctx.OtherModuleName(module))
|
|
}
|
|
})
|
|
}
|
|
|
|
if ctx.Failed() {
|
|
return
|
|
}
|
|
|
|
for _, tool := range g.properties.Tool_files {
|
|
toolPath := android.PathForModuleSrc(ctx, tool)
|
|
g.deps = append(g.deps, toolPath)
|
|
if _, exists := tools[tool]; !exists {
|
|
tools[tool] = toolPath
|
|
} else {
|
|
ctx.ModuleErrorf("multiple tools for %q, %q and %q", tool, tools[tool], toolPath.String())
|
|
}
|
|
}
|
|
|
|
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) {
|
|
switch name {
|
|
case "location":
|
|
if len(g.properties.Tools) > 0 {
|
|
return tools[g.properties.Tools[0]].String(), nil
|
|
} else {
|
|
return tools[g.properties.Tool_files[0]].String(), nil
|
|
}
|
|
case "in":
|
|
return "${in}", nil
|
|
case "out":
|
|
return "__SBOX_OUT_FILES__", nil
|
|
case "depfile":
|
|
referencedDepfile = true
|
|
if !Bool(g.properties.Depfile) {
|
|
return "", fmt.Errorf("$(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 "", fmt.Errorf("unknown location label %q", label)
|
|
}
|
|
}
|
|
return "", fmt.Errorf("unknown variable '$(%s)'", name)
|
|
}
|
|
})
|
|
|
|
if Bool(g.properties.Depfile) && !referencedDepfile {
|
|
ctx.PropertyErrorf("cmd", "specified depfile=true but did not include a reference to '${depfile}' in cmd")
|
|
}
|
|
|
|
if err != nil {
|
|
ctx.PropertyErrorf("cmd", "%s", err.Error())
|
|
return
|
|
}
|
|
|
|
// 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)
|
|
sandboxCommand := fmt.Sprintf("$sboxCmd --sandbox-path %s --output-root %s -c %q %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.out.Strings(), " "),
|
|
},
|
|
}
|
|
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)
|
|
}
|
|
}
|
|
|
|
func generatorFactory(taskGenerator taskFunc, props ...interface{}) *Module {
|
|
module := &Module{
|
|
taskGenerator: taskGenerator,
|
|
}
|
|
|
|
module.AddProperties(props...)
|
|
module.AddProperties(&module.properties)
|
|
|
|
return module
|
|
}
|
|
|
|
func NewGenSrcs() *Module {
|
|
properties := &genSrcsProperties{}
|
|
|
|
taskGenerator := func(ctx android.ModuleContext, rawCommand string, srcFiles android.Paths) generateTask {
|
|
commands := []string{}
|
|
outFiles := android.WritablePaths{}
|
|
genPath := android.PathForModuleGen(ctx).String()
|
|
for _, in := range srcFiles {
|
|
outFile := android.GenPathWithExt(ctx, "", in, String(properties.Output_extension))
|
|
outFiles = append(outFiles, outFile)
|
|
|
|
// replace "out" with "__SBOX_OUT_DIR__/<the value of ${out}>"
|
|
relOut, err := filepath.Rel(genPath, outFile.String())
|
|
if err != nil {
|
|
panic(fmt.Sprintf("Could not make ${out} relative: %v", err))
|
|
}
|
|
sandboxOutfile := filepath.Join("__SBOX_OUT_DIR__", relOut)
|
|
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,
|
|
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))
|
|
for i, out := range properties.Out {
|
|
outs[i] = android.PathForModuleGen(ctx, out)
|
|
}
|
|
return generateTask{
|
|
in: srcFiles,
|
|
out: outs,
|
|
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
|