Add depfile support to RuleBuilder
Allow rules built with RuleBuilder to use depfiles. Ninja only supports a single depfile with single output. If there are multiple outputs in a rule, move all but the first to implicit outputs. If multiple depfiles are specified, use new support in dep_fixer to combine additional depfiles into the first depfile. Test: rule_builder_test.go Change-Id: I7dd4ba7fdf9feaf89b3dd2b7abb0e79006e06018
This commit is contained in:
parent
92b7d584c8
commit
1d2cf0494a
|
@ -171,6 +171,20 @@ func (r *RuleBuilder) Outputs() WritablePaths {
|
||||||
return outputList
|
return outputList
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// DepFiles returns the list of paths that were passed to the RuleBuilderCommand methods that take depfile paths, such
|
||||||
|
// as RuleBuilderCommand.DepFile or RuleBuilderCommand.FlagWithDepFile.
|
||||||
|
func (r *RuleBuilder) DepFiles() WritablePaths {
|
||||||
|
var depFiles WritablePaths
|
||||||
|
|
||||||
|
for _, c := range r.commands {
|
||||||
|
for _, depFile := range c.depFiles {
|
||||||
|
depFiles = append(depFiles, depFile)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return depFiles
|
||||||
|
}
|
||||||
|
|
||||||
// Installs returns the list of tuples passed to Install.
|
// Installs returns the list of tuples passed to Install.
|
||||||
func (r *RuleBuilder) Installs() RuleBuilderInstalls {
|
func (r *RuleBuilder) Installs() RuleBuilderInstalls {
|
||||||
return append(RuleBuilderInstalls(nil), r.installs...)
|
return append(RuleBuilderInstalls(nil), r.installs...)
|
||||||
|
@ -222,9 +236,17 @@ type BuilderContext interface {
|
||||||
var _ BuilderContext = ModuleContext(nil)
|
var _ BuilderContext = ModuleContext(nil)
|
||||||
var _ BuilderContext = SingletonContext(nil)
|
var _ BuilderContext = SingletonContext(nil)
|
||||||
|
|
||||||
|
func (r *RuleBuilder) depFileMergerCmd(ctx PathContext, depFiles WritablePaths) *RuleBuilderCommand {
|
||||||
|
return (&RuleBuilderCommand{}).
|
||||||
|
Tool(ctx.Config().HostToolPath(ctx, "dep_fixer")).
|
||||||
|
Flags(depFiles.Strings())
|
||||||
|
}
|
||||||
|
|
||||||
// Build adds the built command line to the build graph, with dependencies on Inputs and Tools, and output files for
|
// Build adds the built command line to the build graph, with dependencies on Inputs and Tools, and output files for
|
||||||
// Outputs.
|
// Outputs.
|
||||||
func (r *RuleBuilder) Build(pctx PackageContext, ctx BuilderContext, name string, desc string) {
|
func (r *RuleBuilder) Build(pctx PackageContext, ctx BuilderContext, name string, desc string) {
|
||||||
|
name = ninjaNameEscape(name)
|
||||||
|
|
||||||
if len(r.missingDeps) > 0 {
|
if len(r.missingDeps) > 0 {
|
||||||
ctx.Build(pctx, BuildParams{
|
ctx.Build(pctx, BuildParams{
|
||||||
Rule: ErrorRule,
|
Rule: ErrorRule,
|
||||||
|
@ -237,16 +259,45 @@ func (r *RuleBuilder) Build(pctx PackageContext, ctx BuilderContext, name string
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(r.Commands()) > 0 {
|
tools := r.Tools()
|
||||||
|
commands := r.Commands()
|
||||||
|
|
||||||
|
var depFile WritablePath
|
||||||
|
var depFormat blueprint.Deps
|
||||||
|
if depFiles := r.DepFiles(); len(depFiles) > 0 {
|
||||||
|
depFile = depFiles[0]
|
||||||
|
depFormat = blueprint.DepsGCC
|
||||||
|
if len(depFiles) > 1 {
|
||||||
|
// Add a command locally that merges all depfiles together into the first depfile.
|
||||||
|
cmd := r.depFileMergerCmd(ctx, depFiles)
|
||||||
|
commands = append(commands, string(cmd.buf))
|
||||||
|
tools = append(tools, cmd.tools...)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ninja doesn't like multiple outputs when depfiles are enabled, move all but the first output to
|
||||||
|
// ImplicitOutputs. RuleBuilder never uses "$out", so the distinction between Outputs and ImplicitOutputs
|
||||||
|
// doesn't matter.
|
||||||
|
var output WritablePath
|
||||||
|
var implicitOutputs WritablePaths
|
||||||
|
if outputs := r.Outputs(); len(outputs) > 0 {
|
||||||
|
output = outputs[0]
|
||||||
|
implicitOutputs = outputs[1:]
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(commands) > 0 {
|
||||||
ctx.Build(pctx, BuildParams{
|
ctx.Build(pctx, BuildParams{
|
||||||
Rule: ctx.Rule(pctx, name, blueprint.RuleParams{
|
Rule: ctx.Rule(pctx, name, blueprint.RuleParams{
|
||||||
Command: strings.Join(proptools.NinjaEscapeList(r.Commands()), " && "),
|
Command: strings.Join(proptools.NinjaEscapeList(commands), " && "),
|
||||||
CommandDeps: r.Tools().Strings(),
|
CommandDeps: tools.Strings(),
|
||||||
Restat: r.restat,
|
Restat: r.restat,
|
||||||
}),
|
}),
|
||||||
Implicits: r.Inputs(),
|
Implicits: r.Inputs(),
|
||||||
Outputs: r.Outputs(),
|
Output: output,
|
||||||
Description: desc,
|
ImplicitOutputs: implicitOutputs,
|
||||||
|
Depfile: depFile,
|
||||||
|
Deps: depFormat,
|
||||||
|
Description: desc,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -256,10 +307,11 @@ func (r *RuleBuilder) Build(pctx PackageContext, ctx BuilderContext, name string
|
||||||
// RuleBuilderCommand, so they can be used chained or unchained. All methods that add text implicitly add a single
|
// RuleBuilderCommand, so they can be used chained or unchained. All methods that add text implicitly add a single
|
||||||
// space as a separator from the previous method.
|
// space as a separator from the previous method.
|
||||||
type RuleBuilderCommand struct {
|
type RuleBuilderCommand struct {
|
||||||
buf []byte
|
buf []byte
|
||||||
inputs Paths
|
inputs Paths
|
||||||
outputs WritablePaths
|
outputs WritablePaths
|
||||||
tools Paths
|
depFiles WritablePaths
|
||||||
|
tools Paths
|
||||||
}
|
}
|
||||||
|
|
||||||
// Text adds the specified raw text to the command line. The text should not contain input or output paths or the
|
// Text adds the specified raw text to the command line. The text should not contain input or output paths or the
|
||||||
|
@ -369,6 +421,14 @@ func (c *RuleBuilderCommand) Outputs(paths WritablePaths) *RuleBuilderCommand {
|
||||||
return c
|
return c
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// DepFile adds the specified depfile path to the paths returned by RuleBuilder.DepFiles and adds it to the command
|
||||||
|
// line, and causes RuleBuilder.Build file to set the depfile flag for ninja. If multiple depfiles are added to
|
||||||
|
// commands in a single RuleBuilder then RuleBuilder.Build will add an extra command to merge the depfiles together.
|
||||||
|
func (c *RuleBuilderCommand) DepFile(path WritablePath) *RuleBuilderCommand {
|
||||||
|
c.depFiles = append(c.depFiles, path)
|
||||||
|
return c.Text(path.String())
|
||||||
|
}
|
||||||
|
|
||||||
// ImplicitOutput adds the specified output path to the dependencies returned by RuleBuilder.Outputs without modifying
|
// ImplicitOutput adds the specified output path to the dependencies returned by RuleBuilder.Outputs without modifying
|
||||||
// the command line.
|
// the command line.
|
||||||
func (c *RuleBuilderCommand) ImplicitOutput(path WritablePath) *RuleBuilderCommand {
|
func (c *RuleBuilderCommand) ImplicitOutput(path WritablePath) *RuleBuilderCommand {
|
||||||
|
@ -383,6 +443,15 @@ func (c *RuleBuilderCommand) ImplicitOutputs(paths WritablePaths) *RuleBuilderCo
|
||||||
return c
|
return c
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ImplicitDepFile adds the specified depfile path to the paths returned by RuleBuilder.DepFiles without modifying
|
||||||
|
// the command line, and causes RuleBuilder.Build file to set the depfile flag for ninja. If multiple depfiles
|
||||||
|
// are added to commands in a single RuleBuilder then RuleBuilder.Build will add an extra command to merge the
|
||||||
|
// depfiles together.
|
||||||
|
func (c *RuleBuilderCommand) ImplicitDepFile(path WritablePath) *RuleBuilderCommand {
|
||||||
|
c.depFiles = append(c.depFiles, path)
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
|
||||||
// FlagWithInput adds the specified flag and input path to the command line, with no separator between them. The path
|
// FlagWithInput adds the specified flag and input path to the command line, with no separator between them. The path
|
||||||
// will also be added to the dependencies returned by RuleBuilder.Inputs.
|
// will also be added to the dependencies returned by RuleBuilder.Inputs.
|
||||||
func (c *RuleBuilderCommand) FlagWithInput(flag string, path Path) *RuleBuilderCommand {
|
func (c *RuleBuilderCommand) FlagWithInput(flag string, path Path) *RuleBuilderCommand {
|
||||||
|
@ -415,7 +484,35 @@ func (c *RuleBuilderCommand) FlagWithOutput(flag string, path WritablePath) *Rul
|
||||||
return c.Text(flag + path.String())
|
return c.Text(flag + path.String())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// FlagWithDepFile adds the specified flag and depfile path to the command line, with no separator between them. The path
|
||||||
|
// will also be added to the outputs returned by RuleBuilder.Outputs.
|
||||||
|
func (c *RuleBuilderCommand) FlagWithDepFile(flag string, path WritablePath) *RuleBuilderCommand {
|
||||||
|
c.depFiles = append(c.depFiles, path)
|
||||||
|
return c.Text(flag + path.String())
|
||||||
|
}
|
||||||
|
|
||||||
// String returns the command line.
|
// String returns the command line.
|
||||||
func (c *RuleBuilderCommand) String() string {
|
func (c *RuleBuilderCommand) String() string {
|
||||||
return string(c.buf)
|
return string(c.buf)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func ninjaNameEscape(s string) string {
|
||||||
|
b := []byte(s)
|
||||||
|
escaped := false
|
||||||
|
for i, c := range b {
|
||||||
|
valid := (c >= 'a' && c <= 'z') ||
|
||||||
|
(c >= 'A' && c <= 'Z') ||
|
||||||
|
(c >= '0' && c <= '9') ||
|
||||||
|
(c == '_') ||
|
||||||
|
(c == '-') ||
|
||||||
|
(c == '.')
|
||||||
|
if !valid {
|
||||||
|
b[i] = '_'
|
||||||
|
escaped = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if escaped {
|
||||||
|
s = string(b)
|
||||||
|
}
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
|
|
@ -237,23 +237,27 @@ func TestRuleBuilder(t *testing.T) {
|
||||||
rule := NewRuleBuilder()
|
rule := NewRuleBuilder()
|
||||||
|
|
||||||
fs := map[string][]byte{
|
fs := map[string][]byte{
|
||||||
"input": nil,
|
"dep_fixer": nil,
|
||||||
"Implicit": nil,
|
"input": nil,
|
||||||
"Input": nil,
|
"Implicit": nil,
|
||||||
"Tool": nil,
|
"Input": nil,
|
||||||
"input2": nil,
|
"Tool": nil,
|
||||||
"tool2": nil,
|
"input2": nil,
|
||||||
"input3": nil,
|
"tool2": nil,
|
||||||
|
"input3": nil,
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx := PathContextForTesting(TestConfig("out", nil), fs)
|
ctx := PathContextForTesting(TestConfig("out", nil), fs)
|
||||||
|
|
||||||
cmd := rule.Command().
|
cmd := rule.Command().
|
||||||
|
DepFile(PathForOutput(ctx, "DepFile")).
|
||||||
Flag("Flag").
|
Flag("Flag").
|
||||||
FlagWithArg("FlagWithArg=", "arg").
|
FlagWithArg("FlagWithArg=", "arg").
|
||||||
|
FlagWithDepFile("FlagWithDepFile=", PathForOutput(ctx, "depfile")).
|
||||||
FlagWithInput("FlagWithInput=", PathForSource(ctx, "input")).
|
FlagWithInput("FlagWithInput=", PathForSource(ctx, "input")).
|
||||||
FlagWithOutput("FlagWithOutput=", PathForOutput(ctx, "output")).
|
FlagWithOutput("FlagWithOutput=", PathForOutput(ctx, "output")).
|
||||||
Implicit(PathForSource(ctx, "Implicit")).
|
Implicit(PathForSource(ctx, "Implicit")).
|
||||||
|
ImplicitDepFile(PathForOutput(ctx, "ImplicitDepFile")).
|
||||||
ImplicitOutput(PathForOutput(ctx, "ImplicitOutput")).
|
ImplicitOutput(PathForOutput(ctx, "ImplicitOutput")).
|
||||||
Input(PathForSource(ctx, "Input")).
|
Input(PathForSource(ctx, "Input")).
|
||||||
Output(PathForOutput(ctx, "Output")).
|
Output(PathForOutput(ctx, "Output")).
|
||||||
|
@ -262,6 +266,7 @@ func TestRuleBuilder(t *testing.T) {
|
||||||
|
|
||||||
rule.Command().
|
rule.Command().
|
||||||
Text("command2").
|
Text("command2").
|
||||||
|
DepFile(PathForOutput(ctx, "depfile2")).
|
||||||
Input(PathForSource(ctx, "input2")).
|
Input(PathForSource(ctx, "input2")).
|
||||||
Output(PathForOutput(ctx, "output2")).
|
Output(PathForOutput(ctx, "output2")).
|
||||||
Tool(PathForSource(ctx, "tool2"))
|
Tool(PathForSource(ctx, "tool2"))
|
||||||
|
@ -279,25 +284,37 @@ func TestRuleBuilder(t *testing.T) {
|
||||||
Output(PathForOutput(ctx, "output3"))
|
Output(PathForOutput(ctx, "output3"))
|
||||||
|
|
||||||
wantCommands := []string{
|
wantCommands := []string{
|
||||||
"Flag FlagWithArg=arg FlagWithInput=input FlagWithOutput=out/output Input out/Output Text Tool after command2 old cmd",
|
"out/DepFile Flag FlagWithArg=arg FlagWithDepFile=out/depfile FlagWithInput=input FlagWithOutput=out/output Input out/Output Text Tool after command2 old cmd",
|
||||||
"command2 input2 out/output2 tool2",
|
"command2 out/depfile2 input2 out/output2 tool2",
|
||||||
"command3 input3 out/output2 out/output3",
|
"command3 input3 out/output2 out/output3",
|
||||||
}
|
}
|
||||||
|
|
||||||
|
wantDepMergerCommand := "out/host/" + ctx.Config().PrebuiltOS() + "/bin/dep_fixer out/DepFile out/depfile out/ImplicitDepFile out/depfile2"
|
||||||
|
|
||||||
wantInputs := PathsForSource(ctx, []string{"Implicit", "Input", "input", "input2", "input3"})
|
wantInputs := PathsForSource(ctx, []string{"Implicit", "Input", "input", "input2", "input3"})
|
||||||
wantOutputs := PathsForOutput(ctx, []string{"ImplicitOutput", "Output", "output", "output2", "output3"})
|
wantOutputs := PathsForOutput(ctx, []string{"ImplicitOutput", "Output", "output", "output2", "output3"})
|
||||||
|
wantDepFiles := PathsForOutput(ctx, []string{"DepFile", "depfile", "ImplicitDepFile", "depfile2"})
|
||||||
wantTools := PathsForSource(ctx, []string{"Tool", "tool2"})
|
wantTools := PathsForSource(ctx, []string{"Tool", "tool2"})
|
||||||
|
|
||||||
if !reflect.DeepEqual(rule.Commands(), wantCommands) {
|
if g, w := rule.Commands(), wantCommands; !reflect.DeepEqual(g, w) {
|
||||||
t.Errorf("\nwant rule.Commands() = %#v\n got %#v", wantCommands, rule.Commands())
|
t.Errorf("\nwant rule.Commands() = %#v\n got %#v", w, g)
|
||||||
}
|
}
|
||||||
if !reflect.DeepEqual(rule.Inputs(), wantInputs) {
|
|
||||||
t.Errorf("\nwant rule.Inputs() = %#v\n got %#v", wantInputs, rule.Inputs())
|
if g, w := rule.depFileMergerCmd(ctx, rule.DepFiles()).String(), wantDepMergerCommand; g != w {
|
||||||
|
t.Errorf("\nwant rule.depFileMergerCmd() = %#v\n got %#v", w, g)
|
||||||
}
|
}
|
||||||
if !reflect.DeepEqual(rule.Outputs(), wantOutputs) {
|
|
||||||
t.Errorf("\nwant rule.Outputs() = %#v\n got %#v", wantOutputs, rule.Outputs())
|
if g, w := rule.Inputs(), wantInputs; !reflect.DeepEqual(w, g) {
|
||||||
|
t.Errorf("\nwant rule.Inputs() = %#v\n got %#v", w, g)
|
||||||
}
|
}
|
||||||
if !reflect.DeepEqual(rule.Tools(), wantTools) {
|
if g, w := rule.Outputs(), wantOutputs; !reflect.DeepEqual(w, g) {
|
||||||
t.Errorf("\nwant rule.Tools() = %#v\n got %#v", wantTools, rule.Tools())
|
t.Errorf("\nwant rule.Outputs() = %#v\n got %#v", w, g)
|
||||||
|
}
|
||||||
|
if g, w := rule.DepFiles(), wantDepFiles; !reflect.DeepEqual(w, g) {
|
||||||
|
t.Errorf("\nwant rule.DepFiles() = %#v\n got %#v", w, g)
|
||||||
|
}
|
||||||
|
if g, w := rule.Tools(), wantTools; !reflect.DeepEqual(w, g) {
|
||||||
|
t.Errorf("\nwant rule.Tools() = %#v\n got %#v", w, g)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -383,8 +400,8 @@ func TestRuleBuilder_Build(t *testing.T) {
|
||||||
t.Errorf("want Implicits = [%q], got %q", "bar", params.Implicits.Strings())
|
t.Errorf("want Implicits = [%q], got %q", "bar", params.Implicits.Strings())
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(params.Outputs) != 1 || params.Outputs[0].String() != wantOutput {
|
if params.Output.String() != wantOutput {
|
||||||
t.Errorf("want Outputs = [%q], got %q", wantOutput, params.Outputs.Strings())
|
t.Errorf("want Output = %q, got %q", wantOutput, params.Output)
|
||||||
}
|
}
|
||||||
|
|
||||||
if !params.RuleParams.Restat {
|
if !params.RuleParams.Restat {
|
||||||
|
|
|
@ -196,6 +196,7 @@ func maybeBuildParamsFromOutput(provider testBuildProvider, file string) (Testin
|
||||||
var searchedOutputs []string
|
var searchedOutputs []string
|
||||||
for _, p := range provider.BuildParamsForTests() {
|
for _, p := range provider.BuildParamsForTests() {
|
||||||
outputs := append(WritablePaths(nil), p.Outputs...)
|
outputs := append(WritablePaths(nil), p.Outputs...)
|
||||||
|
outputs = append(outputs, p.ImplicitOutputs...)
|
||||||
if p.Output != nil {
|
if p.Output != nil {
|
||||||
outputs = append(outputs, p.Output)
|
outputs = append(outputs, p.Output)
|
||||||
}
|
}
|
||||||
|
@ -222,6 +223,7 @@ func allOutputs(provider testBuildProvider) []string {
|
||||||
var outputFullPaths []string
|
var outputFullPaths []string
|
||||||
for _, p := range provider.BuildParamsForTests() {
|
for _, p := range provider.BuildParamsForTests() {
|
||||||
outputs := append(WritablePaths(nil), p.Outputs...)
|
outputs := append(WritablePaths(nil), p.Outputs...)
|
||||||
|
outputs = append(outputs, p.ImplicitOutputs...)
|
||||||
if p.Output != nil {
|
if p.Output != nil {
|
||||||
outputs = append(outputs, p.Output)
|
outputs = append(outputs, p.Output)
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,30 +29,42 @@ import (
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
flag.Usage = func() {
|
flag.Usage = func() {
|
||||||
fmt.Fprintf(os.Stderr, "Usage: %s <depfile.d>", os.Args[0])
|
fmt.Fprintf(os.Stderr, "Usage: %s [-o <output>] <depfile.d> [<depfile.d>...]", os.Args[0])
|
||||||
flag.PrintDefaults()
|
flag.PrintDefaults()
|
||||||
}
|
}
|
||||||
output := flag.String("o", "", "Optional output file (defaults to rewriting source if necessary)")
|
output := flag.String("o", "", "Optional output file (defaults to rewriting source if necessary)")
|
||||||
flag.Parse()
|
flag.Parse()
|
||||||
|
|
||||||
if flag.NArg() != 1 {
|
if flag.NArg() < 1 {
|
||||||
log.Fatal("Expected a single file as an argument")
|
log.Fatal("Expected at least one input file as an argument")
|
||||||
}
|
}
|
||||||
|
|
||||||
old, err := ioutil.ReadFile(flag.Arg(0))
|
var mergedDeps *Deps
|
||||||
if err != nil {
|
var firstInput []byte
|
||||||
log.Fatalf("Error opening %q: %v", flag.Arg(0), err)
|
|
||||||
|
for i, arg := range flag.Args() {
|
||||||
|
input, err := ioutil.ReadFile(arg)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("Error opening %q: %v", arg, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
deps, err := Parse(arg, bytes.NewBuffer(append([]byte(nil), input...)))
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("Failed to parse: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if i == 0 {
|
||||||
|
mergedDeps = deps
|
||||||
|
firstInput = input
|
||||||
|
} else {
|
||||||
|
mergedDeps.Inputs = append(mergedDeps.Inputs, deps.Inputs...)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
deps, err := Parse(flag.Arg(0), bytes.NewBuffer(append([]byte(nil), old...)))
|
new := mergedDeps.Print()
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Failed to parse: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
new := deps.Print()
|
|
||||||
|
|
||||||
if *output == "" || *output == flag.Arg(0) {
|
if *output == "" || *output == flag.Arg(0) {
|
||||||
if !bytes.Equal(old, new) {
|
if !bytes.Equal(firstInput, new) {
|
||||||
err := ioutil.WriteFile(flag.Arg(0), new, 0666)
|
err := ioutil.WriteFile(flag.Arg(0), new, 0666)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to write: %v", err)
|
log.Fatalf("Failed to write: %v", err)
|
||||||
|
|
|
@ -103,7 +103,7 @@ func TestDexpreoptBootJars(t *testing.T) {
|
||||||
expectedOutputs[i] = filepath.Join(buildDir, "test_device", expectedOutputs[i])
|
expectedOutputs[i] = filepath.Join(buildDir, "test_device", expectedOutputs[i])
|
||||||
}
|
}
|
||||||
|
|
||||||
outputs := bootArt.Outputs.Strings()
|
outputs := append(android.WritablePaths{bootArt.Output}, bootArt.ImplicitOutputs...).Strings()
|
||||||
sort.Strings(outputs)
|
sort.Strings(outputs)
|
||||||
sort.Strings(expectedOutputs)
|
sort.Strings(expectedOutputs)
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue