This leads to an arm64 apex being used on a device that is mixed x86_64 with nativebridge=arm64. A device like that doesn't appear to work with arm64 binaries. For example, the boringssl-self-check binary crashes on boot. Bug: 260115309 Test: unit test Test: boot emulator with this combination Change-Id: Ic4a91974290a05b1799f755fcf52ef226d68f4c2
		
			
				
	
	
		
			179 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			179 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2020 Google Inc. All rights reserved.
 | |
| //
 | |
| // Licensed under the Apache License, Version 2.0 (the "License");
 | |
| // you may not use this file except in compliance with the License.
 | |
| // You may obtain a copy of the License at
 | |
| //
 | |
| //     http://www.apache.org/licenses/LICENSE-2.0
 | |
| //
 | |
| // Unless required by applicable law or agreed to in writing, software
 | |
| // distributed under the License is distributed on an "AS IS" BASIS,
 | |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
| // See the License for the specific language governing permissions and
 | |
| // limitations under the License.
 | |
| 
 | |
| package java
 | |
| 
 | |
| // This file contains the module implementation for android_app_set.
 | |
| 
 | |
| import (
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 
 | |
| 	"github.com/google/blueprint/proptools"
 | |
| 
 | |
| 	"android/soong/android"
 | |
| )
 | |
| 
 | |
| func init() {
 | |
| 	RegisterAppSetBuildComponents(android.InitRegistrationContext)
 | |
| }
 | |
| 
 | |
| func RegisterAppSetBuildComponents(ctx android.RegistrationContext) {
 | |
| 	ctx.RegisterModuleType("android_app_set", AndroidAppSetFactory)
 | |
| }
 | |
| 
 | |
| type AndroidAppSetProperties struct {
 | |
| 	// APK Set path
 | |
| 	Set *string
 | |
| 
 | |
| 	// Specifies that this app should be installed to the priv-app directory,
 | |
| 	// where the system will grant it additional privileges not available to
 | |
| 	// normal apps.
 | |
| 	Privileged *bool
 | |
| 
 | |
| 	// APKs in this set use prerelease SDK version
 | |
| 	Prerelease *bool
 | |
| 
 | |
| 	// Names of modules to be overridden. Listed modules can only be other apps
 | |
| 	//	(in Make or Soong).
 | |
| 	Overrides []string
 | |
| }
 | |
| 
 | |
| type AndroidAppSet struct {
 | |
| 	android.ModuleBase
 | |
| 	android.DefaultableModuleBase
 | |
| 	prebuilt android.Prebuilt
 | |
| 
 | |
| 	properties    AndroidAppSetProperties
 | |
| 	packedOutput  android.WritablePath
 | |
| 	primaryOutput android.WritablePath
 | |
| 	apkcertsFile  android.ModuleOutPath
 | |
| }
 | |
| 
 | |
| func (as *AndroidAppSet) Name() string {
 | |
| 	return as.prebuilt.Name(as.ModuleBase.Name())
 | |
| }
 | |
| 
 | |
| func (as *AndroidAppSet) IsInstallable() bool {
 | |
| 	return true
 | |
| }
 | |
| 
 | |
| func (as *AndroidAppSet) Prebuilt() *android.Prebuilt {
 | |
| 	return &as.prebuilt
 | |
| }
 | |
| 
 | |
| func (as *AndroidAppSet) Privileged() bool {
 | |
| 	return Bool(as.properties.Privileged)
 | |
| }
 | |
| 
 | |
| func (as *AndroidAppSet) OutputFile() android.Path {
 | |
| 	return as.primaryOutput
 | |
| }
 | |
| 
 | |
| func (as *AndroidAppSet) PackedAdditionalOutputs() android.Path {
 | |
| 	return as.packedOutput
 | |
| }
 | |
| 
 | |
| func (as *AndroidAppSet) APKCertsFile() android.Path {
 | |
| 	return as.apkcertsFile
 | |
| }
 | |
| 
 | |
| var TargetCpuAbi = map[string]string{
 | |
| 	"arm":   "ARMEABI_V7A",
 | |
| 	"arm64": "ARM64_V8A",
 | |
| 	// TODO: use "RISCV64" when that is supported in bundles
 | |
| 	"riscv64": "ARM64_V8A",
 | |
| 	"x86":     "X86",
 | |
| 	"x86_64":  "X86_64",
 | |
| }
 | |
| 
 | |
