| // Copyright 2016 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 android | 
 |  | 
 | import ( | 
 | 	"fmt" | 
 | 	"testing" | 
 |  | 
 | 	"github.com/google/blueprint" | 
 | ) | 
 |  | 
 | var prebuiltsTests = []struct { | 
 | 	name     string | 
 | 	modules  string | 
 | 	prebuilt bool | 
 | }{ | 
 | 	{ | 
 | 		name: "no prebuilt", | 
 | 		modules: ` | 
 | 			source { | 
 | 				name: "bar", | 
 | 			}`, | 
 | 		prebuilt: false, | 
 | 	}, | 
 | 	{ | 
 | 		name: "no source prebuilt not preferred", | 
 | 		modules: ` | 
 | 			prebuilt { | 
 | 				name: "bar", | 
 | 				prefer: false, | 
 | 				srcs: ["prebuilt_file"], | 
 | 			}`, | 
 | 		prebuilt: true, | 
 | 	}, | 
 | 	{ | 
 | 		name: "no source prebuilt preferred", | 
 | 		modules: ` | 
 | 			prebuilt { | 
 | 				name: "bar", | 
 | 				prefer: true, | 
 | 				srcs: ["prebuilt_file"], | 
 | 			}`, | 
 | 		prebuilt: true, | 
 | 	}, | 
 | 	{ | 
 | 		name: "prebuilt not preferred", | 
 | 		modules: ` | 
 | 			source { | 
 | 				name: "bar", | 
 | 			} | 
 | 			 | 
 | 			prebuilt { | 
 | 				name: "bar", | 
 | 				prefer: false, | 
 | 				srcs: ["prebuilt_file"], | 
 | 			}`, | 
 | 		prebuilt: false, | 
 | 	}, | 
 | 	{ | 
 | 		name: "prebuilt preferred", | 
 | 		modules: ` | 
 | 			source { | 
 | 				name: "bar", | 
 | 			} | 
 | 			 | 
 | 			prebuilt { | 
 | 				name: "bar", | 
 | 				prefer: true, | 
 | 				srcs: ["prebuilt_file"], | 
 | 			}`, | 
 | 		prebuilt: true, | 
 | 	}, | 
 | 	{ | 
 | 		name: "prebuilt no file not preferred", | 
 | 		modules: ` | 
 | 			source { | 
 | 				name: "bar", | 
 | 			} | 
 | 			 | 
 | 			prebuilt { | 
 | 				name: "bar", | 
 | 				prefer: false, | 
 | 			}`, | 
 | 		prebuilt: false, | 
 | 	}, | 
 | 	{ | 
 | 		name: "prebuilt no file preferred", | 
 | 		modules: ` | 
 | 			source { | 
 | 				name: "bar", | 
 | 			} | 
 | 			 | 
 | 			prebuilt { | 
 | 				name: "bar", | 
 | 				prefer: true, | 
 | 			}`, | 
 | 		prebuilt: false, | 
 | 	}, | 
 | 	{ | 
 | 		name: "prebuilt file from filegroup preferred", | 
 | 		modules: ` | 
 | 			filegroup { | 
 | 				name: "fg", | 
 | 				srcs: ["prebuilt_file"], | 
 | 			} | 
 | 			prebuilt { | 
 | 				name: "bar", | 
 | 				prefer: true, | 
 | 				srcs: [":fg"], | 
 | 			}`, | 
 | 		prebuilt: true, | 
 | 	}, | 
 | } | 
 |  | 
 | func TestPrebuilts(t *testing.T) { | 
 | 	fs := map[string][]byte{ | 
 | 		"prebuilt_file": nil, | 
 | 		"source_file":   nil, | 
 | 	} | 
 |  | 
 | 	for _, test := range prebuiltsTests { | 
 | 		t.Run(test.name, func(t *testing.T) { | 
 | 			bp := ` | 
 | 				source { | 
 | 					name: "foo", | 
 | 					deps: [":bar"], | 
 | 				} | 
 | 				` + test.modules | 
 | 			config := TestConfig(buildDir, nil, bp, fs) | 
 |  | 
 | 			ctx := NewTestContext() | 
 | 			registerTestPrebuiltBuildComponents(ctx) | 
 | 			ctx.RegisterModuleType("filegroup", FileGroupFactory) | 
 | 			ctx.Register(config) | 
 |  | 
 | 			_, errs := ctx.ParseBlueprintsFiles("Android.bp") | 
 | 			FailIfErrored(t, errs) | 
 | 			_, errs = ctx.PrepareBuildActions(config) | 
 | 			FailIfErrored(t, errs) | 
 |  | 
 | 			foo := ctx.ModuleForTests("foo", "") | 
 |  | 
 | 			var dependsOnSourceModule, dependsOnPrebuiltModule bool | 
 | 			ctx.VisitDirectDeps(foo.Module(), func(m blueprint.Module) { | 
 | 				if _, ok := m.(*sourceModule); ok { | 
 | 					dependsOnSourceModule = true | 
 | 				} | 
 | 				if p, ok := m.(*prebuiltModule); ok { | 
 | 					dependsOnPrebuiltModule = true | 
 | 					if !p.Prebuilt().properties.UsePrebuilt { | 
 | 						t.Errorf("dependency on prebuilt module not marked used") | 
 | 					} | 
 | 				} | 
 | 			}) | 
 |  | 
 | 			deps := foo.Module().(*sourceModule).deps | 
 | 			if deps == nil || len(deps) != 1 { | 
 | 				t.Errorf("deps does not have single path, but is %v", deps) | 
 | 			} | 
 | 			var usingSourceFile, usingPrebuiltFile bool | 
 | 			if deps[0].String() == "source_file" { | 
 | 				usingSourceFile = true | 
 | 			} | 
 | 			if deps[0].String() == "prebuilt_file" { | 
 | 				usingPrebuiltFile = true | 
 | 			} | 
 |  | 
 | 			if test.prebuilt { | 
 | 				if !dependsOnPrebuiltModule { | 
 | 					t.Errorf("doesn't depend on prebuilt module") | 
 | 				} | 
 | 				if !usingPrebuiltFile { | 
 | 					t.Errorf("doesn't use prebuilt_file") | 
 | 				} | 
 |  | 
 | 				if dependsOnSourceModule { | 
 | 					t.Errorf("depends on source module") | 
 | 				} | 
 | 				if usingSourceFile { | 
 | 					t.Errorf("using source_file") | 
 | 				} | 
 | 			} else { | 
 | 				if dependsOnPrebuiltModule { | 
 | 					t.Errorf("depends on prebuilt module") | 
 | 				} | 
 | 				if usingPrebuiltFile { | 
 | 					t.Errorf("using prebuilt_file") | 
 | 				} | 
 |  | 
 | 				if !dependsOnSourceModule { | 
 | 					t.Errorf("doesn't depend on source module") | 
 | 				} | 
 | 				if !usingSourceFile { | 
 | 					t.Errorf("doesn't use source_file") | 
 | 				} | 
 | 			} | 
 | 		}) | 
 | 	} | 
 | } | 
 |  | 
 | func registerTestPrebuiltBuildComponents(ctx RegistrationContext) { | 
 | 	ctx.RegisterModuleType("prebuilt", newPrebuiltModule) | 
 | 	ctx.RegisterModuleType("source", newSourceModule) | 
 |  | 
 | 	RegisterPrebuiltMutators(ctx) | 
 | } | 
 |  | 
 | type prebuiltModule struct { | 
 | 	ModuleBase | 
 | 	prebuilt   Prebuilt | 
 | 	properties struct { | 
 | 		Srcs []string `android:"path"` | 
 | 	} | 
 | 	src Path | 
 | } | 
 |  | 
 | func newPrebuiltModule() Module { | 
 | 	m := &prebuiltModule{} | 
 | 	m.AddProperties(&m.properties) | 
 | 	InitPrebuiltModule(m, &m.properties.Srcs) | 
 | 	InitAndroidModule(m) | 
 | 	return m | 
 | } | 
 |  | 
 | func (p *prebuiltModule) Name() string { | 
 | 	return p.prebuilt.Name(p.ModuleBase.Name()) | 
 | } | 
 |  | 
 | func (p *prebuiltModule) GenerateAndroidBuildActions(ctx ModuleContext) { | 
 | 	if len(p.properties.Srcs) >= 1 { | 
 | 		p.src = p.prebuilt.SingleSourcePath(ctx) | 
 | 	} | 
 | } | 
 |  | 
 | func (p *prebuiltModule) Prebuilt() *Prebuilt { | 
 | 	return &p.prebuilt | 
 | } | 
 |  | 
 | func (p *prebuiltModule) OutputFiles(tag string) (Paths, error) { | 
 | 	switch tag { | 
 | 	case "": | 
 | 		return Paths{p.src}, nil | 
 | 	default: | 
 | 		return nil, fmt.Errorf("unsupported module reference tag %q", tag) | 
 | 	} | 
 | } | 
 |  | 
 | type sourceModule struct { | 
 | 	ModuleBase | 
 | 	properties struct { | 
 | 		Deps []string `android:"path"` | 
 | 	} | 
 | 	dependsOnSourceModule, dependsOnPrebuiltModule bool | 
 | 	deps                                           Paths | 
 | 	src                                            Path | 
 | } | 
 |  | 
 | func newSourceModule() Module { | 
 | 	m := &sourceModule{} | 
 | 	m.AddProperties(&m.properties) | 
 | 	InitAndroidModule(m) | 
 | 	return m | 
 | } | 
 |  | 
 | func (s *sourceModule) DepsMutator(ctx BottomUpMutatorContext) { | 
 | 	// s.properties.Deps are annotated with android:path, so they are | 
 | 	// automatically added to the dependency by pathDeps mutator | 
 | } | 
 |  | 
 | func (s *sourceModule) GenerateAndroidBuildActions(ctx ModuleContext) { | 
 | 	s.deps = PathsForModuleSrc(ctx, s.properties.Deps) | 
 | 	s.src = PathForModuleSrc(ctx, "source_file") | 
 | } | 
 |  | 
 | func (s *sourceModule) Srcs() Paths { | 
 | 	return Paths{s.src} | 
 | } |