crt_pad_segment adds a NOTE to the ELF which is used by the binoic loader to determine whether it should pad segments when mapping them into the virtual address space, such that there are no gaps between mappings of consecutive segments. This avoids an increase in unreclaimable kernel slab memory usage for VMAs on devices where the runtime-page-size > elf-segment-p_align. Since -fandroid-pad-segment [1] respects -nostdlib used in android platform builds, soong must link in crt_pad_segment to platform shared libraries. For simplicity, link crt_pad_segment everywhere that crtend_so is applicable, ignoring nocrt property, as there is no other reason to track these separately. Example: ❯ readelf -WS /system/lib64/libc++.so [Output simplified] ... Section Headers: [Nr] Name Type Address Off Size ES Flg Lk Inf Al ... [ 2] .note.android.pad_segment NOTE 0000000000000288 000288 000018 00 A 0 0 4 ... [1] https://github.com/llvm/llvm-project/pull/77244 Bug: 316403210 Test: readelf -WS <lib>.so Change-Id: Icc06611376cfd5ee4de7281b4134f9f8ffe7ca60 Signed-off-by: Kalesh Singh <kaleshsingh@google.com>
		
			
				
	
	
		
			1574 lines
		
	
	
		
			55 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			1574 lines
		
	
	
		
			55 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 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 rust
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"path/filepath"
 | |
| 	"reflect"
 | |
| 	"strings"
 | |
| 	"testing"
 | |
| 
 | |
| 	"android/soong/android"
 | |
| 	"android/soong/cc"
 | |
| )
 | |
| 
 | |
| func TestVendorSnapshotCapture(t *testing.T) {
 | |
| 	bp := `
 | |
| 	rust_ffi {
 | |
| 		name: "libffivendor_available",
 | |
| 		crate_name: "ffivendor_available",
 | |
| 		srcs: ["lib.rs"],
 | |
| 		vendor_available: true,
 | |
| 		include_dirs: ["rust_headers/"],
 | |
| 	}
 | |
| 
 | |
| 	rust_ffi {
 | |
| 		name: "libffivendor",
 | |
| 		crate_name: "ffivendor",
 | |
| 		srcs: ["lib.rs"],
 | |
| 		vendor: true,
 | |
| 		include_dirs: ["rust_headers/"],
 | |
| 	}
 | |
| 
 | |
| 	rust_library {
 | |
| 		name: "librustvendor_available",
 | |
| 		crate_name: "rustvendor_available",
 | |
| 		srcs: ["lib.rs"],
 | |
| 		vendor_available: true,
 | |
| 	}
 | |
| 
 | |
| 	rust_library {
 | |
| 		name: "librustvendor",
 | |
| 		crate_name: "rustvendor",
 | |
| 		srcs: ["lib.rs"],
 | |
| 		vendor: true,
 | |
| 	}
 | |
| 
 | |
| 	rust_binary {
 | |
| 		name: "vendor_available_bin",
 | |
| 		vendor_available: true,
 | |
| 		srcs: ["srcs/lib.rs"],
 | |
| 	}
 | |
| 
 | |
| 	rust_binary {
 | |
| 		name: "vendor_bin",
 | |
| 		vendor: true,
 | |
| 		srcs: ["srcs/lib.rs"],
 | |
| 	}
 | |
|     `
 | |
| 	skipTestIfOsNotSupported(t)
 | |
| 	result := android.GroupFixturePreparers(
 | |
| 		prepareForRustTest,
 | |
| 		rustMockedFiles.AddToFixture(),
 | |
| 		android.FixtureModifyProductVariables(
 | |
| 			func(variables android.FixtureProductVariables) {
 | |
| 				variables.DeviceVndkVersion = StringPtr("current")
 | |
| 				variables.Platform_vndk_version = StringPtr("29")
 | |
| 			},
 | |
| 		),
 | |
| 	).RunTestWithBp(t, bp)
 | |
| 	ctx := result.TestContext
 | |
| 
 | |
| 	// Check Vendor snapshot output.
 | |
| 
 | |
| 	snapshotDir := "vendor-snapshot"
 | |
| 	snapshotVariantPath := filepath.Join("out/soong", snapshotDir, "arm64")
 | |
| 	snapshotSingleton := ctx.SingletonForTests("vendor-snapshot")
 | |
| 	var jsonFiles []string
 | |
| 	for _, arch := range [][]string{
 | |
| 		[]string{"arm64", "armv8-a"},
 | |
| 		[]string{"arm", "armv7-a-neon"},
 | |
| 	} {
 | |
| 		archType := arch[0]
 | |
| 		archVariant := arch[1]
 | |
| 		archDir := fmt.Sprintf("arch-%s-%s", archType, archVariant)
 | |
| 
 | |
| 		// For shared libraries, only non-VNDK vendor_available modules are captured
 | |
| 		sharedVariant := fmt.Sprintf("android_vendor.29_%s_%s_shared", archType, archVariant)
 | |
| 		sharedDir := filepath.Join(snapshotVariantPath, archDir, "shared")
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "libffivendor_available", "libffivendor_available.so", sharedDir, sharedVariant)
 | |
| 		jsonFiles = append(jsonFiles,
 | |
| 			filepath.Join(sharedDir, "libffivendor_available.so.json"))
 | |
| 
 | |
| 		// For static libraries, all vendor:true and vendor_available modules (including VNDK) are captured.
 | |
| 		staticVariant := fmt.Sprintf("android_vendor.29_%s_%s_static", archType, archVariant)
 | |
| 		staticDir := filepath.Join(snapshotVariantPath, archDir, "static")
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "libffivendor_available", "libffivendor_available.a", staticDir, staticVariant)
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "libffivendor", "libffivendor.a", staticDir, staticVariant)
 | |
| 		jsonFiles = append(jsonFiles,
 | |
| 			filepath.Join(staticDir, "libffivendor_available.a.json"))
 | |
| 		jsonFiles = append(jsonFiles,
 | |
| 			filepath.Join(staticDir, "libffivendor.a.json"))
 | |
| 
 | |
| 		// For rlib libraries, all vendor:true and vendor_available modules (including VNDK) are captured.
 | |
| 		rlibVariant := fmt.Sprintf("android_vendor.29_%s_%s_rlib_dylib-std", archType, archVariant)
 | |
| 		rlibDir := filepath.Join(snapshotVariantPath, archDir, "rlib")
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librustvendor_available", "librustvendor_available.rlib", rlibDir, rlibVariant)
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librustvendor", "librustvendor.rlib", rlibDir, rlibVariant)
 | |
| 		jsonFiles = append(jsonFiles,
 | |
| 			filepath.Join(rlibDir, "librustvendor_available.rlib.json"))
 | |
| 		jsonFiles = append(jsonFiles,
 | |
| 			filepath.Join(rlibDir, "librustvendor.rlib.json"))
 | |
| 
 | |
| 		// For rlib libraries, all rlib-std variants vendor:true and vendor_available modules (including VNDK) are captured.
 | |
| 		rlibStdVariant := fmt.Sprintf("android_vendor.29_%s_%s_rlib_rlib-std", archType, archVariant)
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librustvendor_available", "librustvendor_available.rlib-std.rlib", rlibDir, rlibStdVariant)
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librustvendor", "librustvendor.rlib-std.rlib", rlibDir, rlibStdVariant)
 | |
| 		jsonFiles = append(jsonFiles,
 | |
| 			filepath.Join(rlibDir, "librustvendor_available.rlib.json"))
 | |
| 		jsonFiles = append(jsonFiles,
 | |
| 			filepath.Join(rlibDir, "librustvendor.rlib.json"))
 | |
| 
 | |
| 		// For dylib libraries, all vendor:true and vendor_available modules (including VNDK) are captured.
 | |
| 		dylibVariant := fmt.Sprintf("android_vendor.29_%s_%s_dylib", archType, archVariant)
 | |
| 		dylibDir := filepath.Join(snapshotVariantPath, archDir, "dylib")
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librustvendor_available", "librustvendor_available.dylib.so", dylibDir, dylibVariant)
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librustvendor", "librustvendor.dylib.so", dylibDir, dylibVariant)
 | |
| 		jsonFiles = append(jsonFiles,
 | |
| 			filepath.Join(dylibDir, "librustvendor_available.dylib.so.json"))
 | |
| 		jsonFiles = append(jsonFiles,
 | |
| 			filepath.Join(dylibDir, "librustvendor.dylib.so.json"))
 | |
| 
 | |
| 		// For binary executables, all vendor:true and vendor_available modules are captured.
 | |
