* changes: Generate app profiles even if dexpreopt is disabled. Generate boot image profiles even if dexpreopt is disabled.
		
			
				
	
	
		
			1409 lines
		
	
	
		
			45 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			1409 lines
		
	
	
		
			45 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright (C) 2021 The Android Open Source Project
 | |
| //
 | |
| // 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 apex
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"path"
 | |
| 	"sort"
 | |
| 	"strings"
 | |
| 	"testing"
 | |
| 
 | |
| 	"android/soong/android"
 | |
| 	"android/soong/dexpreopt"
 | |
| 	"android/soong/java"
 | |
| 
 | |
| 	"github.com/google/blueprint/proptools"
 | |
| )
 | |
| 
 | |
| // Contains tests for bootclasspath_fragment logic from java/bootclasspath_fragment.go as the ART
 | |
| // bootclasspath_fragment requires modules from the ART apex.
 | |
| 
 | |
| var prepareForTestWithBootclasspathFragment = android.GroupFixturePreparers(
 | |
| 	java.PrepareForTestWithDexpreopt,
 | |
| 	PrepareForTestWithApexBuildComponents,
 | |
| )
 | |
| 
 | |
| // Some additional files needed for the art apex.
 | |
| var prepareForTestWithArtApex = android.GroupFixturePreparers(
 | |
| 	android.FixtureMergeMockFs(android.MockFS{
 | |
| 		"com.android.art.avbpubkey":                          nil,
 | |
| 		"com.android.art.pem":                                nil,
 | |
| 		"system/sepolicy/apex/com.android.art-file_contexts": nil,
 | |
| 	}),
 | |
| 	dexpreopt.FixtureSetBootImageProfiles("art/build/boot/boot-image-profile.txt"),
 | |
| )
 | |
| 
 | |
| func TestBootclasspathFragments(t *testing.T) {
 | |
| 	result := android.GroupFixturePreparers(
 | |
| 		prepareForTestWithBootclasspathFragment,
 | |
| 		// Configure some libraries in the art bootclasspath_fragment and platform_bootclasspath.
 | |
| 		java.FixtureConfigureBootJars("com.android.art:baz", "com.android.art:quuz", "platform:foo", "platform:bar"),
 | |
| 		prepareForTestWithArtApex,
 | |
| 
 | |
| 		java.PrepareForTestWithJavaSdkLibraryFiles,
 | |
| 		java.FixtureWithLastReleaseApis("foo"),
 | |
| 	).RunTestWithBp(t, `
 | |
| 		java_sdk_library {
 | |
| 			name: "foo",
 | |
| 			srcs: ["b.java"],
 | |
| 		}
 | |
| 
 | |
| 		java_library {
 | |
| 			name: "bar",
 | |
| 			srcs: ["b.java"],
 | |
| 			installable: true,
 | |
| 		}
 | |
| 
 | |
| 		apex {
 | |
| 			name: "com.android.art",
 | |
| 			key: "com.android.art.key",
 | |
| 			bootclasspath_fragments: ["art-bootclasspath-fragment"],
 | |
| 			updatable: false,
 | |
| 		}
 | |
| 
 | |
| 		apex_key {
 | |
| 			name: "com.android.art.key",
 | |
| 			public_key: "com.android.art.avbpubkey",
 | |
| 			private_key: "com.android.art.pem",
 | |
| 		}
 | |
| 
 | |
| 		java_library {
 | |
| 			name: "baz",
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 			srcs: ["b.java"],
 | |
| 			compile_dex: true,
 | |
| 		}
 | |
| 
 | |
| 		java_library {
 | |
| 			name: "quuz",
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 			srcs: ["b.java"],
 | |
| 			compile_dex: true,
 | |
| 		}
 | |
| 
 | |
| 		bootclasspath_fragment {
 | |
| 			name: "art-bootclasspath-fragment",
 | |
| 			image_name: "art",
 | |
| 			// Must match the "com.android.art:" entries passed to FixtureConfigureBootJars above.
 | |
| 			contents: ["baz", "quuz"],
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 			hidden_api: {
 | |
| 				split_packages: ["*"],
 | |
| 			},
 | |
| 		}
 | |
| `,
 | |
| 	)
 | |
| 
 | |
| 	// Make sure that the art-bootclasspath-fragment is using the correct configuration.
 | |
| 	checkBootclasspathFragment(t, result, "art-bootclasspath-fragment", "android_common_apex10000",
 | |
| 		"com.android.art:baz,com.android.art:quuz", `
 | |
| dexpreopt_arm64/dex_artjars/android/apex/art_boot_images/javalib/arm/boot.art
 | |
| dexpreopt_arm64/dex_artjars/android/apex/art_boot_images/javalib/arm/boot.oat
 | |
| dexpreopt_arm64/dex_artjars/android/apex/art_boot_images/javalib/arm/boot.vdex
 | |
| dexpreopt_arm64/dex_artjars/android/apex/art_boot_images/javalib/arm/boot-quuz.art
 | |
| dexpreopt_arm64/dex_artjars/android/apex/art_boot_images/javalib/arm/boot-quuz.oat
 | |
| dexpreopt_arm64/dex_artjars/android/apex/art_boot_images/javalib/arm/boot-quuz.vdex
 | |
| dexpreopt_arm64/dex_artjars/android/apex/art_boot_images/javalib/arm64/boot.art
 | |
| dexpreopt_arm64/dex_artjars/android/apex/art_boot_images/javalib/arm64/boot.oat
 | |
| dexpreopt_arm64/dex_artjars/android/apex/art_boot_images/javalib/arm64/boot.vdex
 | |
| dexpreopt_arm64/dex_artjars/android/apex/art_boot_images/javalib/arm64/boot-quuz.art
 | |
| dexpreopt_arm64/dex_artjars/android/apex/art_boot_images/javalib/arm64/boot-quuz.oat
 | |
| dexpreopt_arm64/dex_artjars/android/apex/art_boot_images/javalib/arm64/boot-quuz.vdex
 | |
| `)
 | |
| }
 | |
| 
 | |
