|  | // Copyright 2019 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 java | 
|  |  | 
|  | import ( | 
|  | "reflect" | 
|  | "testing" | 
|  |  | 
|  | "android/soong/android" | 
|  | "android/soong/cc" | 
|  |  | 
|  | "github.com/google/blueprint/proptools" | 
|  | ) | 
|  |  | 
|  | func TestRequired(t *testing.T) { | 
|  | ctx, _ := testJava(t, ` | 
|  | java_library { | 
|  | name: "foo", | 
|  | srcs: ["a.java"], | 
|  | required: ["libfoo"], | 
|  | } | 
|  | `) | 
|  |  | 
|  | mod := ctx.ModuleForTests("foo", "android_common").Module() | 
|  | entries := android.AndroidMkEntriesForTest(t, ctx, mod)[0] | 
|  |  | 
|  | expected := []string{"libfoo"} | 
|  | actual := entries.EntryMap["LOCAL_REQUIRED_MODULES"] | 
|  | if !reflect.DeepEqual(expected, actual) { | 
|  | t.Errorf("Unexpected required modules - expected: %q, actual: %q", expected, actual) | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestHostdex(t *testing.T) { | 
|  | ctx, _ := testJava(t, ` | 
|  | java_library { | 
|  | name: "foo", | 
|  | srcs: ["a.java"], | 
|  | hostdex: true, | 
|  | } | 
|  | `) | 
|  |  | 
|  | mod := ctx.ModuleForTests("foo", "android_common").Module() | 
|  | entriesList := android.AndroidMkEntriesForTest(t, ctx, mod) | 
|  | if len(entriesList) != 2 { | 
|  | t.Errorf("two entries are expected, but got %d", len(entriesList)) | 
|  | } | 
|  |  | 
|  | mainEntries := &entriesList[0] | 
|  | expected := []string{"foo"} | 
|  | actual := mainEntries.EntryMap["LOCAL_MODULE"] | 
|  | if !reflect.DeepEqual(expected, actual) { | 
|  | t.Errorf("Unexpected module name - expected: %q, actual: %q", expected, actual) | 
|  | } | 
|  |  | 
|  | subEntries := &entriesList[1] | 
|  | expected = []string{"foo-hostdex"} | 
|  | actual = subEntries.EntryMap["LOCAL_MODULE"] | 
|  | if !reflect.DeepEqual(expected, actual) { | 
|  | t.Errorf("Unexpected module name - expected: %q, actual: %q", expected, actual) | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestHostdexRequired(t *testing.T) { | 
|  | ctx, _ := testJava(t, ` | 
|  | java_library { | 
|  | name: "foo", | 
|  | srcs: ["a.java"], | 
|  | hostdex: true, | 
|  | required: ["libfoo"], | 
|  | } | 
|  | `) | 
|  |  | 
|  | mod := ctx.ModuleForTests("foo", "android_common").Module() | 
|  | entriesList := android.AndroidMkEntriesForTest(t, ctx, mod) | 
|  | if len(entriesList) != 2 { | 
|  | t.Errorf("two entries are expected, but got %d", len(entriesList)) | 
|  | } | 
|  |  | 
|  | mainEntries := &entriesList[0] | 
|  | expected := []string{"libfoo"} | 
|  | actual := mainEntries.EntryMap["LOCAL_REQUIRED_MODULES"] | 
|  | if !reflect.DeepEqual(expected, actual) { | 
|  | t.Errorf("Unexpected required modules - expected: %q, actual: %q", expected, actual) | 
|  | } | 
|  |  | 
|  | subEntries := &entriesList[1] | 
|  | expected = []string{"libfoo"} | 
|  | actual = subEntries.EntryMap["LOCAL_REQUIRED_MODULES"] | 
|  | if !reflect.DeepEqual(expected, actual) { | 
|  | t.Errorf("Unexpected required modules - expected: %q, actual: %q", expected, actual) | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestHostdexSpecificRequired(t *testing.T) { | 
|  | ctx, _ := testJava(t, ` | 
|  | java_library { | 
|  | name: "foo", | 
|  | srcs: ["a.java"], | 
|  | hostdex: true, | 
|  | target: { | 
|  | hostdex: { | 
|  | required: ["libfoo"], | 
|  | }, | 
|  | }, | 
|  | } | 
|  | `) | 
|  |  | 
|  | mod := ctx.ModuleForTests("foo", "android_common").Module() | 
|  | entriesList := android.AndroidMkEntriesForTest(t, ctx, mod) | 
|  | if len(entriesList) != 2 { | 
|  | t.Errorf("two entries are expected, but got %d", len(entriesList)) | 
|  | } | 
|  |  | 
|  | mainEntries := &entriesList[0] | 
|  | if r, ok := mainEntries.EntryMap["LOCAL_REQUIRED_MODULES"]; ok { | 
|  | t.Errorf("Unexpected required modules: %q", r) | 
|  | } | 
|  |  | 
|  | subEntries := &entriesList[1] | 
|  | expected := []string{"libfoo"} | 
|  | actual := subEntries.EntryMap["LOCAL_REQUIRED_MODULES"] | 
|  | if !reflect.DeepEqual(expected, actual) { | 
|  | t.Errorf("Unexpected required modules - expected: %q, actual: %q", expected, actual) | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestJavaSdkLibrary_RequireXmlPermissionFile(t *testing.T) { | 
|  | result := android.GroupFixturePreparers( | 
|  | prepareForJavaTest, | 
|  | PrepareForTestWithJavaSdkLibraryFiles, | 
|  | FixtureWithLastReleaseApis("foo-shared_library", "foo-no_shared_library"), | 
|  | ).RunTestWithBp(t, ` | 
|  | java_sdk_library { | 
|  | name: "foo-shared_library", | 
|  | srcs: ["a.java"], | 
|  | } | 
|  | java_sdk_library { | 
|  | name: "foo-no_shared_library", | 
|  | srcs: ["a.java"], | 
|  | shared_library: false, | 
|  | } | 
|  | `) | 
|  |  | 
|  | // Verify the existence of internal modules | 
|  | result.ModuleForTests("foo-shared_library.xml", "android_common") | 
|  |  | 
|  | testCases := []struct { | 
|  | moduleName string | 
|  | expected   []string | 
|  | }{ | 
|  | {"foo-shared_library", []string{"foo-shared_library.xml"}}, | 
|  | {"foo-no_shared_library", nil}, | 
|  | } | 
|  | for _, tc := range testCases { | 
|  | mod := result.ModuleForTests(tc.moduleName, "android_common").Module() | 
|  | entries := android.AndroidMkEntriesForTest(t, result.TestContext, mod)[0] | 
|  | actual := entries.EntryMap["LOCAL_REQUIRED_MODULES"] | 
|  | if !reflect.DeepEqual(tc.expected, actual) { | 
|  | t.Errorf("Unexpected required modules - expected: %q, actual: %q", tc.expected, actual) | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestImportSoongDexJar(t *testing.T) { | 
|  | result := PrepareForTestWithJavaDefaultModules.RunTestWithBp(t, ` | 
|  | java_import { | 
|  | name: "my-java-import", | 
|  | jars: ["a.jar"], | 
|  | prefer: true, | 
|  | compile_dex: true, | 
|  | } | 
|  | `) | 
|  |  | 
|  | mod := result.Module("my-java-import", "android_common") | 
|  | entries := android.AndroidMkEntriesForTest(t, result.TestContext, mod)[0] | 
|  | expectedSoongDexJar := "out/soong/.intermediates/my-java-import/android_common/dex/my-java-import.jar" | 
|  | actualSoongDexJar := entries.EntryMap["LOCAL_SOONG_DEX_JAR"] | 
|  |  | 
|  | android.AssertStringPathsRelativeToTopEquals(t, "LOCAL_SOONG_DEX_JAR", result.Config, []string{expectedSoongDexJar}, actualSoongDexJar) | 
|  | } | 
|  |  | 
|  | func TestAndroidTestHelperApp_LocalDisableTestConfig(t *testing.T) { | 
|  | ctx, _ := testJava(t, ` | 
|  | android_test_helper_app { | 
|  | name: "foo", | 
|  | srcs: ["a.java"], | 
|  | } | 
|  | `) | 
|  |  | 
|  | mod := ctx.ModuleForTests("foo", "android_common").Module() | 
|  | entries := android.AndroidMkEntriesForTest(t, ctx, mod)[0] | 
|  |  | 
|  | expected := []string{"true"} | 
|  | actual := entries.EntryMap["LOCAL_DISABLE_TEST_CONFIG"] | 
|  | if !reflect.DeepEqual(expected, actual) { | 
|  | t.Errorf("Unexpected flag value - expected: %q, actual: %q", expected, actual) | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestGetOverriddenPackages(t *testing.T) { | 
|  | ctx, _ := testJava( | 
|  | t, ` | 
|  | android_app { | 
|  | name: "foo", | 
|  | srcs: ["a.java"], | 
|  | sdk_version: "current", | 
|  | overrides: ["qux"] | 
|  | } | 
|  |  | 
|  | override_android_app { | 
|  | name: "foo_override", | 
|  | base: "foo", | 
|  | overrides: ["bar"] | 
|  | } | 
|  | `) | 
|  |  | 
|  | expectedVariants := []struct { | 
|  | name        string | 
|  | moduleName  string | 
|  | variantName string | 
|  | overrides   []string | 
|  | }{ | 
|  | { | 
|  | name:        "foo", | 
|  | moduleName:  "foo", | 
|  | variantName: "android_common", | 
|  | overrides:   []string{"qux"}, | 
|  | }, | 
|  | { | 
|  | name:        "foo", | 
|  | moduleName:  "foo_override", | 
|  | variantName: "android_common_foo_override", | 
|  | overrides:   []string{"bar", "foo"}, | 
|  | }, | 
|  | } | 
|  |  | 
|  | for _, expected := range expectedVariants { | 
|  | mod := ctx.ModuleForTests(expected.name, expected.variantName).Module() | 
|  | entries := android.AndroidMkEntriesForTest(t, ctx, mod)[0] | 
|  | actual := entries.EntryMap["LOCAL_OVERRIDES_PACKAGES"] | 
|  |  | 
|  | android.AssertDeepEquals(t, "overrides property", expected.overrides, actual) | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestJniPartition(t *testing.T) { | 
|  | bp := ` | 
|  | cc_library { | 
|  | name: "libjni_system", | 
|  | system_shared_libs: [], | 
|  | sdk_version: "current", | 
|  | stl: "none", | 
|  | } | 
|  |  | 
|  | cc_library { | 
|  | name: "libjni_system_ext", | 
|  | system_shared_libs: [], | 
|  | sdk_version: "current", | 
|  | stl: "none", | 
|  | system_ext_specific: true, | 
|  | } | 
|  |  | 
|  | cc_library { | 
|  | name: "libjni_odm", | 
|  | system_shared_libs: [], | 
|  | sdk_version: "current", | 
|  | stl: "none", | 
|  | device_specific: true, | 
|  | } | 
|  |  | 
|  | cc_library { | 
|  | name: "libjni_product", | 
|  | system_shared_libs: [], | 
|  | sdk_version: "current", | 
|  | stl: "none", | 
|  | product_specific: true, | 
|  | } | 
|  |  | 
|  | cc_library { | 
|  | name: "libjni_vendor", | 
|  | system_shared_libs: [], | 
|  | sdk_version: "current", | 
|  | stl: "none", | 
|  | soc_specific: true, | 
|  | } | 
|  |  | 
|  | android_app { | 
|  | name: "test_app_system_jni_system", | 
|  | privileged: true, | 
|  | platform_apis: true, | 
|  | certificate: "platform", | 
|  | jni_libs: ["libjni_system"], | 
|  | } | 
|  |  | 
|  | android_app { | 
|  | name: "test_app_system_jni_system_ext", | 
|  | privileged: true, | 
|  | platform_apis: true, | 
|  | certificate: "platform", | 
|  | jni_libs: ["libjni_system_ext"], | 
|  | } | 
|  |  | 
|  | android_app { | 
|  | name: "test_app_system_ext_jni_system", | 
|  | privileged: true, | 
|  | platform_apis: true, | 
|  | certificate: "platform", | 
|  | jni_libs: ["libjni_system"], | 
|  | system_ext_specific: true | 
|  | } | 
|  |  | 
|  | android_app { | 
|  | name: "test_app_system_ext_jni_system_ext", | 
|  | sdk_version: "core_platform", | 
|  | jni_libs: ["libjni_system_ext"], | 
|  | system_ext_specific: true | 
|  | } | 
|  |  | 
|  | android_app { | 
|  | name: "test_app_product_jni_product", | 
|  | sdk_version: "core_platform", | 
|  | jni_libs: ["libjni_product"], | 
|  | product_specific: true | 
|  | } | 
|  |  | 
|  | android_app { | 
|  | name: "test_app_vendor_jni_odm", | 
|  | sdk_version: "core_platform", | 
|  | jni_libs: ["libjni_odm"], | 
|  | soc_specific: true | 
|  | } | 
|  |  | 
|  | android_app { | 
|  | name: "test_app_odm_jni_vendor", | 
|  | sdk_version: "core_platform", | 
|  | jni_libs: ["libjni_vendor"], | 
|  | device_specific: true | 
|  | } | 
|  | android_app { | 
|  | name: "test_app_system_jni_multiple", | 
|  | privileged: true, | 
|  | platform_apis: true, | 
|  | certificate: "platform", | 
|  | jni_libs: ["libjni_system", "libjni_system_ext"], | 
|  | } | 
|  | android_app { | 
|  | name: "test_app_vendor_jni_multiple", | 
|  | sdk_version: "core_platform", | 
|  | jni_libs: ["libjni_odm", "libjni_vendor"], | 
|  | soc_specific: true | 
|  | } | 
|  | ` | 
|  | arch := "arm64" | 
|  | ctx := android.GroupFixturePreparers( | 
|  | PrepareForTestWithJavaDefaultModules, | 
|  | cc.PrepareForTestWithCcDefaultModules, | 
|  | android.PrepareForTestWithAndroidMk, | 
|  | android.FixtureModifyConfig(func(config android.Config) { | 
|  | config.TestProductVariables.DeviceArch = proptools.StringPtr(arch) | 
|  | }), | 
|  | ). | 
|  | RunTestWithBp(t, bp) | 
|  | testCases := []struct { | 
|  | name           string | 
|  | partitionNames []string | 
|  | partitionTags  []string | 
|  | }{ | 
|  | {"test_app_system_jni_system", []string{"libjni_system"}, []string{""}}, | 
|  | {"test_app_system_jni_system_ext", []string{"libjni_system_ext"}, []string{"_SYSTEM_EXT"}}, | 
|  | {"test_app_system_ext_jni_system", []string{"libjni_system"}, []string{""}}, | 
|  | {"test_app_system_ext_jni_system_ext", []string{"libjni_system_ext"}, []string{"_SYSTEM_EXT"}}, | 
|  | {"test_app_product_jni_product", []string{"libjni_product"}, []string{"_PRODUCT"}}, | 
|  | {"test_app_vendor_jni_odm", []string{"libjni_odm"}, []string{"_ODM"}}, | 
|  | {"test_app_odm_jni_vendor", []string{"libjni_vendor"}, []string{"_VENDOR"}}, | 
|  | {"test_app_system_jni_multiple", []string{"libjni_system", "libjni_system_ext"}, []string{"", "_SYSTEM_EXT"}}, | 
|  | {"test_app_vendor_jni_multiple", []string{"libjni_odm", "libjni_vendor"}, []string{"_ODM", "_VENDOR"}}, | 
|  | } | 
|  |  | 
|  | for _, test := range testCases { | 
|  | t.Run(test.name, func(t *testing.T) { | 
|  | mod := ctx.ModuleForTests(test.name, "android_common").Module() | 
|  | entry := android.AndroidMkEntriesForTest(t, ctx.TestContext, mod)[0] | 
|  | for i := range test.partitionNames { | 
|  | actual := entry.EntryMap["LOCAL_SOONG_JNI_LIBS_PARTITION_"+arch][i] | 
|  | expected := test.partitionNames[i] + ":" + test.partitionTags[i] | 
|  | android.AssertStringEquals(t, "Expected and actual differ", expected, actual) | 
|  | } | 
|  | }) | 
|  | } | 
|  | } |