| 		if archType == "arm64" {
 | |
| 			binaryVariant := fmt.Sprintf("android_vendor.29_%s_%s", archType, archVariant)
 | |
| 			binaryDir := filepath.Join(snapshotVariantPath, archDir, "binary")
 | |
| 			cc.CheckSnapshot(t, ctx, snapshotSingleton, "vendor_available_bin", "vendor_available_bin", binaryDir, binaryVariant)
 | |
| 			cc.CheckSnapshot(t, ctx, snapshotSingleton, "vendor_bin", "vendor_bin", binaryDir, binaryVariant)
 | |
| 			jsonFiles = append(jsonFiles,
 | |
| 				filepath.Join(binaryDir, "vendor_available_bin.json"))
 | |
| 			jsonFiles = append(jsonFiles,
 | |
| 				filepath.Join(binaryDir, "vendor_bin.json"))
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	for _, jsonFile := range jsonFiles {
 | |
| 		// verify all json files exist
 | |
| 		if snapshotSingleton.MaybeOutput(jsonFile).Rule == nil {
 | |
| 			t.Errorf("%q expected but not found; #%v", jsonFile, jsonFiles)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// fake snapshot should have all outputs in the normal snapshot.
 | |
| 	fakeSnapshotSingleton := ctx.SingletonForTests("vendor-fake-snapshot")
 | |
| 
 | |
| 	for _, output := range snapshotSingleton.AllOutputs() {
 | |
| 		fakeOutput := strings.Replace(output, "/vendor-snapshot/", "/fake/vendor-snapshot/", 1)
 | |
| 		if fakeSnapshotSingleton.MaybeOutput(fakeOutput).Rule == nil {
 | |
| 			t.Errorf("%q expected but not found", fakeOutput)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestVendorSnapshotDirected(t *testing.T) {
 | |
| 	bp := `
 | |
| 	rust_ffi_shared {
 | |
| 		name: "libffivendor_available",
 | |
| 		crate_name: "ffivendor_available",
 | |
| 		srcs: ["lib.rs"],
 | |
| 		vendor_available: true,
 | |
| 	}
 | |
| 
 | |
| 	rust_library {
 | |
| 		name: "librustvendor_available",
 | |
| 		crate_name: "rustvendor_available",
 | |
| 		srcs: ["lib.rs"],
 | |
| 		vendor_available: true,
 | |
| 	}
 | |
| 
 | |
| 	rust_ffi_shared {
 | |
| 		name: "libffivendor_exclude",
 | |
| 		crate_name: "ffivendor_exclude",
 | |
| 		srcs: ["lib.rs"],
 | |
| 		vendor_available: true,
 | |
| 	}
 | |
| 
 | |
| 	rust_library {
 | |
| 		name: "librustvendor_exclude",
 | |
| 		crate_name: "rustvendor_exclude",
 | |
| 		srcs: ["lib.rs"],
 | |
| 		vendor_available: true,
 | |
| 	}
 | |
| `
 | |
| 	ctx := testRustVndk(t, bp)
 | |
| 	ctx.Config().TestProductVariables.VendorSnapshotModules = make(map[string]bool)
 | |
| 	ctx.Config().TestProductVariables.VendorSnapshotModules["librustvendor_available"] = true
 | |
| 	ctx.Config().TestProductVariables.VendorSnapshotModules["libffivendor_available"] = true
 | |
| 	ctx.Config().TestProductVariables.DirectedVendorSnapshot = true
 | |
| 
 | |
| 	// Check Vendor snapshot output.
 | |
| 
 | |
| 	snapshotDir := "vendor-snapshot"
 | |
| 	snapshotVariantPath := filepath.Join("out/soong", snapshotDir, "arm64")
 | |
| 	snapshotSingleton := ctx.SingletonForTests("vendor-snapshot")
 | |
| 
 | |
| 	var includeJsonFiles []string
 | |
| 
 | |
| 	for _, arch := range [][]string{
 | |
| 		[]string{"arm64", "armv8-a"},
 | |
| 		[]string{"arm", "armv7-a-neon"},
 | |
| 	} {
 | |
| 		archType := arch[0]
 | |
| 		archVariant := arch[1]
 | |
| 		archDir := fmt.Sprintf("arch-%s-%s", archType, archVariant)
 | |
| 
 | |
| 		sharedVariant := fmt.Sprintf("android_vendor.29_%s_%s_shared", archType, archVariant)
 | |
| 		rlibVariant := fmt.Sprintf("android_vendor.29_%s_%s_rlib_dylib-std", archType, archVariant)
 | |
| 		rlibRlibStdVariant := fmt.Sprintf("android_vendor.29_%s_%s_rlib_rlib-std", archType, archVariant)
 | |
| 		sharedDir := filepath.Join(snapshotVariantPath, archDir, "shared")
 | |
| 		rlibDir := filepath.Join(snapshotVariantPath, archDir, "rlib")
 | |
| 		dylibVariant := fmt.Sprintf("android_vendor.29_%s_%s_dylib", archType, archVariant)
 | |
| 		dylibDir := filepath.Join(snapshotVariantPath, archDir, "dylib")
 | |
| 
 | |
| 		// Included modules
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librustvendor_available", "librustvendor_available.rlib", rlibDir, rlibVariant)
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librustvendor_available", "librustvendor_available.rlib-std.rlib", rlibDir, rlibRlibStdVariant)
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librustvendor_available", "librustvendor_available.dylib.so", dylibDir, dylibVariant)
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "libffivendor_available", "libffivendor_available.so", sharedDir, sharedVariant)
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(rlibDir, "librustvendor_available.rlib.json"))
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(rlibDir, "librustvendor_available.rlib-std.rlib.json"))
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(dylibDir, "librustvendor_available.dylib.so.json"))
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(sharedDir, "libffivendor_available.so.json"))
 | |
| 
 | |
| 		// Excluded modules. Modules not included in the directed vendor snapshot
 | |
| 		// are still include as fake modules.
 | |
| 		cc.CheckSnapshotRule(t, ctx, snapshotSingleton, "librustvendor_exclude", "librustvendor_exclude.rlib", rlibDir, rlibVariant)
 | |
| 		cc.CheckSnapshotRule(t, ctx, snapshotSingleton, "librustvendor_exclude", "librustvendor_exclude.rlib-std.rlib", rlibDir, rlibRlibStdVariant)
 | |
| 		cc.CheckSnapshotRule(t, ctx, snapshotSingleton, "librustvendor_exclude", "librustvendor_exclude.dylib.so", dylibDir, dylibVariant)
 | |
| 		cc.CheckSnapshotRule(t, ctx, snapshotSingleton, "libffivendor_exclude", "libffivendor_exclude.so", sharedDir, sharedVariant)
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(rlibDir, "librustvendor_exclude.rlib.json"))
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(rlibDir, "librustvendor_exclude.rlib-std.rlib.json"))
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(dylibDir, "librustvendor_exclude.dylib.so.json"))
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(sharedDir, "libffivendor_exclude.so.json"))
 | |
| 	}
 | |
| 
 | |
| 	// Verify that each json file for an included module has a rule.
 | |
| 	for _, jsonFile := range includeJsonFiles {
 | |
| 		if snapshotSingleton.MaybeOutput(jsonFile).Rule == nil {
 | |
| 			t.Errorf("include json file %q not found", jsonFile)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestVendorSnapshotExclude(t *testing.T) {
 | |
| 
 | |
| 	// This test verifies that the exclude_from_vendor_snapshot property
 | |
| 	// makes its way from the Android.bp source file into the module data
 | |
| 	// structure. It also verifies that modules are correctly included or
 | |
| 	// excluded in the vendor snapshot based on their path (framework or
 | |
| 	// vendor) and the exclude_from_vendor_snapshot property.
 | |
| 
 | |
| 	frameworkBp := `
 | |
| 		rust_ffi_shared {
 | |
| 			name: "libinclude",
 | |
| 			crate_name: "include",
 | |
| 			srcs: ["include.rs"],
 | |
| 			vendor_available: true,
 | |
| 		}
 | |
| 
 | |
| 		rust_ffi_shared {
 | |
| 			name: "libexclude",
 | |
| 			crate_name: "exclude",
 | |
| 			srcs: ["exclude.rs"],
 | |
| 			vendor: true,
 | |
| 			exclude_from_vendor_snapshot: true,
 | |
| 		}
 | |
| 
 | |
| 		rust_ffi_shared {
 | |
| 			name: "libavailable_exclude",
 | |
| 			crate_name: "available_exclude",
 | |
| 			srcs: ["lib.rs"],
 | |
| 			vendor_available: true,
 | |
| 			exclude_from_vendor_snapshot: true,
 | |
| 		}
 | |
| 
 | |
| 		rust_library {
 | |
| 			name: "librust_include",
 | |
| 			crate_name: "rust_include",
 | |
| 			srcs: ["include.rs"],
 | |
| 			vendor_available: true,
 | |
| 		}
 | |
| 
 | |
| 		rust_library {
 | |
| 			name: "librust_exclude",
 | |
| 			crate_name: "rust_exclude",
 | |
| 			srcs: ["exclude.rs"],
 | |
| 			vendor: true,
 | |
| 			exclude_from_vendor_snapshot: true,
 | |
| 		}
 | |
| 
 | |
| 		rust_library {
 | |
| 			name: "librust_available_exclude",
 | |
| 			crate_name: "rust_available_exclude",
 | |
| 			srcs: ["lib.rs"],
 | |
| 			vendor_available: true,
 | |
| 			exclude_from_vendor_snapshot: true,
 | |
| 		}
 | |
| 	`
 | |
| 
 | |
| 	mockFS := map[string][]byte{
 | |
| 		"framework/Android.bp": []byte(frameworkBp),
 | |
| 		"framework/include.rs": nil,
 | |
| 		"framework/exclude.rs": nil,
 | |
| 	}
 | |
| 
 | |
| 	ctx := testRustVndkFs(t, "", mockFS)
 | |
| 
 | |
| 	// Test an include and exclude framework module.
 | |
| 	cc.AssertExcludeFromVendorSnapshotIs(t, ctx, "libinclude", false, sharedVendorVariant)
 | |
| 	cc.AssertExcludeFromVendorSnapshotIs(t, ctx, "libexclude", true, sharedVendorVariant)
 | |
| 	cc.AssertExcludeFromVendorSnapshotIs(t, ctx, "libavailable_exclude", true, sharedVendorVariant)
 | |
| 
 | |
| 	cc.AssertExcludeFromVendorSnapshotIs(t, ctx, "librust_include", false, rlibVendorVariant)
 | |
| 	cc.AssertExcludeFromVendorSnapshotIs(t, ctx, "librust_exclude", true, rlibVendorVariant)
 | |
| 	cc.AssertExcludeFromVendorSnapshotIs(t, ctx, "librust_available_exclude", true, rlibVendorVariant)
 | |
| 
 | |
| 	cc.AssertExcludeFromVendorSnapshotIs(t, ctx, "librust_include", false, rlibDylibStdVendorVariant)
 | |
| 	cc.AssertExcludeFromVendorSnapshotIs(t, ctx, "librust_exclude", true, rlibDylibStdVendorVariant)
 | |
| 	cc.AssertExcludeFromVendorSnapshotIs(t, ctx, "librust_available_exclude", true, rlibDylibStdVendorVariant)
 | |
| 
 | |
| 	cc.AssertExcludeFromVendorSnapshotIs(t, ctx, "librust_include", false, dylibVendorVariant)
 | |
| 	cc.AssertExcludeFromVendorSnapshotIs(t, ctx, "librust_exclude", true, dylibVendorVariant)
 | |
| 	cc.AssertExcludeFromVendorSnapshotIs(t, ctx, "librust_available_exclude", true, dylibVendorVariant)
 | |
| 
 | |
| 	// Verify the content of the vendor snapshot.
 | |
| 
 | |
| 	snapshotDir := "vendor-snapshot"
 | |
| 	snapshotVariantPath := filepath.Join("out/soong", snapshotDir, "arm64")
 | |
| 	snapshotSingleton := ctx.SingletonForTests("vendor-snapshot")
 | |
| 
 | |
| 	var includeJsonFiles []string
 | |
| 	var excludeJsonFiles []string
 | |
| 
 | |
| 	for _, arch := range [][]string{
 | |
| 		[]string{"arm64", "armv8-a"},
 | |
| 		[]string{"arm", "armv7-a-neon"},
 | |
| 	} {
 | |
| 		archType := arch[0]
 | |
| 		archVariant := arch[1]
 | |
| 		archDir := fmt.Sprintf("arch-%s-%s", archType, archVariant)
 | |
| 
 | |
| 		sharedVariant := fmt.Sprintf("android_vendor.29_%s_%s_shared", archType, archVariant)
 | |
| 		sharedDir := filepath.Join(snapshotVariantPath, archDir, "shared")
 | |
| 
 | |
| 		rlibVariant := fmt.Sprintf("android_vendor.29_%s_%s_rlib_dylib-std", archType, archVariant)
 | |
| 		rlibRlibStdVariant := fmt.Sprintf("android_vendor.29_%s_%s_rlib_rlib-std", archType, archVariant)
 | |
| 		rlibDir := filepath.Join(snapshotVariantPath, archDir, "rlib")
 | |
| 		dylibVariant := fmt.Sprintf("android_vendor.29_%s_%s_dylib", archType, archVariant)
 | |
| 		dylibDir := filepath.Join(snapshotVariantPath, archDir, "dylib")
 | |
| 
 | |
| 		// Included modules
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "libinclude", "libinclude.so", sharedDir, sharedVariant)
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(sharedDir, "libinclude.so.json"))
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librust_include", "librust_include.rlib", rlibDir, rlibVariant)
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(rlibDir, "librust_include.rlib.json"))
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librust_include", "librust_include.rlib-std.rlib", rlibDir, rlibRlibStdVariant)
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(rlibDir, "librust_include.rlib-std.rlib.json"))
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librust_include", "librust_include.dylib.so", dylibDir, dylibVariant)
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(dylibDir, "librust_include.dylib.so.json"))
 | |
