|  | // 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 android | 
|  |  | 
|  | import ( | 
|  | "reflect" | 
|  | "strconv" | 
|  | "testing" | 
|  |  | 
|  | "github.com/google/blueprint/proptools" | 
|  | ) | 
|  |  | 
|  | type printfIntoPropertyTestCase struct { | 
|  | in  string | 
|  | val interface{} | 
|  | out string | 
|  | err bool | 
|  | } | 
|  |  | 
|  | var printfIntoPropertyTestCases = []printfIntoPropertyTestCase{ | 
|  | { | 
|  | in:  "%d", | 
|  | val: 0, | 
|  | out: "0", | 
|  | }, | 
|  | { | 
|  | in:  "%d", | 
|  | val: 1, | 
|  | out: "1", | 
|  | }, | 
|  | { | 
|  | in:  "%d", | 
|  | val: 2, | 
|  | out: "2", | 
|  | }, | 
|  | { | 
|  | in:  "%d", | 
|  | val: false, | 
|  | out: "0", | 
|  | }, | 
|  | { | 
|  | in:  "%d", | 
|  | val: true, | 
|  | out: "1", | 
|  | }, | 
|  | { | 
|  | in:  "%d", | 
|  | val: -1, | 
|  | out: "-1", | 
|  | }, | 
|  |  | 
|  | { | 
|  | in:  "-DA=%d", | 
|  | val: 1, | 
|  | out: "-DA=1", | 
|  | }, | 
|  | { | 
|  | in:  "-DA=%du", | 
|  | val: 1, | 
|  | out: "-DA=1u", | 
|  | }, | 
|  | { | 
|  | in:  "-DA=%s", | 
|  | val: "abc", | 
|  | out: "-DA=abc", | 
|  | }, | 
|  | { | 
|  | in:  `-DA="%s"`, | 
|  | val: "abc", | 
|  | out: `-DA="abc"`, | 
|  | }, | 
|  |  | 
|  | { | 
|  | in:  "%%", | 
|  | err: true, | 
|  | }, | 
|  | { | 
|  | in:  "%d%s", | 
|  | err: true, | 
|  | }, | 
|  | { | 
|  | in:  "%d,%s", | 
|  | err: true, | 
|  | }, | 
|  | { | 
|  | in:  "%d", | 
|  | val: "", | 
|  | err: true, | 
|  | }, | 
|  | { | 
|  | in:  "%d", | 
|  | val: 1.5, | 
|  | err: true, | 
|  | }, | 
|  | { | 
|  | in:  "%f", | 
|  | val: 1.5, | 
|  | err: true, | 
|  | }, | 
|  | } | 
|  |  | 
|  | func TestPrintfIntoProperty(t *testing.T) { | 
|  | for _, testCase := range printfIntoPropertyTestCases { | 
|  | s := testCase.in | 
|  | v := reflect.ValueOf(&s).Elem() | 
|  | err := printfIntoProperty(v, testCase.val) | 
|  | if err != nil && !testCase.err { | 
|  | t.Errorf("unexpected error %s", err) | 
|  | } else if err == nil && testCase.err { | 
|  | t.Errorf("expected error") | 
|  | } else if err == nil && v.String() != testCase.out { | 
|  | t.Errorf("expected %q got %q", testCase.out, v.String()) | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | type testProductVariableModule struct { | 
|  | ModuleBase | 
|  | } | 
|  |  | 
|  | func (m *testProductVariableModule) GenerateAndroidBuildActions(ctx ModuleContext) { | 
|  | } | 
|  |  | 
|  | var testProductVariableProperties = struct { | 
|  | Product_variables struct { | 
|  | Eng struct { | 
|  | Srcs   []string | 
|  | Cflags []string | 
|  | } | 
|  | } | 
|  | }{} | 
|  |  | 
|  | func testProductVariableModuleFactoryFactory(props interface{}) func() Module { | 
|  | return func() Module { | 
|  | m := &testProductVariableModule{} | 
|  | clonedProps := proptools.CloneProperties(reflect.ValueOf(props)).Interface() | 
|  | m.AddProperties(clonedProps) | 
|  |  | 
|  | // Set a default soongConfigVariableProperties, this will be used as the input to the property struct filter | 
|  | // for this test module. | 
|  | m.variableProperties = testProductVariableProperties | 
|  | InitAndroidModule(m) | 
|  | return m | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestProductVariables(t *testing.T) { | 
|  | // Test that a module can use one product variable even if it doesn't have all the properties | 
|  | // supported by that product variable. | 
|  | bp := ` | 
|  | module1 { | 
|  | name: "foo", | 
|  | product_variables: { | 
|  | eng: { | 
|  | srcs: ["foo.c"], | 
|  | }, | 
|  | }, | 
|  | } | 
|  | module2 { | 
|  | name: "bar", | 
|  | product_variables: { | 
|  | eng: { | 
|  | cflags: ["-DBAR"], | 
|  | }, | 
|  | }, | 
|  | } | 
|  |  | 
|  | module3 { | 
|  | name: "baz", | 
|  | } | 
|  | ` | 
|  |  | 
|  | GroupFixturePreparers( | 
|  | FixtureModifyProductVariables(func(variables FixtureProductVariables) { | 
|  | variables.Eng = proptools.BoolPtr(true) | 
|  | }), | 
|  | FixtureRegisterWithContext(func(ctx RegistrationContext) { | 
|  | // A module type that has a srcs property but not a cflags property. | 
|  | ctx.RegisterModuleType("module1", testProductVariableModuleFactoryFactory(&struct { | 
|  | Srcs []string | 
|  | }{})) | 
|  | // A module type that has a cflags property but not a srcs property. | 
|  | ctx.RegisterModuleType("module2", testProductVariableModuleFactoryFactory(&struct { | 
|  | Cflags []string | 
|  | }{})) | 
|  | // A module type that does not have any properties that match product_variables. | 
|  | ctx.RegisterModuleType("module3", testProductVariableModuleFactoryFactory(&struct { | 
|  | Foo []string | 
|  | }{})) | 
|  | ctx.PreDepsMutators(func(ctx RegisterMutatorsContext) { | 
|  | ctx.BottomUp("variable", VariableMutator).Parallel() | 
|  | }) | 
|  | }), | 
|  | FixtureWithRootAndroidBp(bp), | 
|  | ).RunTest(t) | 
|  | } | 
|  |  | 
|  | var testProductVariableDefaultsProperties = struct { | 
|  | Product_variables struct { | 
|  | Eng struct { | 
|  | Foo []string | 
|  | Bar []string | 
|  | } | 
|  | } | 
|  | }{} | 
|  |  | 
|  | type productVariablesDefaultsTestProperties struct { | 
|  | Foo []string | 
|  | } | 
|  |  | 
|  | type productVariablesDefaultsTestProperties2 struct { | 
|  | Foo []string | 
|  | Bar []string | 
|  | } | 
|  |  | 
|  | type productVariablesDefaultsTestModule struct { | 
|  | ModuleBase | 
|  | DefaultableModuleBase | 
|  | properties productVariablesDefaultsTestProperties | 
|  | } | 
|  |  | 
|  | func (d *productVariablesDefaultsTestModule) GenerateAndroidBuildActions(ctx ModuleContext) { | 
|  | ctx.Build(pctx, BuildParams{ | 
|  | Rule:   Touch, | 
|  | Output: PathForModuleOut(ctx, "out"), | 
|  | }) | 
|  | } | 
|  |  | 
|  | func productVariablesDefaultsTestModuleFactory() Module { | 
|  | module := &productVariablesDefaultsTestModule{} | 
|  | module.AddProperties(&module.properties) | 
|  | module.variableProperties = testProductVariableDefaultsProperties | 
|  | InitAndroidModule(module) | 
|  | InitDefaultableModule(module) | 
|  | return module | 
|  | } | 
|  |  | 
|  | type productVariablesDefaultsTestDefaults struct { | 
|  | ModuleBase | 
|  | DefaultsModuleBase | 
|  | } | 
|  |  | 
|  | func productVariablesDefaultsTestDefaultsFactory() Module { | 
|  | defaults := &productVariablesDefaultsTestDefaults{} | 
|  | defaults.AddProperties(&productVariablesDefaultsTestProperties{}) | 
|  | defaults.AddProperties(&productVariablesDefaultsTestProperties2{}) | 
|  | defaults.variableProperties = testProductVariableDefaultsProperties | 
|  | InitDefaultsModule(defaults) | 
|  | return defaults | 
|  | } | 
|  |  | 
|  | // Test a defaults module that supports more product variable properties than the target module. | 
|  | func TestProductVariablesDefaults(t *testing.T) { | 
|  | bp := ` | 
|  | defaults { | 
|  | name: "defaults", | 
|  | product_variables: { | 
|  | eng: { | 
|  | foo: ["product_variable_defaults"], | 
|  | bar: ["product_variable_defaults"], | 
|  | }, | 
|  | }, | 
|  | foo: ["defaults"], | 
|  | bar: ["defaults"], | 
|  | } | 
|  |  | 
|  | test { | 
|  | name: "foo", | 
|  | defaults: ["defaults"], | 
|  | foo: ["module"], | 
|  | product_variables: { | 
|  | eng: { | 
|  | foo: ["product_variable_module"], | 
|  | }, | 
|  | }, | 
|  | } | 
|  | ` | 
|  |  | 
|  | result := GroupFixturePreparers( | 
|  | FixtureModifyProductVariables(func(variables FixtureProductVariables) { | 
|  | variables.Eng = boolPtr(true) | 
|  | }), | 
|  | PrepareForTestWithDefaults, | 
|  | PrepareForTestWithVariables, | 
|  | FixtureRegisterWithContext(func(ctx RegistrationContext) { | 
|  | ctx.RegisterModuleType("test", productVariablesDefaultsTestModuleFactory) | 
|  | ctx.RegisterModuleType("defaults", productVariablesDefaultsTestDefaultsFactory) | 
|  | }), | 
|  | FixtureWithRootAndroidBp(bp), | 
|  | ).RunTest(t) | 
|  |  | 
|  | foo := result.ModuleForTests("foo", "").Module().(*productVariablesDefaultsTestModule) | 
|  |  | 
|  | want := []string{"defaults", "module", "product_variable_defaults", "product_variable_module"} | 
|  | AssertDeepEquals(t, "foo", want, foo.properties.Foo) | 
|  | } | 
|  |  | 
|  | func BenchmarkSliceToTypeArray(b *testing.B) { | 
|  | for _, n := range []int{1, 2, 4, 8, 100} { | 
|  | var propStructs []interface{} | 
|  | for i := 0; i < n; i++ { | 
|  | propStructs = append(propStructs, &struct { | 
|  | A *string | 
|  | B string | 
|  | }{}) | 
|  |  | 
|  | } | 
|  | b.Run(strconv.Itoa(n), func(b *testing.B) { | 
|  | for i := 0; i < b.N; i++ { | 
|  | _ = sliceToTypeArray(propStructs) | 
|  | } | 
|  | }) | 
|  | } | 
|  | } |