| func TestBootclasspathFragments_FragmentDependency(t *testing.T) {
 | |
| 	result := android.GroupFixturePreparers(
 | |
| 		prepareForTestWithBootclasspathFragment,
 | |
| 		// Configure some libraries in the art bootclasspath_fragment and platform_bootclasspath.
 | |
| 		java.FixtureConfigureBootJars("com.android.art:baz", "com.android.art:quuz"),
 | |
| 		java.FixtureConfigureApexBootJars("someapex:foo", "someapex:bar"),
 | |
| 		prepareForTestWithArtApex,
 | |
| 
 | |
| 		java.PrepareForTestWithJavaSdkLibraryFiles,
 | |
| 		java.FixtureWithLastReleaseApis("foo", "baz"),
 | |
| 	).RunTestWithBp(t, `
 | |
| 		java_sdk_library {
 | |
| 			name: "foo",
 | |
| 			srcs: ["b.java"],
 | |
| 			shared_library: false,
 | |
| 			public: {
 | |
| 				enabled: true,
 | |
| 			},
 | |
| 			system: {
 | |
| 				enabled: true,
 | |
| 			},
 | |
| 		}
 | |
| 
 | |
| 		java_library {
 | |
| 			name: "bar",
 | |
| 			srcs: ["b.java"],
 | |
| 			installable: true,
 | |
| 		}
 | |
| 
 | |
| 		apex {
 | |
| 			name: "com.android.art",
 | |
| 			key: "com.android.art.key",
 | |
| 			bootclasspath_fragments: ["art-bootclasspath-fragment"],
 | |
| 			updatable: false,
 | |
| 		}
 | |
| 
 | |
| 		apex_key {
 | |
| 			name: "com.android.art.key",
 | |
| 			public_key: "com.android.art.avbpubkey",
 | |
| 			private_key: "com.android.art.pem",
 | |
| 		}
 | |
| 
 | |
| 		java_sdk_library {
 | |
| 			name: "baz",
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 			srcs: ["b.java"],
 | |
| 			shared_library: false,
 | |
| 			public: {
 | |
| 				enabled: true,
 | |
| 			},
 | |
| 			system: {
 | |
| 				enabled: true,
 | |
| 			},
 | |
| 			test: {
 | |
| 				enabled: true,
 | |
| 			},
 | |
| 		}
 | |
| 
 | |
| 		java_library {
 | |
| 			name: "quuz",
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 			srcs: ["b.java"],
 | |
| 			compile_dex: true,
 | |
| 		}
 | |
| 
 | |
| 		bootclasspath_fragment {
 | |
| 			name: "art-bootclasspath-fragment",
 | |
| 			image_name: "art",
 | |
| 			// Must match the "com.android.art:" entries passed to FixtureConfigureBootJars above.
 | |
| 			contents: ["baz", "quuz"],
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 			hidden_api: {
 | |
| 				split_packages: ["*"],
 | |
| 			},
 | |
| 		}
 | |
| 
 | |
| 		bootclasspath_fragment {
 | |
| 			name: "other-bootclasspath-fragment",
 | |
| 			contents: ["foo", "bar"],
 | |
| 			fragments: [
 | |
| 					{
 | |
| 							apex: "com.android.art",
 | |
| 							module: "art-bootclasspath-fragment",
 | |
| 					},
 | |
| 			],
 | |
| 			hidden_api: {
 | |
| 				split_packages: ["*"],
 | |
| 			},
 | |
| 		}
 | |
| `,
 | |
| 	)
 | |
| 
 | |
| 	checkAPIScopeStubs := func(message string, info java.HiddenAPIInfo, apiScope *java.HiddenAPIScope, expectedPaths ...string) {
 | |
| 		t.Helper()
 | |
| 		paths := info.TransitiveStubDexJarsByScope.StubDexJarsForScope(apiScope)
 | |
| 		android.AssertPathsRelativeToTopEquals(t, fmt.Sprintf("%s %s", message, apiScope), expectedPaths, paths)
 | |
| 	}
 | |
| 
 | |
| 	// Check stub dex paths exported by art.
 | |
| 	artFragment := result.Module("art-bootclasspath-fragment", "android_common")
 | |
| 	artInfo := result.ModuleProvider(artFragment, java.HiddenAPIInfoProvider).(java.HiddenAPIInfo)
 | |
| 
 | |
| 	bazPublicStubs := "out/soong/.intermediates/baz.stubs/android_common/dex/baz.stubs.jar"
 | |
| 	bazSystemStubs := "out/soong/.intermediates/baz.stubs.system/android_common/dex/baz.stubs.system.jar"
 | |
| 	bazTestStubs := "out/soong/.intermediates/baz.stubs.test/android_common/dex/baz.stubs.test.jar"
 | |
| 
 | |
| 	checkAPIScopeStubs("art", artInfo, java.PublicHiddenAPIScope, bazPublicStubs)
 | |
| 	checkAPIScopeStubs("art", artInfo, java.SystemHiddenAPIScope, bazSystemStubs)
 | |
| 	checkAPIScopeStubs("art", artInfo, java.TestHiddenAPIScope, bazTestStubs)
 | |
| 	checkAPIScopeStubs("art", artInfo, java.CorePlatformHiddenAPIScope)
 | |
| 
 | |
| 	// Check stub dex paths exported by other.
 | |
| 	otherFragment := result.Module("other-bootclasspath-fragment", "android_common")
 | |
| 	otherInfo := result.ModuleProvider(otherFragment, java.HiddenAPIInfoProvider).(java.HiddenAPIInfo)
 | |
| 
 | |
| 	fooPublicStubs := "out/soong/.intermediates/foo.stubs/android_common/dex/foo.stubs.jar"
 | |
| 	fooSystemStubs := "out/soong/.intermediates/foo.stubs.system/android_common/dex/foo.stubs.system.jar"
 | |
| 
 | |
| 	checkAPIScopeStubs("other", otherInfo, java.PublicHiddenAPIScope, bazPublicStubs, fooPublicStubs)
 | |
| 	checkAPIScopeStubs("other", otherInfo, java.SystemHiddenAPIScope, bazSystemStubs, fooSystemStubs)
 | |
| 	checkAPIScopeStubs("other", otherInfo, java.TestHiddenAPIScope, bazTestStubs, fooSystemStubs)
 | |
| 	checkAPIScopeStubs("other", otherInfo, java.CorePlatformHiddenAPIScope)
 | |
| }
 | |
| 
 | |