| 
 | |
| 		// Excluded modules
 | |
| 		cc.CheckSnapshotExclude(t, ctx, snapshotSingleton, "libexclude", "libexclude.so", sharedDir, sharedVariant)
 | |
| 		excludeJsonFiles = append(excludeJsonFiles, filepath.Join(sharedDir, "libexclude.so.json"))
 | |
| 		cc.CheckSnapshotExclude(t, ctx, snapshotSingleton, "libavailable_exclude", "libavailable_exclude.so", sharedDir, sharedVariant)
 | |
| 		excludeJsonFiles = append(excludeJsonFiles, filepath.Join(sharedDir, "libavailable_exclude.so.json"))
 | |
| 		cc.CheckSnapshotExclude(t, ctx, snapshotSingleton, "librust_exclude", "librust_exclude.rlib", rlibDir, rlibVariant)
 | |
| 		excludeJsonFiles = append(excludeJsonFiles, filepath.Join(rlibDir, "librust_exclude.rlib.json"))
 | |
| 		cc.CheckSnapshotExclude(t, ctx, snapshotSingleton, "librust_available_exclude", "librust_available_exclude.rlib", rlibDir, rlibVariant)
 | |
| 		excludeJsonFiles = append(excludeJsonFiles, filepath.Join(rlibDir, "librust_available_exclude.rlib.json"))
 | |
| 		cc.CheckSnapshotExclude(t, ctx, snapshotSingleton, "librust_available_exclude", "librust_available_exclude.rlib-std.rlib", rlibDir, rlibRlibStdVariant)
 | |
| 		excludeJsonFiles = append(excludeJsonFiles, filepath.Join(rlibDir, "librust_available_exclude.rlib.rlib-std.json"))
 | |
| 		cc.CheckSnapshotExclude(t, ctx, snapshotSingleton, "librust_exclude", "librust_exclude.dylib.so", dylibDir, dylibVariant)
 | |
| 		excludeJsonFiles = append(excludeJsonFiles, filepath.Join(dylibDir, "librust_exclude.dylib.so.json"))
 | |
| 		cc.CheckSnapshotExclude(t, ctx, snapshotSingleton, "librust_available_exclude", "librust_available_exclude.dylib.so", dylibDir, dylibVariant)
 | |
| 		excludeJsonFiles = append(excludeJsonFiles, filepath.Join(dylibDir, "librust_available_exclude.dylib.so.json"))
 | |
| 	}
 | |
| 
 | |
| 	// Verify that each json file for an included module has a rule.
 | |