| func SupportedAbis(ctx android.ModuleContext, excludeNativeBridgeAbis bool) []string {
 | |
| 	abiName := func(targetIdx int, deviceArch string) string {
 | |
| 		if abi, found := TargetCpuAbi[deviceArch]; found {
 | |
| 			return abi
 | |
| 		}
 | |
| 		ctx.ModuleErrorf("Target %d has invalid Arch: %s", targetIdx, deviceArch)
 | |
| 		return "BAD_ABI"
 | |
| 	}
 | |
| 
 | |
| 	var result []string
 | |
| 	for i, target := range ctx.Config().Targets[android.Android] {
 | |
| 		if target.NativeBridge == android.NativeBridgeEnabled && excludeNativeBridgeAbis {
 | |
| 			continue
 | |
| 		}
 | |
| 		result = append(result, abiName(i, target.Arch.ArchType.String()))
 | |
| 	}
 | |
| 	return result
 | |
| }
 | |
| 
 | |
| func (as *AndroidAppSet) GenerateAndroidBuildActions(ctx android.ModuleContext) {
 | |
| 	as.packedOutput = android.PathForModuleOut(ctx, ctx.ModuleName()+".zip")
 | |
| 	as.primaryOutput = android.PathForModuleOut(ctx, as.BaseModuleName()+".apk")
 | |
| 	as.apkcertsFile = android.PathForModuleOut(ctx, "apkcerts.txt")
 | |
| 	// We are assuming here that the install file in the APK
 | |
| 	// set has `.apk` suffix. If it doesn't the build will fail.
 | |
| 	// APK sets containing APEX files are handled elsewhere.
 | |
| 	screenDensities := "all"
 | |
| 	if dpis := ctx.Config().ProductAAPTPrebuiltDPI(); len(dpis) > 0 {
 | |
| 		screenDensities = strings.ToUpper(strings.Join(dpis, ","))
 | |
| 	}
 | |
| 	// TODO(asmundak): handle locales.
 | |
| 	// TODO(asmundak): do we support device features
 | |
| 	ctx.Build(pctx,
 | |
| 		android.BuildParams{
 | |
| 			Rule:            extractMatchingApks,
 | |
| 			Description:     "Extract APKs from APK set",
 | |
| 			Output:          as.primaryOutput,
 | |
| 			ImplicitOutputs: android.WritablePaths{as.packedOutput, as.apkcertsFile},
 | |
| 			Inputs:          android.Paths{as.prebuilt.SingleSourcePath(ctx)},
 | |
| 			Args: map[string]string{
 | |
| 				"abis":              strings.Join(SupportedAbis(ctx, false), ","),
 | |
| 				"allow-prereleased": strconv.FormatBool(proptools.Bool(as.properties.Prerelease)),
 | |
| 				"screen-densities":  screenDensities,
 | |
| 				"sdk-version":       ctx.Config().PlatformSdkVersion().String(),
 | |
| 				"stem":              as.BaseModuleName(),
 | |
| 				"apkcerts":          as.apkcertsFile.String(),
 | |
| 				"partition":         as.PartitionTag(ctx.DeviceConfig()),
 | |
| 				"zip":               as.packedOutput.String(),
 | |
| 			},
 | |
| 		})
 | |
| 
 | |
| 	var installDir android.InstallPath
 | |
| 	if as.Privileged() {
 | |
| 		installDir = android.PathForModuleInstall(ctx, "priv-app", as.BaseModuleName())
 | |
| 	} else {
 | |
| 		installDir = android.PathForModuleInstall(ctx, "app", as.BaseModuleName())
 | |
| 	}
 | |
| 	ctx.InstallFileWithExtraFilesZip(installDir, as.BaseModuleName()+".apk", as.primaryOutput, as.packedOutput)
 | |
| }
 | |
| 
 | |
| func (as *AndroidAppSet) InstallBypassMake() bool { return true }
 | |
| 
 | |
| // android_app_set extracts a set of APKs based on the target device
 | |
| // configuration and installs this set as "split APKs".
 | |
| // The extracted set always contains an APK whose name is
 | |
| // _module_name_.apk and every split APK matching target device.
 | |
| // The extraction of the density-specific splits depends on
 | |
| // PRODUCT_AAPT_PREBUILT_DPI variable. If present (its value should
 | |
| // be a list density names: LDPI, MDPI, HDPI, etc.), only listed
 | |
| // splits will be extracted. Otherwise all density-specific splits
 | |
| // will be extracted.
 | |
| func AndroidAppSetFactory() android.Module {
 | |
| 	module := &AndroidAppSet{}
 | |
| 	module.AddProperties(&module.properties)
 | |
| 	InitJavaModule(module, android.DeviceSupported)
 | |
| 	android.InitSingleSourcePrebuiltModule(module, &module.properties, "Set")
 | |
| 	return module
 | |
| }
 |