| func checkBootclasspathFragment(t *testing.T, result *android.TestResult, moduleName, variantName string, expectedConfiguredModules string, expectedBootclasspathFragmentFiles string) {
 | |
| 	t.Helper()
 | |
| 
 | |
| 	bootclasspathFragment := result.ModuleForTests(moduleName, variantName).Module().(*java.BootclasspathFragmentModule)
 | |
| 
 | |
| 	bootclasspathFragmentInfo := result.ModuleProvider(bootclasspathFragment, java.BootclasspathFragmentApexContentInfoProvider).(java.BootclasspathFragmentApexContentInfo)
 | |
| 	modules := bootclasspathFragmentInfo.Modules()
 | |
| 	android.AssertStringEquals(t, "invalid modules for "+moduleName, expectedConfiguredModules, modules.String())
 | |
| 
 | |
| 	// Get a list of all the paths in the boot image sorted by arch type.
 | |
| 	allPaths := []string{}
 | |
| 	bootImageFilesByArchType := bootclasspathFragmentInfo.AndroidBootImageFilesByArchType()
 | |
| 	for _, archType := range android.ArchTypeList() {
 | |
| 		if paths, ok := bootImageFilesByArchType[archType]; ok {
 | |
| 			for _, path := range paths {
 | |
| 				allPaths = append(allPaths, android.NormalizePathForTesting(path))
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	android.AssertTrimmedStringEquals(t, "invalid paths for "+moduleName, expectedBootclasspathFragmentFiles, strings.Join(allPaths, "\n"))
 | |
| }
 | |
| 
 | |
| func TestBootclasspathFragmentInArtApex(t *testing.T) {
 | |
| 	commonPreparer := android.GroupFixturePreparers(
 | |
| 		prepareForTestWithBootclasspathFragment,
 | |
| 		prepareForTestWithArtApex,
 | |
| 
 | |
| 		android.FixtureWithRootAndroidBp(`
 | |
| 		apex {
 | |
| 			name: "com.android.art",
 | |
| 			key: "com.android.art.key",
 | |
| 			bootclasspath_fragments: [
 | |
| 				"mybootclasspathfragment",
 | |
| 			],
 | |
| 			// bar (like foo) should be transitively included in this apex because it is part of the
 | |
| 			// mybootclasspathfragment bootclasspath_fragment.
 | |
| 			updatable: false,
 | |
| 		}
 | |
| 
 | |
| 		apex_key {
 | |
| 			name: "com.android.art.key",
 | |
| 			public_key: "testkey.avbpubkey",
 | |
| 			private_key: "testkey.pem",
 | |
| 		}
 | |
| 
 | |
| 		java_library {
 | |
| 			name: "foo",
 | |
| 			srcs: ["b.java"],
 | |
| 			installable: true,
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 		}
 | |
| 
 | |
| 		java_library {
 | |
| 			name: "bar",
 | |
| 			srcs: ["b.java"],
 | |
| 			installable: true,
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 		}
 | |
| 
 | |
| 		java_import {
 | |
| 			name: "foo",
 | |
| 			jars: ["foo.jar"],
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 			compile_dex: true,
 | |
| 		}
 | |
| 
 | |
| 		java_import {
 | |
| 			name: "bar",
 | |
| 			jars: ["bar.jar"],
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 			compile_dex: true,
 | |
| 		}
 | |
| 	`),
 | |
| 	)
 | |
| 
 | |
| 	contentsInsert := func(contents []string) string {
 | |
| 		insert := ""
 | |
| 		if contents != nil {
 | |
| 			insert = fmt.Sprintf(`contents: ["%s"],`, strings.Join(contents, `", "`))
 | |
| 		}
 | |
| 		return insert
 | |
| 	}
 | |
| 
 | |
| 	addSource := func(contents ...string) android.FixturePreparer {
 | |
| 		text := fmt.Sprintf(`
 | |
| 			bootclasspath_fragment {
 | |
| 				name: "mybootclasspathfragment",
 | |
| 				image_name: "art",
 | |
| 				%s
 | |
| 				apex_available: [
 | |
| 					"com.android.art",
 | |
| 				],
 | |
| 				hidden_api: {
 | |
| 					split_packages: ["*"],
 | |
| 				},
 | |
| 			}
 | |
| 		`, contentsInsert(contents))
 | |
| 
 | |
| 		return android.FixtureAddTextFile("art/build/boot/Android.bp", text)
 | |
| 	}
 | |
| 
 | |
| 	addPrebuilt := func(prefer bool, contents ...string) android.FixturePreparer {
 | |
| 		text := fmt.Sprintf(`
 | |
| 			prebuilt_apex {
 | |
| 				name: "com.android.art",
 | |
| 				arch: {
 | |
| 					arm64: {
 | |
| 						src: "com.android.art-arm64.apex",
 | |
| 					},
 | |
| 					arm: {
 | |
| 						src: "com.android.art-arm.apex",
 | |
| 					},
 | |
| 				},
 | |
| 				exported_bootclasspath_fragments: ["mybootclasspathfragment"],
 | |
| 			}
 | |
| 
 | |
| 			prebuilt_bootclasspath_fragment {
 | |
| 				name: "mybootclasspathfragment",
 | |
| 				image_name: "art",
 | |
| 				%s
 | |
| 				prefer: %t,
 | |
| 				apex_available: [
 | |
| 					"com.android.art",
 | |
| 				],
 | |
| 				hidden_api: {
 | |
| 					annotation_flags: "mybootclasspathfragment/annotation-flags.csv",
 | |
| 					metadata: "mybootclasspathfragment/metadata.csv",
 | |
| 					index: "mybootclasspathfragment/index.csv",
 | |
| 					stub_flags: "mybootclasspathfragment/stub-flags.csv",
 | |
| 					all_flags: "mybootclasspathfragment/all-flags.csv",
 | |
| 				},
 | |
| 			}
 | |
| 		`, contentsInsert(contents), prefer)
 | |
| 		return android.FixtureAddTextFile("prebuilts/module_sdk/art/Android.bp", text)
 | |
| 	}
 | |
| 
 | |
| 	t.Run("boot image files from source", func(t *testing.T) {
 | |
| 		result := android.GroupFixturePreparers(
 | |
| 			commonPreparer,
 | |
| 
 | |
| 			// Configure some libraries in the art bootclasspath_fragment that match the source
 | |
| 			// bootclasspath_fragment's contents property.
 | |
| 			java.FixtureConfigureBootJars("com.android.art:foo", "com.android.art:bar"),
 | |
| 			addSource("foo", "bar"),
 | |
| 			java.FixtureSetBootImageInstallDirOnDevice("art", "apex/com.android.art/javalib"),
 | |
| 		).RunTest(t)
 | |
| 
 | |
| 		ensureExactContents(t, result.TestContext, "com.android.art", "android_common_com.android.art_image", []string{
 | |
| 			"etc/boot-image.prof",
 | |
| 			"etc/classpaths/bootclasspath.pb",
 | |
| 			"javalib/arm/boot.art",
 | |
| 			"javalib/arm/boot.oat",
 | |
| 			"javalib/arm/boot.vdex",
 | |
| 			"javalib/arm/boot-bar.art",
 | |
| 			"javalib/arm/boot-bar.oat",
 | |
| 			"javalib/arm/boot-bar.vdex",
 | |
| 			"javalib/arm64/boot.art",
 | |
| 			"javalib/arm64/boot.oat",
 | |
| 			"javalib/arm64/boot.vdex",
 | |
| 			"javalib/arm64/boot-bar.art",
 | |
| 			"javalib/arm64/boot-bar.oat",
 | |
| 			"javalib/arm64/boot-bar.vdex",
 | |
| 			"javalib/bar.jar",
 | |
| 			"javalib/foo.jar",
 | |
| 		})
 | |
| 
 | |
| 		java.CheckModuleDependencies(t, result.TestContext, "com.android.art", "android_common_com.android.art_image", []string{
 | |
| 			`com.android.art.key`,
 | |
| 			`mybootclasspathfragment`,
 | |
| 		})
 | |
| 
 | |
| 		// The boot images are installed in the APEX by Soong, so there shouldn't be any dexpreopt-related Make modules.
 | |
| 		ensureDoesNotContainRequiredDeps(t, result.TestContext, "com.android.art", "android_common_com.android.art_image", []string{
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot.art",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot.oat",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot.vdex",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot-bar.art",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot-bar.oat",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot-bar.vdex",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot.art",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot.oat",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot.vdex",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot-bar.art",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot-bar.oat",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot-bar.vdex",
 | |
| 		})
 | |
| 
 | |
| 		// Make sure that the source bootclasspath_fragment copies its dex files to the predefined
 | |
| 		// locations for the art image.
 | |
| 		module := result.ModuleForTests("mybootclasspathfragment", "android_common_apex10000")
 | |
| 		checkCopiesToPredefinedLocationForArt(t, result.Config, module, "bar", "foo")
 | |
| 	})
 | |
| 
 | |
| 	t.Run("boot image files from source no boot image in apex", func(t *testing.T) {
 | |
| 		result := android.GroupFixturePreparers(
 | |
| 			commonPreparer,
 | |
| 
 | |
| 			// Configure some libraries in the art bootclasspath_fragment that match the source
 | |
| 			// bootclasspath_fragment's contents property.
 | |
| 			java.FixtureConfigureBootJars("com.android.art:foo", "com.android.art:bar"),
 | |
| 			addSource("foo", "bar"),
 | |
| 			java.FixtureSetBootImageInstallDirOnDevice("art", "system/framework"),
 | |
| 		).RunTest(t)
 | |
| 
 | |
| 		ensureExactContents(t, result.TestContext, "com.android.art", "android_common_com.android.art_image", []string{
 | |
| 			"etc/boot-image.prof",
 | |
| 			"etc/classpaths/bootclasspath.pb",
 | |
| 			"javalib/bar.jar",
 | |
| 			"javalib/foo.jar",
 | |
| 		})
 | |
| 
 | |
| 		ensureContainsRequiredDeps(t, result.TestContext, "com.android.art", "android_common_com.android.art_image", []string{
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot.art",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot.oat",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot.vdex",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot-bar.art",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot-bar.oat",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot-bar.vdex",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot.art",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot.oat",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot.vdex",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot-bar.art",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot-bar.oat",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot-bar.vdex",
 | |
| 		})
 | |
| 	})
 | |
| 
 | |
| 	t.Run("generate boot image profile even if dexpreopt is disabled", func(t *testing.T) {
 | |
| 		result := android.GroupFixturePreparers(
 | |
| 			commonPreparer,
 | |
| 
 | |
| 			// Configure some libraries in the art bootclasspath_fragment that match the source
 | |
| 			// bootclasspath_fragment's contents property.
 | |
| 			java.FixtureConfigureBootJars("com.android.art:foo", "com.android.art:bar"),
 | |
| 			addSource("foo", "bar"),
 | |
| 			java.FixtureSetBootImageInstallDirOnDevice("art", "system/framework"),
 | |
| 			dexpreopt.FixtureDisableDexpreoptBootImages(true),
 | |
| 		).RunTest(t)
 | |
| 
 | |
| 		ensureExactContents(t, result.TestContext, "com.android.art", "android_common_com.android.art_image", []string{
 | |
| 			"etc/boot-image.prof",
 | |
| 			"etc/classpaths/bootclasspath.pb",
 | |
| 			"javalib/bar.jar",
 | |
| 			"javalib/foo.jar",
 | |
| 		})
 | |
| 	})
 | |
| 
 | |
| 	t.Run("boot image disable generate profile", func(t *testing.T) {
 | |
| 		result := android.GroupFixturePreparers(
 | |
| 			commonPreparer,
 | |
| 
 | |
| 			// Configure some libraries in the art bootclasspath_fragment that match the source
 | |
| 			// bootclasspath_fragment's contents property.
 | |
| 			java.FixtureConfigureBootJars("com.android.art:foo", "com.android.art:bar"),
 | |
| 			addSource("foo", "bar"),
 | |
| 			dexpreopt.FixtureDisableGenerateProfile(true),
 | |
| 		).RunTest(t)
 | |
| 
 | |
| 		files := getFiles(t, result.TestContext, "com.android.art", "android_common_com.android.art_image")
 | |
| 		for _, file := range files {
 | |
| 			matched, _ := path.Match("etc/boot-image.prof", file.path)
 | |
| 			android.AssertBoolEquals(t, "\"etc/boot-image.prof\" should not be in the APEX", matched, false)
 | |
| 		}
 | |
| 	})
 | |
| 
 | |
| 	t.Run("boot image files with preferred prebuilt", func(t *testing.T) {
 | |
| 		result := android.GroupFixturePreparers(
 | |
| 			commonPreparer,
 | |
| 
 | |
| 			// Configure some libraries in the art bootclasspath_fragment that match the source
 | |
| 			// bootclasspath_fragment's contents property.
 | |
| 			java.FixtureConfigureBootJars("com.android.art:foo", "com.android.art:bar"),
 | |
| 			addSource("foo", "bar"),
 | |
| 
 | |
| 			// Make sure that a preferred prebuilt with consistent contents doesn't affect the apex.
 | |
| 			addPrebuilt(true, "foo", "bar"),
 | |
| 
 | |
| 			java.FixtureSetBootImageInstallDirOnDevice("art", "apex/com.android.art/javalib"),
 | |
| 		).RunTest(t)
 | |
| 
 | |
| 		ensureExactDeapexedContents(t, result.TestContext, "com.android.art", "android_common", []string{
 | |
| 			"etc/boot-image.prof",
 | |
| 			"javalib/arm/boot.art",
 | |
| 			"javalib/arm/boot.oat",
 | |
| 			"javalib/arm/boot.vdex",
 | |
| 			"javalib/arm/boot-bar.art",
 | |
| 			"javalib/arm/boot-bar.oat",
 | |
| 			"javalib/arm/boot-bar.vdex",
 | |
| 			"javalib/arm64/boot.art",
 | |
| 			"javalib/arm64/boot.oat",
 | |
| 			"javalib/arm64/boot.vdex",
 | |
| 			"javalib/arm64/boot-bar.art",
 | |
| 			"javalib/arm64/boot-bar.oat",
 | |
| 			"javalib/arm64/boot-bar.vdex",
 | |
| 			"javalib/bar.jar",
 | |
| 			"javalib/foo.jar",
 | |
| 		})
 | |
| 
 | |
| 		java.CheckModuleDependencies(t, result.TestContext, "com.android.art", "android_common_com.android.art_image", []string{
 | |
| 			`com.android.art.key`,
 | |
| 			`mybootclasspathfragment`,
 | |
| 			`prebuilt_com.android.art`,
 | |
| 		})
 | |
| 
 | |
| 		// The boot images are installed in the APEX by Soong, so there shouldn't be any dexpreopt-related Make modules.
 | |
| 		ensureDoesNotContainRequiredDeps(t, result.TestContext, "com.android.art", "android_common_com.android.art_image", []string{
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot.art",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot.oat",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot.vdex",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot-bar.art",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot-bar.oat",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot-bar.vdex",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot.art",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot.oat",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot.vdex",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot-bar.art",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot-bar.oat",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot-bar.vdex",
 | |
| 		})
 | |
| 
 | |
| 		// Make sure that the prebuilt bootclasspath_fragment copies its dex files to the predefined
 | |
| 		// locations for the art image.
 | |
| 		module := result.ModuleForTests("prebuilt_mybootclasspathfragment", "android_common_com.android.art")
 | |
| 		checkCopiesToPredefinedLocationForArt(t, result.Config, module, "bar", "foo")
 | |
| 	})
 | |
| 
 | |
| 	t.Run("boot image files from preferred prebuilt no boot image in apex", func(t *testing.T) {
 | |
| 		result := android.GroupFixturePreparers(
 | |
| 			commonPreparer,
 | |
| 
 | |
| 			// Configure some libraries in the art bootclasspath_fragment that match the source
 | |
| 			// bootclasspath_fragment's contents property.
 | |
| 			java.FixtureConfigureBootJars("com.android.art:foo", "com.android.art:bar"),
 | |
| 			addSource("foo", "bar"),
 | |
| 
 | |
| 			// Make sure that a preferred prebuilt with consistent contents doesn't affect the apex.
 | |
| 			addPrebuilt(true, "foo", "bar"),
 | |
| 
 | |
| 			java.FixtureSetBootImageInstallDirOnDevice("art", "system/framework"),
 | |
| 		).RunTest(t)
 | |
| 
 | |
| 		ensureExactDeapexedContents(t, result.TestContext, "com.android.art", "android_common", []string{
 | |
| 			"etc/boot-image.prof",
 | |
| 			"javalib/bar.jar",
 | |
| 			"javalib/foo.jar",
 | |
| 		})
 | |
| 
 | |
| 		ensureContainsRequiredDeps(t, result.TestContext, "com.android.art", "android_common_com.android.art_image", []string{
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot.art",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot.oat",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot.vdex",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot-bar.art",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot-bar.oat",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm64-boot-bar.vdex",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot.art",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot.oat",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot.vdex",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot-bar.art",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot-bar.oat",
 | |
| 			"mybootclasspathfragment-dexpreopt-arm-boot-bar.vdex",
 | |
| 		})
 | |
| 	})
 | |
| 
 | |
| 	t.Run("source with inconsistency between config and contents", func(t *testing.T) {
 | |
| 		android.GroupFixturePreparers(
 | |
| 			commonPreparer,
 | |
| 
 | |
| 			// Create an inconsistency between the ArtApexJars configuration and the art source
 | |
| 			// bootclasspath_fragment module's contents property.
 | |
| 			java.FixtureConfigureBootJars("com.android.art:foo"),
 | |
| 			addSource("foo", "bar"),
 | |
| 		).
 | |
| 			ExtendWithErrorHandler(android.FixtureExpectsAtLeastOneErrorMatchingPattern(`\QArtApexJars configuration specifies []string{"foo"}, contents property specifies []string{"foo", "bar"}\E`)).
 | |
| 			RunTest(t)
 | |
| 	})
 | |
| 
 | |
| 	t.Run("prebuilt with inconsistency between config and contents", func(t *testing.T) {
 | |
| 		android.GroupFixturePreparers(
 | |
| 			commonPreparer,
 | |
| 
 | |
| 			// Create an inconsistency between the ArtApexJars configuration and the art
 | |
| 			// prebuilt_bootclasspath_fragment module's contents property.
 | |
| 			java.FixtureConfigureBootJars("com.android.art:foo"),
 | |
| 			addPrebuilt(false, "foo", "bar"),
 | |
| 		).
 | |
| 			ExtendWithErrorHandler(android.FixtureExpectsAtLeastOneErrorMatchingPattern(`\QArtApexJars configuration specifies []string{"foo"}, contents property specifies []string{"foo", "bar"}\E`)).
 | |
| 			RunTest(t)
 | |
| 	})
 | |
| 
 | |
| 	t.Run("preferred prebuilt with inconsistency between config and contents", func(t *testing.T) {
 | |
| 		android.GroupFixturePreparers(
 | |
| 			commonPreparer,
 | |
| 
 | |
| 			// Create an inconsistency between the ArtApexJars configuration and the art
 | |
| 			// prebuilt_bootclasspath_fragment module's contents property.
 | |
| 			java.FixtureConfigureBootJars("com.android.art:foo"),
 | |
| 			addPrebuilt(true, "foo", "bar"),
 | |
| 
 | |
| 			// Source contents property is consistent with the config.
 | |
| 			addSource("foo"),
 | |
| 		).
 | |
| 			ExtendWithErrorHandler(android.FixtureExpectsAtLeastOneErrorMatchingPattern(`\QArtApexJars configuration specifies []string{"foo"}, contents property specifies []string{"foo", "bar"}\E`)).
 | |
| 			RunTest(t)
 | |
| 	})
 | |
| 
 | |
| 	t.Run("source preferred and prebuilt with inconsistency between config and contents", func(t *testing.T) {
 | |
| 		android.GroupFixturePreparers(
 | |
| 			commonPreparer,
 | |
| 
 | |
| 			// Create an inconsistency between the ArtApexJars configuration and the art
 | |
| 			// prebuilt_bootclasspath_fragment module's contents property.
 | |
| 			java.FixtureConfigureBootJars("com.android.art:foo"),
 | |
| 			addPrebuilt(false, "foo", "bar"),
 | |
| 
 | |
| 			// Source contents property is consistent with the config.
 | |
| 			addSource("foo"),
 | |
| 
 | |
| 			// This should pass because while the prebuilt is inconsistent with the configuration it is
 | |
| 			// not actually used.
 | |
| 		).RunTest(t)
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func TestBootclasspathFragmentInPrebuiltArtApex(t *testing.T) {
 | |
| 	preparers := android.GroupFixturePreparers(
 | |
| 		prepareForTestWithBootclasspathFragment,
 | |
| 		prepareForTestWithArtApex,
 | |
| 
 | |
| 		android.FixtureMergeMockFs(android.MockFS{
 | |
| 			"com.android.art-arm64.apex": nil,
 | |
| 			"com.android.art-arm.apex":   nil,
 | |
| 		}),
 | |
| 
 | |
| 		// Configure some libraries in the art bootclasspath_fragment.
 | |
| 		java.FixtureConfigureBootJars("com.android.art:foo", "com.android.art:bar"),
 | |
| 		java.FixtureSetBootImageInstallDirOnDevice("art", "apex/com.android.art/javalib"),
 | |
| 	)
 | |
| 
 | |
| 	bp := `
 | |
| 		prebuilt_apex {
 | |
| 			name: "com.android.art",
 | |
| 			arch: {
 | |
| 				arm64: {
 | |
| 					src: "com.android.art-arm64.apex",
 | |
| 				},
 | |
| 				arm: {
 | |
| 					src: "com.android.art-arm.apex",
 | |
| 				},
 | |
| 			},
 | |
| 			exported_bootclasspath_fragments: ["mybootclasspathfragment"],
 | |
| 		}
 | |
| 
 | |
| 		java_import {
 | |
| 			name: "foo",
 | |
| 			jars: ["foo.jar"],
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 		}
 | |
| 
 | |
| 		java_import {
 | |
| 			name: "bar",
 | |
| 			jars: ["bar.jar"],
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 		}
 | |
| 
 | |
| 		prebuilt_bootclasspath_fragment {
 | |
| 			name: "mybootclasspathfragment",
 | |
| 			image_name: "art",
 | |
| 			// Must match the "com.android.art:" entries passed to FixtureConfigureBootJars above.
 | |
| 			contents: ["foo", "bar"],
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 			hidden_api: {
 | |
| 				annotation_flags: "mybootclasspathfragment/annotation-flags.csv",
 | |
| 				metadata: "mybootclasspathfragment/metadata.csv",
 | |
| 				index: "mybootclasspathfragment/index.csv",
 | |
| 				stub_flags: "mybootclasspathfragment/stub-flags.csv",
 | |
| 				all_flags: "mybootclasspathfragment/all-flags.csv",
 | |
| 			},
 | |
| 		}
 | |
| 
 | |
| 		// A prebuilt apex with the same apex_name that shouldn't interfere when it isn't enabled.
 | |
| 		prebuilt_apex {
 | |
| 			name: "com.mycompany.android.art",
 | |
| 			apex_name: "com.android.art",
 | |
| 			%s
 | |
| 			src: "com.mycompany.android.art.apex",
 | |
| 			exported_bootclasspath_fragments: ["mybootclasspathfragment"],
 | |
| 		}
 | |
| 	`
 | |
| 
 | |
| 	t.Run("disabled alternative APEX", func(t *testing.T) {
 | |
| 		result := preparers.RunTestWithBp(t, fmt.Sprintf(bp, "enabled: false,"))
 | |
| 
 | |
| 		java.CheckModuleDependencies(t, result.TestContext, "com.android.art", "android_common_com.android.art", []string{
 | |
| 			`com.android.art.apex.selector`,
 | |
| 			`prebuilt_mybootclasspathfragment`,
 | |
| 		})
 | |
| 
 | |
| 		java.CheckModuleDependencies(t, result.TestContext, "mybootclasspathfragment", "android_common_com.android.art", []string{
 | |
| 			`com.android.art.deapexer`,
 | |
| 			`dex2oatd`,
 | |
| 			`prebuilt_bar`,
 | |
| 			`prebuilt_foo`,
 | |
| 		})
 | |
| 
 | |
| 		module := result.ModuleForTests("mybootclasspathfragment", "android_common_com.android.art")
 | |
| 		checkCopiesToPredefinedLocationForArt(t, result.Config, module, "bar", "foo")
 | |
| 
 | |
| 		// Check that the right deapexer module was chosen for a boot image.
 | |
| 		param := module.Output("out/soong/dexpreopt_arm64/dex_artjars/android/apex/art_boot_images/javalib/arm64/boot.art")
 | |
| 		android.AssertStringDoesContain(t, "didn't find the expected deapexer in the input path", param.Input.String(), "/com.android.art.deapexer")
 | |
| 	})
 | |
| 
 | |
| 	t.Run("enabled alternative APEX", func(t *testing.T) {
 | |
| 		preparers.ExtendWithErrorHandler(android.FixtureExpectsAtLeastOneErrorMatchingPattern(
 | |
| 			"Multiple installable prebuilt APEXes provide ambiguous deapexers: com.android.art and com.mycompany.android.art")).
 | |
| 			RunTestWithBp(t, fmt.Sprintf(bp, ""))
 | |
| 	})
 | |
| }
 | |
| 
 | |
| // checkCopiesToPredefinedLocationForArt checks that the supplied modules are copied to the
 | |
| // predefined locations of boot dex jars used as inputs for the ART boot image.
 | |
| func checkCopiesToPredefinedLocationForArt(t *testing.T, config android.Config, module android.TestingModule, modules ...string) {
 | |
| 	t.Helper()
 | |
| 	bootJarLocations := []string{}
 | |
| 	for _, output := range module.AllOutputs() {
 | |
| 		output = android.StringRelativeToTop(config, output)
 | |
| 		if strings.HasPrefix(output, "out/soong/dexpreopt_arm64/dex_artjars_input/") {
 | |
| 			bootJarLocations = append(bootJarLocations, output)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	sort.Strings(bootJarLocations)
 | |
| 	expected := []string{}
 | |
| 	for _, m := range modules {
 | |
| 		expected = append(expected, fmt.Sprintf("out/soong/dexpreopt_arm64/dex_artjars_input/%s.jar", m))
 | |
| 	}
 | |
| 	sort.Strings(expected)
 | |
| 
 | |
| 	android.AssertArrayString(t, "copies to predefined locations for art", expected, bootJarLocations)
 | |
| }
 | |
| 
 | |
| func TestBootclasspathFragmentContentsNoName(t *testing.T) {
 | |
| 	result := android.GroupFixturePreparers(
 | |
| 		prepareForTestWithBootclasspathFragment,
 | |
| 		prepareForTestWithMyapex,
 | |
| 		// Configure bootclasspath jars to ensure that hidden API encoding is performed on them.
 | |
| 		java.FixtureConfigureApexBootJars("myapex:foo", "myapex:bar"),
 | |
| 		// Make sure that the frameworks/base/Android.bp file exists as otherwise hidden API encoding
 | |
| 		// is disabled.
 | |
| 		android.FixtureAddTextFile("frameworks/base/Android.bp", ""),
 | |
| 
 | |
| 		java.PrepareForTestWithJavaSdkLibraryFiles,
 | |
| 		java.FixtureWithLastReleaseApis("foo"),
 | |
| 	).RunTestWithBp(t, `
 | |
| 		apex {
 | |
| 			name: "myapex",
 | |
| 			key: "myapex.key",
 | |
| 			bootclasspath_fragments: [
 | |
| 				"mybootclasspathfragment",
 | |
| 			],
 | |
| 			updatable: false,
 | |
| 		}
 | |
| 
 | |
| 		apex_key {
 | |
| 			name: "myapex.key",
 | |
| 			public_key: "testkey.avbpubkey",
 | |
| 			private_key: "testkey.pem",
 | |
| 		}
 | |
| 
 | |
| 		java_sdk_library {
 | |
| 			name: "foo",
 | |
| 			srcs: ["b.java"],
 | |
| 			shared_library: false,
 | |
| 			public: {enabled: true},
 | |
| 			apex_available: [
 | |
| 				"myapex",
 | |
| 			],
 | |
| 		}
 | |
| 
 | |
| 		java_library {
 | |
| 			name: "bar",
 | |
| 			srcs: ["b.java"],
 | |
| 			installable: true,
 | |
| 			apex_available: [
 | |
| 				"myapex",
 | |
| 			],
 | |
| 		}
 | |
| 
 | |
| 		bootclasspath_fragment {
 | |
| 			name: "mybootclasspathfragment",
 | |
| 			contents: [
 | |
| 				"foo",
 | |
| 				"bar",
 | |
| 			],
 | |
| 			apex_available: [
 | |
| 				"myapex",
 | |
| 			],
 | |
| 			hidden_api: {
 | |
| 				split_packages: ["*"],
 | |
| 			},
 | |
| 		}
 | |
| 	`)
 | |
| 
 | |
| 	ensureExactContents(t, result.TestContext, "myapex", "android_common_myapex_image", []string{
 | |
| 		// This does not include art, oat or vdex files as they are only included for the art boot
 | |
| 		// image.
 | |
| 		"etc/classpaths/bootclasspath.pb",
 | |
| 		"javalib/bar.jar",
 | |
| 		"javalib/foo.jar",
 | |
| 	})
 | |
| 
 | |
| 	java.CheckModuleDependencies(t, result.TestContext, "myapex", "android_common_myapex_image", []string{
 | |
| 		`myapex.key`,
 | |
| 		`mybootclasspathfragment`,
 | |
| 	})
 | |
| 
 | |
| 	apex := result.ModuleForTests("myapex", "android_common_myapex_image")
 | |
| 	apexRule := apex.Rule("apexRule")
 | |
| 	copyCommands := apexRule.Args["copy_commands"]
 | |
| 
 | |
| 	// Make sure that the fragment provides the hidden API encoded dex jars to the APEX.
 | |
| 	fragment := result.Module("mybootclasspathfragment", "android_common_apex10000")
 | |
| 
 | |
| 	info := result.ModuleProvider(fragment, java.BootclasspathFragmentApexContentInfoProvider).(java.BootclasspathFragmentApexContentInfo)
 | |
| 
 | |
| 	checkFragmentExportedDexJar := func(name string, expectedDexJar string) {
 | |
| 		module := result.Module(name, "android_common_apex10000")
 | |
| 		dexJar, err := info.DexBootJarPathForContentModule(module)
 | |
| 		if err != nil {
 | |
| 			t.Error(err)
 | |
| 		}
 | |
| 		android.AssertPathRelativeToTopEquals(t, name+" dex", expectedDexJar, dexJar)
 | |
| 
 | |
| 		expectedCopyCommand := fmt.Sprintf("&& cp -f %s out/soong/.intermediates/myapex/android_common_myapex_image/image.apex/javalib/%s.jar", expectedDexJar, name)
 | |
| 		android.AssertStringDoesContain(t, name+" apex copy command", copyCommands, expectedCopyCommand)
 | |
| 	}
 | |
| 
 | |
| 	checkFragmentExportedDexJar("foo", "out/soong/.intermediates/mybootclasspathfragment/android_common_apex10000/hiddenapi-modular/encoded/foo.jar")
 | |
| 	checkFragmentExportedDexJar("bar", "out/soong/.intermediates/mybootclasspathfragment/android_common_apex10000/hiddenapi-modular/encoded/bar.jar")
 | |
| }
 | |
| 
 | |
| func getDexJarPath(result *android.TestResult, name string) string {
 | |
| 	module := result.Module(name, "android_common")
 | |
| 	return module.(java.UsesLibraryDependency).DexJarBuildPath().Path().RelativeToTop().String()
 | |
| }
 | |
| 
 | |
| // TestBootclasspathFragment_HiddenAPIList checks to make sure that the correct parameters are
 | |
| // passed to the hiddenapi list tool.
 | |
| func TestBootclasspathFragment_HiddenAPIList(t *testing.T) {
 | |
| 	result := android.GroupFixturePreparers(
 | |
| 		prepareForTestWithBootclasspathFragment,
 | |
| 		prepareForTestWithArtApex,
 | |
| 		prepareForTestWithMyapex,
 | |
| 		// Configure bootclasspath jars to ensure that hidden API encoding is performed on them.
 | |
| 		java.FixtureConfigureBootJars("com.android.art:baz", "com.android.art:quuz"),
 | |
| 		java.FixtureConfigureApexBootJars("myapex:foo", "myapex:bar"),
 | |
| 		// Make sure that the frameworks/base/Android.bp file exists as otherwise hidden API encoding
 | |
| 		// is disabled.
 | |
| 		android.FixtureAddTextFile("frameworks/base/Android.bp", ""),
 | |
| 
 | |
| 		java.PrepareForTestWithJavaSdkLibraryFiles,
 | |
| 		java.FixtureWithLastReleaseApis("foo", "quuz"),
 | |
| 	).RunTestWithBp(t, `
 | |
| 		apex {
 | |
| 			name: "com.android.art",
 | |
| 			key: "com.android.art.key",
 | |
| 			bootclasspath_fragments: ["art-bootclasspath-fragment"],
 | |
| 			updatable: false,
 | |
| 		}
 | |
| 
 | |
| 		apex_key {
 | |
| 			name: "com.android.art.key",
 | |
| 			public_key: "com.android.art.avbpubkey",
 | |
| 			private_key: "com.android.art.pem",
 | |
| 		}
 | |
| 
 | |
| 		java_library {
 | |
| 			name: "baz",
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 			srcs: ["b.java"],
 | |
| 			compile_dex: true,
 | |
| 		}
 | |
| 
 | |
| 		java_sdk_library {
 | |
| 			name: "quuz",
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 			srcs: ["b.java"],
 | |
| 			compile_dex: true,
 | |
| 			public: {enabled: true},
 | |
| 			system: {enabled: true},
 | |
| 			test: {enabled: true},
 | |
| 			module_lib: {enabled: true},
 | |
| 		}
 | |
| 
 | |
| 		bootclasspath_fragment {
 | |
| 			name: "art-bootclasspath-fragment",
 | |
| 			image_name: "art",
 | |
| 			// Must match the "com.android.art:" entries passed to FixtureConfigureBootJars above.
 | |
| 			contents: ["baz", "quuz"],
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 			hidden_api: {
 | |
| 				split_packages: ["*"],
 | |
| 			},
 | |
| 		}
 | |
| 
 | |
| 		apex {
 | |
| 			name: "myapex",
 | |
| 			key: "myapex.key",
 | |
| 			bootclasspath_fragments: [
 | |
| 				"mybootclasspathfragment",
 | |
| 			],
 | |
| 			updatable: false,
 | |
| 		}
 | |
| 
 | |
| 		apex_key {
 | |
| 			name: "myapex.key",
 | |
| 			public_key: "testkey.avbpubkey",
 | |
| 			private_key: "testkey.pem",
 | |
| 		}
 | |
| 
 | |
| 		java_sdk_library {
 | |
| 			name: "foo",
 | |
| 			srcs: ["b.java"],
 | |
| 			shared_library: false,
 | |
| 			public: {enabled: true},
 | |
| 			apex_available: [
 | |
| 				"myapex",
 | |
| 			],
 | |
| 		}
 | |
| 
 | |
| 		java_library {
 | |
| 			name: "bar",
 | |
| 			srcs: ["b.java"],
 | |
| 			installable: true,
 | |
| 			apex_available: [
 | |
| 				"myapex",
 | |
| 			],
 | |
| 		}
 | |
| 
 | |
| 		bootclasspath_fragment {
 | |
| 			name: "mybootclasspathfragment",
 | |
| 			contents: [
 | |
| 				"foo",
 | |
| 				"bar",
 | |
| 			],
 | |
| 			apex_available: [
 | |
| 				"myapex",
 | |
| 			],
 | |
| 			fragments: [
 | |
| 				{
 | |
| 					apex: "com.android.art",
 | |
| 					module: "art-bootclasspath-fragment",
 | |
| 				},
 | |
| 			],
 | |
| 			hidden_api: {
 | |
| 				split_packages: ["*"],
 | |
| 			},
 | |
| 		}
 | |
| 	`)
 | |
| 
 | |
| 	java.CheckModuleDependencies(t, result.TestContext, "mybootclasspathfragment", "android_common_apex10000", []string{
 | |
| 		"art-bootclasspath-fragment",
 | |
| 		"bar",
 | |
| 		"dex2oatd",
 | |
| 		"foo",
 | |
| 	})
 | |
| 
 | |
| 	fooStubs := getDexJarPath(result, "foo.stubs")
 | |
| 	quuzPublicStubs := getDexJarPath(result, "quuz.stubs")
 | |
| 	quuzSystemStubs := getDexJarPath(result, "quuz.stubs.system")
 | |
| 	quuzTestStubs := getDexJarPath(result, "quuz.stubs.test")
 | |
| 	quuzModuleLibStubs := getDexJarPath(result, "quuz.stubs.module_lib")
 | |
| 
 | |
| 	// Make sure that the fragment uses the quuz stub dex jars when generating the hidden API flags.
 | |
| 	fragment := result.ModuleForTests("mybootclasspathfragment", "android_common_apex10000")
 | |
| 
 | |
| 	rule := fragment.Rule("modularHiddenAPIStubFlagsFile")
 | |
| 	command := rule.RuleParams.Command
 | |
| 	android.AssertStringDoesContain(t, "check correct rule", command, "hiddenapi list")
 | |
| 
 | |
| 	// Make sure that the quuz stubs are available for resolving references from the implementation
 | |
| 	// boot dex jars provided by this module.
 | |
| 	android.AssertStringDoesContain(t, "quuz widest", command, "--dependency-stub-dex="+quuzModuleLibStubs)
 | |
| 
 | |
| 	// Make sure that the quuz stubs are available for resolving references from the different API
 | |
| 	// stubs provided by this module.
 | |
| 	android.AssertStringDoesContain(t, "public", command, "--public-stub-classpath="+quuzPublicStubs+":"+fooStubs)
 | |
| 	android.AssertStringDoesContain(t, "system", command, "--system-stub-classpath="+quuzSystemStubs+":"+fooStubs)
 | |
| 	android.AssertStringDoesContain(t, "test", command, "--test-stub-classpath="+quuzTestStubs+":"+fooStubs)
 | |
| }
 | |
| 
 | |
| // TestBootclasspathFragment_AndroidNonUpdatable checks to make sure that setting
 | |
| // additional_stubs: ["android-non-updatable"] causes the source android-non-updatable modules to be
 | |
| // added to the hiddenapi list tool.
 | |
| func TestBootclasspathFragment_AndroidNonUpdatable(t *testing.T) {
 | |
| 	result := android.GroupFixturePreparers(
 | |
| 		prepareForTestWithBootclasspathFragment,
 | |
| 		prepareForTestWithArtApex,
 | |
| 		prepareForTestWithMyapex,
 | |
| 		// Configure bootclasspath jars to ensure that hidden API encoding is performed on them.
 | |
| 		java.FixtureConfigureBootJars("com.android.art:baz", "com.android.art:quuz"),
 | |
| 		java.FixtureConfigureApexBootJars("myapex:foo", "myapex:bar"),
 | |
| 		// Make sure that the frameworks/base/Android.bp file exists as otherwise hidden API encoding
 | |
| 		// is disabled.
 | |
| 		android.FixtureAddTextFile("frameworks/base/Android.bp", ""),
 | |
| 
 | |
| 		java.PrepareForTestWithJavaSdkLibraryFiles,
 | |
| 		java.FixtureWithLastReleaseApis("foo", "android-non-updatable"),
 | |
| 	).RunTestWithBp(t, `
 | |
| 		java_sdk_library {
 | |
| 			name: "android-non-updatable",
 | |
| 			srcs: ["b.java"],
 | |
| 			compile_dex: true,
 | |
| 			public: {
 | |
| 				enabled: true,
 | |
| 			},
 | |
| 			system: {
 | |
| 				enabled: true,
 | |
| 			},
 | |
| 			test: {
 | |
| 				enabled: true,
 | |
| 			},
 | |
| 			module_lib: {
 | |
| 				enabled: true,
 | |
| 			},
 | |
| 		}
 | |
| 
 | |
| 		apex {
 | |
| 			name: "com.android.art",
 | |
| 			key: "com.android.art.key",
 | |
| 			bootclasspath_fragments: ["art-bootclasspath-fragment"],
 | |
| 			updatable: false,
 | |
| 		}
 | |
| 
 | |
| 		apex_key {
 | |
| 			name: "com.android.art.key",
 | |
| 			public_key: "com.android.art.avbpubkey",
 | |
| 			private_key: "com.android.art.pem",
 | |
| 		}
 | |
| 
 | |
| 		java_library {
 | |
| 			name: "baz",
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 			srcs: ["b.java"],
 | |
| 			compile_dex: true,
 | |
| 		}
 | |
| 
 | |
| 		java_library {
 | |
| 			name: "quuz",
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 			srcs: ["b.java"],
 | |
| 			compile_dex: true,
 | |
| 		}
 | |
| 
 | |
| 		bootclasspath_fragment {
 | |
| 			name: "art-bootclasspath-fragment",
 | |
| 			image_name: "art",
 | |
| 			// Must match the "com.android.art:" entries passed to FixtureConfigureBootJars above.
 | |
| 			contents: ["baz", "quuz"],
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 			hidden_api: {
 | |
| 				split_packages: ["*"],
 | |
| 			},
 | |
| 		}
 | |
| 
 | |
| 		apex {
 | |
| 			name: "myapex",
 | |
| 			key: "myapex.key",
 | |
| 			bootclasspath_fragments: [
 | |
| 				"mybootclasspathfragment",
 | |
| 			],
 | |
| 			updatable: false,
 | |
| 		}
 | |
| 
 | |
| 		apex_key {
 | |
| 			name: "myapex.key",
 | |
| 			public_key: "testkey.avbpubkey",
 | |
| 			private_key: "testkey.pem",
 | |
| 		}
 | |
| 
 | |
| 		java_sdk_library {
 | |
| 			name: "foo",
 | |
| 			srcs: ["b.java"],
 | |
| 			shared_library: false,
 | |
| 			public: {enabled: true},
 | |
| 			apex_available: [
 | |
| 				"myapex",
 | |
| 			],
 | |
| 		}
 | |
| 
 | |
| 		java_library {
 | |
| 			name: "bar",
 | |
| 			srcs: ["b.java"],
 | |
| 			installable: true,
 | |
| 			apex_available: [
 | |
| 				"myapex",
 | |
| 			],
 | |
| 		}
 | |
| 
 | |
| 		bootclasspath_fragment {
 | |
| 			name: "mybootclasspathfragment",
 | |
| 			contents: [
 | |
| 				"foo",
 | |
| 				"bar",
 | |
| 			],
 | |
| 			apex_available: [
 | |
| 				"myapex",
 | |
| 			],
 | |
| 			additional_stubs: ["android-non-updatable"],
 | |
| 			fragments: [
 | |
| 				{
 | |
| 					apex: "com.android.art",
 | |
| 					module: "art-bootclasspath-fragment",
 | |
| 				},
 | |
| 			],
 | |
| 			hidden_api: {
 | |
| 				split_packages: ["*"],
 | |
| 			},
 | |
| 		}
 | |
| 	`)
 | |
| 
 | |
| 	java.CheckModuleDependencies(t, result.TestContext, "mybootclasspathfragment", "android_common_apex10000", []string{
 | |
| 		"android-non-updatable.stubs",
 | |
| 		"android-non-updatable.stubs.module_lib",
 | |
| 		"android-non-updatable.stubs.system",
 | |
| 		"android-non-updatable.stubs.test",
 | |
| 		"art-bootclasspath-fragment",
 | |
| 		"bar",
 | |
| 		"dex2oatd",
 | |
| 		"foo",
 | |
| 	})
 | |
| 
 | |
| 	nonUpdatablePublicStubs := getDexJarPath(result, "android-non-updatable.stubs")
 | |
| 	nonUpdatableSystemStubs := getDexJarPath(result, "android-non-updatable.stubs.system")
 | |
| 	nonUpdatableTestStubs := getDexJarPath(result, "android-non-updatable.stubs.test")
 | |
| 	nonUpdatableModuleLibStubs := getDexJarPath(result, "android-non-updatable.stubs.module_lib")
 | |
| 
 | |
| 	// Make sure that the fragment uses the android-non-updatable modules when generating the hidden
 | |
| 	// API flags.
 | |
| 	fragment := result.ModuleForTests("mybootclasspathfragment", "android_common_apex10000")
 | |
| 
 | |
| 	rule := fragment.Rule("modularHiddenAPIStubFlagsFile")
 | |
| 	command := rule.RuleParams.Command
 | |
| 	android.AssertStringDoesContain(t, "check correct rule", command, "hiddenapi list")
 | |
| 
 | |
| 	// Make sure that the module_lib non-updatable stubs are available for resolving references from
 | |
| 	// the implementation boot dex jars provided by this module.
 | |
| 	android.AssertStringDoesContain(t, "android-non-updatable widest", command, "--dependency-stub-dex="+nonUpdatableModuleLibStubs)
 | |
| 
 | |
| 	// Make sure that the appropriate non-updatable stubs are available for resolving references from
 | |
| 	// the different API stubs provided by this module.
 | |
| 	android.AssertStringDoesContain(t, "public", command, "--public-stub-classpath="+nonUpdatablePublicStubs)
 | |
| 	android.AssertStringDoesContain(t, "system", command, "--system-stub-classpath="+nonUpdatableSystemStubs)
 | |
| 	android.AssertStringDoesContain(t, "test", command, "--test-stub-classpath="+nonUpdatableTestStubs)
 | |
| }
 | |
| 
 | |
| // TestBootclasspathFragment_AndroidNonUpdatable_AlwaysUsePrebuiltSdks checks to make sure that
 | |
| // setting additional_stubs: ["android-non-updatable"] causes the prebuilt android-non-updatable
 | |
| // modules to be added to the hiddenapi list tool.
 | |
| func TestBootclasspathFragment_AndroidNonUpdatable_AlwaysUsePrebuiltSdks(t *testing.T) {
 | |
| 	result := android.GroupFixturePreparers(
 | |
| 		prepareForTestWithBootclasspathFragment,
 | |
| 		java.PrepareForTestWithJavaDefaultModules,
 | |
| 		prepareForTestWithArtApex,
 | |
| 		prepareForTestWithMyapex,
 | |
| 		// Configure bootclasspath jars to ensure that hidden API encoding is performed on them.
 | |
| 		java.FixtureConfigureBootJars("com.android.art:baz", "com.android.art:quuz"),
 | |
| 		java.FixtureConfigureApexBootJars("myapex:foo", "myapex:bar"),
 | |
| 		// Make sure that the frameworks/base/Android.bp file exists as otherwise hidden API encoding
 | |
| 		// is disabled.
 | |
| 		android.FixtureAddTextFile("frameworks/base/Android.bp", ""),
 | |
| 
 | |
| 		android.FixtureModifyProductVariables(func(variables android.FixtureProductVariables) {
 | |
| 			variables.Always_use_prebuilt_sdks = proptools.BoolPtr(true)
 | |
| 		}),
 | |
| 
 | |
| 		java.PrepareForTestWithJavaSdkLibraryFiles,
 | |
| 		java.FixtureWithPrebuiltApis(map[string][]string{
 | |
| 			"current": {"android-non-updatable"},
 | |
| 			"30":      {"foo"},
 | |
| 		}),
 | |
| 	).RunTestWithBp(t, `
 | |
| 		apex {
 | |
| 			name: "com.android.art",
 | |
| 			key: "com.android.art.key",
 | |
| 			bootclasspath_fragments: ["art-bootclasspath-fragment"],
 | |
| 			updatable: false,
 | |
| 		}
 | |
| 
 | |
| 		apex_key {
 | |
| 			name: "com.android.art.key",
 | |
| 			public_key: "com.android.art.avbpubkey",
 | |
| 			private_key: "com.android.art.pem",
 | |
| 		}
 | |
| 
 | |
| 		java_library {
 | |
| 			name: "baz",
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 			srcs: ["b.java"],
 | |
| 			compile_dex: true,
 | |
| 		}
 | |
| 
 | |
| 		java_library {
 | |
| 			name: "quuz",
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 			srcs: ["b.java"],
 | |
| 			compile_dex: true,
 | |
| 		}
 | |
| 
 | |
| 		bootclasspath_fragment {
 | |
| 			name: "art-bootclasspath-fragment",
 | |
| 			image_name: "art",
 | |
| 			// Must match the "com.android.art:" entries passed to FixtureConfigureBootJars above.
 | |
| 			contents: ["baz", "quuz"],
 | |
| 			apex_available: [
 | |
| 				"com.android.art",
 | |
| 			],
 | |
| 			hidden_api: {
 | |
| 				split_packages: ["*"],
 | |
| 			},
 | |
| 		}
 | |
| 
 | |
| 		apex {
 | |
| 			name: "myapex",
 | |
| 			key: "myapex.key",
 | |
| 			bootclasspath_fragments: [
 | |
| 				"mybootclasspathfragment",
 | |
| 			],
 | |
| 			updatable: false,
 | |
| 		}
 | |
| 
 | |
| 		apex_key {
 | |
| 			name: "myapex.key",
 | |
| 			public_key: "testkey.avbpubkey",
 | |
| 			private_key: "testkey.pem",
 | |
| 		}
 | |
| 
 | |
| 		java_sdk_library {
 | |
| 			name: "foo",
 | |
| 			srcs: ["b.java"],
 | |
| 			shared_library: false,
 | |
| 			public: {enabled: true},
 | |
| 			apex_available: [
 | |
| 				"myapex",
 | |
| 			],
 | |
| 		}
 | |
| 
 | |
| 		java_library {
 | |
| 			name: "bar",
 | |
| 			srcs: ["b.java"],
 | |
| 			installable: true,
 | |
| 			apex_available: [
 | |
| 				"myapex",
 | |
| 			],
 | |
| 		}
 | |
| 
 | |
| 		bootclasspath_fragment {
 | |
| 			name: "mybootclasspathfragment",
 | |
| 			contents: [
 | |
| 				"foo",
 | |
| 				"bar",
 | |
| 			],
 | |
| 			apex_available: [
 | |
| 				"myapex",
 | |
| 			],
 | |
| 			additional_stubs: ["android-non-updatable"],
 | |
| 			fragments: [
 | |
| 				{
 | |
| 					apex: "com.android.art",
 | |
| 					module: "art-bootclasspath-fragment",
 | |
| 				},
 | |
| 			],
 | |
| 			hidden_api: {
 | |
| 				split_packages: ["*"],
 | |
| 			},
 | |
| 		}
 | |
| 	`)
 | |
| 
 | |
| 	java.CheckModuleDependencies(t, result.TestContext, "mybootclasspathfragment", "android_common_apex10000", []string{
 | |
| 		"art-bootclasspath-fragment",
 | |
| 		"bar",
 | |
| 		"dex2oatd",
 | |
| 		"foo",
 | |
| 		"prebuilt_sdk_module-lib_current_android-non-updatable",
 | |
| 		"prebuilt_sdk_public_current_android-non-updatable",
 | |
| 		"prebuilt_sdk_system_current_android-non-updatable",
 | |
| 		"prebuilt_sdk_test_current_android-non-updatable",
 | |
| 	})
 | |
| 
 | |
| 	nonUpdatablePublicStubs := getDexJarPath(result, "sdk_public_current_android-non-updatable")
 | |
| 	nonUpdatableSystemStubs := getDexJarPath(result, "sdk_system_current_android-non-updatable")
 | |
| 	nonUpdatableTestStubs := getDexJarPath(result, "sdk_test_current_android-non-updatable")
 | |
| 	nonUpdatableModuleLibStubs := getDexJarPath(result, "sdk_module-lib_current_android-non-updatable")
 | |
| 
 | |
| 	// Make sure that the fragment uses the android-non-updatable modules when generating the hidden
 | |
| 	// API flags.
 | |
| 	fragment := result.ModuleForTests("mybootclasspathfragment", "android_common_apex10000")
 | |
| 
 | |
| 	rule := fragment.Rule("modularHiddenAPIStubFlagsFile")
 | |
| 	command := rule.RuleParams.Command
 | |
| 	android.AssertStringDoesContain(t, "check correct rule", command, "hiddenapi list")
 | |
| 
 | |
| 	// Make sure that the module_lib non-updatable stubs are available for resolving references from
 | |
| 	// the implementation boot dex jars provided by this module.
 | |
| 	android.AssertStringDoesContain(t, "android-non-updatable widest", command, "--dependency-stub-dex="+nonUpdatableModuleLibStubs)
 | |
| 
 | |
| 	// Make sure that the appropriate non-updatable stubs are available for resolving references from
 | |
| 	// the different API stubs provided by this module.
 | |
| 	android.AssertStringDoesContain(t, "public", command, "--public-stub-classpath="+nonUpdatablePublicStubs)
 | |
| 	android.AssertStringDoesContain(t, "system", command, "--system-stub-classpath="+nonUpdatableSystemStubs)
 | |
| 	android.AssertStringDoesContain(t, "test", command, "--test-stub-classpath="+nonUpdatableTestStubs)
 | |
| }
 | |
| 
 | |
| // TODO(b/177892522) - add test for host apex.
 |