| // Copyright 2015 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 implementations for android_app, android_test, and some more | 
 | // related module types, including their override variants. | 
 |  | 
 | import ( | 
 | 	"fmt" | 
 | 	"path/filepath" | 
 | 	"strings" | 
 |  | 
 | 	"android/soong/testing" | 
 | 	"github.com/google/blueprint" | 
 | 	"github.com/google/blueprint/proptools" | 
 |  | 
 | 	"android/soong/android" | 
 | 	"android/soong/bazel" | 
 | 	"android/soong/cc" | 
 | 	"android/soong/dexpreopt" | 
 | 	"android/soong/genrule" | 
 | 	"android/soong/tradefed" | 
 | 	"android/soong/ui/metrics/bp2build_metrics_proto" | 
 | ) | 
 |  | 
 | func init() { | 
 | 	RegisterAppBuildComponents(android.InitRegistrationContext) | 
 | 	pctx.HostBinToolVariable("ModifyAllowlistCmd", "modify_permissions_allowlist") | 
 | } | 
 |  | 
 | var ( | 
 | 	modifyAllowlist = pctx.AndroidStaticRule("modifyAllowlist", | 
 | 		blueprint.RuleParams{ | 
 | 			Command:     "${ModifyAllowlistCmd} $in $packageName $out", | 
 | 			CommandDeps: []string{"${ModifyAllowlistCmd}"}, | 
 | 		}, "packageName") | 
 | ) | 
 |  | 
 | func RegisterAppBuildComponents(ctx android.RegistrationContext) { | 
 | 	ctx.RegisterModuleType("android_app", AndroidAppFactory) | 
 | 	ctx.RegisterModuleType("android_test", AndroidTestFactory) | 
 | 	ctx.RegisterModuleType("android_test_helper_app", AndroidTestHelperAppFactory) | 
 | 	ctx.RegisterModuleType("android_app_certificate", AndroidAppCertificateFactory) | 
 | 	ctx.RegisterModuleType("override_android_app", OverrideAndroidAppModuleFactory) | 
 | 	ctx.RegisterModuleType("override_android_test", OverrideAndroidTestModuleFactory) | 
 | } | 
 |  | 
 | // AndroidManifest.xml merging | 
 | // package splits | 
 |  | 
 | type appProperties struct { | 
 | 	// Names of extra android_app_certificate modules to sign the apk with in the form ":module". | 
 | 	Additional_certificates []string | 
 |  | 
 | 	// If set, create package-export.apk, which other packages can | 
 | 	// use to get PRODUCT-agnostic resource data like IDs and type definitions. | 
 | 	Export_package_resources *bool | 
 |  | 
 | 	// 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 | 
 |  | 
 | 	// list of resource labels to generate individual resource packages | 
 | 	Package_splits []string | 
 |  | 
 | 	// list of native libraries that will be provided in or alongside the resulting jar | 
 | 	Jni_libs []string `android:"arch_variant"` | 
 |  | 
 | 	// if true, use JNI libraries that link against platform APIs even if this module sets | 
 | 	// sdk_version. | 
 | 	Jni_uses_platform_apis *bool | 
 |  | 
 | 	// if true, use JNI libraries that link against SDK APIs even if this module does not set | 
 | 	// sdk_version. | 
 | 	Jni_uses_sdk_apis *bool | 
 |  | 
 | 	// STL library to use for JNI libraries. | 
 | 	Stl *string `android:"arch_variant"` | 
 |  | 
 | 	// Store native libraries uncompressed in the APK and set the android:extractNativeLibs="false" manifest | 
 | 	// flag so that they are used from inside the APK at runtime.  Defaults to true for android_test modules unless | 
 | 	// sdk_version or min_sdk_version is set to a version that doesn't support it (<23), defaults to true for | 
 | 	// android_app modules that are embedded to APEXes, defaults to false for other module types where the native | 
 | 	// libraries are generally preinstalled outside the APK. | 
 | 	Use_embedded_native_libs *bool | 
 |  | 
 | 	// Store dex files uncompressed in the APK and set the android:useEmbeddedDex="true" manifest attribute so that | 
 | 	// they are used from inside the APK at runtime. | 
 | 	Use_embedded_dex *bool | 
 |  | 
 | 	// Forces native libraries to always be packaged into the APK, | 
 | 	// Use_embedded_native_libs still selects whether they are stored uncompressed and aligned or compressed. | 
 | 	// True for android_test* modules. | 
 | 	AlwaysPackageNativeLibs bool `blueprint:"mutated"` | 
 |  | 
 | 	// If set, find and merge all NOTICE files that this module and its dependencies have and store | 
 | 	// it in the APK as an asset. | 
 | 	Embed_notices *bool | 
 |  | 
 | 	// cc.Coverage related properties | 
 | 	PreventInstall    bool `blueprint:"mutated"` | 
 | 	IsCoverageVariant bool `blueprint:"mutated"` | 
 |  | 
 | 	// It can be set to test the behaviour of default target sdk version. | 
 | 	// Only required when updatable: false. It is an error if updatable: true and this is false. | 
 | 	Enforce_default_target_sdk_version *bool | 
 |  | 
 | 	// If set, the targetSdkVersion for the target is set to the latest default API level. | 
 | 	// This would be by default false, unless updatable: true or | 
 | 	// enforce_default_target_sdk_version: true in which case this defaults to true. | 
 | 	EnforceDefaultTargetSdkVersion bool `blueprint:"mutated"` | 
 |  | 
 | 	// Whether this app is considered mainline updatable or not. When set to true, this will enforce | 
 | 	// additional rules to make sure an app can safely be updated. Default is false. | 
 | 	// Prefer using other specific properties if build behaviour must be changed; avoid using this | 
 | 	// flag for anything but neverallow rules (unless the behaviour change is invisible to owners). | 
 | 	Updatable *bool | 
 |  | 
 | 	// Specifies the file that contains the allowlist for this app. | 
 | 	Privapp_allowlist *string `android:"path"` | 
 |  | 
 | 	// If set, create an RRO package which contains only resources having PRODUCT_CHARACTERISTICS | 
 | 	// and install the RRO package to /product partition, instead of passing --product argument | 
 | 	// to aapt2. Default is false. | 
 | 	// Setting this will make this APK identical to all targets, regardless of | 
 | 	// PRODUCT_CHARACTERISTICS. | 
 | 	Generate_product_characteristics_rro *bool | 
 |  | 
 | 	ProductCharacteristicsRROPackageName        *string `blueprint:"mutated"` | 
 | 	ProductCharacteristicsRROManifestModuleName *string `blueprint:"mutated"` | 
 | } | 
 |  | 
 | // android_app properties that can be overridden by override_android_app | 
 | type overridableAppProperties struct { | 
 | 	// The name of a certificate in the default certificate directory, blank to use the default product certificate, | 
 | 	// or an android_app_certificate module name in the form ":module". | 
 | 	Certificate *string | 
 |  | 
 | 	// Name of the signing certificate lineage file or filegroup module. | 
 | 	Lineage *string `android:"path"` | 
 |  | 
 | 	// For overriding the --rotation-min-sdk-version property of apksig | 
 | 	RotationMinSdkVersion *string | 
 |  | 
 | 	// the package name of this app. The package name in the manifest file is used if one was not given. | 
 | 	Package_name *string | 
 |  | 
 | 	// the logging parent of this app. | 
 | 	Logging_parent *string | 
 |  | 
 | 	// Whether to rename the package in resources to the override name rather than the base name. Defaults to true. | 
 | 	Rename_resources_package *bool | 
 |  | 
 | 	// Names of modules to be overridden. Listed modules can only be other binaries | 
 | 	// (in Make or Soong). | 
 | 	// This does not completely prevent installation of the overridden binaries, but if both | 
 | 	// binaries would be installed by default (in PRODUCT_PACKAGES) the other binary will be removed | 
 | 	// from PRODUCT_PACKAGES. | 
 | 	Overrides []string | 
 | } | 
 |  | 
 | type AndroidApp struct { | 
 | 	android.BazelModuleBase | 
 | 	Library | 
 | 	aapt | 
 | 	android.OverridableModuleBase | 
 |  | 
 | 	certificate Certificate | 
 |  | 
 | 	appProperties appProperties | 
 |  | 
 | 	overridableAppProperties overridableAppProperties | 
 |  | 
 | 	jniLibs                  []jniLib | 
 | 	installPathForJNISymbols android.Path | 
 | 	embeddedJniLibs          bool | 
 | 	jniCoverageOutputs       android.Paths | 
 |  | 
 | 	bundleFile android.Path | 
 |  | 
 | 	// the install APK name is normally the same as the module name, but can be overridden with PRODUCT_PACKAGE_NAME_OVERRIDES. | 
 | 	installApkName string | 
 |  | 
 | 	installDir android.InstallPath | 
 |  | 
 | 	onDeviceDir string | 
 |  | 
 | 	additionalAaptFlags []string | 
 |  | 
 | 	overriddenManifestPackageName string | 
 |  | 
 | 	android.ApexBundleDepsInfo | 
 |  | 
 | 	javaApiUsedByOutputFile android.ModuleOutPath | 
 |  | 
 | 	privAppAllowlist android.OptionalPath | 
 | } | 
 |  | 
 | func (a *AndroidApp) IsInstallable() bool { | 
 | 	return Bool(a.properties.Installable) | 
 | } | 
 |  | 
 | func (a *AndroidApp) ResourcesNodeDepSet() *android.DepSet[*resourcesNode] { | 
 | 	return a.aapt.resourcesNodesDepSet | 
 | } | 
 |  | 
 | func (a *AndroidApp) OutputFile() android.Path { | 
 | 	return a.outputFile | 
 | } | 
 |  | 
 | func (a *AndroidApp) Certificate() Certificate { | 
 | 	return a.certificate | 
 | } | 
 |  | 
 | func (a *AndroidApp) JniCoverageOutputs() android.Paths { | 
 | 	return a.jniCoverageOutputs | 
 | } | 
 |  | 
 | func (a *AndroidApp) PrivAppAllowlist() android.OptionalPath { | 
 | 	return a.privAppAllowlist | 
 | } | 
 |  | 
 | var _ AndroidLibraryDependency = (*AndroidApp)(nil) | 
 |  | 
 | type Certificate struct { | 
 | 	Pem, Key  android.Path | 
 | 	presigned bool | 
 | } | 
 |  | 
 | var PresignedCertificate = Certificate{presigned: true} | 
 |  | 
 | func (c Certificate) AndroidMkString() string { | 
 | 	if c.presigned { | 
 | 		return "PRESIGNED" | 
 | 	} else { | 
 | 		return c.Pem.String() | 
 | 	} | 
 | } | 
 |  | 
 | func (a *AndroidApp) DepsMutator(ctx android.BottomUpMutatorContext) { | 
 | 	a.Module.deps(ctx) | 
 |  | 
 | 	if String(a.appProperties.Stl) == "c++_shared" && !a.SdkVersion(ctx).Specified() { | 
 | 		ctx.PropertyErrorf("stl", "sdk_version must be set in order to use c++_shared") | 
 | 	} | 
 |  | 
 | 	sdkDep := decodeSdkDep(ctx, android.SdkContext(a)) | 
 | 	if sdkDep.hasFrameworkLibs() { | 
 | 		a.aapt.deps(ctx, sdkDep) | 
 | 	} | 
 |  | 
 | 	usesSDK := a.SdkVersion(ctx).Specified() && a.SdkVersion(ctx).Kind != android.SdkCorePlatform | 
 |  | 
 | 	if usesSDK && Bool(a.appProperties.Jni_uses_sdk_apis) { | 
 | 		ctx.PropertyErrorf("jni_uses_sdk_apis", | 
 | 			"can only be set for modules that do not set sdk_version") | 
 | 	} else if !usesSDK && Bool(a.appProperties.Jni_uses_platform_apis) { | 
 | 		ctx.PropertyErrorf("jni_uses_platform_apis", | 
 | 			"can only be set for modules that set sdk_version") | 
 | 	} | 
 |  | 
 | 	for _, jniTarget := range ctx.MultiTargets() { | 
 | 		variation := append(jniTarget.Variations(), | 
 | 			blueprint.Variation{Mutator: "link", Variation: "shared"}) | 
 |  | 
 | 		// If the app builds against an Android SDK use the SDK variant of JNI dependencies | 
 | 		// unless jni_uses_platform_apis is set. | 
 | 		// Don't require the SDK variant for apps that are shipped on vendor, etc., as they already | 
 | 		// have stable APIs through the VNDK. | 
 | 		if (usesSDK && !a.RequiresStableAPIs(ctx) && | 
 | 			!Bool(a.appProperties.Jni_uses_platform_apis)) || | 
 | 			Bool(a.appProperties.Jni_uses_sdk_apis) { | 
 | 			variation = append(variation, blueprint.Variation{Mutator: "sdk", Variation: "sdk"}) | 
 | 		} | 
 | 		ctx.AddFarVariationDependencies(variation, jniLibTag, a.appProperties.Jni_libs...) | 
 | 	} | 
 |  | 
 | 	a.usesLibrary.deps(ctx, sdkDep.hasFrameworkLibs()) | 
 | } | 
 |  | 
 | func (a *AndroidApp) OverridablePropertiesDepsMutator(ctx android.BottomUpMutatorContext) { | 
 | 	cert := android.SrcIsModule(a.getCertString(ctx)) | 
 | 	if cert != "" { | 
 | 		ctx.AddDependency(ctx.Module(), certificateTag, cert) | 
 | 	} | 
 |  | 
 | 	if a.appProperties.Privapp_allowlist != nil && !Bool(a.appProperties.Privileged) { | 
 | 		// There are a few uids that are explicitly considered privileged regardless of their | 
 | 		// app's location. Bluetooth is one such app. It should arguably be moved to priv-app, | 
 | 		// but for now, allow it not to be in priv-app. | 
 | 		privilegedBecauseOfUid := ctx.ModuleName() == "Bluetooth" | 
 | 		if !privilegedBecauseOfUid { | 
 | 			ctx.PropertyErrorf("privapp_allowlist", "privileged must be set in order to use privapp_allowlist (with a few exceptions)") | 
 | 		} | 
 | 	} | 
 |  | 
 | 	for _, cert := range a.appProperties.Additional_certificates { | 
 | 		cert = android.SrcIsModule(cert) | 
 | 		if cert != "" { | 
 | 			ctx.AddDependency(ctx.Module(), certificateTag, cert) | 
 | 		} else { | 
 | 			ctx.PropertyErrorf("additional_certificates", | 
 | 				`must be names of android_app_certificate modules in the form ":module"`) | 
 | 		} | 
 | 	} | 
 | } | 
 |  | 
 | func (a *AndroidTestHelperApp) GenerateAndroidBuildActions(ctx android.ModuleContext) { | 
 | 	applicationId := a.appTestHelperAppProperties.Manifest_values.ApplicationId | 
 | 	if applicationId != nil { | 
 | 		if a.overridableAppProperties.Package_name != nil { | 
 | 			ctx.PropertyErrorf("manifest_values.applicationId", "property is not supported when property package_name is set.") | 
 | 		} | 
 | 		a.aapt.manifestValues.applicationId = *applicationId | 
 | 	} | 
 | 	a.generateAndroidBuildActions(ctx) | 
 | } | 
 |  | 
 | func (a *AndroidApp) GenerateAndroidBuildActions(ctx android.ModuleContext) { | 
 | 	a.checkAppSdkVersions(ctx) | 
 | 	a.generateAndroidBuildActions(ctx) | 
 | 	a.generateJavaUsedByApex(ctx) | 
 | } | 
 |  | 
 | func (a *AndroidApp) MinSdkVersion(ctx android.EarlyModuleContext) android.ApiLevel { | 
 | 	defaultMinSdkVersion := a.Module.MinSdkVersion(ctx) | 
 | 	if proptools.Bool(a.appProperties.Updatable) { | 
 | 		overrideApiLevel := android.MinSdkVersionFromValue(ctx, ctx.DeviceConfig().ApexGlobalMinSdkVersionOverride()) | 
 | 		if !overrideApiLevel.IsNone() && overrideApiLevel.CompareTo(defaultMinSdkVersion) > 0 { | 
 | 			return overrideApiLevel | 
 | 		} | 
 | 	} | 
 | 	return defaultMinSdkVersion | 
 | } | 
 |  | 
 | func (a *AndroidApp) checkAppSdkVersions(ctx android.ModuleContext) { | 
 | 	if a.Updatable() { | 
 | 		if !a.SdkVersion(ctx).Stable() { | 
 | 			ctx.PropertyErrorf("sdk_version", "Updatable apps must use stable SDKs, found %v", a.SdkVersion(ctx)) | 
 | 		} | 
 | 		if String(a.deviceProperties.Min_sdk_version) == "" { | 
 | 			ctx.PropertyErrorf("updatable", "updatable apps must set min_sdk_version.") | 
 | 		} | 
 |  | 
 | 		if minSdkVersion, err := a.MinSdkVersion(ctx).EffectiveVersion(ctx); err == nil { | 
 | 			a.checkJniLibsSdkVersion(ctx, minSdkVersion) | 
 | 			android.CheckMinSdkVersion(ctx, minSdkVersion, a.WalkPayloadDeps) | 
 | 		} else { | 
 | 			ctx.PropertyErrorf("min_sdk_version", "%s", err.Error()) | 
 | 		} | 
 |  | 
 | 		if !BoolDefault(a.appProperties.Enforce_default_target_sdk_version, true) { | 
 | 			ctx.PropertyErrorf("enforce_default_target_sdk_version", "Updatable apps must enforce default target sdk version") | 
 | 		} | 
 | 		// TODO(b/227460469) after all the modules removes the target sdk version, throw an error if the target sdk version is explicitly set. | 
 | 		if a.deviceProperties.Target_sdk_version == nil { | 
 | 			a.SetEnforceDefaultTargetSdkVersion(true) | 
 | 		} | 
 | 	} | 
 |  | 
 | 	a.checkPlatformAPI(ctx) | 
 | 	a.checkSdkVersions(ctx) | 
 | } | 
 |  | 
 | // If an updatable APK sets min_sdk_version, min_sdk_vesion of JNI libs should match with it. | 
 | // This check is enforced for "updatable" APKs (including APK-in-APEX). | 
 | func (a *AndroidApp) checkJniLibsSdkVersion(ctx android.ModuleContext, minSdkVersion android.ApiLevel) { | 
 | 	// It's enough to check direct JNI deps' sdk_version because all transitive deps from JNI deps are checked in cc.checkLinkType() | 
 | 	ctx.VisitDirectDeps(func(m android.Module) { | 
 | 		if !IsJniDepTag(ctx.OtherModuleDependencyTag(m)) { | 
 | 			return | 
 | 		} | 
 | 		dep, _ := m.(*cc.Module) | 
 | 		// The domain of cc.sdk_version is "current" and <number> | 
 | 		// We can rely on android.SdkSpec to convert it to <number> so that "current" is | 
 | 		// handled properly regardless of sdk finalization. | 
 | 		jniSdkVersion, err := android.SdkSpecFrom(ctx, dep.MinSdkVersion()).EffectiveVersion(ctx) | 
 | 		if err != nil || minSdkVersion.LessThan(jniSdkVersion) { | 
 | 			ctx.OtherModuleErrorf(dep, "min_sdk_version(%v) is higher than min_sdk_version(%v) of the containing android_app(%v)", | 
 | 				dep.MinSdkVersion(), minSdkVersion, ctx.ModuleName()) | 
 | 			return | 
 | 		} | 
 |  | 
 | 	}) | 
 | } | 
 |  | 
 | // Returns true if the native libraries should be stored in the APK uncompressed and the | 
 | // extractNativeLibs application flag should be set to false in the manifest. | 
 | func (a *AndroidApp) useEmbeddedNativeLibs(ctx android.ModuleContext) bool { | 
 | 	minSdkVersion, err := a.MinSdkVersion(ctx).EffectiveVersion(ctx) | 
 | 	if err != nil { | 
 | 		ctx.PropertyErrorf("min_sdk_version", "invalid value %q: %s", a.MinSdkVersion(ctx), err) | 
 | 	} | 
 |  | 
 | 	apexInfo := ctx.Provider(android.ApexInfoProvider).(android.ApexInfo) | 
 | 	return (minSdkVersion.FinalOrFutureInt() >= 23 && Bool(a.appProperties.Use_embedded_native_libs)) || | 
 | 		!apexInfo.IsForPlatform() | 
 | } | 
 |  | 
 | // Returns whether this module should have the dex file stored uncompressed in the APK. | 
 | func (a *AndroidApp) shouldUncompressDex(ctx android.ModuleContext) bool { | 
 | 	if Bool(a.appProperties.Use_embedded_dex) { | 
 | 		return true | 
 | 	} | 
 |  | 
 | 	// Uncompress dex in APKs of privileged apps (even for unbundled builds, they may | 
 | 	// be preinstalled as prebuilts). | 
 | 	if ctx.Config().UncompressPrivAppDex() && a.Privileged() { | 
 | 		return true | 
 | 	} | 
 |  | 
 | 	if ctx.Config().UnbundledBuild() { | 
 | 		return false | 
 | 	} | 
 |  | 
 | 	return shouldUncompressDex(ctx, &a.dexpreopter) | 
 | } | 
 |  | 
 | func (a *AndroidApp) shouldEmbedJnis(ctx android.BaseModuleContext) bool { | 
 | 	apexInfo := ctx.Provider(android.ApexInfoProvider).(android.ApexInfo) | 
 | 	return ctx.Config().UnbundledBuild() || Bool(a.appProperties.Use_embedded_native_libs) || | 
 | 		!apexInfo.IsForPlatform() || a.appProperties.AlwaysPackageNativeLibs | 
 | } | 
 |  | 
 | func generateAaptRenamePackageFlags(packageName string, renameResourcesPackage bool) []string { | 
 | 	aaptFlags := []string{"--rename-manifest-package " + packageName} | 
 | 	if renameResourcesPackage { | 
 | 		// Required to rename the package name in the resources table. | 
 | 		aaptFlags = append(aaptFlags, "--rename-resources-package "+packageName) | 
 | 	} | 
 | 	return aaptFlags | 
 | } | 
 |  | 
 | func (a *AndroidApp) OverriddenManifestPackageName() string { | 
 | 	return a.overriddenManifestPackageName | 
 | } | 
 |  | 
 | func (a *AndroidApp) renameResourcesPackage() bool { | 
 | 	return proptools.BoolDefault(a.overridableAppProperties.Rename_resources_package, true) | 
 | } | 
 |  | 
 | func (a *AndroidApp) aaptBuildActions(ctx android.ModuleContext) { | 
 | 	usePlatformAPI := proptools.Bool(a.Module.deviceProperties.Platform_apis) | 
 | 	if ctx.Module().(android.SdkContext).SdkVersion(ctx).Kind == android.SdkModule { | 
 | 		usePlatformAPI = true | 
 | 	} | 
 | 	a.aapt.usesNonSdkApis = usePlatformAPI | 
 |  | 
 | 	// Ask manifest_fixer to add or update the application element indicating this app has no code. | 
 | 	a.aapt.hasNoCode = !a.hasCode(ctx) | 
 |  | 
 | 	aaptLinkFlags := []string{} | 
 |  | 
 | 	// Add TARGET_AAPT_CHARACTERISTICS values to AAPT link flags if they exist and --product flags were not provided. | 
 | 	autogenerateRRO := proptools.Bool(a.appProperties.Generate_product_characteristics_rro) | 
 | 	hasProduct := android.PrefixInList(a.aaptProperties.Aaptflags, "--product") | 
 | 	if !autogenerateRRO && !hasProduct && len(ctx.Config().ProductAAPTCharacteristics()) > 0 { | 
 | 		aaptLinkFlags = append(aaptLinkFlags, "--product", ctx.Config().ProductAAPTCharacteristics()) | 
 | 	} | 
 |  | 
 | 	if !Bool(a.aaptProperties.Aapt_include_all_resources) { | 
 | 		// Product AAPT config | 
 | 		for _, aaptConfig := range ctx.Config().ProductAAPTConfig() { | 
 | 			aaptLinkFlags = append(aaptLinkFlags, "-c", aaptConfig) | 
 | 		} | 
 |  | 
 | 		// Product AAPT preferred config | 
 | 		if len(ctx.Config().ProductAAPTPreferredConfig()) > 0 { | 
 | 			aaptLinkFlags = append(aaptLinkFlags, "--preferred-density", ctx.Config().ProductAAPTPreferredConfig()) | 
 | 		} | 
 | 	} | 
 |  | 
 | 	manifestPackageName, overridden := ctx.DeviceConfig().OverrideManifestPackageNameFor(ctx.ModuleName()) | 
 | 	if overridden || a.overridableAppProperties.Package_name != nil { | 
 | 		// The product override variable has a priority over the package_name property. | 
 | 		if !overridden { | 
 | 			manifestPackageName = *a.overridableAppProperties.Package_name | 
 | 		} | 
 | 		aaptLinkFlags = append(aaptLinkFlags, generateAaptRenamePackageFlags(manifestPackageName, a.renameResourcesPackage())...) | 
 | 		a.overriddenManifestPackageName = manifestPackageName | 
 | 	} | 
 |  | 
 | 	aaptLinkFlags = append(aaptLinkFlags, a.additionalAaptFlags...) | 
 |  | 
 | 	a.aapt.splitNames = a.appProperties.Package_splits | 
 | 	a.aapt.LoggingParent = String(a.overridableAppProperties.Logging_parent) | 
 | 	if a.Updatable() { | 
 | 		a.aapt.defaultManifestVersion = android.DefaultUpdatableModuleVersion | 
 | 	} | 
 | 	a.aapt.buildActions(ctx, | 
 | 		aaptBuildActionOptions{ | 
 | 			android.SdkContext(a), | 
 | 			a.classLoaderContexts, | 
 | 			a.usesLibraryProperties.Exclude_uses_libs, | 
 | 			a.enforceDefaultTargetSdkVersion(), | 
 | 			aaptLinkFlags, | 
 | 		}, | 
 | 	) | 
 |  | 
 | 	// apps manifests are handled by aapt, don't let Module see them | 
 | 	a.properties.Manifest = nil | 
 | } | 
 |  | 
 | func (a *AndroidApp) proguardBuildActions(ctx android.ModuleContext) { | 
 | 	var staticLibProguardFlagFiles android.Paths | 
 | 	ctx.VisitDirectDeps(func(m android.Module) { | 
 | 		depProguardInfo := ctx.OtherModuleProvider(m, ProguardSpecInfoProvider).(ProguardSpecInfo) | 
 | 		staticLibProguardFlagFiles = append(staticLibProguardFlagFiles, depProguardInfo.UnconditionallyExportedProguardFlags.ToList()...) | 
 | 		if ctx.OtherModuleDependencyTag(m) == staticLibTag { | 
 | 			staticLibProguardFlagFiles = append(staticLibProguardFlagFiles, depProguardInfo.ProguardFlagsFiles.ToList()...) | 
 | 		} | 
 | 	}) | 
 |  | 
 | 	staticLibProguardFlagFiles = android.FirstUniquePaths(staticLibProguardFlagFiles) | 
 |  | 
 | 	a.Module.extraProguardFlagFiles = append(a.Module.extraProguardFlagFiles, staticLibProguardFlagFiles...) | 
 | 	a.Module.extraProguardFlagFiles = append(a.Module.extraProguardFlagFiles, a.proguardOptionsFile) | 
 | } | 
 |  | 
 | func (a *AndroidApp) installPath(ctx android.ModuleContext) android.InstallPath { | 
 | 	var installDir string | 
 | 	if ctx.ModuleName() == "framework-res" { | 
 | 		// framework-res.apk is installed as system/framework/framework-res.apk | 
 | 		installDir = "framework" | 
 | 	} else if a.Privileged() { | 
 | 		installDir = filepath.Join("priv-app", a.installApkName) | 
 | 	} else { | 
 | 		installDir = filepath.Join("app", a.installApkName) | 
 | 	} | 
 |  | 
 | 	return android.PathForModuleInstall(ctx, installDir, a.installApkName+".apk") | 
 | } | 
 |  | 
 | func (a *AndroidApp) dexBuildActions(ctx android.ModuleContext) android.Path { | 
 | 	a.dexpreopter.installPath = a.installPath(ctx) | 
 | 	a.dexpreopter.isApp = true | 
 | 	if a.dexProperties.Uncompress_dex == nil { | 
 | 		// If the value was not force-set by the user, use reasonable default based on the module. | 
 | 		a.dexProperties.Uncompress_dex = proptools.BoolPtr(a.shouldUncompressDex(ctx)) | 
 | 	} | 
 | 	a.dexpreopter.uncompressedDex = *a.dexProperties.Uncompress_dex | 
 | 	a.dexpreopter.enforceUsesLibs = a.usesLibrary.enforceUsesLibraries() | 
 | 	a.dexpreopter.classLoaderContexts = a.classLoaderContexts | 
 | 	a.dexpreopter.manifestFile = a.mergedManifestFile | 
 | 	a.dexpreopter.preventInstall = a.appProperties.PreventInstall | 
 |  | 
 | 	if ctx.ModuleName() != "framework-res" { | 
 | 		var extraSrcJars android.Paths | 
 | 		var extraClasspathJars android.Paths | 
 | 		var extraCombinedJars android.Paths | 
 | 		if a.useResourceProcessorBusyBox() { | 
 | 			// When building an app with ResourceProcessorBusyBox enabled ResourceProcessorBusyBox has already | 
 | 			// created R.class files that provide IDs for resources in busybox/R.jar.  Pass that file in the | 
 | 			// classpath when compiling everything else, and add it to the final classes jar. | 
 | 			extraClasspathJars = android.Paths{a.aapt.rJar} | 
 | 			extraCombinedJars = android.Paths{a.aapt.rJar} | 
 | 		} else { | 
 | 			// When building an app without ResourceProcessorBusyBox the aapt2 rule creates R.srcjar containing | 
 | 			// R.java files for the app's package and the packages from all transitive static android_library | 
 | 			// dependencies.  Compile the srcjar alongside the rest of the sources. | 
 | 			extraSrcJars = android.Paths{a.aapt.aaptSrcJar} | 
 | 		} | 
 |  | 
 | 		a.Module.compile(ctx, extraSrcJars, extraClasspathJars, extraCombinedJars) | 
 | 	} | 
 |  | 
 | 	return a.dexJarFile.PathOrNil() | 
 | } | 
 |  | 
 | func (a *AndroidApp) jniBuildActions(jniLibs []jniLib, prebuiltJniPackages android.Paths, ctx android.ModuleContext) android.WritablePath { | 
 | 	var jniJarFile android.WritablePath | 
 | 	if len(jniLibs) > 0 || len(prebuiltJniPackages) > 0 { | 
 | 		a.jniLibs = jniLibs | 
 | 		if a.shouldEmbedJnis(ctx) { | 
 | 			jniJarFile = android.PathForModuleOut(ctx, "jnilibs.zip") | 
 | 			a.installPathForJNISymbols = a.installPath(ctx) | 
 | 			TransformJniLibsToJar(ctx, jniJarFile, jniLibs, prebuiltJniPackages, a.useEmbeddedNativeLibs(ctx)) | 
 | 			for _, jni := range jniLibs { | 
 | 				if jni.coverageFile.Valid() { | 
 | 					// Only collect coverage for the first target arch if this is a multilib target. | 
 | 					// TODO(jungjw): Ideally, we want to collect both reports, but that would cause coverage | 
 | 					// data file path collisions since the current coverage file path format doesn't contain | 
 | 					// arch-related strings. This is fine for now though; the code coverage team doesn't use | 
 | 					// multi-arch targets such as test_suite_* for coverage collections yet. | 
 | 					// | 
 | 					// Work with the team to come up with a new format that handles multilib modules properly | 
 | 					// and change this. | 
 | 					if len(ctx.Config().Targets[android.Android]) == 1 || | 
 | 						ctx.Config().AndroidFirstDeviceTarget.Arch.ArchType == jni.target.Arch.ArchType { | 
 | 						a.jniCoverageOutputs = append(a.jniCoverageOutputs, jni.coverageFile.Path()) | 
 | 					} | 
 | 				} | 
 | 			} | 
 | 			a.embeddedJniLibs = true | 
 | 		} | 
 | 	} | 
 | 	return jniJarFile | 
 | } | 
 |  | 
 | func (a *AndroidApp) JNISymbolsInstalls(installPath string) android.RuleBuilderInstalls { | 
 | 	var jniSymbols android.RuleBuilderInstalls | 
 | 	for _, jniLib := range a.jniLibs { | 
 | 		if jniLib.unstrippedFile != nil { | 
 | 			jniSymbols = append(jniSymbols, android.RuleBuilderInstall{ | 
 | 				From: jniLib.unstrippedFile, | 
 | 				To:   filepath.Join(installPath, targetToJniDir(jniLib.target), jniLib.unstrippedFile.Base()), | 
 | 			}) | 
 | 		} | 
 | 	} | 
 | 	return jniSymbols | 
 | } | 
 |  | 
 | // Reads and prepends a main cert from the default cert dir if it hasn't been set already, i.e. it | 
 | // isn't a cert module reference. Also checks and enforces system cert restriction if applicable. | 
 | func processMainCert(m android.ModuleBase, certPropValue string, certificates []Certificate, | 
 | 	ctx android.ModuleContext) (mainCertificate Certificate, allCertificates []Certificate) { | 
 | 	if android.SrcIsModule(certPropValue) == "" { | 
 | 		var mainCert Certificate | 
 | 		if certPropValue != "" { | 
 | 			defaultDir := ctx.Config().DefaultAppCertificateDir(ctx) | 
 | 			mainCert = Certificate{ | 
 | 				Pem: defaultDir.Join(ctx, certPropValue+".x509.pem"), | 
 | 				Key: defaultDir.Join(ctx, certPropValue+".pk8"), | 
 | 			} | 
 | 		} else { | 
 | 			pem, key := ctx.Config().DefaultAppCertificate(ctx) | 
 | 			mainCert = Certificate{ | 
 | 				Pem: pem, | 
 | 				Key: key, | 
 | 			} | 
 | 		} | 
 | 		certificates = append([]Certificate{mainCert}, certificates...) | 
 | 	} | 
 |  | 
 | 	if len(certificates) > 0 { | 
 | 		mainCertificate = certificates[0] | 
 | 	} else { | 
 | 		// This can be reached with an empty certificate list if AllowMissingDependencies is set | 
 | 		// and the certificate property for this module is a module reference to a missing module. | 
 | 		if !ctx.Config().AllowMissingDependencies() && len(ctx.GetMissingDependencies()) > 0 { | 
 | 			panic("Should only get here if AllowMissingDependencies set and there are missing dependencies") | 
 | 		} | 
 | 		// Set a certificate to avoid panics later when accessing it. | 
 | 		mainCertificate = Certificate{ | 
 | 			Key: android.PathForModuleOut(ctx, "missing.pk8"), | 
 | 			Pem: android.PathForModuleOut(ctx, "missing.x509.pem"), | 
 | 		} | 
 | 	} | 
 |  | 
 | 	if !m.Platform() { | 
 | 		certPath := mainCertificate.Pem.String() | 
 | 		systemCertPath := ctx.Config().DefaultAppCertificateDir(ctx).String() | 
 | 		if strings.HasPrefix(certPath, systemCertPath) { | 
 | 			enforceSystemCert := ctx.Config().EnforceSystemCertificate() | 
 | 			allowed := ctx.Config().EnforceSystemCertificateAllowList() | 
 |  | 
 | 			if enforceSystemCert && !inList(m.Name(), allowed) { | 
 | 				ctx.PropertyErrorf("certificate", "The module in product partition cannot be signed with certificate in system.") | 
 | 			} | 
 | 		} | 
 | 	} | 
 |  | 
 | 	return mainCertificate, certificates | 
 | } | 
 |  | 
 | func (a *AndroidApp) InstallApkName() string { | 
 | 	return a.installApkName | 
 | } | 
 |  | 
 | func (a *AndroidApp) createPrivappAllowlist(ctx android.ModuleContext) android.Path { | 
 | 	if a.appProperties.Privapp_allowlist == nil { | 
 | 		return nil | 
 | 	} | 
 |  | 
 | 	isOverrideApp := a.GetOverriddenBy() != "" | 
 | 	if !isOverrideApp { | 
 | 		// if this is not an override, we don't need to rewrite the existing privapp allowlist | 
 | 		return android.PathForModuleSrc(ctx, *a.appProperties.Privapp_allowlist) | 
 | 	} | 
 |  | 
 | 	if a.overridableAppProperties.Package_name == nil { | 
 | 		ctx.PropertyErrorf("privapp_allowlist", "package_name must be set to use privapp_allowlist") | 
 | 	} | 
 |  | 
 | 	packageName := *a.overridableAppProperties.Package_name | 
 | 	fileName := "privapp_allowlist_" + packageName + ".xml" | 
 | 	outPath := android.PathForModuleOut(ctx, fileName).OutputPath | 
 | 	ctx.Build(pctx, android.BuildParams{ | 
 | 		Rule:   modifyAllowlist, | 
 | 		Input:  android.PathForModuleSrc(ctx, *a.appProperties.Privapp_allowlist), | 
 | 		Output: outPath, | 
 | 		Args: map[string]string{ | 
 | 			"packageName": packageName, | 
 | 		}, | 
 | 	}) | 
 | 	return &outPath | 
 | } | 
 |  | 
 | func (a *AndroidApp) generateAndroidBuildActions(ctx android.ModuleContext) { | 
 | 	var apkDeps android.Paths | 
 |  | 
 | 	if !ctx.Provider(android.ApexInfoProvider).(android.ApexInfo).IsForPlatform() { | 
 | 		a.hideApexVariantFromMake = true | 
 | 	} | 
 |  | 
 | 	a.aapt.useEmbeddedNativeLibs = a.useEmbeddedNativeLibs(ctx) | 
 | 	a.aapt.useEmbeddedDex = Bool(a.appProperties.Use_embedded_dex) | 
 |  | 
 | 	// Unlike installApkName, a.stem should respect base module name for override_android_app. | 
 | 	// Therefore, use ctx.ModuleName() instead of a.Name(). | 
 | 	a.stem = proptools.StringDefault(a.overridableDeviceProperties.Stem, ctx.ModuleName()) | 
 |  | 
 | 	// Check if the install APK name needs to be overridden. | 
 | 	// Both android_app and override_android_app module are expected to possess | 
 | 	// its module bound apk path. However, override_android_app inherits ctx.ModuleName() | 
 | 	// from the base module. Therefore, use a.Name() which represents | 
 | 	// the module name for both android_app and override_android_app. | 
 | 	a.installApkName = ctx.DeviceConfig().OverridePackageNameFor( | 
 | 		proptools.StringDefault(a.overridableDeviceProperties.Stem, a.Name())) | 
 |  | 
 | 	if ctx.ModuleName() == "framework-res" { | 
 | 		// framework-res.apk is installed as system/framework/framework-res.apk | 
 | 		a.installDir = android.PathForModuleInstall(ctx, "framework") | 
 | 	} else if a.Privileged() { | 
 | 		a.installDir = android.PathForModuleInstall(ctx, "priv-app", a.installApkName) | 
 | 	} else if ctx.InstallInTestcases() { | 
 | 		a.installDir = android.PathForModuleInstall(ctx, a.installApkName, ctx.DeviceConfig().DeviceArch()) | 
 | 	} else { | 
 | 		a.installDir = android.PathForModuleInstall(ctx, "app", a.installApkName) | 
 | 	} | 
 | 	a.onDeviceDir = android.InstallPathToOnDevicePath(ctx, a.installDir) | 
 |  | 
 | 	a.classLoaderContexts = a.usesLibrary.classLoaderContextForUsesLibDeps(ctx) | 
 |  | 
 | 	var noticeAssetPath android.WritablePath | 
 | 	if Bool(a.appProperties.Embed_notices) || ctx.Config().IsEnvTrue("ALWAYS_EMBED_NOTICES") { | 
 | 		// The rule to create the notice file can't be generated yet, as the final output path | 
 | 		// for the apk isn't known yet.  Add the path where the notice file will be generated to the | 
 | 		// aapt rules now before calling aaptBuildActions, the rule to create the notice file will | 
 | 		// be generated later. | 
 | 		noticeAssetPath = android.PathForModuleOut(ctx, "NOTICE", "NOTICE.html.gz") | 
 | 		a.aapt.noticeFile = android.OptionalPathForPath(noticeAssetPath) | 
 | 	} | 
 |  | 
 | 	// For apps targeting latest target_sdk_version | 
 | 	if Bool(a.appProperties.Enforce_default_target_sdk_version) { | 
 | 		a.SetEnforceDefaultTargetSdkVersion(true) | 
 | 	} | 
 |  | 
 | 	// Process all building blocks, from AAPT to certificates. | 
 | 	a.aaptBuildActions(ctx) | 
 |  | 
 | 	// The decision to enforce <uses-library> checks is made before adding implicit SDK libraries. | 
 | 	a.usesLibrary.freezeEnforceUsesLibraries() | 
 |  | 
 | 	// Add implicit SDK libraries to <uses-library> list. | 
 | 	requiredUsesLibs, optionalUsesLibs := a.classLoaderContexts.UsesLibs() | 
 | 	for _, usesLib := range requiredUsesLibs { | 
 | 		a.usesLibrary.addLib(usesLib, false) | 
 | 	} | 
 | 	for _, usesLib := range optionalUsesLibs { | 
 | 		a.usesLibrary.addLib(usesLib, true) | 
 | 	} | 
 |  | 
 | 	// Check that the <uses-library> list is coherent with the manifest. | 
 | 	if a.usesLibrary.enforceUsesLibraries() { | 
 | 		manifestCheckFile := a.usesLibrary.verifyUsesLibrariesManifest(ctx, a.mergedManifestFile) | 
 | 		apkDeps = append(apkDeps, manifestCheckFile) | 
 | 	} | 
 |  | 
 | 	a.proguardBuildActions(ctx) | 
 |  | 
 | 	a.linter.mergedManifest = a.aapt.mergedManifestFile | 
 | 	a.linter.manifest = a.aapt.manifestPath | 
 | 	a.linter.resources = a.aapt.resourceFiles | 
 | 	a.linter.buildModuleReportZip = ctx.Config().UnbundledBuildApps() | 
 |  | 
 | 	dexJarFile := a.dexBuildActions(ctx) | 
 |  | 
 | 	jniLibs, prebuiltJniPackages, certificates := collectAppDeps(ctx, a, a.shouldEmbedJnis(ctx), !Bool(a.appProperties.Jni_uses_platform_apis)) | 
 | 	jniJarFile := a.jniBuildActions(jniLibs, prebuiltJniPackages, ctx) | 
 |  | 
 | 	if ctx.Failed() { | 
 | 		return | 
 | 	} | 
 |  | 
 | 	a.certificate, certificates = processMainCert(a.ModuleBase, a.getCertString(ctx), certificates, ctx) | 
 |  | 
 | 	// Build a final signed app package. | 
 | 	packageFile := android.PathForModuleOut(ctx, a.installApkName+".apk") | 
 | 	v4SigningRequested := Bool(a.Module.deviceProperties.V4_signature) | 
 | 	var v4SignatureFile android.WritablePath = nil | 
 | 	if v4SigningRequested { | 
 | 		v4SignatureFile = android.PathForModuleOut(ctx, a.installApkName+".apk.idsig") | 
 | 	} | 
 | 	var lineageFile android.Path | 
 | 	if lineage := String(a.overridableAppProperties.Lineage); lineage != "" { | 
 | 		lineageFile = android.PathForModuleSrc(ctx, lineage) | 
 | 	} | 
 | 	rotationMinSdkVersion := String(a.overridableAppProperties.RotationMinSdkVersion) | 
 |  | 
 | 	CreateAndSignAppPackage(ctx, packageFile, a.exportPackage, jniJarFile, dexJarFile, certificates, apkDeps, v4SignatureFile, lineageFile, rotationMinSdkVersion, Bool(a.dexProperties.Optimize.Shrink_resources)) | 
 | 	a.outputFile = packageFile | 
 | 	if v4SigningRequested { | 
 | 		a.extraOutputFiles = append(a.extraOutputFiles, v4SignatureFile) | 
 | 	} | 
 |  | 
 | 	if a.aapt.noticeFile.Valid() { | 
 | 		// Generating the notice file rule has to be here after a.outputFile is known. | 
 | 		noticeFile := android.PathForModuleOut(ctx, "NOTICE.html.gz") | 
 | 		android.BuildNoticeHtmlOutputFromLicenseMetadata( | 
 | 			ctx, noticeFile, "", "", | 
 | 			[]string{ | 
 | 				a.installDir.String() + "/", | 
 | 				android.PathForModuleInstall(ctx).String() + "/", | 
 | 				a.outputFile.String(), | 
 | 			}) | 
 | 		builder := android.NewRuleBuilder(pctx, ctx) | 
 | 		builder.Command().Text("cp"). | 
 | 			Input(noticeFile). | 
 | 			Output(noticeAssetPath) | 
 | 		builder.Build("notice_dir", "Building notice dir") | 
 | 	} | 
 |  | 
 | 	for _, split := range a.aapt.splits { | 
 | 		// Sign the split APKs | 
 | 		packageFile := android.PathForModuleOut(ctx, a.installApkName+"_"+split.suffix+".apk") | 
 | 		if v4SigningRequested { | 
 | 			v4SignatureFile = android.PathForModuleOut(ctx, a.installApkName+"_"+split.suffix+".apk.idsig") | 
 | 		} | 
 | 		CreateAndSignAppPackage(ctx, packageFile, split.path, nil, nil, certificates, apkDeps, v4SignatureFile, lineageFile, rotationMinSdkVersion, false) | 
 | 		a.extraOutputFiles = append(a.extraOutputFiles, packageFile) | 
 | 		if v4SigningRequested { | 
 | 			a.extraOutputFiles = append(a.extraOutputFiles, v4SignatureFile) | 
 | 		} | 
 | 	} | 
 |  | 
 | 	// Build an app bundle. | 
 | 	bundleFile := android.PathForModuleOut(ctx, "base.zip") | 
 | 	BuildBundleModule(ctx, bundleFile, a.exportPackage, jniJarFile, dexJarFile) | 
 | 	a.bundleFile = bundleFile | 
 |  | 
 | 	allowlist := a.createPrivappAllowlist(ctx) | 
 | 	if allowlist != nil { | 
 | 		a.privAppAllowlist = android.OptionalPathForPath(allowlist) | 
 | 	} | 
 |  | 
 | 	apexInfo := ctx.Provider(android.ApexInfoProvider).(android.ApexInfo) | 
 |  | 
 | 	// Install the app package. | 
 | 	shouldInstallAppPackage := (Bool(a.Module.properties.Installable) || ctx.Host()) && apexInfo.IsForPlatform() && !a.appProperties.PreventInstall | 
 | 	if shouldInstallAppPackage { | 
 | 		if a.privAppAllowlist.Valid() { | 
 | 			allowlistInstallPath := android.PathForModuleInstall(ctx, "etc", "permissions") | 
 | 			allowlistInstallFilename := a.installApkName + ".xml" | 
 | 			ctx.InstallFile(allowlistInstallPath, allowlistInstallFilename, a.privAppAllowlist.Path()) | 
 | 		} | 
 |  | 
 | 		var extraInstalledPaths android.Paths | 
 | 		for _, extra := range a.extraOutputFiles { | 
 | 			installed := ctx.InstallFile(a.installDir, extra.Base(), extra) | 
 | 			extraInstalledPaths = append(extraInstalledPaths, installed) | 
 | 		} | 
 | 		ctx.InstallFile(a.installDir, a.outputFile.Base(), a.outputFile, extraInstalledPaths...) | 
 | 	} | 
 |  | 
 | 	a.buildAppDependencyInfo(ctx) | 
 | } | 
 |  | 
 | type appDepsInterface interface { | 
 | 	SdkVersion(ctx android.EarlyModuleContext) android.SdkSpec | 
 | 	MinSdkVersion(ctx android.EarlyModuleContext) android.ApiLevel | 
 | 	RequiresStableAPIs(ctx android.BaseModuleContext) bool | 
 | } | 
 |  | 
 | func collectAppDeps(ctx android.ModuleContext, app appDepsInterface, | 
 | 	shouldCollectRecursiveNativeDeps bool, | 
 | 	checkNativeSdkVersion bool) ([]jniLib, android.Paths, []Certificate) { | 
 |  | 
 | 	var jniLibs []jniLib | 
 | 	var prebuiltJniPackages android.Paths | 
 | 	var certificates []Certificate | 
 | 	seenModulePaths := make(map[string]bool) | 
 |  | 
 | 	if checkNativeSdkVersion { | 
 | 		checkNativeSdkVersion = app.SdkVersion(ctx).Specified() && | 
 | 			app.SdkVersion(ctx).Kind != android.SdkCorePlatform && !app.RequiresStableAPIs(ctx) | 
 | 	} | 
 |  | 
 | 	ctx.WalkDeps(func(module android.Module, parent android.Module) bool { | 
 | 		otherName := ctx.OtherModuleName(module) | 
 | 		tag := ctx.OtherModuleDependencyTag(module) | 
 |  | 
 | 		if IsJniDepTag(tag) || cc.IsSharedDepTag(tag) { | 
 | 			if dep, ok := module.(cc.LinkableInterface); ok { | 
 | 				if dep.IsNdk(ctx.Config()) || dep.IsStubs() { | 
 | 					return false | 
 | 				} | 
 |  | 
 | 				lib := dep.OutputFile() | 
 | 				if lib.Valid() { | 
 | 					path := lib.Path() | 
 | 					if seenModulePaths[path.String()] { | 
 | 						return false | 
 | 					} | 
 | 					seenModulePaths[path.String()] = true | 
 |  | 
 | 					if checkNativeSdkVersion && dep.SdkVersion() == "" { | 
 | 						ctx.PropertyErrorf("jni_libs", "JNI dependency %q uses platform APIs, but this module does not", | 
 | 							otherName) | 
 | 					} | 
 |  | 
 | 					jniLibs = append(jniLibs, jniLib{ | 
 | 						name:           ctx.OtherModuleName(module), | 
 | 						path:           path, | 
 | 						target:         module.Target(), | 
 | 						coverageFile:   dep.CoverageOutputFile(), | 
 | 						unstrippedFile: dep.UnstrippedOutputFile(), | 
 | 						partition:      dep.Partition(), | 
 | 					}) | 
 | 				} else if ctx.Config().AllowMissingDependencies() { | 
 | 					ctx.AddMissingDependencies([]string{otherName}) | 
 | 				} else { | 
 | 					ctx.ModuleErrorf("dependency %q missing output file", otherName) | 
 | 				} | 
 | 			} else { | 
 | 				ctx.ModuleErrorf("jni_libs dependency %q must be a cc library", otherName) | 
 | 			} | 
 |  | 
 | 			return shouldCollectRecursiveNativeDeps | 
 | 		} | 
 |  | 
 | 		if info, ok := ctx.OtherModuleProvider(module, JniPackageProvider).(JniPackageInfo); ok { | 
 | 			prebuiltJniPackages = append(prebuiltJniPackages, info.JniPackages...) | 
 | 		} | 
 |  | 
 | 		if tag == certificateTag { | 
 | 			if dep, ok := module.(*AndroidAppCertificate); ok { | 
 | 				certificates = append(certificates, dep.Certificate) | 
 | 			} else { | 
 | 				ctx.ModuleErrorf("certificate dependency %q must be an android_app_certificate module", otherName) | 
 | 			} | 
 | 		} | 
 |  | 
 | 		return false | 
 | 	}) | 
 |  | 
 | 	return jniLibs, prebuiltJniPackages, certificates | 
 | } | 
 |  | 
 | func (a *AndroidApp) WalkPayloadDeps(ctx android.ModuleContext, do android.PayloadDepsCallback) { | 
 | 	ctx.WalkDeps(func(child, parent android.Module) bool { | 
 | 		isExternal := !a.DepIsInSameApex(ctx, child) | 
 | 		if am, ok := child.(android.ApexModule); ok { | 
 | 			if !do(ctx, parent, am, isExternal) { | 
 | 				return false | 
 | 			} | 
 | 		} | 
 | 		return !isExternal | 
 | 	}) | 
 | } | 
 |  | 
 | func (a *AndroidApp) buildAppDependencyInfo(ctx android.ModuleContext) { | 
 | 	if ctx.Host() { | 
 | 		return | 
 | 	} | 
 |  | 
 | 	depsInfo := android.DepNameToDepInfoMap{} | 
 | 	a.WalkPayloadDeps(ctx, func(ctx android.ModuleContext, from blueprint.Module, to android.ApexModule, externalDep bool) bool { | 
 | 		depName := to.Name() | 
 |  | 
 | 		// Skip dependencies that are only available to APEXes; they are developed with updatability | 
 | 		// in mind and don't need manual approval. | 
 | 		if to.(android.ApexModule).NotAvailableForPlatform() { | 
 | 			return true | 
 | 		} | 
 |  | 
 | 		if info, exist := depsInfo[depName]; exist { | 
 | 			info.From = append(info.From, from.Name()) | 
 | 			info.IsExternal = info.IsExternal && externalDep | 
 | 			depsInfo[depName] = info | 
 | 		} else { | 
 | 			toMinSdkVersion := "(no version)" | 
 | 			if m, ok := to.(interface { | 
 | 				MinSdkVersion(ctx android.EarlyModuleContext) android.ApiLevel | 
 | 			}); ok { | 
 | 				if v := m.MinSdkVersion(ctx); !v.IsNone() { | 
 | 					toMinSdkVersion = v.String() | 
 | 				} | 
 | 			} else if m, ok := to.(interface{ MinSdkVersion() string }); ok { | 
 | 				// TODO(b/175678607) eliminate the use of MinSdkVersion returning | 
 | 				// string | 
 | 				if v := m.MinSdkVersion(); v != "" { | 
 | 					toMinSdkVersion = v | 
 | 				} | 
 | 			} | 
 | 			depsInfo[depName] = android.ApexModuleDepInfo{ | 
 | 				To:            depName, | 
 | 				From:          []string{from.Name()}, | 
 | 				IsExternal:    externalDep, | 
 | 				MinSdkVersion: toMinSdkVersion, | 
 | 			} | 
 | 		} | 
 | 		return true | 
 | 	}) | 
 |  | 
 | 	a.ApexBundleDepsInfo.BuildDepsInfoLists(ctx, a.MinSdkVersion(ctx).String(), depsInfo) | 
 | } | 
 |  | 
 | func (a *AndroidApp) enforceDefaultTargetSdkVersion() bool { | 
 | 	return a.appProperties.EnforceDefaultTargetSdkVersion | 
 | } | 
 |  | 
 | func (a *AndroidApp) SetEnforceDefaultTargetSdkVersion(val bool) { | 
 | 	a.appProperties.EnforceDefaultTargetSdkVersion = val | 
 | } | 
 |  | 
 | func (a *AndroidApp) Updatable() bool { | 
 | 	return Bool(a.appProperties.Updatable) | 
 | } | 
 |  | 
 | func (a *AndroidApp) SetUpdatable(val bool) { | 
 | 	a.appProperties.Updatable = &val | 
 | } | 
 |  | 
 | func (a *AndroidApp) getCertString(ctx android.BaseModuleContext) string { | 
 | 	certificate, overridden := ctx.DeviceConfig().OverrideCertificateFor(ctx.ModuleName()) | 
 | 	if overridden { | 
 | 		return ":" + certificate | 
 | 	} | 
 | 	return String(a.overridableAppProperties.Certificate) | 
 | } | 
 |  | 
 | func (a *AndroidApp) DepIsInSameApex(ctx android.BaseModuleContext, dep android.Module) bool { | 
 | 	if IsJniDepTag(ctx.OtherModuleDependencyTag(dep)) { | 
 | 		return true | 
 | 	} | 
 | 	return a.Library.DepIsInSameApex(ctx, dep) | 
 | } | 
 |  | 
 | // For OutputFileProducer interface | 
 | func (a *AndroidApp) OutputFiles(tag string) (android.Paths, error) { | 
 | 	switch tag { | 
 | 	// In some instances, it can be useful to reference the aapt-generated flags from another | 
 | 	// target, e.g., system server implements services declared in the framework-res manifest. | 
 | 	case ".aapt.proguardOptionsFile": | 
 | 		return []android.Path{a.proguardOptionsFile}, nil | 
 | 	case ".aapt.srcjar": | 
 | 		if a.aaptSrcJar != nil { | 
 | 			return []android.Path{a.aaptSrcJar}, nil | 
 | 		} | 
 | 	case ".aapt.jar": | 
 | 		if a.rJar != nil { | 
 | 			return []android.Path{a.rJar}, nil | 
 | 		} | 
 | 	case ".export-package.apk": | 
 | 		return []android.Path{a.exportPackage}, nil | 
 | 	case ".manifest.xml": | 
 | 		return []android.Path{a.aapt.manifestPath}, nil | 
 | 	} | 
 | 	return a.Library.OutputFiles(tag) | 
 | } | 
 |  | 
 | func (a *AndroidApp) Privileged() bool { | 
 | 	return Bool(a.appProperties.Privileged) | 
 | } | 
 |  | 
 | func (a *AndroidApp) IsNativeCoverageNeeded(ctx android.BaseModuleContext) bool { | 
 | 	return ctx.Device() && ctx.DeviceConfig().NativeCoverageEnabled() | 
 | } | 
 |  | 
 | func (a *AndroidApp) SetPreventInstall() { | 
 | 	a.appProperties.PreventInstall = true | 
 | } | 
 |  | 
 | func (a *AndroidApp) MarkAsCoverageVariant(coverage bool) { | 
 | 	a.appProperties.IsCoverageVariant = coverage | 
 | } | 
 |  | 
 | func (a *AndroidApp) EnableCoverageIfNeeded() {} | 
 |  | 
 | var _ cc.Coverage = (*AndroidApp)(nil) | 
 |  | 
 | func (a *AndroidApp) IDEInfo(dpInfo *android.IdeInfo) { | 
 | 	a.Library.IDEInfo(dpInfo) | 
 | 	a.aapt.IDEInfo(dpInfo) | 
 | } | 
 |  | 
 | func (a *AndroidApp) productCharacteristicsRROPackageName() string { | 
 | 	return proptools.String(a.appProperties.ProductCharacteristicsRROPackageName) | 
 | } | 
 |  | 
 | func (a *AndroidApp) productCharacteristicsRROManifestModuleName() string { | 
 | 	return proptools.String(a.appProperties.ProductCharacteristicsRROManifestModuleName) | 
 | } | 
 |  | 
 | // android_app compiles sources and Android resources into an Android application package `.apk` file. | 
 | func AndroidAppFactory() android.Module { | 
 | 	module := &AndroidApp{} | 
 |  | 
 | 	module.Module.dexProperties.Optimize.EnabledByDefault = true | 
 | 	module.Module.dexProperties.Optimize.Shrink = proptools.BoolPtr(true) | 
 | 	module.Module.dexProperties.Optimize.Proguard_compatibility = proptools.BoolPtr(false) | 
 |  | 
 | 	module.Module.properties.Instrument = true | 
 | 	module.Module.properties.Supports_static_instrumentation = true | 
 | 	module.Module.properties.Installable = proptools.BoolPtr(true) | 
 |  | 
 | 	module.addHostAndDeviceProperties() | 
 | 	module.AddProperties( | 
 | 		&module.aaptProperties, | 
 | 		&module.appProperties, | 
 | 		&module.overridableAppProperties) | 
 |  | 
 | 	module.usesLibrary.enforce = true | 
 |  | 
 | 	android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibCommon) | 
 | 	android.InitDefaultableModule(module) | 
 | 	android.InitOverridableModule(module, &module.overridableAppProperties.Overrides) | 
 | 	android.InitApexModule(module) | 
 | 	android.InitBazelModule(module) | 
 |  | 
 | 	android.AddLoadHook(module, func(ctx android.LoadHookContext) { | 
 | 		a := ctx.Module().(*AndroidApp) | 
 |  | 
 | 		characteristics := ctx.Config().ProductAAPTCharacteristics() | 
 | 		if characteristics == "default" || characteristics == "" { | 
 | 			module.appProperties.Generate_product_characteristics_rro = nil | 
 | 			// no need to create RRO | 
 | 			return | 
 | 		} | 
 |  | 
 | 		if !proptools.Bool(module.appProperties.Generate_product_characteristics_rro) { | 
 | 			return | 
 | 		} | 
 |  | 
 | 		rroPackageName := a.Name() + "__" + strings.ReplaceAll(characteristics, ",", "_") + "__auto_generated_characteristics_rro" | 
 | 		rroManifestName := rroPackageName + "_manifest" | 
 |  | 
 | 		a.appProperties.ProductCharacteristicsRROPackageName = proptools.StringPtr(rroPackageName) | 
 | 		a.appProperties.ProductCharacteristicsRROManifestModuleName = proptools.StringPtr(rroManifestName) | 
 |  | 
 | 		rroManifestProperties := struct { | 
 | 			Name  *string | 
 | 			Tools []string | 
 | 			Out   []string | 
 | 			Srcs  []string | 
 | 			Cmd   *string | 
 | 		}{ | 
 | 			Name:  proptools.StringPtr(rroManifestName), | 
 | 			Tools: []string{"characteristics_rro_generator"}, | 
 | 			Out:   []string{"AndroidManifest.xml"}, | 
 | 			Srcs:  []string{":" + a.Name() + "{.manifest.xml}"}, | 
 | 			Cmd:   proptools.StringPtr("$(location characteristics_rro_generator) $(in) $(out)"), | 
 | 		} | 
 | 		ctx.CreateModule(genrule.GenRuleFactory, &rroManifestProperties) | 
 |  | 
 | 		rroProperties := struct { | 
 | 			Name           *string | 
 | 			Filter_product *string | 
 | 			Aaptflags      []string | 
 | 			Manifest       *string | 
 | 			Resource_dirs  []string | 
 | 		}{ | 
 | 			Name:           proptools.StringPtr(rroPackageName), | 
 | 			Filter_product: proptools.StringPtr(characteristics), | 
 | 			Aaptflags:      []string{"--auto-add-overlay"}, | 
 | 			Manifest:       proptools.StringPtr(":" + rroManifestName), | 
 | 			Resource_dirs:  a.aaptProperties.Resource_dirs, | 
 | 		} | 
 | 		ctx.CreateModule(RuntimeResourceOverlayFactory, &rroProperties) | 
 | 	}) | 
 |  | 
 | 	return module | 
 | } | 
 |  | 
 | // A dictionary of values to be overridden in the manifest. | 
 | type Manifest_values struct { | 
 | 	// Overrides the value of package_name in the manifest | 
 | 	ApplicationId *string | 
 | } | 
 |  | 
 | type appTestProperties struct { | 
 | 	// The name of the android_app module that the tests will run against. | 
 | 	Instrumentation_for *string | 
 |  | 
 | 	// If specified, the instrumentation target package name in the manifest is overwritten by it. | 
 | 	Instrumentation_target_package *string | 
 |  | 
 | 	// If specified, the mainline module package name in the test config is overwritten by it. | 
 | 	Mainline_package_name *string | 
 |  | 
 | 	Manifest_values Manifest_values | 
 | } | 
 |  | 
 | type AndroidTest struct { | 
 | 	AndroidApp | 
 |  | 
 | 	appTestProperties appTestProperties | 
 |  | 
 | 	testProperties testProperties | 
 |  | 
 | 	testConfig       android.Path | 
 | 	extraTestConfigs android.Paths | 
 | 	data             android.Paths | 
 |  | 
 | 	android.BazelModuleBase | 
 | } | 
 |  | 
 | func (a *AndroidTest) InstallInTestcases() bool { | 
 | 	return true | 
 | } | 
 |  | 
 | type androidTestApp interface { | 
 | 	includedInTestSuite(searchPrefix string) bool | 
 | } | 
 |  | 
 | func (a *AndroidTest) includedInTestSuite(searchPrefix string) bool { | 
 | 	return android.PrefixInList(a.testProperties.Test_suites, searchPrefix) | 
 | } | 
 |  | 
 | func (a *AndroidTestHelperApp) includedInTestSuite(searchPrefix string) bool { | 
 | 	return android.PrefixInList(a.appTestHelperAppProperties.Test_suites, searchPrefix) | 
 | } | 
 |  | 
 | func (a *AndroidTest) GenerateAndroidBuildActions(ctx android.ModuleContext) { | 
 | 	var configs []tradefed.Config | 
 | 	if a.appTestProperties.Instrumentation_target_package != nil { | 
 | 		a.additionalAaptFlags = append(a.additionalAaptFlags, | 
 | 			"--rename-instrumentation-target-package "+*a.appTestProperties.Instrumentation_target_package) | 
 | 	} else if a.appTestProperties.Instrumentation_for != nil { | 
 | 		// Check if the instrumentation target package is overridden. | 
 | 		manifestPackageName, overridden := ctx.DeviceConfig().OverrideManifestPackageNameFor(*a.appTestProperties.Instrumentation_for) | 
 | 		if overridden { | 
 | 			a.additionalAaptFlags = append(a.additionalAaptFlags, "--rename-instrumentation-target-package "+manifestPackageName) | 
 | 		} | 
 | 	} | 
 | 	applicationId := a.appTestProperties.Manifest_values.ApplicationId | 
 | 	if applicationId != nil { | 
 | 		if a.overridableAppProperties.Package_name != nil { | 
 | 			ctx.PropertyErrorf("manifest_values.applicationId", "property is not supported when property package_name is set.") | 
 | 		} | 
 | 		a.aapt.manifestValues.applicationId = *applicationId | 
 | 	} | 
 | 	a.generateAndroidBuildActions(ctx) | 
 |  | 
 | 	for _, module := range a.testProperties.Test_mainline_modules { | 
 | 		configs = append(configs, tradefed.Option{Name: "config-descriptor:metadata", Key: "mainline-param", Value: module}) | 
 | 	} | 
 |  | 
 | 	testConfig := tradefed.AutoGenInstrumentationTestConfig(ctx, a.testProperties.Test_config, | 
 | 		a.testProperties.Test_config_template, a.manifestPath, a.testProperties.Test_suites, a.testProperties.Auto_gen_config, configs) | 
 | 	a.testConfig = a.FixTestConfig(ctx, testConfig) | 
 | 	a.extraTestConfigs = android.PathsForModuleSrc(ctx, a.testProperties.Test_options.Extra_test_configs) | 
 | 	a.data = android.PathsForModuleSrc(ctx, a.testProperties.Data) | 
 | 	ctx.SetProvider(testing.TestModuleProviderKey, testing.TestModuleProviderData{}) | 
 | } | 
 |  | 
 | func (a *AndroidTest) FixTestConfig(ctx android.ModuleContext, testConfig android.Path) android.Path { | 
 | 	if testConfig == nil { | 
 | 		return nil | 
 | 	} | 
 |  | 
 | 	fixedConfig := android.PathForModuleOut(ctx, "test_config_fixer", "AndroidTest.xml") | 
 | 	rule := android.NewRuleBuilder(pctx, ctx) | 
 | 	command := rule.Command().BuiltTool("test_config_fixer").Input(testConfig).Output(fixedConfig) | 
 | 	fixNeeded := false | 
 |  | 
 | 	// Auto-generated test config uses `ModuleName` as the APK name. So fix it if it is not the case. | 
 | 	if ctx.ModuleName() != a.installApkName { | 
 | 		fixNeeded = true | 
 | 		command.FlagWithArg("--test-file-name ", a.installApkName+".apk") | 
 | 	} | 
 |  | 
 | 	if a.overridableAppProperties.Package_name != nil { | 
 | 		fixNeeded = true | 
 | 		command.FlagWithInput("--manifest ", a.manifestPath). | 
 | 			FlagWithArg("--package-name ", *a.overridableAppProperties.Package_name) | 
 | 	} | 
 |  | 
 | 	if a.appTestProperties.Mainline_package_name != nil { | 
 | 		fixNeeded = true | 
 | 		command.FlagWithArg("--mainline-package-name ", *a.appTestProperties.Mainline_package_name) | 
 | 	} | 
 |  | 
 | 	if fixNeeded { | 
 | 		rule.Build("fix_test_config", "fix test config") | 
 | 		return fixedConfig | 
 | 	} | 
 | 	return testConfig | 
 | } | 
 |  | 
 | func (a *AndroidTest) DepsMutator(ctx android.BottomUpMutatorContext) { | 
 | 	a.AndroidApp.DepsMutator(ctx) | 
 | } | 
 |  | 
 | func (a *AndroidTest) OverridablePropertiesDepsMutator(ctx android.BottomUpMutatorContext) { | 
 | 	a.AndroidApp.OverridablePropertiesDepsMutator(ctx) | 
 | 	if a.appTestProperties.Instrumentation_for != nil { | 
 | 		// The android_app dependency listed in instrumentation_for needs to be added to the classpath for javac, | 
 | 		// but not added to the aapt2 link includes like a normal android_app or android_library dependency, so | 
 | 		// use instrumentationForTag instead of libTag. | 
 | 		ctx.AddVariationDependencies(nil, instrumentationForTag, String(a.appTestProperties.Instrumentation_for)) | 
 | 	} | 
 | } | 
 |  | 
 | // android_test compiles test sources and Android resources into an Android application package `.apk` file and | 
 | // creates an `AndroidTest.xml` file to allow running the test with `atest` or a `TEST_MAPPING` file. | 
 | func AndroidTestFactory() android.Module { | 
 | 	module := &AndroidTest{} | 
 |  | 
 | 	module.Module.dexProperties.Optimize.EnabledByDefault = false | 
 |  | 
 | 	module.Module.properties.Instrument = true | 
 | 	module.Module.properties.Supports_static_instrumentation = true | 
 | 	module.Module.properties.Installable = proptools.BoolPtr(true) | 
 | 	module.appProperties.Use_embedded_native_libs = proptools.BoolPtr(true) | 
 | 	module.appProperties.AlwaysPackageNativeLibs = true | 
 | 	module.Module.dexpreopter.isTest = true | 
 | 	module.Module.linter.properties.Lint.Test = proptools.BoolPtr(true) | 
 |  | 
 | 	module.addHostAndDeviceProperties() | 
 | 	module.AddProperties( | 
 | 		&module.aaptProperties, | 
 | 		&module.appProperties, | 
 | 		&module.appTestProperties, | 
 | 		&module.overridableAppProperties, | 
 | 		&module.testProperties) | 
 |  | 
 | 	android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibCommon) | 
 | 	android.InitDefaultableModule(module) | 
 | 	android.InitOverridableModule(module, &module.overridableAppProperties.Overrides) | 
 |  | 
 | 	android.InitBazelModule(module) | 
 | 	return module | 
 | } | 
 |  | 
 | type appTestHelperAppProperties struct { | 
 | 	// list of compatibility suites (for example "cts", "vts") that the module should be | 
 | 	// installed into. | 
 | 	Test_suites []string `android:"arch_variant"` | 
 |  | 
 | 	// Flag to indicate whether or not to create test config automatically. If AndroidTest.xml | 
 | 	// doesn't exist next to the Android.bp, this attribute doesn't need to be set to true | 
 | 	// explicitly. | 
 | 	Auto_gen_config *bool | 
 |  | 
 | 	// Install the test into a folder named for the module in all test suites. | 
 | 	Per_testcase_directory *bool | 
 |  | 
 | 	Manifest_values Manifest_values | 
 | } | 
 |  | 
 | type AndroidTestHelperApp struct { | 
 | 	AndroidApp | 
 |  | 
 | 	appTestHelperAppProperties appTestHelperAppProperties | 
 |  | 
 | 	android.BazelModuleBase | 
 | } | 
 |  | 
 | func (a *AndroidTestHelperApp) InstallInTestcases() bool { | 
 | 	return true | 
 | } | 
 |  | 
 | // android_test_helper_app compiles sources and Android resources into an Android application package `.apk` file that | 
 | // will be used by tests, but does not produce an `AndroidTest.xml` file so the module will not be run directly as a | 
 | // test. | 
 | func AndroidTestHelperAppFactory() android.Module { | 
 | 	module := &AndroidTestHelperApp{} | 
 |  | 
 | 	// TODO(b/192032291): Disable by default after auditing downstream usage. | 
 | 	module.Module.dexProperties.Optimize.EnabledByDefault = true | 
 |  | 
 | 	module.Module.properties.Installable = proptools.BoolPtr(true) | 
 | 	module.appProperties.Use_embedded_native_libs = proptools.BoolPtr(true) | 
 | 	module.appProperties.AlwaysPackageNativeLibs = true | 
 | 	module.Module.dexpreopter.isTest = true | 
 | 	module.Module.linter.properties.Lint.Test = proptools.BoolPtr(true) | 
 |  | 
 | 	module.addHostAndDeviceProperties() | 
 | 	module.AddProperties( | 
 | 		&module.aaptProperties, | 
 | 		&module.appProperties, | 
 | 		&module.appTestHelperAppProperties, | 
 | 		&module.overridableAppProperties) | 
 |  | 
 | 	android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibCommon) | 
 | 	android.InitDefaultableModule(module) | 
 | 	android.InitApexModule(module) | 
 | 	android.InitBazelModule(module) | 
 | 	return module | 
 | } | 
 |  | 
 | type AndroidAppCertificate struct { | 
 | 	android.ModuleBase | 
 | 	android.BazelModuleBase | 
 |  | 
 | 	properties  AndroidAppCertificateProperties | 
 | 	Certificate Certificate | 
 | } | 
 |  | 
 | type AndroidAppCertificateProperties struct { | 
 | 	// Name of the certificate files.  Extensions .x509.pem and .pk8 will be added to the name. | 
 | 	Certificate *string | 
 | } | 
 |  | 
 | // android_app_certificate modules can be referenced by the certificates property of android_app modules to select | 
 | // the signing key. | 
 | func AndroidAppCertificateFactory() android.Module { | 
 | 	module := &AndroidAppCertificate{} | 
 | 	module.AddProperties(&module.properties) | 
 | 	android.InitAndroidModule(module) | 
 | 	android.InitBazelModule(module) | 
 | 	return module | 
 | } | 
 |  | 
 | func (c *AndroidAppCertificate) GenerateAndroidBuildActions(ctx android.ModuleContext) { | 
 | 	cert := String(c.properties.Certificate) | 
 | 	c.Certificate = Certificate{ | 
 | 		Pem: android.PathForModuleSrc(ctx, cert+".x509.pem"), | 
 | 		Key: android.PathForModuleSrc(ctx, cert+".pk8"), | 
 | 	} | 
 | } | 
 |  | 
 | type OverrideAndroidApp struct { | 
 | 	android.ModuleBase | 
 | 	android.OverrideModuleBase | 
 | } | 
 |  | 
 | func (i *OverrideAndroidApp) GenerateAndroidBuildActions(_ android.ModuleContext) { | 
 | 	// All the overrides happen in the base module. | 
 | 	// TODO(jungjw): Check the base module type. | 
 | } | 
 |  | 
 | // override_android_app is used to create an android_app module based on another android_app by overriding | 
 | // some of its properties. | 
 | func OverrideAndroidAppModuleFactory() android.Module { | 
 | 	m := &OverrideAndroidApp{} | 
 | 	m.AddProperties( | 
 | 		&OverridableDeviceProperties{}, | 
 | 		&overridableAppProperties{}, | 
 | 	) | 
 |  | 
 | 	android.InitAndroidMultiTargetsArchModule(m, android.DeviceSupported, android.MultilibCommon) | 
 | 	android.InitOverrideModule(m) | 
 | 	return m | 
 | } | 
 |  | 
 | type OverrideAndroidTest struct { | 
 | 	android.ModuleBase | 
 | 	android.OverrideModuleBase | 
 | } | 
 |  | 
 | func (i *OverrideAndroidTest) GenerateAndroidBuildActions(_ android.ModuleContext) { | 
 | 	// All the overrides happen in the base module. | 
 | 	// TODO(jungjw): Check the base module type. | 
 | } | 
 |  | 
 | // override_android_test is used to create an android_app module based on another android_test by overriding | 
 | // some of its properties. | 
 | func OverrideAndroidTestModuleFactory() android.Module { | 
 | 	m := &OverrideAndroidTest{} | 
 | 	m.AddProperties(&overridableAppProperties{}) | 
 | 	m.AddProperties(&appTestProperties{}) | 
 |  | 
 | 	android.InitAndroidMultiTargetsArchModule(m, android.DeviceSupported, android.MultilibCommon) | 
 | 	android.InitOverrideModule(m) | 
 | 	return m | 
 | } | 
 |  | 
 | type UsesLibraryProperties struct { | 
 | 	// A list of shared library modules that will be listed in uses-library tags in the AndroidManifest.xml file. | 
 | 	Uses_libs []string | 
 |  | 
 | 	// A list of shared library modules that will be listed in uses-library tags in the AndroidManifest.xml file with | 
 | 	// required=false. | 
 | 	Optional_uses_libs []string | 
 |  | 
 | 	// If true, the list of uses_libs and optional_uses_libs modules must match the AndroidManifest.xml file.  Defaults | 
 | 	// to true if either uses_libs or optional_uses_libs is set.  Will unconditionally default to true in the future. | 
 | 	Enforce_uses_libs *bool | 
 |  | 
 | 	// Optional name of the <uses-library> provided by this module. This is needed for non-SDK | 
 | 	// libraries, because SDK ones are automatically picked up by Soong. The <uses-library> name | 
 | 	// normally is the same as the module name, but there are exceptions. | 
 | 	Provides_uses_lib *string | 
 |  | 
 | 	// A list of shared library names to exclude from the classpath of the APK. Adding a library here | 
 | 	// will prevent it from being used when precompiling the APK and prevent it from being implicitly | 
 | 	// added to the APK's manifest's <uses-library> elements. | 
 | 	// | 
 | 	// Care must be taken when using this as it could result in runtime errors if the APK actually | 
 | 	// uses classes provided by the library and which are not provided in any other way. | 
 | 	// | 
 | 	// This is primarily intended for use by various CTS tests that check the runtime handling of the | 
 | 	// android.test.base shared library (and related libraries) but which depend on some common | 
 | 	// libraries that depend on the android.test.base library. Without this those tests will end up | 
 | 	// with a <uses-library android:name="android.test.base"/> in their manifest which would either | 
 | 	// render the tests worthless (as they would be testing the wrong behavior), or would break the | 
 | 	// test altogether by providing access to classes that the tests were not expecting. Those tests | 
 | 	// provide the android.test.base statically and use jarjar to rename them so they do not collide | 
 | 	// with the classes provided by the android.test.base library. | 
 | 	Exclude_uses_libs []string | 
 | } | 
 |  | 
 | // usesLibrary provides properties and helper functions for AndroidApp and AndroidAppImport to verify that the | 
 | // <uses-library> tags that end up in the manifest of an APK match the ones known to the build system through the | 
 | // uses_libs and optional_uses_libs properties.  The build system's values are used by dexpreopt to preopt apps | 
 | // with knowledge of their shared libraries. | 
 | type usesLibrary struct { | 
 | 	usesLibraryProperties UsesLibraryProperties | 
 |  | 
 | 	// Whether to enforce verify_uses_library check. | 
 | 	enforce bool | 
 | } | 
 |  | 
 | func (u *usesLibrary) addLib(lib string, optional bool) { | 
 | 	if !android.InList(lib, u.usesLibraryProperties.Uses_libs) && !android.InList(lib, u.usesLibraryProperties.Optional_uses_libs) { | 
 | 		if optional { | 
 | 			u.usesLibraryProperties.Optional_uses_libs = append(u.usesLibraryProperties.Optional_uses_libs, lib) | 
 | 		} else { | 
 | 			u.usesLibraryProperties.Uses_libs = append(u.usesLibraryProperties.Uses_libs, lib) | 
 | 		} | 
 | 	} | 
 | } | 
 |  | 
 | func (u *usesLibrary) deps(ctx android.BottomUpMutatorContext, addCompatDeps bool) { | 
 | 	if !ctx.Config().UnbundledBuild() || ctx.Config().UnbundledBuildImage() { | 
 | 		ctx.AddVariationDependencies(nil, usesLibReqTag, u.usesLibraryProperties.Uses_libs...) | 
 | 		ctx.AddVariationDependencies(nil, usesLibOptTag, u.presentOptionalUsesLibs(ctx)...) | 
 | 		// Only add these extra dependencies if the module is an app that depends on framework | 
 | 		// libs. This avoids creating a cyclic dependency: | 
 | 		//     e.g. framework-res -> org.apache.http.legacy -> ... -> framework-res. | 
 | 		if addCompatDeps { | 
 | 			// Dexpreopt needs paths to the dex jars of these libraries in order to construct | 
 | 			// class loader context for dex2oat. Add them as a dependency with a special tag. | 
 | 			ctx.AddVariationDependencies(nil, usesLibCompat29ReqTag, dexpreopt.CompatUsesLibs29...) | 
 | 			ctx.AddVariationDependencies(nil, usesLibCompat28OptTag, dexpreopt.OptionalCompatUsesLibs28...) | 
 | 			ctx.AddVariationDependencies(nil, usesLibCompat30OptTag, dexpreopt.OptionalCompatUsesLibs30...) | 
 | 		} | 
 | 	} else { | 
 | 		ctx.AddVariationDependencies(nil, r8LibraryJarTag, u.usesLibraryProperties.Uses_libs...) | 
 | 		ctx.AddVariationDependencies(nil, r8LibraryJarTag, u.presentOptionalUsesLibs(ctx)...) | 
 | 	} | 
 | } | 
 |  | 
 | // presentOptionalUsesLibs returns optional_uses_libs after filtering out libraries that don't exist in the source tree. | 
 | func (u *usesLibrary) presentOptionalUsesLibs(ctx android.BaseModuleContext) []string { | 
 | 	optionalUsesLibs := android.FilterListPred(u.usesLibraryProperties.Optional_uses_libs, func(s string) bool { | 
 | 		exists := ctx.OtherModuleExists(s) | 
 | 		if !exists && !android.InList(ctx.ModuleName(), ctx.Config().BuildWarningBadOptionalUsesLibsAllowlist()) { | 
 | 			fmt.Printf("Warning: Module '%s' depends on non-existing optional_uses_libs '%s'\n", ctx.ModuleName(), s) | 
 | 		} | 
 | 		return exists | 
 | 	}) | 
 | 	return optionalUsesLibs | 
 | } | 
 |  | 
 | // Helper function to replace string in a list. | 
 | func replaceInList(list []string, oldstr, newstr string) { | 
 | 	for i, str := range list { | 
 | 		if str == oldstr { | 
 | 			list[i] = newstr | 
 | 		} | 
 | 	} | 
 | } | 
 |  | 
 | // Returns a map of module names of shared library dependencies to the paths to their dex jars on | 
 | // host and on device. | 
 | func (u *usesLibrary) classLoaderContextForUsesLibDeps(ctx android.ModuleContext) dexpreopt.ClassLoaderContextMap { | 
 | 	clcMap := make(dexpreopt.ClassLoaderContextMap) | 
 |  | 
 | 	// Skip when UnbundledBuild() is true, but UnbundledBuildImage() is false. With | 
 | 	// UnbundledBuildImage() it is necessary to generate dexpreopt.config for post-dexpreopting. | 
 | 	if ctx.Config().UnbundledBuild() && !ctx.Config().UnbundledBuildImage() { | 
 | 		return clcMap | 
 | 	} | 
 |  | 
 | 	ctx.VisitDirectDeps(func(m android.Module) { | 
 | 		tag, isUsesLibTag := ctx.OtherModuleDependencyTag(m).(usesLibraryDependencyTag) | 
 | 		if !isUsesLibTag { | 
 | 			return | 
 | 		} | 
 |  | 
 | 		dep := android.RemoveOptionalPrebuiltPrefix(ctx.OtherModuleName(m)) | 
 |  | 
 | 		// Skip stub libraries. A dependency on the implementation library has been added earlier, | 
 | 		// so it will be added to CLC, but the stub shouldn't be. Stub libraries can be distingushed | 
 | 		// from implementation libraries by their name, which is different as it has a suffix. | 
 | 		if comp, ok := m.(SdkLibraryComponentDependency); ok { | 
 | 			if impl := comp.OptionalSdkLibraryImplementation(); impl != nil && *impl != dep { | 
 | 				return | 
 | 			} | 
 | 		} | 
 |  | 
 | 		if lib, ok := m.(UsesLibraryDependency); ok { | 
 | 			libName := dep | 
 | 			if ulib, ok := m.(ProvidesUsesLib); ok && ulib.ProvidesUsesLib() != nil { | 
 | 				libName = *ulib.ProvidesUsesLib() | 
 | 				// Replace module name with library name in `uses_libs`/`optional_uses_libs` in | 
 | 				// order to pass verify_uses_libraries check (which compares these properties | 
 | 				// against library names written in the manifest). | 
 | 				replaceInList(u.usesLibraryProperties.Uses_libs, dep, libName) | 
 | 				replaceInList(u.usesLibraryProperties.Optional_uses_libs, dep, libName) | 
 | 			} | 
 | 			clcMap.AddContext(ctx, tag.sdkVersion, libName, tag.optional, | 
 | 				lib.DexJarBuildPath().PathOrNil(), lib.DexJarInstallPath(), | 
 | 				lib.ClassLoaderContexts()) | 
 | 		} else if ctx.Config().AllowMissingDependencies() { | 
 | 			ctx.AddMissingDependencies([]string{dep}) | 
 | 		} else { | 
 | 			ctx.ModuleErrorf("module %q in uses_libs or optional_uses_libs must be a java library", dep) | 
 | 		} | 
 | 	}) | 
 | 	return clcMap | 
 | } | 
 |  | 
 | // enforceUsesLibraries returns true of <uses-library> tags should be checked against uses_libs and optional_uses_libs | 
 | // properties.  Defaults to true if either of uses_libs or optional_uses_libs is specified.  Will default to true | 
 | // unconditionally in the future. | 
 | func (u *usesLibrary) enforceUsesLibraries() bool { | 
 | 	defaultEnforceUsesLibs := len(u.usesLibraryProperties.Uses_libs) > 0 || | 
 | 		len(u.usesLibraryProperties.Optional_uses_libs) > 0 | 
 | 	return BoolDefault(u.usesLibraryProperties.Enforce_uses_libs, u.enforce || defaultEnforceUsesLibs) | 
 | } | 
 |  | 
 | // Freeze the value of `enforce_uses_libs` based on the current values of `uses_libs` and `optional_uses_libs`. | 
 | func (u *usesLibrary) freezeEnforceUsesLibraries() { | 
 | 	enforce := u.enforceUsesLibraries() | 
 | 	u.usesLibraryProperties.Enforce_uses_libs = &enforce | 
 | } | 
 |  | 
 | // verifyUsesLibraries checks the <uses-library> tags in the manifest against the ones specified | 
 | // in the `uses_libs`/`optional_uses_libs` properties. The input can be either an XML manifest, or | 
 | // an APK with the manifest embedded in it (manifest_check will know which one it is by the file | 
 | // extension: APKs are supposed to end with '.apk'). | 
 | func (u *usesLibrary) verifyUsesLibraries(ctx android.ModuleContext, inputFile android.Path, | 
 | 	outputFile android.WritablePath) android.Path { | 
 |  | 
 | 	statusFile := dexpreopt.UsesLibrariesStatusFile(ctx) | 
 |  | 
 | 	// Disable verify_uses_libraries check if dexpreopt is globally disabled. Without dexpreopt the | 
 | 	// check is not necessary, and although it is good to have, it is difficult to maintain on | 
 | 	// non-linux build platforms where dexpreopt is generally disabled (the check may fail due to | 
 | 	// various unrelated reasons, such as a failure to get manifest from an APK). | 
 | 	global := dexpreopt.GetGlobalConfig(ctx) | 
 | 	if global.DisablePreopt || global.OnlyPreoptArtBootImage { | 
 | 		return inputFile | 
 | 	} | 
 |  | 
 | 	rule := android.NewRuleBuilder(pctx, ctx) | 
 | 	cmd := rule.Command().BuiltTool("manifest_check"). | 
 | 		Flag("--enforce-uses-libraries"). | 
 | 		Input(inputFile). | 
 | 		FlagWithOutput("--enforce-uses-libraries-status ", statusFile). | 
 | 		FlagWithInput("--aapt ", ctx.Config().HostToolPath(ctx, "aapt2")) | 
 |  | 
 | 	if outputFile != nil { | 
 | 		cmd.FlagWithOutput("-o ", outputFile) | 
 | 	} | 
 |  | 
 | 	if dexpreopt.GetGlobalConfig(ctx).RelaxUsesLibraryCheck { | 
 | 		cmd.Flag("--enforce-uses-libraries-relax") | 
 | 	} | 
 |  | 
 | 	for _, lib := range u.usesLibraryProperties.Uses_libs { | 
 | 		cmd.FlagWithArg("--uses-library ", lib) | 
 | 	} | 
 |  | 
 | 	for _, lib := range u.usesLibraryProperties.Optional_uses_libs { | 
 | 		cmd.FlagWithArg("--optional-uses-library ", lib) | 
 | 	} | 
 |  | 
 | 	rule.Build("verify_uses_libraries", "verify <uses-library>") | 
 | 	return outputFile | 
 | } | 
 |  | 
 | // verifyUsesLibrariesManifest checks the <uses-library> tags in an AndroidManifest.xml against | 
 | // the build system and returns the path to a copy of the manifest. | 
 | func (u *usesLibrary) verifyUsesLibrariesManifest(ctx android.ModuleContext, manifest android.Path) android.Path { | 
 | 	outputFile := android.PathForModuleOut(ctx, "manifest_check", "AndroidManifest.xml") | 
 | 	return u.verifyUsesLibraries(ctx, manifest, outputFile) | 
 | } | 
 |  | 
 | // verifyUsesLibrariesAPK checks the <uses-library> tags in the manifest of an APK against the build | 
 | // system and returns the path to a copy of the APK. | 
 | func (u *usesLibrary) verifyUsesLibrariesAPK(ctx android.ModuleContext, apk android.Path) { | 
 | 	u.verifyUsesLibraries(ctx, apk, nil) // for APKs manifest_check does not write output file | 
 | } | 
 |  | 
 | // For Bazel / bp2build | 
 |  | 
 | type bazelAndroidAppCertificateAttributes struct { | 
 | 	Certificate string | 
 | } | 
 |  | 
 | func (m *AndroidAppCertificate) ConvertWithBp2build(ctx android.Bp2buildMutatorContext) { | 
 | 	androidAppCertificateBp2Build(ctx, m) | 
 | } | 
 |  | 
 | func androidAppCertificateBp2Build(ctx android.Bp2buildMutatorContext, module *AndroidAppCertificate) { | 
 | 	var certificate string | 
 | 	if module.properties.Certificate != nil { | 
 | 		certificate = *module.properties.Certificate | 
 | 	} | 
 |  | 
 | 	attrs := &bazelAndroidAppCertificateAttributes{ | 
 | 		Certificate: certificate, | 
 | 	} | 
 |  | 
 | 	props := bazel.BazelTargetModuleProperties{ | 
 | 		Rule_class:        "android_app_certificate", | 
 | 		Bzl_load_location: "//build/bazel/rules/android:android_app_certificate.bzl", | 
 | 	} | 
 |  | 
 | 	ctx.CreateBazelTargetModule(props, android.CommonAttributes{Name: module.Name()}, attrs) | 
 | } | 
 |  | 
 | type manifestValueAttribute struct { | 
 | 	MinSdkVersion    *string | 
 | 	TargetSdkVersion *string | 
 | } | 
 |  | 
 | type bazelAndroidAppAttributes struct { | 
 | 	*javaCommonAttributes | 
 | 	*bazelAapt | 
 | 	Deps             bazel.LabelListAttribute | 
 | 	Custom_package   *string | 
 | 	Certificate      bazel.LabelAttribute | 
 | 	Certificate_name bazel.StringAttribute | 
 | 	Manifest_values  *manifestValueAttribute | 
 | 	Optimize         *bool | 
 | 	Proguard_specs   bazel.LabelListAttribute | 
 | 	Updatable        *bool | 
 | } | 
 |  | 
 | func (b bazelAapt) ConvertJavaResources(ctx android.Bp2buildMutatorContext, javaAttrs *javaCommonAttributes) bool { | 
 | 	// TODO (b/300470246) bp2build support for java_resources & java_resource_dirs in android rules | 
 | 	hasJavaResources := !javaAttrs.javaResourcesAttributes.Resources.IsEmpty() | 
 | 	if hasJavaResources { | 
 | 		ctx.MarkBp2buildUnconvertible(bp2build_metrics_proto.UnconvertedReasonType_UNSUPPORTED, "(b/300470246) java resources in android_* module") | 
 | 	} | 
 | 	return hasJavaResources | 
 | } | 
 |  | 
 | func convertWithBp2build(ctx android.Bp2buildMutatorContext, a *AndroidApp) (bool, android.CommonAttributes, *bazelAndroidAppAttributes) { | 
 | 	aapt, supported := a.convertAaptAttrsWithBp2Build(ctx) | 
 | 	if !supported { | 
 | 		return false, android.CommonAttributes{}, &bazelAndroidAppAttributes{} | 
 | 	} | 
 | 	if a.appProperties.Jni_uses_platform_apis != nil { | 
 | 		ctx.MarkBp2buildUnconvertible( | 
 | 			bp2build_metrics_proto.UnconvertedReasonType_UNSUPPORTED, | 
 | 			"TODO - b/299360988: Add bp2build support for jni_uses_platform_apis", | 
 | 		) | 
 | 		return false, android.CommonAttributes{}, &bazelAndroidAppAttributes{} | 
 | 	} | 
 | 	if a.appProperties.Jni_uses_sdk_apis != nil { | 
 | 		ctx.MarkBp2buildUnconvertible( | 
 | 			bp2build_metrics_proto.UnconvertedReasonType_UNSUPPORTED, | 
 | 			"TODO - b/299360988: Add bp2build support for jni_uses_sdk_apis", | 
 | 		) | 
 | 		return false, android.CommonAttributes{}, &bazelAndroidAppAttributes{} | 
 | 	} | 
 |  | 
 | 	certificate, certificateName := android.BazelStringOrLabelFromProp(ctx, a.overridableAppProperties.Certificate) | 
 |  | 
 | 	manifestValues := &manifestValueAttribute{ | 
 | 		MinSdkVersion:    a.deviceProperties.Min_sdk_version, | 
 | 		TargetSdkVersion: a.deviceProperties.Target_sdk_version, | 
 | 	} | 
 |  | 
 | 	appAttrs := &bazelAndroidAppAttributes{ | 
 | 		// TODO(b/209576404): handle package name override by product variable PRODUCT_MANIFEST_PACKAGE_NAME_OVERRIDES | 
 | 		Custom_package:   a.overridableAppProperties.Package_name, | 
 | 		Certificate:      certificate, | 
 | 		Certificate_name: certificateName, | 
 | 		Manifest_values:  manifestValues, | 
 | 		Updatable:        a.appProperties.Updatable, | 
 | 	} | 
 |  | 
 | 	// As framework-res has no sources, no deps in the Bazel sense, and java compilation, dexing and optimization is skipped by | 
 | 	// Soong specifically for it, return early here before any of the conversion work for the above is attempted. | 
 | 	if ctx.ModuleName() == "framework-res" { | 
 | 		appAttrs.bazelAapt = aapt | 
 | 		return true, android.CommonAttributes{Name: a.Name(), SkipData: proptools.BoolPtr(true)}, appAttrs | 
 | 	} | 
 |  | 
 | 	// Optimization is.. | 
 | 	// - enabled by default for android_app, android_test_helper_app | 
 | 	// - disabled by default for android_test | 
 | 	// | 
 | 	// TODO(b/192032291): Disable android_test_helper_app optimization by | 
 | 	// default after auditing downstream usage. | 
 | 	if a.dexProperties.Optimize.EnabledByDefault != a.dexer.effectiveOptimizeEnabled() { | 
 | 		// Property is explicitly defined by default from default, so emit the Bazel attribute. | 
 | 		appAttrs.Optimize = proptools.BoolPtr(a.dexer.effectiveOptimizeEnabled()) | 
 | 	} | 
 |  | 
 | 	if a.dexer.effectiveOptimizeEnabled() { | 
 | 		handCraftedFlags := "" | 
 | 		if Bool(a.dexProperties.Optimize.Ignore_warnings) { | 
 | 			handCraftedFlags += "-ignorewarning " | 
 | 		} | 
 | 		if !Bool(a.dexProperties.Optimize.Shrink) { | 
 | 			handCraftedFlags += "-dontshrink " | 
 | 		} | 
 | 		if !Bool(a.dexProperties.Optimize.Optimize) { | 
 | 			handCraftedFlags += "-dontoptimize " | 
 | 		} | 
 | 		if !Bool(a.dexProperties.Optimize.Obfuscate) { | 
 | 			handCraftedFlags += "-dontobfuscate " | 
 | 		} | 
 | 		appAttrs.Proguard_specs = bazel.MakeLabelListAttribute(android.BazelLabelForModuleSrc(ctx, a.dexProperties.Optimize.Proguard_flags_files)) | 
 | 		if handCraftedFlags != "" { | 
 | 			generatedFlagFileRuleName := a.Name() + "_proguard_flags" | 
 | 			ctx.CreateBazelTargetModule(bazel.BazelTargetModuleProperties{ | 
 | 				Rule_class: "genrule", | 
 | 			}, android.CommonAttributes{ | 
 | 				Name:     generatedFlagFileRuleName, | 
 | 				SkipData: proptools.BoolPtr(true), | 
 | 			}, &genrule.BazelGenruleAttributes{ | 
 | 				Outs: []string{a.Name() + "_proguard.flags"}, | 
 | 				Cmd: bazel.StringAttribute{ | 
 | 					Value: proptools.StringPtr("echo " + handCraftedFlags + "> $(OUTS)"), | 
 | 				}, | 
 | 			}) | 
 | 			appAttrs.Proguard_specs.Add(bazel.MakeLabelAttribute(":" + generatedFlagFileRuleName)) | 
 | 		} | 
 | 	} | 
 |  | 
 | 	commonAttrs, bp2BuildInfo, supported := a.convertLibraryAttrsBp2Build(ctx) | 
 | 	if !supported { | 
 | 		return false, android.CommonAttributes{}, &bazelAndroidAppAttributes{} | 
 | 	} | 
 | 	if hasJavaResources := aapt.ConvertJavaResources(ctx, commonAttrs); hasJavaResources { | 
 | 		return false, android.CommonAttributes{}, &bazelAndroidAppAttributes{} | 
 | 	} | 
 |  | 
 | 	depLabels := bp2BuildInfo.DepLabels | 
 |  | 
 | 	deps := depLabels.Deps | 
 | 	deps.Append(depLabels.StaticDeps) | 
 |  | 
 | 	var jniDeps bazel.LabelListAttribute | 
 | 	archVariantProps := a.GetArchVariantProperties(ctx, &appProperties{}) | 
 | 	for axis, configToProps := range archVariantProps { | 
 | 		for config, _props := range configToProps { | 
 | 			if archProps, ok := _props.(*appProperties); ok { | 
 | 				archJniLibs := android.BazelLabelForModuleDeps( | 
 | 					ctx, | 
 | 					android.LastUniqueStrings(android.CopyOf(archProps.Jni_libs))) | 
 | 				jniDeps.SetSelectValue(axis, config, archJniLibs) | 
 | 			} | 
 | 		} | 
 | 	} | 
 | 	deps.Append(jniDeps) | 
 |  | 
 | 	if !bp2BuildInfo.hasKotlin { | 
 | 		appAttrs.javaCommonAttributes = commonAttrs | 
 | 		appAttrs.bazelAapt = aapt | 
 | 		appAttrs.Deps = deps | 
 | 	} else { | 
 | 		ktName := a.Name() + "_kt" | 
 | 		ctx.CreateBazelTargetModule( | 
 | 			AndroidLibraryBazelTargetModuleProperties(), | 
 | 			android.CommonAttributes{Name: ktName}, | 
 | 			&bazelAndroidLibrary{ | 
 | 				javaLibraryAttributes: &javaLibraryAttributes{ | 
 | 					javaCommonAttributes: commonAttrs, | 
 | 					Deps:                 deps, | 
 | 				}, | 
 | 				bazelAapt: aapt, | 
 | 			}, | 
 | 		) | 
 |  | 
 | 		appAttrs.bazelAapt = &bazelAapt{Manifest: aapt.Manifest} | 
 | 		appAttrs.Deps = bazel.MakeSingleLabelListAttribute(bazel.Label{Label: ":" + ktName}) | 
 | 		appAttrs.javaCommonAttributes = &javaCommonAttributes{ | 
 | 			Sdk_version: commonAttrs.Sdk_version, | 
 | 		} | 
 | 	} | 
 |  | 
 | 	return true, android.CommonAttributes{Name: a.Name(), SkipData: proptools.BoolPtr(true)}, appAttrs | 
 | } | 
 |  | 
 | // ConvertWithBp2build is used to convert android_app to Bazel. | 
 | func (a *AndroidApp) ConvertWithBp2build(ctx android.Bp2buildMutatorContext) { | 
 | 	if ok, commonAttrs, appAttrs := convertWithBp2build(ctx, a); ok { | 
 | 		var props bazel.BazelTargetModuleProperties | 
 | 		if ctx.ModuleName() == "framework-res" { | 
 | 			props = bazel.BazelTargetModuleProperties{ | 
 | 				Rule_class:        "framework_resources", | 
 | 				Bzl_load_location: "//build/bazel/rules/android:framework_resources.bzl", | 
 | 			} | 
 | 		} else { | 
 | 			props = bazel.BazelTargetModuleProperties{ | 
 | 				Rule_class:        "android_binary", | 
 | 				Bzl_load_location: "//build/bazel/rules/android:android_binary.bzl", | 
 | 			} | 
 | 		} | 
 | 		ctx.CreateBazelTargetModule(props, commonAttrs, appAttrs) | 
 | 	} | 
 |  | 
 | } | 
 |  | 
 | // ConvertWithBp2build is used to convert android_test to Bazel. | 
 | func (at *AndroidTest) ConvertWithBp2build(ctx android.Bp2buildMutatorContext) { | 
 | 	if ok, commonAttrs, appAttrs := convertWithBp2build(ctx, &at.AndroidApp); ok { | 
 | 		props := bazel.BazelTargetModuleProperties{ | 
 | 			Rule_class:        "android_test", | 
 | 			Bzl_load_location: "//build/bazel/rules/android:android_test.bzl", | 
 | 		} | 
 |  | 
 | 		ctx.CreateBazelTargetModule(props, commonAttrs, appAttrs) | 
 | 	} | 
 |  | 
 | } | 
 |  | 
 | func (atha *AndroidTestHelperApp) ConvertWithBp2build(ctx android.Bp2buildMutatorContext) { | 
 | 	if ok, commonAttrs, appAttrs := convertWithBp2build(ctx, &atha.AndroidApp); ok { | 
 | 		// an android_test_helper_app is an android_binary with testonly = True | 
 | 		commonAttrs.Testonly = proptools.BoolPtr(true) | 
 |  | 
 | 		// android_test_helper_app sets default values differently to android_app, | 
 | 		// https://cs.android.com/android/platform/superproject/main/+/main:build/soong/java/app.go;l=1273-1279;drc=e12c083198403ec694af6c625aed11327eb2bf7f | 
 | 		// | 
 | 		// installable: true (settable prop) | 
 | 		// use_embedded_native_libs: true (settable prop) | 
 | 		// lint.test: true (settable prop) | 
 | 		// AlwaysPackageNativeLibs: true (blueprint mutated prop) | 
 | 		// dexpreopt isTest: true (not prop) | 
 |  | 
 | 		props := bazel.BazelTargetModuleProperties{ | 
 | 			Rule_class:        "android_binary", | 
 | 			Bzl_load_location: "//build/bazel/rules/android:android_binary.bzl", | 
 | 		} | 
 |  | 
 | 		ctx.CreateBazelTargetModule(props, commonAttrs, appAttrs) | 
 | 	} | 
 | } |