2015-03-19 04:28:46 +08:00
|
|
|
// 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 (
|
2016-11-05 06:32:58 +08:00
|
|
|
"fmt"
|
|
|
|
"strings"
|
2016-09-29 07:19:10 +08:00
|
|
|
|
2015-03-24 03:57:34 +08:00
|
|
|
"github.com/google/blueprint"
|
2015-03-19 04:28:46 +08:00
|
|
|
|
2016-05-19 06:37:25 +08:00
|
|
|
"android/soong/android"
|
2015-03-19 04:28:46 +08:00
|
|
|
)
|
|
|
|
|
2015-06-18 05:20:06 +08:00
|
|
|
func init() {
|
2016-10-13 05:28:16 +08:00
|
|
|
android.RegisterModuleType("gensrcs", GenSrcsFactory)
|
|
|
|
android.RegisterModuleType("genrule", GenRuleFactory)
|
2015-06-18 05:20:06 +08:00
|
|
|
}
|
|
|
|
|
2015-03-19 04:28:46 +08:00
|
|
|
var (
|
2016-05-19 06:37:25 +08:00
|
|
|
pctx = android.NewPackageContext("android/soong/genrule")
|
2015-03-19 04:28:46 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
type SourceFileGenerator interface {
|
2016-05-19 06:37:25 +08:00
|
|
|
GeneratedSourceFiles() android.Paths
|
|
|
|
GeneratedHeaderDir() android.Path
|
2015-03-19 04:28:46 +08:00
|
|
|
}
|
|
|
|
|
2015-04-29 04:25:36 +08:00
|
|
|
type HostToolProvider interface {
|
2016-05-19 06:37:25 +08:00
|
|
|
HostToolPath() android.OptionalPath
|
2015-04-29 04:25:36 +08:00
|
|
|
}
|
2015-03-19 04:28:46 +08:00
|
|
|
|
2015-05-12 04:39:40 +08:00
|
|
|
type generatorProperties struct {
|
|
|
|
// command to run on one or more input files. Available variables for substitution:
|
2016-11-05 06:32:58 +08:00
|
|
|
// $(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
|
|
|
|
// $(genDir): the sandbox directory for this tool; contains $(out)
|
|
|
|
// $$: a literal $
|
|
|
|
//
|
|
|
|
// DO NOT directly reference paths to files in the source tree, or the
|
|
|
|
// command will be missing proper dependencies to re-run if the files
|
|
|
|
// change.
|
2015-05-12 04:39:40 +08:00
|
|
|
Cmd string
|
|
|
|
|
2016-11-05 06:32:58 +08:00
|
|
|
// name of the modules (if any) that produces the host executable. Leave empty for
|
2015-05-12 04:39:40 +08:00
|
|
|
// prebuilts or scripts that do not need a module to build them.
|
2016-11-05 06:32:58 +08:00
|
|
|
Tools []string
|
2016-04-21 05:54:32 +08:00
|
|
|
|
|
|
|
// Local file that is used as the tool
|
2016-11-05 06:32:58 +08:00
|
|
|
Tool_files []string
|
2015-05-12 04:39:40 +08:00
|
|
|
}
|
|
|
|
|
2015-04-29 04:25:36 +08:00
|
|
|
type generator struct {
|
2016-05-19 06:37:25 +08:00
|
|
|
android.ModuleBase
|
2015-03-19 04:28:46 +08:00
|
|
|
|
2015-05-12 04:39:40 +08:00
|
|
|
properties generatorProperties
|
2015-04-29 04:25:36 +08:00
|
|
|
|
|
|
|
tasks taskFunc
|
|
|
|
|
2016-05-19 06:37:25 +08:00
|
|
|
deps android.Paths
|
2015-04-29 04:25:36 +08:00
|
|
|
rule blueprint.Rule
|
|
|
|
|
2016-05-19 06:37:25 +08:00
|
|
|
genPath android.Path
|
2016-04-21 05:21:14 +08:00
|
|
|
|
2016-05-19 06:37:25 +08:00
|
|
|
outputFiles android.Paths
|
2015-03-19 04:28:46 +08:00
|
|
|
}
|
|
|
|
|
2016-05-19 06:37:25 +08:00
|
|
|
type taskFunc func(ctx android.ModuleContext) []generateTask
|
2015-03-19 04:28:46 +08:00
|
|
|
|
2015-04-29 04:25:36 +08:00
|
|
|
type generateTask struct {
|
2016-05-19 06:37:25 +08:00
|
|
|
in android.Paths
|
2016-09-29 07:21:00 +08:00
|
|
|
out android.WritablePaths
|
2015-03-19 04:28:46 +08:00
|
|
|
}
|
|
|
|
|
2016-05-19 06:37:25 +08:00
|
|
|
func (g *generator) GeneratedSourceFiles() android.Paths {
|
2015-04-29 04:25:36 +08:00
|
|
|
return g.outputFiles
|
|
|
|
}
|
2015-03-19 04:28:46 +08:00
|
|
|
|
2016-05-19 06:37:25 +08:00
|
|
|
func (g *generator) GeneratedHeaderDir() android.Path {
|
2016-04-21 05:21:14 +08:00
|
|
|
return g.genPath
|
|
|
|
}
|
|
|
|
|
2016-10-13 05:38:15 +08:00
|
|
|
func (g *generator) DepsMutator(ctx android.BottomUpMutatorContext) {
|
2015-10-30 06:25:03 +08:00
|
|
|
if g, ok := ctx.Module().(*generator); ok {
|
2016-11-05 06:32:58 +08:00
|
|
|
if len(g.properties.Tools) > 0 {
|
2015-11-25 09:53:15 +08:00
|
|
|
ctx.AddFarVariationDependencies([]blueprint.Variation{
|
2016-06-02 08:09:44 +08:00
|
|
|
{"arch", ctx.AConfig().BuildOsVariant},
|
2016-11-05 06:32:58 +08:00
|
|
|
}, nil, g.properties.Tools...)
|
2015-10-30 06:25:03 +08:00
|
|
|
}
|
2015-04-29 04:25:36 +08:00
|
|
|
}
|
2015-03-19 04:28:46 +08:00
|
|
|
}
|
|
|
|
|
2016-05-19 06:37:25 +08:00
|
|
|
func (g *generator) GenerateAndroidBuildActions(ctx android.ModuleContext) {
|
2016-11-05 06:32:58 +08:00
|
|
|
if len(g.properties.Tools) == 0 && len(g.properties.Tool_files) == 0 {
|
|
|
|
ctx.ModuleErrorf("at least one `tools` or `tool_files` is required")
|
2016-04-21 05:54:32 +08:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-09-29 07:19:10 +08:00
|
|
|
g.genPath = android.PathForModuleGen(ctx, "")
|
|
|
|
|
2016-11-05 06:32:58 +08:00
|
|
|
tools := map[string]android.Path{}
|
2016-09-29 07:19:10 +08:00
|
|
|
|
2016-11-05 06:32:58 +08:00
|
|
|
if len(g.properties.Tools) > 0 {
|
2016-04-21 05:54:32 +08:00
|
|
|
ctx.VisitDirectDeps(func(module blueprint.Module) {
|
|
|
|
if t, ok := module.(HostToolProvider); ok {
|
|
|
|
p := t.HostToolPath()
|
|
|
|
if p.Valid() {
|
|
|
|
g.deps = append(g.deps, p.Path())
|
2016-11-05 06:32:58 +08:00
|
|
|
tool := ctx.OtherModuleName(module)
|
|
|
|
if _, exists := tools[tool]; !exists {
|
|
|
|
tools[tool] = p.Path()
|
|
|
|
} else {
|
|
|
|
ctx.ModuleErrorf("multiple tools for %q, %q and %q", tool, tools[tool], p.Path().String())
|
|
|
|
}
|
2016-04-21 05:54:32 +08:00
|
|
|
} else {
|
|
|
|
ctx.ModuleErrorf("host tool %q missing output file", ctx.OtherModuleName(module))
|
|
|
|
}
|
2015-04-29 04:25:36 +08:00
|
|
|
} else {
|
2016-04-21 05:54:32 +08:00
|
|
|
ctx.ModuleErrorf("unknown dependency %q", ctx.OtherModuleName(module))
|
2015-04-29 04:25:36 +08:00
|
|
|
}
|
2016-04-21 05:54:32 +08:00
|
|
|
})
|
|
|
|
}
|
2015-03-19 04:28:46 +08:00
|
|
|
|
2016-11-05 06:32:58 +08:00
|
|
|
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())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cmd, err := android.Expand(g.properties.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 "${out}", nil
|
|
|
|
case "genDir":
|
|
|
|
return g.genPath.String(), 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 err != nil {
|
|
|
|
ctx.PropertyErrorf("cmd", "%s", err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
g.rule = ctx.Rule(pctx, "generator", blueprint.RuleParams{
|
|
|
|
Command: cmd,
|
|
|
|
})
|
|
|
|
|
2015-04-29 04:25:36 +08:00
|
|
|
for _, task := range g.tasks(ctx) {
|
2016-11-05 06:32:58 +08:00
|
|
|
g.generateSourceFile(ctx, task)
|
2015-04-29 04:25:36 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-05 06:32:58 +08:00
|
|
|
func (g *generator) generateSourceFile(ctx android.ModuleContext, task generateTask) {
|
2016-05-19 06:37:25 +08:00
|
|
|
ctx.ModuleBuild(pctx, android.ModuleBuildParams{
|
2015-04-29 04:25:36 +08:00
|
|
|
Rule: g.rule,
|
2016-09-29 07:21:00 +08:00
|
|
|
Outputs: task.out,
|
2015-04-29 04:25:36 +08:00
|
|
|
Inputs: task.in,
|
|
|
|
Implicits: g.deps,
|
|
|
|
})
|
2015-03-19 04:28:46 +08:00
|
|
|
|
2016-09-29 07:21:00 +08:00
|
|
|
for _, outputFile := range task.out {
|
|
|
|
g.outputFiles = append(g.outputFiles, outputFile)
|
|
|
|
}
|
2015-04-29 04:25:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func generatorFactory(tasks taskFunc, props ...interface{}) (blueprint.Module, []interface{}) {
|
|
|
|
module := &generator{
|
|
|
|
tasks: tasks,
|
2015-03-19 04:28:46 +08:00
|
|
|
}
|
2015-04-29 04:25:36 +08:00
|
|
|
|
|
|
|
props = append(props, &module.properties)
|
|
|
|
|
2016-05-19 06:37:25 +08:00
|
|
|
return android.InitAndroidModule(module, props...)
|
2015-03-19 04:28:46 +08:00
|
|
|
}
|
|
|
|
|
2015-04-29 04:25:36 +08:00
|
|
|
func GenSrcsFactory() (blueprint.Module, []interface{}) {
|
|
|
|
properties := &genSrcsProperties{}
|
|
|
|
|
2016-05-19 06:37:25 +08:00
|
|
|
tasks := func(ctx android.ModuleContext) []generateTask {
|
2015-07-01 09:15:24 +08:00
|
|
|
srcFiles := ctx.ExpandSources(properties.Srcs, nil)
|
2015-04-29 04:25:36 +08:00
|
|
|
tasks := make([]generateTask, 0, len(srcFiles))
|
|
|
|
for _, in := range srcFiles {
|
2015-09-24 06:26:20 +08:00
|
|
|
tasks = append(tasks, generateTask{
|
2016-05-19 06:37:25 +08:00
|
|
|
in: android.Paths{in},
|
2016-11-03 11:43:13 +08:00
|
|
|
out: android.WritablePaths{android.GenPathWithExt(ctx, "", in, properties.Output_extension)},
|
2015-09-24 06:26:20 +08:00
|
|
|
})
|
2015-04-29 04:25:36 +08:00
|
|
|
}
|
|
|
|
return tasks
|
|
|
|
}
|
2015-03-19 04:28:46 +08:00
|
|
|
|
2015-04-29 04:25:36 +08:00
|
|
|
return generatorFactory(tasks, properties)
|
|
|
|
}
|
|
|
|
|
|
|
|
type genSrcsProperties struct {
|
2015-05-12 04:39:40 +08:00
|
|
|
// list of input files
|
2015-04-29 04:25:36 +08:00
|
|
|
Srcs []string
|
|
|
|
|
2015-05-12 04:39:40 +08:00
|
|
|
// extension that will be substituted for each output file
|
2015-04-29 04:25:36 +08:00
|
|
|
Output_extension string
|
|
|
|
}
|
|
|
|
|
|
|
|
func GenRuleFactory() (blueprint.Module, []interface{}) {
|
|
|
|
properties := &genRuleProperties{}
|
|
|
|
|
2016-05-19 06:37:25 +08:00
|
|
|
tasks := func(ctx android.ModuleContext) []generateTask {
|
2016-09-29 07:21:00 +08:00
|
|
|
outs := make(android.WritablePaths, len(properties.Out))
|
|
|
|
for i, out := range properties.Out {
|
|
|
|
outs[i] = android.PathForModuleGen(ctx, out)
|
|
|
|
}
|
2015-04-29 04:25:36 +08:00
|
|
|
return []generateTask{
|
|
|
|
{
|
2015-07-01 09:15:24 +08:00
|
|
|
in: ctx.ExpandSources(properties.Srcs, nil),
|
2016-09-29 07:21:00 +08:00
|
|
|
out: outs,
|
2015-04-29 04:25:36 +08:00
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return generatorFactory(tasks, properties)
|
|
|
|
}
|
|
|
|
|
|
|
|
type genRuleProperties struct {
|
2015-05-12 04:39:40 +08:00
|
|
|
// list of input files
|
2015-04-29 04:25:36 +08:00
|
|
|
Srcs []string
|
|
|
|
|
2016-09-29 07:21:00 +08:00
|
|
|
// names of the output files that will be generated
|
|
|
|
Out []string
|
2015-03-19 04:28:46 +08:00
|
|
|
}
|