// 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 { return 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))) } 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: "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 } 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) }