| 	for _, jsonFile := range includeJsonFiles {
 | |
| 		if snapshotSingleton.MaybeOutput(jsonFile).Rule == nil {
 | |
| 			t.Errorf("include json file %q not found", jsonFile)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// Verify that each json file for an excluded module has no rule.
 | |
| 	for _, jsonFile := range excludeJsonFiles {
 | |
| 		if snapshotSingleton.MaybeOutput(jsonFile).Rule != nil {
 | |
| 			t.Errorf("exclude json file %q found", jsonFile)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestVendorSnapshotUse(t *testing.T) {
 | |
| 	frameworkBp := `
 | |
| 	cc_library {
 | |
| 		name: "libvndk",
 | |
| 		vendor_available: true,
 | |
| 		product_available: true,
 | |
| 		vndk: {
 | |
| 			enabled: true,
 | |
| 		},
 | |
| 		nocrt: true,
 | |
| 	}
 | |
| 
 | |
| 	cc_library {
 | |
| 		name: "libvendor",
 | |
| 		vendor: true,
 | |
| 		nocrt: true,
 | |
| 		no_libcrt: true,
 | |
| 		stl: "none",
 | |
| 		system_shared_libs: [],
 | |
| 	}
 | |
| 
 | |
| 	cc_library {
 | |
| 		name: "libvendor_available",
 | |
| 		vendor_available: true,
 | |
| 		nocrt: true,
 | |
| 		no_libcrt: true,
 | |
| 		stl: "none",
 | |
| 		system_shared_libs: [],
 | |
| 	}
 | |
| 
 | |
| 	cc_library {
 | |
| 		name: "lib32",
 | |
| 		vendor: true,
 | |
| 		nocrt: true,
 | |
| 		no_libcrt: true,
 | |
| 		stl: "none",
 | |
| 		system_shared_libs: [],
 | |
| 		compile_multilib: "32",
 | |
| 	}
 | |
| 
 | |
| 	cc_library {
 | |
| 		name: "lib64",
 | |
| 		vendor: true,
 | |
| 		nocrt: true,
 | |
| 		no_libcrt: true,
 | |
| 		stl: "none",
 | |
| 		system_shared_libs: [],
 | |
| 		compile_multilib: "64",
 | |
| 	}
 | |
| 
 | |
| 	rust_binary {
 | |
| 		name: "bin",
 | |
| 		vendor: true,
 | |
| 		srcs: ["bin.rs"],
 | |
| 	}
 | |
| 
 | |
| 	rust_binary {
 | |
| 		name: "bin32",
 | |
| 		vendor: true,
 | |
| 		compile_multilib: "32",
 | |
| 		srcs: ["bin.rs"],
 | |
| 	}
 | |
| 
 | |
| 	rust_library {
 | |
| 		name: "librust_vendor_available",
 | |
| 		crate_name: "rust_vendor",
 | |
| 		vendor_available: true,
 | |
| 		srcs: ["client.rs"],
 | |
| 	}
 | |
| 
 | |
| `
 | |
| 
 | |
| 	vndkBp := `
 | |
| 	vndk_prebuilt_shared {
 | |
| 		name: "libvndk",
 | |
| 		version: "30",
 | |
| 		target_arch: "arm64",
 | |
| 		vendor_available: true,
 | |
| 		product_available: true,
 | |
| 		vndk: {
 | |
| 			enabled: true,
 | |
| 		},
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				srcs: ["libvndk.so"],
 | |
| 				export_include_dirs: ["include/libvndk"],
 | |
| 			},
 | |
| 			arm: {
 | |
| 				srcs: ["libvndk.so"],
 | |
| 				export_include_dirs: ["include/libvndk"],
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	// old snapshot module which has to be ignored
 | |
| 	vndk_prebuilt_shared {
 | |
| 		name: "libvndk",
 | |
| 		version: "26",
 | |
| 		target_arch: "arm64",
 | |
| 		vendor_available: true,
 | |
| 		product_available: true,
 | |
| 		vndk: {
 | |
| 			enabled: true,
 | |
| 		},
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				srcs: ["libvndk.so"],
 | |
| 				export_include_dirs: ["include/libvndk"],
 | |
| 			},
 | |
| 			arm: {
 | |
| 				srcs: ["libvndk.so"],
 | |
| 				export_include_dirs: ["include/libvndk"],
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	// different arch snapshot which has to be ignored
 | |
| 	vndk_prebuilt_shared {
 | |
| 		name: "libvndk",
 | |
| 		version: "30",
 | |
| 		target_arch: "arm",
 | |
| 		vendor_available: true,
 | |
| 		product_available: true,
 | |
| 		vndk: {
 | |
| 			enabled: true,
 | |
| 		},
 | |
| 		arch: {
 | |
| 			arm: {
 | |
| 				srcs: ["libvndk.so"],
 | |
| 				export_include_dirs: ["include/libvndk"],
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| `
 | |
| 
 | |
| 	vendorProprietaryBp := `
 | |
| 	cc_library {
 | |
| 		name: "libvendor_without_snapshot",
 | |
| 		vendor: true,
 | |
| 		nocrt: true,
 | |
| 		no_libcrt: true,
 | |
| 		no_crt_pad_segment: true,
 | |
| 		stl: "none",
 | |
| 		system_shared_libs: [],
 | |
| 	}
 | |
| 
 | |
| 	rust_ffi_shared {
 | |
| 		name: "libclient",
 | |
| 		crate_name: "client",
 | |
| 		vendor: true,
 | |
| 		shared_libs: ["libvndk", "libvendor_available"],
 | |
| 		static_libs: ["libvendor", "libvendor_without_snapshot"],
 | |
| 		rustlibs: ["librust_vendor_available"],
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				shared_libs: ["lib64"],
 | |
| 			},
 | |
| 			arm: {
 | |
| 				shared_libs: ["lib32"],
 | |
| 			},
 | |
| 		},
 | |
| 		srcs: ["client.rs"],
 | |
| 	}
 | |
| 
 | |
| 	rust_library {
 | |
| 		name: "libclient_rust",
 | |
| 		crate_name: "client_rust",
 | |
| 		vendor: true,
 | |
| 		shared_libs: ["libvndk", "libvendor_available"],
 | |
| 		static_libs: ["libvendor", "libvendor_without_snapshot"],
 | |
| 		rustlibs: ["librust_vendor_available"],
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				shared_libs: ["lib64"],
 | |
| 			},
 | |
| 			arm: {
 | |
| 				shared_libs: ["lib32"],
 | |
| 			},
 | |
| 		},
 | |
| 		srcs: ["client.rs"],
 | |
| 	}
 | |
| 
 | |
| 	rust_binary {
 | |
| 		name: "bin_without_snapshot",
 | |
| 		vendor: true,
 | |
| 		static_libs: ["libvndk"],
 | |
| 		srcs: ["bin.rs"],
 | |
| 		rustlibs: ["librust_vendor_available"],
 | |
| 	}
 | |
| 
 | |
| 	vendor_snapshot {
 | |
| 		name: "vendor_snapshot",
 | |
| 		version: "30",
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				vndk_libs: [
 | |
| 					"libvndk",
 | |
| 				],
 | |
| 				static_libs: [
 | |
| 					"libvendor",
 | |
| 					"libvndk",
 | |
| 					"libclang_rt.builtins",
 | |
| 					"note_memtag_heap_sync",
 | |
| 				],
 | |
| 				shared_libs: [
 | |
| 					"libvendor_available",
 | |
| 					"lib64",
 | |
| 				],
 | |
| 				rlibs: [
 | |
| 					"libstd",
 | |
| 					"librust_vendor_available",
 | |
| 					"librust_vendor_available.rlib-std"
 | |
| 				],
 | |
| 				dylibs: [
 | |
| 					"libstd",
 | |
| 					"librust_vendor_available",
 | |
| 				],
 | |
| 				binaries: [
 | |
| 					"bin",
 | |
| 				],
 | |
|                 objects: [
 | |
| 				    "crtend_so",
 | |
| 					"crtbegin_so",
 | |
| 					"crtbegin_dynamic",
 | |
| 					"crtend_android"
 | |
| 				],
 | |
| 			},
 | |
| 			arm: {
 | |
| 				vndk_libs: [
 | |
| 					"libvndk",
 | |
| 				],
 | |
| 				static_libs: [
 | |
| 					"libvendor",
 | |
| 					"libvndk",
 | |
| 					"libclang_rt.builtins",
 | |
| 				],
 | |
| 				shared_libs: [
 | |
| 					"libvendor_available",
 | |
| 					"lib32",
 | |
| 				],
 | |
| 				rlibs: [
 | |
| 					"libstd",
 | |
| 					"librust_vendor_available",
 | |
| 				],
 | |
| 				dylibs: [
 | |
| 					"libstd",
 | |
| 					"librust_vendor_available",
 | |
| 				],
 | |
| 				binaries: [
 | |
| 					"bin32",
 | |
| 				],
 | |
|                 objects: [
 | |
| 				    "crtend_so",
 | |
| 					"crtbegin_so",
 | |
| 					"crtbegin_dynamic",
 | |
| 					"crtend_android"
 | |
| 				],
 | |
| 
 | |
| 			},
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	vendor_snapshot_object {
 | |
| 		name: "crtend_so",
 | |
| 		version: "30",
 | |
| 		target_arch: "arm64",
 | |
| 		vendor: true,
 | |
| 		stl: "none",
 | |
| 		crt: true,
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				src: "crtend_so.o",
 | |
| 			},
 | |
| 			arm: {
 | |
| 				src: "crtend_so.o",
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	vendor_snapshot_object {
 | |
| 		name: "crtbegin_so",
 | |
| 		version: "30",
 | |
| 		target_arch: "arm64",
 | |
| 		vendor: true,
 | |
| 		stl: "none",
 | |
| 		crt: true,
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				src: "crtbegin_so.o",
 | |
| 			},
 | |
| 			arm: {
 | |
| 				src: "crtbegin_so.o",
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	vendor_snapshot_rlib {
 | |
| 		name: "libstd",
 | |
| 		version: "30",
 | |
| 		target_arch: "arm64",
 | |
| 		vendor: true,
 | |
| 		sysroot: true,
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				src: "libstd.rlib",
 | |
| 			},
 | |
| 			arm: {
 | |
| 				src: "libstd.rlib",
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	vendor_snapshot_rlib {
 | |
| 		name: "librust_vendor_available",
 | |
| 		version: "30",
 | |
| 		target_arch: "arm64",
 | |
| 		vendor: true,
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				src: "librust_vendor_available.rlib",
 | |
| 			},
 | |
| 			arm: {
 | |
| 				src: "librust_vendor_available.rlib",
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	vendor_snapshot_rlib {
 | |
| 		name: "librust_vendor_available.rlib-std",
 | |
| 		version: "30",
 | |
| 		target_arch: "arm64",
 | |
| 		vendor: true,
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				src: "librust_vendor_available.rlib-std.rlib",
 | |
| 			},
 | |
| 			arm: {
 | |
| 				src: "librust_vendor_available.rlib-std.rlib",
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	vendor_snapshot_dylib {
 | |
| 		name: "libstd",
 | |
| 		version: "30",
 | |
| 		target_arch: "arm64",
 | |
| 		vendor: true,
 | |
| 		sysroot: true,
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				src: "libstd.dylib.so",
 | |
| 			},
 | |
| 			arm: {
 | |
| 				src: "libstd.dylib.so",
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	vendor_snapshot_dylib {
 | |
| 		name: "librust_vendor_available",
 | |
| 		version: "30",
 | |
| 		target_arch: "arm64",
 | |
| 		vendor: true,
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				src: "librust_vendor_available.dylib.so",
 | |
| 			},
 | |
| 			arm: {
 | |
| 				src: "librust_vendor_available.dylib.so",
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	vendor_snapshot_object {
 | |
| 		name: "crtend_android",
 | |
| 		version: "30",
 | |
| 		target_arch: "arm64",
 | |
| 		vendor: true,
 | |
| 		stl: "none",
 | |
| 		crt: true,
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				src: "crtend_so.o",
 | |
| 			},
 | |
| 			arm: {
 | |
| 				src: "crtend_so.o",
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	vendor_snapshot_object {
 | |
| 		name: "crtbegin_dynamic",
 | |
| 		version: "30",
 | |
| 		target_arch: "arm64",
 | |
| 		vendor: true,
 | |
| 		stl: "none",
 | |
| 		crt: true,
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				src: "crtbegin_so.o",
 | |
| 			},
 | |
| 			arm: {
 | |
| 				src: "crtbegin_so.o",
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	vendor_snapshot_static {
 | |
| 		name: "libvndk",
 | |
| 		version: "30",
 | |
| 		target_arch: "arm64",
 | |
| 		compile_multilib: "both",
 | |
| 		vendor: true,
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				src: "libvndk.a",
 | |
| 			},
 | |
| 			arm: {
 | |
| 				src: "libvndk.a",
 | |
| 			},
 | |
| 		},
 | |
| 		shared_libs: ["libvndk"],
 | |
| 		export_shared_lib_headers: ["libvndk"],
 | |
| 	}
 | |
| 
 | |
| 	vendor_snapshot_static {
 | |
| 		name: "libclang_rt.builtins",
 | |
| 		version: "30",
 | |
| 		target_arch: "arm64",
 | |
| 		vendor: true,
 | |
| 		arch: {
 | |
| 			arm: {
 | |
| 				src: "libclang_rt.builtins-arm-android.a",
 | |
| 			},
 | |
| 			arm64: {
 | |
| 				src: "libclang_rt.builtins-aarch64-android.a",
 | |
| 			},
 | |
| 		},
 | |
|     }
 | |
| 
 | |
| 	vendor_snapshot_shared {
 | |
| 		name: "lib32",
 | |
| 		version: "30",
 | |
| 		target_arch: "arm64",
 | |
| 		compile_multilib: "32",
 | |
| 		vendor: true,
 | |
| 		no_crt_pad_segment: true,
 | |
| 		arch: {
 | |
| 			arm: {
 | |
| 				src: "lib32.so",
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	vendor_snapshot_shared {
 | |
| 		name: "lib64",
 | |
| 		version: "30",
 | |
| 		target_arch: "arm64",
 | |
| 		compile_multilib: "64",
 | |
| 		vendor: true,
 | |
| 		no_crt_pad_segment: true,
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				src: "lib64.so",
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 	vendor_snapshot_shared {
 | |
| 		name: "liblog",
 | |
| 		version: "30",
 | |
| 		target_arch: "arm64",
 | |
| 		compile_multilib: "64",
 | |
| 		vendor: true,
 | |
| 		no_crt_pad_segment: true,
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				src: "liblog.so",
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	vendor_snapshot_static {
 | |
| 		name: "libvendor",
 | |
| 		version: "30",
 | |
| 		target_arch: "arm64",
 | |
| 		compile_multilib: "both",
 | |
| 		vendor: true,
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				src: "libvendor.a",
 | |
| 				export_include_dirs: ["include/libvendor"],
 | |
| 			},
 | |
| 			arm: {
 | |
| 				src: "libvendor.a",
 | |
| 				export_include_dirs: ["include/libvendor"],
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	vendor_snapshot_shared {
 | |
| 		name: "libvendor_available",
 | |
| 		version: "30",
 | |
| 		target_arch: "arm64",
 | |
| 		compile_multilib: "both",
 | |
| 		vendor: true,
 | |
| 		no_crt_pad_segment: true,
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				src: "libvendor_available.so",
 | |
| 				export_include_dirs: ["include/libvendor"],
 | |
| 			},
 | |
| 			arm: {
 | |
| 				src: "libvendor_available.so",
 | |
| 				export_include_dirs: ["include/libvendor"],
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	vendor_snapshot_binary {
 | |
| 		name: "bin",
 | |
| 		version: "30",
 | |
| 		target_arch: "arm64",
 | |
| 		compile_multilib: "64",
 | |
| 		vendor: true,
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				src: "bin",
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	vendor_snapshot_binary {
 | |
| 		name: "bin32",
 | |
| 		version: "30",
 | |
| 		target_arch: "arm64",
 | |
| 		compile_multilib: "32",
 | |
| 		vendor: true,
 | |
| 		arch: {
 | |
| 			arm: {
 | |
| 				src: "bin32",
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	// Test sanitizers use the snapshot libraries
 | |
| 	rust_binary {
 | |
| 		name: "memtag_binary",
 | |
| 		srcs: ["vendor/bin.rs"],
 | |
| 		vendor: true,
 | |
| 		compile_multilib: "64",
 | |
| 		sanitize: {
 | |
| 			memtag_heap: true,
 | |
| 			diag: {
 | |
| 				memtag_heap: true,
 | |
| 			}
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	// old snapshot module which has to be ignored
 | |
| 	vendor_snapshot_binary {
 | |
| 		name: "bin",
 | |
| 		version: "26",
 | |
| 		target_arch: "arm64",
 | |
| 		compile_multilib: "first",
 | |
| 		vendor: true,
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				src: "bin",
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	// different arch snapshot which has to be ignored
 | |
| 	vendor_snapshot_binary {
 | |
| 		name: "bin",
 | |
| 		version: "30",
 | |
| 		target_arch: "arm",
 | |
| 		compile_multilib: "first",
 | |
| 		vendor: true,
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				src: "bin",
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	vendor_snapshot_static {
 | |
| 		name: "note_memtag_heap_sync",
 | |
| 		vendor: true,
 | |
| 		target_arch: "arm64",
 | |
| 		version: "30",
 | |
| 		arch: {
 | |
| 			arm64: {
 | |
| 				src: "note_memtag_heap_sync.a",
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| `
 | |
| 
 | |
| 	mockFS := android.MockFS{
 | |
| 		"framework/Android.bp":                          []byte(frameworkBp),
 | |
| 		"framework/bin.rs":                              nil,
 | |
| 		"note_memtag_heap_sync.a":                       nil,
 | |
| 		"vendor/Android.bp":                             []byte(vendorProprietaryBp),
 | |
| 		"vendor/bin":                                    nil,
 | |
| 		"vendor/bin32":                                  nil,
 | |
| 		"vendor/bin.rs":                                 nil,
 | |
| 		"vendor/client.rs":                              nil,
 | |
| 		"vendor/include/libvndk/a.h":                    nil,
 | |
| 		"vendor/include/libvendor/b.h":                  nil,
 | |
| 		"vendor/libvndk.a":                              nil,
 | |
| 		"vendor/libvendor.a":                            nil,
 | |
| 		"vendor/libvendor.so":                           nil,
 | |
| 		"vendor/lib32.so":                               nil,
 | |
| 		"vendor/lib64.so":                               nil,
 | |
| 		"vendor/liblog.so":                              nil,
 | |
| 		"vendor/libstd.rlib":                            nil,
 | |
| 		"vendor/librust_vendor_available.rlib":          nil,
 | |
| 		"vendor/librust_vendor_available.rlib-std.rlib": nil,
 | |
| 		"vendor/libstd.dylib.so":                        nil,
 | |
| 		"vendor/librust_vendor_available.dylib.so":      nil,
 | |
| 		"vendor/crtbegin_so.o":                          nil,
 | |
| 		"vendor/crtend_so.o":                            nil,
 | |
| 		"vendor/libclang_rt.builtins-aarch64-android.a": nil,
 | |
| 		"vendor/libclang_rt.builtins-arm-android.a":     nil,
 | |
| 		"vndk/Android.bp":                               []byte(vndkBp),
 | |
| 		"vndk/include/libvndk/a.h":                      nil,
 | |
| 		"vndk/libvndk.so":                               nil,
 | |
| 	}
 | |
| 
 | |
| 	sharedVariant := "android_vendor.30_arm64_armv8-a_shared"
 | |
| 	rlibVariant := "android_vendor.30_arm64_armv8-a_rlib_dylib-std"
 | |
| 	rlibRlibStdVariant := "android_vendor.30_arm64_armv8-a_rlib_rlib-std"
 | |
| 	dylibVariant := "android_vendor.30_arm64_armv8-a_dylib"
 | |
| 	staticVariant := "android_vendor.30_arm64_armv8-a_static"
 | |
| 	binaryVariant := "android_vendor.30_arm64_armv8-a"
 | |
| 
 | |
| 	shared32Variant := "android_vendor.30_arm_armv7-a-neon_shared"
 | |
| 	binary32Variant := "android_vendor.30_arm_armv7-a-neon"
 | |
| 
 | |
| 	ctx := testRustVndkFsVersions(t, "", mockFS, "30", "current", "31")
 | |
| 
 | |
| 	// libclient uses libvndk.vndk.30.arm64, libvendor.vendor_static.30.arm64, libvendor_without_snapshot
 | |
| 	libclientLdFlags := ctx.ModuleForTests("libclient", sharedVariant).Rule("rustc").Args["linkFlags"]
 | |
| 	for _, input := range [][]string{
 | |
| 		[]string{sharedVariant, "libvndk.vndk.30.arm64"},
 | |
| 		[]string{staticVariant, "libvendor.vendor_static.30.arm64"},
 | |
| 		[]string{staticVariant, "libvendor_without_snapshot"},
 | |
| 	} {
 | |
| 		outputPaths := cc.GetOutputPaths(ctx, input[0] /* variant */, []string{input[1]} /* module name */)
 | |
| 		if !strings.Contains(libclientLdFlags, outputPaths[0].String()) {
 | |
| 			t.Errorf("libflags for libclient must contain %#v, but was %#v", outputPaths[0], libclientLdFlags)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	libclientAndroidMkSharedLibs := ctx.ModuleForTests("libclient", sharedVariant).Module().(*Module).transitiveAndroidMkSharedLibs.ToList()
 | |
| 	if g, w := libclientAndroidMkSharedLibs, []string{"libvndk.vendor", "libvendor_available.vendor", "lib64", "liblog.vendor", "libc.vendor", "libm.vendor", "libdl.vendor"}; !reflect.DeepEqual(g, w) {
 | |
| 		t.Errorf("wanted libclient AndroidMkSharedLibs %q, got %q", w, g)
 | |
| 	}
 | |
| 
 | |
| 	libclientAndroidMkStaticLibs := ctx.ModuleForTests("libclient", sharedVariant).Module().(*Module).Properties.AndroidMkStaticLibs
 | |
| 	if g, w := libclientAndroidMkStaticLibs, []string{"libvendor", "libvendor_without_snapshot", "libclang_rt.builtins.vendor"}; !reflect.DeepEqual(g, w) {
 | |
| 		t.Errorf("wanted libclient AndroidMkStaticLibs %q, got %q", w, g)
 | |
| 	}
 | |
| 
 | |
| 	libclientAndroidMkDylibs := ctx.ModuleForTests("libclient", sharedVariant).Module().(*Module).Properties.AndroidMkDylibs
 | |
| 	if g, w := libclientAndroidMkDylibs, []string{"librust_vendor_available.vendor", "libstd.vendor"}; !reflect.DeepEqual(g, w) {
 | |
| 		t.Errorf("wanted libclient libclientAndroidMkDylibs %q, got %q", w, libclientAndroidMkDylibs)
 | |
| 	}
 | |
| 
 | |
| 	libclient32AndroidMkSharedLibs := ctx.ModuleForTests("libclient", shared32Variant).Module().(*Module).transitiveAndroidMkSharedLibs.ToList()
 | |
| 	if g, w := libclient32AndroidMkSharedLibs, []string{"libvndk.vendor", "libvendor_available.vendor", "lib32", "liblog.vendor", "libc.vendor", "libm.vendor", "libdl.vendor"}; !reflect.DeepEqual(g, w) {
 | |
| 		t.Errorf("wanted libclient32 AndroidMkSharedLibs %q, got %q", w, g)
 | |
| 	}
 | |
| 
 | |
| 	libclientRustAndroidMkRlibs := ctx.ModuleForTests("libclient_rust", rlibVariant).Module().(*Module).Properties.AndroidMkRlibs
 | |
| 	if g, w := libclientRustAndroidMkRlibs, []string{"librust_vendor_available.vendor"}; !reflect.DeepEqual(g, w) {
 | |
| 		t.Errorf("wanted rlib libclient libclientAndroidMkRlibs %q, got %q", w, g)
 | |
| 	}
 | |
| 
 | |
| 	libclientRlibStdRustAndroidMkRlibs := ctx.ModuleForTests("libclient_rust", rlibRlibStdVariant).Module().(*Module).Properties.AndroidMkRlibs
 | |
| 	if g, w := libclientRlibStdRustAndroidMkRlibs, []string{"librust_vendor_available.vendor.rlib-std", "libstd.vendor"}; !reflect.DeepEqual(g, w) {
 | |
| 		t.Errorf("wanted rlib libclient libclientAndroidMkRlibs %q, got %q", w, g)
 | |
| 	}
 | |
| 
 | |
| 	libclientRustDylibAndroidMkDylibs := ctx.ModuleForTests("libclient_rust", dylibVariant).Module().(*Module).Properties.AndroidMkDylibs
 | |
| 	if g, w := libclientRustDylibAndroidMkDylibs, []string{"librust_vendor_available.vendor", "libstd.vendor"}; !reflect.DeepEqual(g, w) {
 | |
| 		t.Errorf("wanted dylib libclient libclientRustDylibAndroidMkDylibs %q, got %q", w, g)
 | |
| 	}
 | |
| 
 | |
| 	// rust vendor snapshot must have ".vendor" suffix in AndroidMk
 | |
| 	librustVendorAvailableSnapshotModule := ctx.ModuleForTests("librust_vendor_available.vendor_rlib.30.arm64", rlibVariant).Module()
 | |
| 	librustVendorSnapshotMkName := android.AndroidMkEntriesForTest(t, ctx, librustVendorAvailableSnapshotModule)[0].EntryMap["LOCAL_MODULE"][0]
 | |
| 	expectedRustVendorSnapshotName := "librust_vendor_available.vendor"
 | |
| 	if librustVendorSnapshotMkName != expectedRustVendorSnapshotName {
 | |
| 		t.Errorf("Unexpected rust vendor snapshot name in AndroidMk: %q, expected: %q\n", librustVendorSnapshotMkName, expectedRustVendorSnapshotName)
 | |
| 	}
 | |
| 
 | |
| 	librustVendorAvailableDylibSnapshotModule := ctx.ModuleForTests("librust_vendor_available.vendor_dylib.30.arm64", dylibVariant).Module()
 | |
| 	librustVendorSnapshotDylibMkName := android.AndroidMkEntriesForTest(t, ctx, librustVendorAvailableDylibSnapshotModule)[0].EntryMap["LOCAL_MODULE"][0]
 | |
| 	expectedRustVendorDylibSnapshotName := "librust_vendor_available.vendor"
 | |
| 	if librustVendorSnapshotDylibMkName != expectedRustVendorDylibSnapshotName {
 | |
| 		t.Errorf("Unexpected rust vendor snapshot name in AndroidMk: %q, expected: %q\n", librustVendorSnapshotDylibMkName, expectedRustVendorDylibSnapshotName)
 | |
| 	}
 | |
| 
 | |
| 	rustVendorBinModule := ctx.ModuleForTests("bin_without_snapshot", binaryVariant).Module()
 | |
| 	rustVendorBinMkDylibName := android.AndroidMkEntriesForTest(t, ctx, rustVendorBinModule)[0].EntryMap["LOCAL_DYLIB_LIBRARIES"][0]
 | |
| 	if rustVendorBinMkDylibName != expectedRustVendorSnapshotName {
 | |
| 		t.Errorf("Unexpected rust rlib name in AndroidMk: %q, expected: %q\n", rustVendorBinMkDylibName, expectedRustVendorSnapshotName)
 | |
| 	}
 | |
| 
 | |
| 	binWithoutSnapshotLdFlags := ctx.ModuleForTests("bin_without_snapshot", binaryVariant).Rule("rustc").Args["linkFlags"]
 | |
| 	libVndkStaticOutputPaths := cc.GetOutputPaths(ctx, staticVariant, []string{"libvndk.vendor_static.30.arm64"})
 | |
| 	if !strings.Contains(binWithoutSnapshotLdFlags, libVndkStaticOutputPaths[0].String()) {
 | |
| 		t.Errorf("libflags for bin_without_snapshot must contain %#v, but was %#v",
 | |
| 			libVndkStaticOutputPaths[0], binWithoutSnapshotLdFlags)
 | |
| 	}
 | |
| 
 | |
| 	// bin is installed by bin.vendor_binary.30.arm64
 | |
| 	ctx.ModuleForTests("bin.vendor_binary.30.arm64", binaryVariant).Output("bin")
 | |
| 
 | |
| 	// bin32 is installed by bin32.vendor_binary.30.arm64
 | |
| 	ctx.ModuleForTests("bin32.vendor_binary.30.arm64", binary32Variant).Output("bin32")
 | |
| 
 | |
| 	// bin_without_snapshot is installed by bin_without_snapshot
 | |
| 	ctx.ModuleForTests("bin_without_snapshot", binaryVariant).Output("bin_without_snapshot")
 | |
| 
 | |
| 	// libvendor, libvendor_available and bin don't have vendor.30 variant
 | |
| 	libvendorVariants := ctx.ModuleVariantsForTests("libvendor")
 | |
| 	if android.InList(sharedVariant, libvendorVariants) {
 | |
| 		t.Errorf("libvendor must not have variant %#v, but it does", sharedVariant)
 | |
| 	}
 | |
| 
 | |
| 	libvendorAvailableVariants := ctx.ModuleVariantsForTests("libvendor_available")
 | |
| 	if android.InList(sharedVariant, libvendorAvailableVariants) {
 | |
| 		t.Errorf("libvendor_available must not have variant %#v, but it does", sharedVariant)
 | |
| 	}
 | |
| 
 | |
| 	binVariants := ctx.ModuleVariantsForTests("bin")
 | |
| 	if android.InList(binaryVariant, binVariants) {
 | |
| 		t.Errorf("bin must not have variant %#v, but it does", sharedVariant)
 | |
| 	}
 | |
| 
 | |
| 	memtagStaticLibs := ctx.ModuleForTests("memtag_binary", "android_vendor.30_arm64_armv8-a").Module().(*Module).Properties.AndroidMkStaticLibs
 | |
| 	if g, w := memtagStaticLibs, []string{"libclang_rt.builtins.vendor", "note_memtag_heap_sync.vendor"}; !reflect.DeepEqual(g, w) {
 | |
| 		t.Errorf("wanted memtag_binary AndroidMkStaticLibs %q, got %q", w, g)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestRecoverySnapshotCapture(t *testing.T) {
 | |
| 	bp := `
 | |
| 	rust_ffi {
 | |
| 		name: "librecovery",
 | |
| 		recovery: true,
 | |
| 		srcs: ["foo.rs"],
 | |
| 		crate_name: "recovery",
 | |
| 	}
 | |
| 
 | |
| 	rust_ffi {
 | |
| 		name: "librecovery_available",
 | |
| 		recovery_available: true,
 | |
| 		srcs: ["foo.rs"],
 | |
| 		crate_name: "recovery_available",
 | |
| 	}
 | |
| 
 | |
| 	rust_library {
 | |
| 		name: "librecovery_rustlib",
 | |
| 		recovery: true,
 | |
| 		srcs: ["foo.rs"],
 | |
| 		crate_name: "recovery_rustlib",
 | |
| 	}
 | |
| 
 | |
| 	rust_library {
 | |
| 		name: "librecovery_available_rustlib",
 | |
| 		recovery_available: true,
 | |
| 		srcs: ["foo.rs"],
 | |
| 		crate_name: "recovery_available_rustlib",
 | |
| 	}
 | |
| 
 | |
| 	rust_binary {
 | |
| 		name: "recovery_bin",
 | |
| 		recovery: true,
 | |
| 		srcs: ["foo.rs"],
 | |
| 	}
 | |
| 
 | |
| 	rust_binary {
 | |
| 		name: "recovery_available_bin",
 | |
| 		recovery_available: true,
 | |
| 		srcs: ["foo.rs"],
 | |
| 	}
 | |
| 
 | |
| `
 | |
| 	// Check Recovery snapshot output.
 | |
| 
 | |
| 	ctx := testRustRecoveryFsVersions(t, bp, rustMockedFiles, "", "29", "current")
 | |
| 	snapshotDir := "recovery-snapshot"
 | |
| 	snapshotVariantPath := filepath.Join("out/soong", snapshotDir, "arm64")
 | |
| 	snapshotSingleton := ctx.SingletonForTests("recovery-snapshot")
 | |
| 
 | |
| 	var jsonFiles []string
 | |
| 
 | |
| 	for _, arch := range [][]string{
 | |
| 		[]string{"arm64", "armv8-a"},
 | |
| 	} {
 | |
| 		archType := arch[0]
 | |
| 		archVariant := arch[1]
 | |
| 		archDir := fmt.Sprintf("arch-%s-%s", archType, archVariant)
 | |
| 
 | |
| 		// For shared libraries, all recovery:true and recovery_available modules are captured.
 | |
| 		sharedVariant := fmt.Sprintf("android_recovery_%s_%s_shared", archType, archVariant)
 | |
| 		sharedDir := filepath.Join(snapshotVariantPath, archDir, "shared")
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librecovery", "librecovery.so", sharedDir, sharedVariant)
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librecovery_available", "librecovery_available.so", sharedDir, sharedVariant)
 | |
| 		jsonFiles = append(jsonFiles,
 | |
| 			filepath.Join(sharedDir, "librecovery.so.json"),
 | |
| 			filepath.Join(sharedDir, "librecovery_available.so.json"))
 | |
| 
 | |
| 		// For static libraries, all recovery:true and recovery_available modules are captured.
 | |
| 		staticVariant := fmt.Sprintf("android_recovery_%s_%s_static", archType, archVariant)
 | |
| 		staticDir := filepath.Join(snapshotVariantPath, archDir, "static")
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librecovery", "librecovery.a", staticDir, staticVariant)
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librecovery_available", "librecovery_available.a", staticDir, staticVariant)
 | |
| 		jsonFiles = append(jsonFiles,
 | |
| 			filepath.Join(staticDir, "librecovery.a.json"),
 | |
| 			filepath.Join(staticDir, "librecovery_available.a.json"))
 | |
| 
 | |
| 		// For rlib libraries, all recovery:true and recovery_available modules are captured.
 | |
| 		rlibVariant := fmt.Sprintf("android_recovery_%s_%s_rlib_dylib-std", archType, archVariant)
 | |
| 		rlibDir := filepath.Join(snapshotVariantPath, archDir, "rlib")
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librecovery_rustlib", "librecovery_rustlib.rlib", rlibDir, rlibVariant)
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librecovery_available_rustlib", "librecovery_available_rustlib.rlib", rlibDir, rlibVariant)
 | |
| 		jsonFiles = append(jsonFiles,
 | |
| 			filepath.Join(rlibDir, "librecovery_rustlib.rlib.json"),
 | |
| 			filepath.Join(rlibDir, "librecovery_available_rustlib.rlib.json"))
 | |
| 
 | |
| 		rlibRlibStdVariant := fmt.Sprintf("android_recovery_%s_%s_rlib_rlib-std", archType, archVariant)
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librecovery_rustlib", "librecovery_rustlib.rlib-std.rlib", rlibDir, rlibRlibStdVariant)
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librecovery_available_rustlib", "librecovery_available_rustlib.rlib-std.rlib", rlibDir, rlibRlibStdVariant)
 | |
| 		jsonFiles = append(jsonFiles,
 | |
| 			filepath.Join(rlibDir, "librecovery_rustlib.rlib-std.rlib.json"),
 | |
| 			filepath.Join(rlibDir, "librecovery_available_rustlib.rlib-std.rlib.json"))
 | |
| 
 | |
| 		// For dylib libraries, all recovery:true and recovery_available modules are captured.
 | |
| 		dylibVariant := fmt.Sprintf("android_recovery_%s_%s_dylib", archType, archVariant)
 | |
| 		dylibDir := filepath.Join(snapshotVariantPath, archDir, "dylib")
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librecovery_rustlib", "librecovery_rustlib.dylib.so", dylibDir, dylibVariant)
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librecovery_available_rustlib", "librecovery_available_rustlib.dylib.so", dylibDir, dylibVariant)
 | |
| 		jsonFiles = append(jsonFiles,
 | |
| 			filepath.Join(dylibDir, "librecovery_rustlib.dylib.so.json"),
 | |
| 			filepath.Join(dylibDir, "librecovery_available_rustlib.dylib.so.json"))
 | |
| 
 | |
| 		// For binary executables, all recovery:true and recovery_available modules are captured.
 | |
| 		if archType == "arm64" {
 | |
| 			binaryVariant := fmt.Sprintf("android_recovery_%s_%s", archType, archVariant)
 | |
| 			binaryDir := filepath.Join(snapshotVariantPath, archDir, "binary")
 | |
| 			cc.CheckSnapshot(t, ctx, snapshotSingleton, "recovery_bin", "recovery_bin", binaryDir, binaryVariant)
 | |
| 			cc.CheckSnapshot(t, ctx, snapshotSingleton, "recovery_available_bin", "recovery_available_bin", binaryDir, binaryVariant)
 | |
| 			jsonFiles = append(jsonFiles,
 | |
| 				filepath.Join(binaryDir, "recovery_bin.json"),
 | |
| 				filepath.Join(binaryDir, "recovery_available_bin.json"))
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	for _, jsonFile := range jsonFiles {
 | |
| 		// verify all json files exist
 | |
| 		if snapshotSingleton.MaybeOutput(jsonFile).Rule == nil {
 | |
| 			t.Errorf("%q expected but not found", jsonFile)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestRecoverySnapshotExclude(t *testing.T) {
 | |
| 	// This test verifies that the exclude_from_recovery_snapshot property
 | |
| 	// makes its way from the Android.bp source file into the module data
 | |
| 	// structure. It also verifies that modules are correctly included or
 | |
| 	// excluded in the recovery snapshot based on their path (framework or
 | |
| 	// vendor) and the exclude_from_recovery_snapshot property.
 | |
| 
 | |
| 	frameworkBp := `
 | |
| 		rust_ffi_shared {
 | |
| 			name: "libinclude",
 | |
| 			srcs: ["src/include.rs"],
 | |
| 			recovery_available: true,
 | |
| 			crate_name: "include",
 | |
| 		}
 | |
| 		rust_ffi_shared {
 | |
| 			name: "libexclude",
 | |
| 			srcs: ["src/exclude.rs"],
 | |
| 			recovery: true,
 | |
| 			exclude_from_recovery_snapshot: true,
 | |
| 			crate_name: "exclude",
 | |
| 		}
 | |
| 		rust_ffi_shared {
 | |
| 			name: "libavailable_exclude",
 | |
| 			srcs: ["src/exclude.rs"],
 | |
| 			recovery_available: true,
 | |
| 			exclude_from_recovery_snapshot: true,
 | |
| 			crate_name: "available_exclude",
 | |
| 		}
 | |
| 		rust_library {
 | |
| 			name: "libinclude_rustlib",
 | |
| 			srcs: ["src/include.rs"],
 | |
| 			recovery_available: true,
 | |
| 			crate_name: "include_rustlib",
 | |
| 		}
 | |
| 		rust_library {
 | |
| 			name: "libexclude_rustlib",
 | |
| 			srcs: ["src/exclude.rs"],
 | |
| 			recovery: true,
 | |
| 			exclude_from_recovery_snapshot: true,
 | |
| 			crate_name: "exclude_rustlib",
 | |
| 		}
 | |
| 		rust_library {
 | |
| 			name: "libavailable_exclude_rustlib",
 | |
| 			srcs: ["src/exclude.rs"],
 | |
| 			recovery_available: true,
 | |
| 			exclude_from_recovery_snapshot: true,
 | |
| 			crate_name: "available_exclude_rustlib",
 | |
| 		}
 | |
| 	`
 | |
| 
 | |
| 	vendorProprietaryBp := `
 | |
| 		rust_ffi_shared {
 | |
| 			name: "librecovery",
 | |
| 			srcs: ["recovery.rs"],
 | |
| 			recovery: true,
 | |
| 			crate_name: "recovery",
 | |
| 		}
 | |
| 		rust_library {
 | |
| 			name: "librecovery_rustlib",
 | |
| 			srcs: ["recovery.rs"],
 | |
| 			recovery: true,
 | |
| 			crate_name: "recovery_rustlib",
 | |
| 		}
 | |
| 	`
 | |
| 
 | |
| 	mockFS := map[string][]byte{
 | |
| 		"framework/Android.bp": []byte(frameworkBp),
 | |
| 		"framework/include.rs": nil,
 | |
| 		"framework/exclude.rs": nil,
 | |
| 		"device/Android.bp":    []byte(vendorProprietaryBp),
 | |
| 		"device/recovery.rs":   nil,
 | |
| 	}
 | |
| 
 | |
| 	ctx := testRustRecoveryFsVersions(t, "", mockFS, "", "29", "current")
 | |
| 
 | |
| 	// Test an include and exclude framework module.
 | |
| 	cc.AssertExcludeFromRecoverySnapshotIs(t, ctx, "libinclude", false, sharedRecoveryVariant)
 | |
| 	cc.AssertExcludeFromRecoverySnapshotIs(t, ctx, "libexclude", true, sharedRecoveryVariant)
 | |
| 	cc.AssertExcludeFromRecoverySnapshotIs(t, ctx, "libavailable_exclude", true, sharedRecoveryVariant)
 | |
| 
 | |
| 	cc.AssertExcludeFromRecoverySnapshotIs(t, ctx, "libinclude_rustlib", false, rlibRecoveryVariant)
 | |
| 	cc.AssertExcludeFromRecoverySnapshotIs(t, ctx, "libexclude_rustlib", true, rlibRecoveryVariant)
 | |
| 	cc.AssertExcludeFromRecoverySnapshotIs(t, ctx, "libavailable_exclude_rustlib", true, rlibRlibStdRecoveryVariant)
 | |
| 
 | |
| 	cc.AssertExcludeFromRecoverySnapshotIs(t, ctx, "libinclude_rustlib", false, rlibRlibStdRecoveryVariant)
 | |
| 	cc.AssertExcludeFromRecoverySnapshotIs(t, ctx, "libexclude_rustlib", true, rlibRlibStdRecoveryVariant)
 | |
| 	cc.AssertExcludeFromRecoverySnapshotIs(t, ctx, "libavailable_exclude_rustlib", true, rlibRlibStdRecoveryVariant)
 | |
| 
 | |
| 	cc.AssertExcludeFromRecoverySnapshotIs(t, ctx, "libinclude_rustlib", false, dylibRecoveryVariant)
 | |
| 	cc.AssertExcludeFromRecoverySnapshotIs(t, ctx, "libexclude_rustlib", true, dylibRecoveryVariant)
 | |
| 	cc.AssertExcludeFromRecoverySnapshotIs(t, ctx, "libavailable_exclude_rustlib", true, dylibRecoveryVariant)
 | |
| 
 | |
| 	// A recovery module is excluded, but by its path not the exclude_from_recovery_snapshot property
 | |
| 	// ('device/' and 'vendor/' are default excluded). See snapshot/recovery_snapshot.go for more detail.
 | |
| 	cc.AssertExcludeFromRecoverySnapshotIs(t, ctx, "librecovery", false, sharedRecoveryVariant)
 | |
| 	cc.AssertExcludeFromRecoverySnapshotIs(t, ctx, "librecovery_rustlib", false, rlibRecoveryVariant)
 | |
| 	cc.AssertExcludeFromRecoverySnapshotIs(t, ctx, "librecovery_rustlib", false, rlibRlibStdRecoveryVariant)
 | |
| 	cc.AssertExcludeFromRecoverySnapshotIs(t, ctx, "librecovery_rustlib", false, dylibRecoveryVariant)
 | |
| 
 | |
| 	// Verify the content of the recovery snapshot.
 | |
| 
 | |
| 	snapshotDir := "recovery-snapshot"
 | |
| 	snapshotVariantPath := filepath.Join("out/soong", snapshotDir, "arm64")
 | |
| 	snapshotSingleton := ctx.SingletonForTests("recovery-snapshot")
 | |
| 
 | |
| 	var includeJsonFiles []string
 | |
| 	var excludeJsonFiles []string
 | |
| 
 | |
| 	for _, arch := range [][]string{
 | |
| 		[]string{"arm64", "armv8-a"},
 | |
| 	} {
 | |
| 		archType := arch[0]
 | |
| 		archVariant := arch[1]
 | |
| 		archDir := fmt.Sprintf("arch-%s-%s", archType, archVariant)
 | |
| 
 | |
| 		sharedVariant := fmt.Sprintf("android_recovery_%s_%s_shared", archType, archVariant)
 | |
| 		rlibVariant := fmt.Sprintf("android_recovery_%s_%s_rlib_dylib-std", archType, archVariant)
 | |
| 		rlibRlibStdVariant := fmt.Sprintf("android_recovery_%s_%s_rlib_rlib-std", archType, archVariant)
 | |
| 		dylibVariant := fmt.Sprintf("android_recovery_%s_%s_dylib", archType, archVariant)
 | |
| 		sharedDir := filepath.Join(snapshotVariantPath, archDir, "shared")
 | |
| 		rlibDir := filepath.Join(snapshotVariantPath, archDir, "rlib")
 | |
| 		dylibDir := filepath.Join(snapshotVariantPath, archDir, "dylib")
 | |
| 
 | |
| 		// Included modules
 | |
| 
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "libinclude", "libinclude.so", sharedDir, sharedVariant)
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(sharedDir, "libinclude.so.json"))
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "libinclude_rustlib", "libinclude_rustlib.rlib", rlibDir, rlibVariant)
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(rlibDir, "libinclude_rustlib.rlib.json"))
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "libinclude_rustlib", "libinclude_rustlib.rlib-std.rlib", rlibDir, rlibRlibStdVariant)
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(rlibDir, "libinclude_rustlib.rlib-std.rlib.json"))
 | |
| 
 | |
| 		// Excluded modules
 | |
| 		cc.CheckSnapshotExclude(t, ctx, snapshotSingleton, "libexclude", "libexclude.so", sharedDir, sharedVariant)
 | |
| 		excludeJsonFiles = append(excludeJsonFiles, filepath.Join(sharedDir, "libexclude.so.json"))
 | |
| 		cc.CheckSnapshotExclude(t, ctx, snapshotSingleton, "librecovery", "librecovery.so", sharedDir, sharedVariant)
 | |
| 		excludeJsonFiles = append(excludeJsonFiles, filepath.Join(sharedDir, "librecovery.so.json"))
 | |
| 		cc.CheckSnapshotExclude(t, ctx, snapshotSingleton, "libavailable_exclude", "libavailable_exclude.so", sharedDir, sharedVariant)
 | |
| 		excludeJsonFiles = append(excludeJsonFiles, filepath.Join(sharedDir, "libavailable_exclude.so.json"))
 | |
| 
 | |
| 		cc.CheckSnapshotExclude(t, ctx, snapshotSingleton, "libexclude_rustlib", "libexclude_rustlib.rlib", rlibDir, rlibVariant)
 | |
| 		excludeJsonFiles = append(excludeJsonFiles, filepath.Join(rlibDir, "libexclude_rustlib.rlib.json"))
 | |
| 		cc.CheckSnapshotExclude(t, ctx, snapshotSingleton, "librecovery_rustlib", "librecovery_rustlib.rlib", rlibDir, rlibVariant)
 | |
| 		excludeJsonFiles = append(excludeJsonFiles, filepath.Join(rlibDir, "librecovery_rustlib.rlib.json"))
 | |
| 		cc.CheckSnapshotExclude(t, ctx, snapshotSingleton, "libavailable_exclude_rustlib", "libavailable_exclude_rustlib.rlib", rlibDir, rlibVariant)
 | |
| 		excludeJsonFiles = append(excludeJsonFiles, filepath.Join(rlibDir, "libavailable_exclude_rustlib.rlib.json"))
 | |
| 
 | |
| 		cc.CheckSnapshotExclude(t, ctx, snapshotSingleton, "libexclude_rustlib", "libexclude_rustlib.rlib-std.rlib", rlibDir, rlibRlibStdVariant)
 | |
| 		excludeJsonFiles = append(excludeJsonFiles, filepath.Join(rlibDir, "libexclude_rustlib.rlib-std.rlib.json"))
 | |
| 		cc.CheckSnapshotExclude(t, ctx, snapshotSingleton, "librecovery_rustlib", "librecovery_rustlib.rlib-std.rlib", rlibDir, rlibRlibStdVariant)
 | |
| 		excludeJsonFiles = append(excludeJsonFiles, filepath.Join(rlibDir, "librecovery_rustlib.rlib-std.rlib.json"))
 | |
| 		cc.CheckSnapshotExclude(t, ctx, snapshotSingleton, "libavailable_exclude_rustlib", "libavailable_exclude_rustlib.rlib-std.rlib", rlibDir, rlibRlibStdVariant)
 | |
| 		excludeJsonFiles = append(excludeJsonFiles, filepath.Join(rlibDir, "libavailable_exclude_rustlib.rlib-std.rlib.json"))
 | |
| 
 | |
| 		cc.CheckSnapshotExclude(t, ctx, snapshotSingleton, "libexclude_rustlib", "libexclude_rustlib.dylib.so", dylibDir, dylibVariant)
 | |
| 		excludeJsonFiles = append(excludeJsonFiles, filepath.Join(rlibDir, "libexclude_rustlib.dylib.so.json"))
 | |
| 		cc.CheckSnapshotExclude(t, ctx, snapshotSingleton, "librecovery_rustlib", "librecovery_rustlib.dylib.so", dylibDir, dylibVariant)
 | |
| 		excludeJsonFiles = append(excludeJsonFiles, filepath.Join(rlibDir, "librecovery_rustlib.dylib.so.json"))
 | |
| 		cc.CheckSnapshotExclude(t, ctx, snapshotSingleton, "libavailable_exclude_rustlib", "libavailable_exclude_rustlib.dylib.so", dylibDir, dylibVariant)
 | |
| 		excludeJsonFiles = append(excludeJsonFiles, filepath.Join(rlibDir, "libavailable_exclude_rustlib.dylib.so.json"))
 | |
| 	}
 | |
| 
 | |
| 	// Verify that each json file for an included module has a rule.
 | |
| 	for _, jsonFile := range includeJsonFiles {
 | |
| 		if snapshotSingleton.MaybeOutput(jsonFile).Rule == nil {
 | |
| 			t.Errorf("include json file %q not found", jsonFile)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// Verify that each json file for an excluded module has no rule.
 | |
| 	for _, jsonFile := range excludeJsonFiles {
 | |
| 		if snapshotSingleton.MaybeOutput(jsonFile).Rule != nil {
 | |
| 			t.Errorf("exclude json file %q found", jsonFile)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestRecoverySnapshotDirected(t *testing.T) {
 | |
| 	bp := `
 | |
| 	rust_ffi_shared {
 | |
| 		name: "librecovery",
 | |
| 		recovery: true,
 | |
| 		crate_name: "recovery",
 | |
| 		srcs: ["foo.rs"],
 | |
| 	}
 | |
| 
 | |
| 	rust_ffi_shared {
 | |
| 		name: "librecovery_available",
 | |
| 		recovery_available: true,
 | |
| 		crate_name: "recovery_available",
 | |
| 		srcs: ["foo.rs"],
 | |
| 	}
 | |
| 
 | |
| 	rust_library {
 | |
| 		name: "librecovery_rustlib",
 | |
| 		recovery: true,
 | |
| 		crate_name: "recovery",
 | |
| 		srcs: ["foo.rs"],
 | |
| 	}
 | |
| 
 | |
| 	rust_library {
 | |
| 		name: "librecovery_available_rustlib",
 | |
| 		recovery_available: true,
 | |
| 		crate_name: "recovery_available",
 | |
| 		srcs: ["foo.rs"],
 | |
| 	}
 | |
| 
 | |
| 	/* TODO: Uncomment when Rust supports the "prefer" property for prebuilts
 | |
| 	rust_library_rlib {
 | |
| 		name: "libfoo_rlib",
 | |
| 		recovery: true,
 | |
| 		crate_name: "foo",
 | |
| 	}
 | |
| 
 | |
| 	rust_prebuilt_rlib {
 | |
| 		name: "libfoo_rlib",
 | |
| 		recovery: true,
 | |
| 		prefer: true,
 | |
| 		srcs: ["libfoo.rlib"],
 | |
| 		crate_name: "foo",
 | |
| 	}
 | |
| 	*/
 | |
| `
 | |
| 	ctx := testRustRecoveryFsVersions(t, bp, rustMockedFiles, "current", "29", "current")
 | |
| 	ctx.Config().TestProductVariables.RecoverySnapshotModules = make(map[string]bool)
 | |
| 	ctx.Config().TestProductVariables.RecoverySnapshotModules["librecovery"] = true
 | |
| 	ctx.Config().TestProductVariables.RecoverySnapshotModules["librecovery_rustlib"] = true
 | |
| 	ctx.Config().TestProductVariables.DirectedRecoverySnapshot = true
 | |
| 
 | |
| 	// Check recovery snapshot output.
 | |
| 	snapshotDir := "recovery-snapshot"
 | |
| 	snapshotVariantPath := filepath.Join("out/soong", snapshotDir, "arm64")
 | |
| 	snapshotSingleton := ctx.SingletonForTests("recovery-snapshot")
 | |
| 
 | |
| 	var includeJsonFiles []string
 | |
| 
 | |
| 	for _, arch := range [][]string{
 | |
| 		[]string{"arm64", "armv8-a"},
 | |
| 	} {
 | |
| 		archType := arch[0]
 | |
| 		archVariant := arch[1]
 | |
| 		archDir := fmt.Sprintf("arch-%s-%s", archType, archVariant)
 | |
| 
 | |
| 		sharedVariant := fmt.Sprintf("android_recovery_%s_%s_shared", archType, archVariant)
 | |
| 		rlibVariant := fmt.Sprintf("android_recovery_%s_%s_rlib_dylib-std", archType, archVariant)
 | |
| 		rlibRlibStdVariant := fmt.Sprintf("android_recovery_%s_%s_rlib_rlib-std", archType, archVariant)
 | |
| 		dylibVariant := fmt.Sprintf("android_recovery_%s_%s_dylib", archType, archVariant)
 | |
| 		sharedDir := filepath.Join(snapshotVariantPath, archDir, "shared")
 | |
| 		rlibDir := filepath.Join(snapshotVariantPath, archDir, "rlib")
 | |
| 		dylibDir := filepath.Join(snapshotVariantPath, archDir, "dylib")
 | |
| 
 | |
| 		// Included modules
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librecovery", "librecovery.so", sharedDir, sharedVariant)
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(sharedDir, "librecovery.so.json"))
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librecovery_rustlib", "librecovery_rustlib.rlib", rlibDir, rlibVariant)
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(rlibDir, "librecovery_rustlib.rlib.json"))
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librecovery_rustlib", "librecovery_rustlib.rlib-std.rlib", rlibDir, rlibRlibStdVariant)
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(rlibDir, "librecovery_rustlib.rlib-std.rlib.json"))
 | |
| 		cc.CheckSnapshot(t, ctx, snapshotSingleton, "librecovery_rustlib", "librecovery_rustlib.dylib.so", dylibDir, dylibVariant)
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(dylibDir, "librecovery_rustlib.dylib.so.json"))
 | |
| 
 | |
| 		// TODO: When Rust supports the "prefer" property for prebuilts, perform this check.
 | |
| 		/*
 | |
| 			// Check that snapshot captures "prefer: true" prebuilt
 | |
| 			cc.CheckSnapshot(t, ctx, snapshotSingleton, "prebuilt_libfoo_rlib", "libfoo_rlib.rlib", rlibDir, rlibVariant)
 | |
| 			includeJsonFiles = append(includeJsonFiles, filepath.Join(sharedDir, "libfoo_rlib.rlib.json"))
 | |
| 		*/
 | |
| 
 | |
| 		// Excluded modules. Modules not included in the directed recovery snapshot
 | |
| 		// are still included as fake modules.
 | |
| 		cc.CheckSnapshotRule(t, ctx, snapshotSingleton, "librecovery_available", "librecovery_available.so", sharedDir, sharedVariant)
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(sharedDir, "librecovery_available.so.json"))
 | |
| 		cc.CheckSnapshotRule(t, ctx, snapshotSingleton, "librecovery_available_rustlib", "librecovery_available_rustlib.rlib", rlibDir, rlibVariant)
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(rlibDir, "librecovery_available_rustlib.rlib.json"))
 | |
| 		cc.CheckSnapshotRule(t, ctx, snapshotSingleton, "librecovery_available_rustlib", "librecovery_available_rustlib.rlib-std.rlib", rlibDir, rlibRlibStdVariant)
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(rlibDir, "librecovery_available_rustlib.rlib-std.rlib.json"))
 | |
| 		cc.CheckSnapshotRule(t, ctx, snapshotSingleton, "librecovery_available_rustlib", "librecovery_available_rustlib.dylib.so", dylibDir, dylibVariant)
 | |
| 		includeJsonFiles = append(includeJsonFiles, filepath.Join(dylibDir, "librecovery_available_rustlib.dylib.so.json"))
 | |
| 	}
 | |
| 
 | |
| 	// Verify that each json file for an included module has a rule.
 | |
| 	for _, jsonFile := range includeJsonFiles {
 | |
| 		if snapshotSingleton.MaybeOutput(jsonFile).Rule == nil {
 | |
| 			t.Errorf("include json file %q not found, %#v", jsonFile, includeJsonFiles)
 | |
| 		}
 | |
| 	}
 | |
| }
 |