|  | // Copyright 2018 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 dexpreopt | 
|  |  | 
|  | import ( | 
|  | "android/soong/android" | 
|  | "fmt" | 
|  | "testing" | 
|  | ) | 
|  |  | 
|  | func testSystemModuleConfig(ctx android.PathContext, name string) *ModuleConfig { | 
|  | return testModuleConfig(ctx, name, "system") | 
|  | } | 
|  |  | 
|  | func testSystemProductModuleConfig(ctx android.PathContext, name string) *ModuleConfig { | 
|  | return testModuleConfig(ctx, name, "system/product") | 
|  | } | 
|  |  | 
|  | func testProductModuleConfig(ctx android.PathContext, name string) *ModuleConfig { | 
|  | return testModuleConfig(ctx, name, "product") | 
|  | } | 
|  |  | 
|  | func testModuleConfig(ctx android.PathContext, name, partition string) *ModuleConfig { | 
|  | return createTestModuleConfig( | 
|  | name, | 
|  | fmt.Sprintf("/%s/app/test/%s.apk", partition, name), | 
|  | android.PathForOutput(ctx, fmt.Sprintf("%s/%s.apk", name, name)), | 
|  | android.PathForOutput(ctx, fmt.Sprintf("%s/dex/%s.jar", name, name)), | 
|  | android.PathForOutput(ctx, fmt.Sprintf("%s/enforce_uses_libraries.status", name))) | 
|  | } | 
|  |  | 
|  | func testApexModuleConfig(ctx android.PathContext, name, apexName string) *ModuleConfig { | 
|  | ret := createTestModuleConfig( | 
|  | name, | 
|  | fmt.Sprintf("/apex/%s/javalib/%s.jar", apexName, name), | 
|  | android.PathForOutput(ctx, fmt.Sprintf("%s/dexpreopt/%s.jar", name, name)), | 
|  | android.PathForOutput(ctx, fmt.Sprintf("%s/aligned/%s.jar", name, name)), | 
|  | android.PathForOutput(ctx, fmt.Sprintf("%s/enforce_uses_libraries.status", name))) | 
|  | ret.ApexPartition = "/system" | 
|  | return ret | 
|  | } | 
|  |  | 
|  | func testPlatformSystemServerModuleConfig(ctx android.PathContext, name string) *ModuleConfig { | 
|  | return createTestModuleConfig( | 
|  | name, | 
|  | fmt.Sprintf("/system/framework/%s.jar", name), | 
|  | android.PathForOutput(ctx, fmt.Sprintf("%s/dexpreopt/%s.jar", name, name)), | 
|  | android.PathForOutput(ctx, fmt.Sprintf("%s/aligned/%s.jar", name, name)), | 
|  | android.PathForOutput(ctx, fmt.Sprintf("%s/enforce_uses_libraries.status", name))) | 
|  | } | 
|  |  | 
|  | func testSystemExtSystemServerModuleConfig(ctx android.PathContext, name string) *ModuleConfig { | 
|  | return createTestModuleConfig( | 
|  | name, | 
|  | fmt.Sprintf("/system_ext/framework/%s.jar", name), | 
|  | android.PathForOutput(ctx, fmt.Sprintf("%s/dexpreopt/%s.jar", name, name)), | 
|  | android.PathForOutput(ctx, fmt.Sprintf("%s/aligned/%s.jar", name, name)), | 
|  | android.PathForOutput(ctx, fmt.Sprintf("%s/enforce_uses_libraries.status", name))) | 
|  | } | 
|  |  | 
|  | func createTestModuleConfig(name, dexLocation string, buildPath, dexPath, enforceUsesLibrariesStatusFile android.OutputPath) *ModuleConfig { | 
|  | return &ModuleConfig{ | 
|  | Name:                            name, | 
|  | DexLocation:                     dexLocation, | 
|  | BuildPath:                       buildPath, | 
|  | DexPath:                         dexPath, | 
|  | UncompressedDex:                 false, | 
|  | HasApkLibraries:                 false, | 
|  | PreoptFlags:                     nil, | 
|  | ProfileClassListing:             android.OptionalPath{}, | 
|  | ProfileIsTextListing:            false, | 
|  | EnforceUsesLibrariesStatusFile:  enforceUsesLibrariesStatusFile, | 
|  | EnforceUsesLibraries:            false, | 
|  | ClassLoaderContexts:             nil, | 
|  | Archs:                           []android.ArchType{android.Arm}, | 
|  | DexPreoptImagesDeps:             []android.OutputPaths{android.OutputPaths{}}, | 
|  | DexPreoptImageLocationsOnHost:   []string{}, | 
|  | PreoptBootClassPathDexFiles:     nil, | 
|  | PreoptBootClassPathDexLocations: nil, | 
|  | NoCreateAppImage:                false, | 
|  | ForceCreateAppImage:             false, | 
|  | PresignedPrebuilt:               false, | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestDexPreopt(t *testing.T) { | 
|  | config := android.TestConfig("out", nil, "", nil) | 
|  | ctx := android.BuilderContextForTesting(config) | 
|  | globalSoong := globalSoongConfigForTests(ctx) | 
|  | global := GlobalConfigForTests(ctx) | 
|  | module := testSystemModuleConfig(ctx, "test") | 
|  | productPackages := android.PathForTesting("product_packages.txt") | 
|  |  | 
|  | rule, err := GenerateDexpreoptRule(ctx, globalSoong, global, module, productPackages, true) | 
|  | if err != nil { | 
|  | t.Fatal(err) | 
|  | } | 
|  |  | 
|  | wantInstalls := android.RuleBuilderInstalls{ | 
|  | {android.PathForOutput(ctx, "test/oat/arm/package.odex"), "/system/app/test/oat/arm/test.odex"}, | 
|  | {android.PathForOutput(ctx, "test/oat/arm/package.vdex"), "/system/app/test/oat/arm/test.vdex"}, | 
|  | } | 
|  |  | 
|  | if rule.Installs().String() != wantInstalls.String() { | 
|  | t.Errorf("\nwant installs:\n   %v\ngot:\n   %v", wantInstalls, rule.Installs()) | 
|  | } | 
|  |  | 
|  | android.AssertStringListContains(t, "", rule.Inputs().RelativeToTop().Strings(), | 
|  | "out/soong/dexpreopt_test/uffd_gc_flag.txt") | 
|  | } | 
|  |  | 
|  | func TestDexPreoptSystemOther(t *testing.T) { | 
|  | config := android.TestConfig("out", nil, "", nil) | 
|  | ctx := android.BuilderContextForTesting(config) | 
|  | globalSoong := globalSoongConfigForTests(ctx) | 
|  | global := GlobalConfigForTests(ctx) | 
|  | systemModule := testSystemModuleConfig(ctx, "Stest") | 
|  | systemProductModule := testSystemProductModuleConfig(ctx, "SPtest") | 
|  | productModule := testProductModuleConfig(ctx, "Ptest") | 
|  | productPackages := android.PathForTesting("product_packages.txt") | 
|  |  | 
|  | global.HasSystemOther = true | 
|  |  | 
|  | type moduleTest struct { | 
|  | module            *ModuleConfig | 
|  | expectedPartition string | 
|  | } | 
|  | tests := []struct { | 
|  | patterns    []string | 
|  | moduleTests []moduleTest | 
|  | }{ | 
|  | { | 
|  | patterns: []string{"app/%"}, | 
|  | moduleTests: []moduleTest{ | 
|  | {module: systemModule, expectedPartition: "system_other/system"}, | 
|  | {module: systemProductModule, expectedPartition: "system/product"}, | 
|  | {module: productModule, expectedPartition: "product"}, | 
|  | }, | 
|  | }, | 
|  | // product/app/% only applies to product apps inside the system partition | 
|  | { | 
|  | patterns: []string{"app/%", "product/app/%"}, | 
|  | moduleTests: []moduleTest{ | 
|  | {module: systemModule, expectedPartition: "system_other/system"}, | 
|  | {module: systemProductModule, expectedPartition: "system_other/system/product"}, | 
|  | {module: productModule, expectedPartition: "system_other/product"}, | 
|  | }, | 
|  | }, | 
|  | } | 
|  |  | 
|  | for _, test := range tests { | 
|  | global.PatternsOnSystemOther = test.patterns | 
|  | for _, mt := range test.moduleTests { | 
|  | rule, err := GenerateDexpreoptRule(ctx, globalSoong, global, mt.module, productPackages, true) | 
|  | if err != nil { | 
|  | t.Fatal(err) | 
|  | } | 
|  |  | 
|  | name := mt.module.Name | 
|  | wantInstalls := android.RuleBuilderInstalls{ | 
|  | {android.PathForOutput(ctx, name+"/oat/arm/package.odex"), fmt.Sprintf("/%s/app/test/oat/arm/%s.odex", mt.expectedPartition, name)}, | 
|  | {android.PathForOutput(ctx, name+"/oat/arm/package.vdex"), fmt.Sprintf("/%s/app/test/oat/arm/%s.vdex", mt.expectedPartition, name)}, | 
|  | } | 
|  |  | 
|  | if rule.Installs().String() != wantInstalls.String() { | 
|  | t.Errorf("\nwant installs:\n   %v\ngot:\n   %v", wantInstalls, rule.Installs()) | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | } | 
|  |  | 
|  | func TestDexPreoptApexSystemServerJars(t *testing.T) { | 
|  | // modify the global variable for test | 
|  | var oldDexpreoptRunningInSoong = DexpreoptRunningInSoong | 
|  | DexpreoptRunningInSoong = true | 
|  |  | 
|  | // test begin | 
|  | config := android.TestConfig("out", nil, "", nil) | 
|  | ctx := android.BuilderContextForTesting(config) | 
|  | globalSoong := globalSoongConfigForTests(ctx) | 
|  | global := GlobalConfigForTests(ctx) | 
|  | module := testApexModuleConfig(ctx, "service-A", "com.android.apex1") | 
|  | productPackages := android.PathForTesting("product_packages.txt") | 
|  |  | 
|  | global.ApexSystemServerJars = android.CreateTestConfiguredJarList( | 
|  | []string{"com.android.apex1:service-A"}) | 
|  |  | 
|  | rule, err := GenerateDexpreoptRule(ctx, globalSoong, global, module, productPackages, true) | 
|  | if err != nil { | 
|  | t.Fatal(err) | 
|  | } | 
|  |  | 
|  | wantInstalls := android.RuleBuilderInstalls{ | 
|  | {android.PathForOutput(ctx, "service-A/dexpreopt/oat/arm/javalib.odex"), "/system/framework/oat/arm/apex@com.android.apex1@javalib@service-A.jar@classes.odex"}, | 
|  | {android.PathForOutput(ctx, "service-A/dexpreopt/oat/arm/javalib.vdex"), "/system/framework/oat/arm/apex@com.android.apex1@javalib@service-A.jar@classes.vdex"}, | 
|  | } | 
|  |  | 
|  | android.AssertStringEquals(t, "installs", wantInstalls.String(), rule.Installs().String()) | 
|  |  | 
|  | android.AssertStringListContains(t, "apex sscp jar copy", rule.Outputs().Strings(), "out/soong/system_server_dexjars/service-A.jar") | 
|  |  | 
|  | // rule with apex sscp cp as false | 
|  | rule, err = GenerateDexpreoptRule(ctx, globalSoong, global, module, productPackages, false) | 
|  | if err != nil { | 
|  | t.Fatal(err) | 
|  | } | 
|  | android.AssertStringListDoesNotContain(t, "apex sscp jar copy", rule.Outputs().Strings(), "out/soong/system_server_dexjars/service-A.jar") | 
|  |  | 
|  | // cleanup the global variable for test | 
|  | DexpreoptRunningInSoong = oldDexpreoptRunningInSoong | 
|  | } | 
|  |  | 
|  | // Same as `TestDexPreoptApexSystemServerJars`, but the apex jar is in /system_ext | 
|  | func TestDexPreoptApexSystemServerJarsSystemExt(t *testing.T) { | 
|  | // modify the global variable for test | 
|  | var oldDexpreoptRunningInSoong = DexpreoptRunningInSoong | 
|  | DexpreoptRunningInSoong = true | 
|  |  | 
|  | // test begin | 
|  | config := android.TestConfig("out", nil, "", nil) | 
|  | ctx := android.BuilderContextForTesting(config) | 
|  | globalSoong := globalSoongConfigForTests(ctx) | 
|  | global := GlobalConfigForTests(ctx) | 
|  | module := testApexModuleConfig(ctx, "service-A", "com.android.apex1") | 
|  | module.ApexPartition = "/system_ext" | 
|  | productPackages := android.PathForTesting("product_packages.txt") | 
|  |  | 
|  | global.ApexSystemServerJars = android.CreateTestConfiguredJarList( | 
|  | []string{"com.android.apex1:service-A"}) | 
|  |  | 
|  | rule, err := GenerateDexpreoptRule(ctx, globalSoong, global, module, productPackages, true) | 
|  | if err != nil { | 
|  | t.Fatal(err) | 
|  | } | 
|  |  | 
|  | wantInstalls := android.RuleBuilderInstalls{ | 
|  | {android.PathForOutput(ctx, "service-A/dexpreopt/oat/arm/javalib.odex"), "/system_ext/framework/oat/arm/apex@com.android.apex1@javalib@service-A.jar@classes.odex"}, | 
|  | {android.PathForOutput(ctx, "service-A/dexpreopt/oat/arm/javalib.vdex"), "/system_ext/framework/oat/arm/apex@com.android.apex1@javalib@service-A.jar@classes.vdex"}, | 
|  | } | 
|  |  | 
|  | android.AssertStringEquals(t, "installs", wantInstalls.String(), rule.Installs().String()) | 
|  |  | 
|  | android.AssertStringListContains(t, "apex sscp jar copy", rule.Outputs().Strings(), "out/soong/system_server_dexjars/service-A.jar") | 
|  |  | 
|  | // rule with apex sscp cp as false | 
|  | rule, err = GenerateDexpreoptRule(ctx, globalSoong, global, module, productPackages, false) | 
|  | if err != nil { | 
|  | t.Fatal(err) | 
|  | } | 
|  | android.AssertStringListDoesNotContain(t, "apex sscp jar copy", rule.Outputs().Strings(), "out/soong/system_server_dexjars/service-A.jar") | 
|  |  | 
|  | // cleanup the global variable for test | 
|  | DexpreoptRunningInSoong = oldDexpreoptRunningInSoong | 
|  | } | 
|  |  | 
|  | func TestDexPreoptStandaloneSystemServerJars(t *testing.T) { | 
|  | config := android.TestConfig("out", nil, "", nil) | 
|  | ctx := android.BuilderContextForTesting(config) | 
|  | globalSoong := globalSoongConfigForTests(ctx) | 
|  | global := GlobalConfigForTests(ctx) | 
|  | module := testPlatformSystemServerModuleConfig(ctx, "service-A") | 
|  | productPackages := android.PathForTesting("product_packages.txt") | 
|  |  | 
|  | global.StandaloneSystemServerJars = android.CreateTestConfiguredJarList( | 
|  | []string{"platform:service-A"}) | 
|  |  | 
|  | rule, err := GenerateDexpreoptRule(ctx, globalSoong, global, module, productPackages, true) | 
|  | if err != nil { | 
|  | t.Fatal(err) | 
|  | } | 
|  |  | 
|  | wantInstalls := android.RuleBuilderInstalls{ | 
|  | {android.PathForOutput(ctx, "service-A/dexpreopt/oat/arm/javalib.odex"), "/system/framework/oat/arm/service-A.odex"}, | 
|  | {android.PathForOutput(ctx, "service-A/dexpreopt/oat/arm/javalib.vdex"), "/system/framework/oat/arm/service-A.vdex"}, | 
|  | } | 
|  |  | 
|  | android.AssertStringEquals(t, "installs", wantInstalls.String(), rule.Installs().String()) | 
|  | } | 
|  |  | 
|  | func TestDexPreoptSystemExtSystemServerJars(t *testing.T) { | 
|  | config := android.TestConfig("out", nil, "", nil) | 
|  | ctx := android.BuilderContextForTesting(config) | 
|  | globalSoong := globalSoongConfigForTests(ctx) | 
|  | global := GlobalConfigForTests(ctx) | 
|  | module := testSystemExtSystemServerModuleConfig(ctx, "service-A") | 
|  | productPackages := android.PathForTesting("product_packages.txt") | 
|  |  | 
|  | global.StandaloneSystemServerJars = android.CreateTestConfiguredJarList( | 
|  | []string{"system_ext:service-A"}) | 
|  |  | 
|  | rule, err := GenerateDexpreoptRule(ctx, globalSoong, global, module, productPackages, true) | 
|  | if err != nil { | 
|  | t.Fatal(err) | 
|  | } | 
|  |  | 
|  | wantInstalls := android.RuleBuilderInstalls{ | 
|  | {android.PathForOutput(ctx, "service-A/dexpreopt/oat/arm/javalib.odex"), "/system_ext/framework/oat/arm/service-A.odex"}, | 
|  | {android.PathForOutput(ctx, "service-A/dexpreopt/oat/arm/javalib.vdex"), "/system_ext/framework/oat/arm/service-A.vdex"}, | 
|  | } | 
|  |  | 
|  | android.AssertStringEquals(t, "installs", wantInstalls.String(), rule.Installs().String()) | 
|  | } | 
|  |  | 
|  | func TestDexPreoptApexStandaloneSystemServerJars(t *testing.T) { | 
|  | config := android.TestConfig("out", nil, "", nil) | 
|  | ctx := android.BuilderContextForTesting(config) | 
|  | globalSoong := globalSoongConfigForTests(ctx) | 
|  | global := GlobalConfigForTests(ctx) | 
|  | module := testApexModuleConfig(ctx, "service-A", "com.android.apex1") | 
|  | productPackages := android.PathForTesting("product_packages.txt") | 
|  |  | 
|  | global.ApexStandaloneSystemServerJars = android.CreateTestConfiguredJarList( | 
|  | []string{"com.android.apex1:service-A"}) | 
|  |  | 
|  | rule, err := GenerateDexpreoptRule(ctx, globalSoong, global, module, productPackages, true) | 
|  | if err != nil { | 
|  | t.Fatal(err) | 
|  | } | 
|  |  | 
|  | wantInstalls := android.RuleBuilderInstalls{ | 
|  | {android.PathForOutput(ctx, "service-A/dexpreopt/oat/arm/javalib.odex"), "/system/framework/oat/arm/apex@com.android.apex1@javalib@service-A.jar@classes.odex"}, | 
|  | {android.PathForOutput(ctx, "service-A/dexpreopt/oat/arm/javalib.vdex"), "/system/framework/oat/arm/apex@com.android.apex1@javalib@service-A.jar@classes.vdex"}, | 
|  | } | 
|  |  | 
|  | android.AssertStringEquals(t, "installs", wantInstalls.String(), rule.Installs().String()) | 
|  | } | 
|  |  | 
|  | func TestDexPreoptProfile(t *testing.T) { | 
|  | config := android.TestConfig("out", nil, "", nil) | 
|  | ctx := android.BuilderContextForTesting(config) | 
|  | globalSoong := globalSoongConfigForTests(ctx) | 
|  | global := GlobalConfigForTests(ctx) | 
|  | module := testSystemModuleConfig(ctx, "test") | 
|  | productPackages := android.PathForTesting("product_packages.txt") | 
|  |  | 
|  | module.ProfileClassListing = android.OptionalPathForPath(android.PathForTesting("profile")) | 
|  |  | 
|  | rule, err := GenerateDexpreoptRule(ctx, globalSoong, global, module, productPackages, true) | 
|  | if err != nil { | 
|  | t.Fatal(err) | 
|  | } | 
|  |  | 
|  | wantInstalls := android.RuleBuilderInstalls{ | 
|  | {android.PathForOutput(ctx, "test/profile.prof"), "/system/app/test/test.apk.prof"}, | 
|  | {android.PathForOutput(ctx, "test/oat/arm/package.art"), "/system/app/test/oat/arm/test.art"}, | 
|  | {android.PathForOutput(ctx, "test/oat/arm/package.odex"), "/system/app/test/oat/arm/test.odex"}, | 
|  | {android.PathForOutput(ctx, "test/oat/arm/package.vdex"), "/system/app/test/oat/arm/test.vdex"}, | 
|  | } | 
|  |  | 
|  | if rule.Installs().String() != wantInstalls.String() { | 
|  | t.Errorf("\nwant installs:\n   %v\ngot:\n   %v", wantInstalls, rule.Installs()) | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestDexPreoptConfigToJson(t *testing.T) { | 
|  | config := android.TestConfig("out", nil, "", nil) | 
|  | ctx := android.BuilderContextForTesting(config) | 
|  | module := testSystemModuleConfig(ctx, "test") | 
|  | data, err := moduleConfigToJSON(module) | 
|  | if err != nil { | 
|  | t.Errorf("Failed to convert module config data to JSON, %v", err) | 
|  | } | 
|  | parsed, err := ParseModuleConfig(ctx, data) | 
|  | if err != nil { | 
|  | t.Errorf("Failed to parse JSON, %v", err) | 
|  | } | 
|  | before := fmt.Sprintf("%v", module) | 
|  | after := fmt.Sprintf("%v", parsed) | 
|  | android.AssertStringEquals(t, "The result must be the same as the original after marshalling and unmarshalling it.", before, after) | 
|  | } | 
|  |  | 
|  | func TestUffdGcFlagForce(t *testing.T) { | 
|  | for _, enableUffdGc := range []string{"true", "false"} { | 
|  | t.Run(enableUffdGc, func(t *testing.T) { | 
|  | preparers := android.GroupFixturePreparers( | 
|  | PrepareForTestWithFakeDex2oatd, | 
|  | PrepareForTestWithDexpreoptConfig, | 
|  | FixtureSetEnableUffdGc(enableUffdGc), | 
|  | ) | 
|  |  | 
|  | result := preparers.RunTest(t) | 
|  | ctx := result.TestContext | 
|  |  | 
|  | ctx.SingletonForTests("dexpreopt-soong-config").Output("out/soong/dexpreopt/uffd_gc_flag.txt") | 
|  | }) | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestUffdGcFlagDefault(t *testing.T) { | 
|  | preparers := android.GroupFixturePreparers( | 
|  | PrepareForTestWithFakeDex2oatd, | 
|  | PrepareForTestWithDexpreoptConfig, | 
|  | FixtureSetEnableUffdGc("default"), | 
|  | ) | 
|  |  | 
|  | result := preparers.RunTest(t) | 
|  | ctx := result.TestContext | 
|  | config := ctx.Config() | 
|  |  | 
|  | rule := ctx.SingletonForTests("dexpreopt-soong-config").Rule("dexpreopt_uffd_gc_flag") | 
|  |  | 
|  | android.AssertStringDoesContain(t, "", rule.RuleParams.Command, "construct_uffd_gc_flag") | 
|  | android.AssertStringPathsRelativeToTopEquals(t, "", config, []string{ | 
|  | "out/soong/dexpreopt/uffd_gc_flag.txt", | 
|  | }, rule.AllOutputs()) | 
|  | android.AssertPathsRelativeToTopEquals(t, "", []string{ | 
|  | "out/soong/dexpreopt/kernel_version_for_uffd_gc.txt", | 
|  | }, rule.Implicits) | 
|  | } | 
|  |  | 
|  | func TestUffdGcFlagBogus(t *testing.T) { | 
|  | preparers := android.GroupFixturePreparers( | 
|  | PrepareForTestWithFakeDex2oatd, | 
|  | PrepareForTestWithDexpreoptConfig, | 
|  | FixtureSetEnableUffdGc("bogus"), | 
|  | ) | 
|  |  | 
|  | preparers. | 
|  | ExtendWithErrorHandler(android.FixtureExpectsAtLeastOneErrorMatchingPattern( | 
|  | "Unknown value of PRODUCT_ENABLE_UFFD_GC: bogus")). | 
|  | RunTest(t) | 
|  | } |