This CL adds java_version and sdk_version support to bp2build converters for - java library - java binary - android library - android binary - android library import Although java import doesn't support java_version and sdk_version, the neverlink java_library wrapper around a java_import must specify a sdk_version when targetting a device. "none" is used by convention. Change-Id: I22a69dea2e351858368df69ed6a703b568d613ea Bug: 215230098 Test: Presubmits
		
			
				
	
	
		
			1135 lines
		
	
	
		
			39 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			1135 lines
		
	
	
		
			39 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2018 Google Inc. All rights reserved.
 | |
| //
 | |
| // Licensed under the Apache License, Version 2.0 (the "License");
 | |
| // you may not use this file except in compliance with the License.
 | |
| // You may obtain a copy of the License at
 | |
| //
 | |
| //     http://www.apache.org/licenses/LICENSE-2.0
 | |
| //
 | |
| // Unless required by applicable law or agreed to in writing, software
 | |
| // distributed under the License is distributed on an "AS IS" BASIS,
 | |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
| // See the License for the specific language governing permissions and
 | |
| // limitations under the License.
 | |
| 
 | |
| package java
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"path/filepath"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 
 | |
| 	"android/soong/android"
 | |
| 	"android/soong/bazel"
 | |
| 	"android/soong/dexpreopt"
 | |
| 
 | |
| 	"github.com/google/blueprint"
 | |
| 	"github.com/google/blueprint/proptools"
 | |
| )
 | |
| 
 | |
| type AndroidLibraryDependency interface {
 | |
| 	LibraryDependency
 | |
| 	ExportPackage() android.Path
 | |
| 	ExportedRRODirs() []rroDir
 | |
| 	ExportedStaticPackages() android.Paths
 | |
| 	ExportedManifests() android.Paths
 | |
| 	ExportedAssets() android.OptionalPath
 | |
| 	SetRROEnforcedForDependent(enforce bool)
 | |
| 	IsRROEnforced(ctx android.BaseModuleContext) bool
 | |
| }
 | |
| 
 | |
| func init() {
 | |
| 	RegisterAARBuildComponents(android.InitRegistrationContext)
 | |
| }
 | |
| 
 | |
| func RegisterAARBuildComponents(ctx android.RegistrationContext) {
 | |
| 	ctx.RegisterModuleType("android_library_import", AARImportFactory)
 | |
| 	ctx.RegisterModuleType("android_library", AndroidLibraryFactory)
 | |
| 	ctx.PostDepsMutators(func(ctx android.RegisterMutatorsContext) {
 | |
| 		ctx.TopDown("propagate_rro_enforcement", propagateRROEnforcementMutator).Parallel()
 | |
| 	})
 | |
| }
 | |
| 
 | |
| //
 | |
| // AAR (android library)
 | |
| //
 | |
| 
 | |
| type androidLibraryProperties struct {
 | |
| 	BuildAAR bool `blueprint:"mutated"`
 | |
| }
 | |
| 
 | |
| type aaptProperties struct {
 | |
| 	// flags passed to aapt when creating the apk
 | |
| 	Aaptflags []string
 | |
| 
 | |
| 	// include all resource configurations, not just the product-configured
 | |
| 	// ones.
 | |
| 	Aapt_include_all_resources *bool
 | |
| 
 | |
| 	// list of directories relative to the Blueprints file containing assets.
 | |
| 	// Defaults to ["assets"] if a directory called assets exists.  Set to []
 | |
| 	// to disable the default.
 | |
| 	Asset_dirs []string
 | |
| 
 | |
| 	// list of directories relative to the Blueprints file containing
 | |
| 	// Android resources.  Defaults to ["res"] if a directory called res exists.
 | |
| 	// Set to [] to disable the default.
 | |
| 	Resource_dirs []string
 | |
| 
 | |
| 	// list of zip files containing Android resources.
 | |
| 	Resource_zips []string `android:"path"`
 | |
| 
 | |
| 	// path to AndroidManifest.xml.  If unset, defaults to "AndroidManifest.xml".
 | |
| 	Manifest *string `android:"path"`
 | |
| 
 | |
| 	// paths to additional manifest files to merge with main manifest.
 | |
| 	Additional_manifests []string `android:"path"`
 | |
| 
 | |
| 	// do not include AndroidManifest from dependent libraries
 | |
| 	Dont_merge_manifests *bool
 | |
| 
 | |
| 	// true if RRO is enforced for any of the dependent modules
 | |
| 	RROEnforcedForDependent bool `blueprint:"mutated"`
 | |
| }
 | |
| 
 | |
| type aapt struct {
 | |
| 	aaptSrcJar              android.Path
 | |
| 	exportPackage           android.Path
 | |
| 	manifestPath            android.Path
 | |
| 	transitiveManifestPaths android.Paths
 | |
| 	proguardOptionsFile     android.Path
 | |
| 	rroDirs                 []rroDir
 | |
| 	rTxt                    android.Path
 | |
| 	extraAaptPackagesFile   android.Path
 | |
| 	mergedManifestFile      android.Path
 | |
| 	noticeFile              android.OptionalPath
 | |
| 	assetPackage            android.OptionalPath
 | |
| 	isLibrary               bool
 | |
| 	defaultManifestVersion  string
 | |
| 	useEmbeddedNativeLibs   bool
 | |
| 	useEmbeddedDex          bool
 | |
| 	usesNonSdkApis          bool
 | |
| 	hasNoCode               bool
 | |
| 	LoggingParent           string
 | |
| 	resourceFiles           android.Paths
 | |
| 
 | |
| 	splitNames []string
 | |
| 	splits     []split
 | |
| 
 | |
| 	aaptProperties aaptProperties
 | |
| }
 | |
| 
 | |
| type split struct {
 | |
| 	name   string
 | |
| 	suffix string
 | |
| 	path   android.Path
 | |
| }
 | |
| 
 | |
| // Propagate RRO enforcement flag to static lib dependencies transitively.
 | |
| func propagateRROEnforcementMutator(ctx android.TopDownMutatorContext) {
 | |
| 	m := ctx.Module()
 | |
| 	if d, ok := m.(AndroidLibraryDependency); ok && d.IsRROEnforced(ctx) {
 | |
| 		ctx.VisitDirectDepsWithTag(staticLibTag, func(d android.Module) {
 | |
| 			if a, ok := d.(AndroidLibraryDependency); ok {
 | |
| 				a.SetRROEnforcedForDependent(true)
 | |
| 			}
 | |
| 		})
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (a *aapt) ExportPackage() android.Path {
 | |
| 	return a.exportPackage
 | |
| }
 | |
| 
 | |
| func (a *aapt) ExportedRRODirs() []rroDir {
 | |
| 	return a.rroDirs
 | |
| }
 | |
| 
 | |
| func (a *aapt) ExportedManifests() android.Paths {
 | |
| 	return a.transitiveManifestPaths
 | |
| }
 | |
| 
 | |
| func (a *aapt) ExportedAssets() android.OptionalPath {
 | |
| 	return a.assetPackage
 | |
| }
 | |
| 
 | |
| func (a *aapt) SetRROEnforcedForDependent(enforce bool) {
 | |
| 	a.aaptProperties.RROEnforcedForDependent = enforce
 | |
| }
 | |
| 
 | |
| func (a *aapt) IsRROEnforced(ctx android.BaseModuleContext) bool {
 | |
| 	// True if RRO is enforced for this module or...
 | |
| 	return ctx.Config().EnforceRROForModule(ctx.ModuleName()) ||
 | |
| 		// if RRO is enforced for any of its dependents.
 | |
| 		a.aaptProperties.RROEnforcedForDependent
 | |
| }
 | |
| 
 | |
| func (a *aapt) aapt2Flags(ctx android.ModuleContext, sdkContext android.SdkContext,
 | |
| 	manifestPath android.Path) (compileFlags, linkFlags []string, linkDeps android.Paths,
 | |
| 	resDirs, overlayDirs []globbedResourceDir, rroDirs []rroDir, resZips android.Paths) {
 | |
| 
 | |
| 	hasVersionCode := android.PrefixInList(a.aaptProperties.Aaptflags, "--version-code")
 | |
| 	hasVersionName := android.PrefixInList(a.aaptProperties.Aaptflags, "--version-name")
 | |
| 
 | |
| 	// Flags specified in Android.bp
 | |
| 	linkFlags = append(linkFlags, a.aaptProperties.Aaptflags...)
 | |
| 
 | |
| 	linkFlags = append(linkFlags, "--no-static-lib-packages")
 | |
| 
 | |
| 	// Find implicit or explicit asset and resource dirs
 | |
| 	assetDirs := android.PathsWithOptionalDefaultForModuleSrc(ctx, a.aaptProperties.Asset_dirs, "assets")
 | |
| 	resourceDirs := android.PathsWithOptionalDefaultForModuleSrc(ctx, a.aaptProperties.Resource_dirs, "res")
 | |
| 	resourceZips := android.PathsForModuleSrc(ctx, a.aaptProperties.Resource_zips)
 | |
| 
 | |
| 	// Glob directories into lists of paths
 | |
| 	for _, dir := range resourceDirs {
 | |
| 		resDirs = append(resDirs, globbedResourceDir{
 | |
| 			dir:   dir,
 | |
| 			files: androidResourceGlob(ctx, dir),
 | |
| 		})
 | |
| 		resOverlayDirs, resRRODirs := overlayResourceGlob(ctx, a, dir)
 | |
| 		overlayDirs = append(overlayDirs, resOverlayDirs...)
 | |
| 		rroDirs = append(rroDirs, resRRODirs...)
 | |
| 	}
 | |
| 
 | |
| 	var assetDeps android.Paths
 | |
| 	for i, dir := range assetDirs {
 | |
| 		// Add a dependency on every file in the asset directory.  This ensures the aapt2
 | |
| 		// rule will be rerun if one of the files in the asset directory is modified.
 | |
| 		assetDeps = append(assetDeps, androidResourceGlob(ctx, dir)...)
 | |
| 
 | |
| 		// Add a dependency on a file that contains a list of all the files in the asset directory.
 | |
| 		// This ensures the aapt2 rule will be run if a file is removed from the asset directory,
 | |
| 		// or a file is added whose timestamp is older than the output of aapt2.
 | |
| 		assetFileListFile := android.PathForModuleOut(ctx, "asset_dir_globs", strconv.Itoa(i)+".glob")
 | |
| 		androidResourceGlobList(ctx, dir, assetFileListFile)
 | |
| 		assetDeps = append(assetDeps, assetFileListFile)
 | |
| 	}
 | |
| 
 | |
| 	assetDirStrings := assetDirs.Strings()
 | |
| 	if a.noticeFile.Valid() {
 | |
| 		assetDirStrings = append(assetDirStrings, filepath.Dir(a.noticeFile.Path().String()))
 | |
| 		assetDeps = append(assetDeps, a.noticeFile.Path())
 | |
| 	}
 | |
| 
 | |
| 	linkFlags = append(linkFlags, "--manifest "+manifestPath.String())
 | |
| 	linkDeps = append(linkDeps, manifestPath)
 | |
| 
 | |
| 	linkFlags = append(linkFlags, android.JoinWithPrefix(assetDirStrings, "-A "))
 | |
| 	linkDeps = append(linkDeps, assetDeps...)
 | |
| 
 | |
| 	// Returns the effective version for {min|target}_sdk_version
 | |
| 	effectiveVersionString := func(sdkVersion android.SdkSpec, minSdkVersion android.ApiLevel) string {
 | |
| 		// If {min|target}_sdk_version is current, use sdk_version to determine the effective level
 | |
| 		// This is necessary for vendor modules.
 | |
| 		// The effective version does not _only_ depend on {min|target}_sdk_version(level),
 | |
| 		// but also on the sdk_version (kind+level)
 | |
| 		if minSdkVersion.IsCurrent() {
 | |
| 			ret, err := sdkVersion.EffectiveVersionString(ctx)
 | |
| 			if err != nil {
 | |
| 				ctx.ModuleErrorf("invalid sdk_version: %s", err)
 | |
| 			}
 | |
| 			return ret
 | |
| 		}
 | |
| 		ret, err := minSdkVersion.EffectiveVersionString(ctx)
 | |
| 		if err != nil {
 | |
| 			ctx.ModuleErrorf("invalid min_sdk_version: %s", err)
 | |
| 		}
 | |
| 		return ret
 | |
| 	}
 | |
| 	// SDK version flags
 | |
| 	sdkVersion := sdkContext.SdkVersion(ctx)
 | |
| 	minSdkVersion := effectiveVersionString(sdkVersion, sdkContext.MinSdkVersion(ctx))
 | |
| 
 | |
| 	linkFlags = append(linkFlags, "--min-sdk-version "+minSdkVersion)
 | |
| 	// Use minSdkVersion for target-sdk-version, even if `target_sdk_version` is set
 | |
| 	// This behavior has been copied from Make.
 | |
| 	linkFlags = append(linkFlags, "--target-sdk-version "+minSdkVersion)
 | |
| 
 | |
| 	// Version code
 | |
| 	if !hasVersionCode {
 | |
| 		linkFlags = append(linkFlags, "--version-code", ctx.Config().PlatformSdkVersion().String())
 | |
| 	}
 | |
| 
 | |
| 	if !hasVersionName {
 | |
| 		var versionName string
 | |
| 		if ctx.ModuleName() == "framework-res" {
 | |
| 			// Some builds set AppsDefaultVersionName() to include the build number ("O-123456").  aapt2 copies the
 | |
| 			// version name of framework-res into app manifests as compileSdkVersionCodename, which confuses things
 | |
| 			// if it contains the build number.  Use the PlatformVersionName instead.
 | |
| 			versionName = ctx.Config().PlatformVersionName()
 | |
| 		} else {
 | |
| 			versionName = ctx.Config().AppsDefaultVersionName()
 | |
| 		}
 | |
| 		versionName = proptools.NinjaEscape(versionName)
 | |
| 		linkFlags = append(linkFlags, "--version-name ", versionName)
 | |
| 	}
 | |
| 
 | |
| 	linkFlags, compileFlags = android.FilterList(linkFlags, []string{"--legacy"})
 | |
| 
 | |
| 	// Always set --pseudo-localize, it will be stripped out later for release
 | |
| 	// builds that don't want it.
 | |
| 	compileFlags = append(compileFlags, "--pseudo-localize")
 | |
| 
 | |
| 	return compileFlags, linkFlags, linkDeps, resDirs, overlayDirs, rroDirs, resourceZips
 | |
| }
 | |
| 
 | |
| func (a *aapt) deps(ctx android.BottomUpMutatorContext, sdkDep sdkDep) {
 | |
| 	if sdkDep.frameworkResModule != "" {
 | |
| 		ctx.AddVariationDependencies(nil, frameworkResTag, sdkDep.frameworkResModule)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| var extractAssetsRule = pctx.AndroidStaticRule("extractAssets",
 | |
| 	blueprint.RuleParams{
 | |
| 		Command:     `${config.Zip2ZipCmd} -i ${in} -o ${out} "assets/**/*"`,
 | |
| 		CommandDeps: []string{"${config.Zip2ZipCmd}"},
 | |
| 	})
 | |
| 
 | |
| func (a *aapt) buildActions(ctx android.ModuleContext, sdkContext android.SdkContext,
 | |
| 	classLoaderContexts dexpreopt.ClassLoaderContextMap, excludedLibs []string,
 | |
| 	enforceDefaultTargetSdkVersion bool, extraLinkFlags ...string) {
 | |
| 
 | |
| 	transitiveStaticLibs, transitiveStaticLibManifests, staticRRODirs, assetPackages, libDeps, libFlags :=
 | |
| 		aaptLibs(ctx, sdkContext, classLoaderContexts)
 | |
| 
 | |
| 	// Exclude any libraries from the supplied list.
 | |
| 	classLoaderContexts = classLoaderContexts.ExcludeLibs(excludedLibs)
 | |
| 
 | |
| 	// App manifest file
 | |
| 	manifestFile := proptools.StringDefault(a.aaptProperties.Manifest, "AndroidManifest.xml")
 | |
| 	manifestSrcPath := android.PathForModuleSrc(ctx, manifestFile)
 | |
| 
 | |
| 	manifestPath := ManifestFixer(ctx, manifestSrcPath, ManifestFixerParams{
 | |
| 		SdkContext:                     sdkContext,
 | |
| 		ClassLoaderContexts:            classLoaderContexts,
 | |
| 		IsLibrary:                      a.isLibrary,
 | |
| 		DefaultManifestVersion:         a.defaultManifestVersion,
 | |
| 		UseEmbeddedNativeLibs:          a.useEmbeddedNativeLibs,
 | |
| 		UsesNonSdkApis:                 a.usesNonSdkApis,
 | |
| 		UseEmbeddedDex:                 a.useEmbeddedDex,
 | |
| 		HasNoCode:                      a.hasNoCode,
 | |
| 		LoggingParent:                  a.LoggingParent,
 | |
| 		EnforceDefaultTargetSdkVersion: enforceDefaultTargetSdkVersion,
 | |
| 	})
 | |
| 
 | |
| 	// Add additional manifest files to transitive manifests.
 | |
| 	additionalManifests := android.PathsForModuleSrc(ctx, a.aaptProperties.Additional_manifests)
 | |
| 	a.transitiveManifestPaths = append(android.Paths{manifestPath}, additionalManifests...)
 | |
| 	a.transitiveManifestPaths = append(a.transitiveManifestPaths, transitiveStaticLibManifests...)
 | |
| 
 | |
| 	if len(a.transitiveManifestPaths) > 1 && !Bool(a.aaptProperties.Dont_merge_manifests) {
 | |
| 		a.mergedManifestFile = manifestMerger(ctx, a.transitiveManifestPaths[0], a.transitiveManifestPaths[1:], a.isLibrary)
 | |
| 		if !a.isLibrary {
 | |
| 			// Only use the merged manifest for applications.  For libraries, the transitive closure of manifests
 | |
| 			// will be propagated to the final application and merged there.  The merged manifest for libraries is
 | |
| 			// only passed to Make, which can't handle transitive dependencies.
 | |
| 			manifestPath = a.mergedManifestFile
 | |
| 		}
 | |
| 	} else {
 | |
| 		a.mergedManifestFile = manifestPath
 | |
| 	}
 | |
| 
 | |
| 	compileFlags, linkFlags, linkDeps, resDirs, overlayDirs, rroDirs, resZips := a.aapt2Flags(ctx, sdkContext, manifestPath)
 | |
| 
 | |
| 	rroDirs = append(rroDirs, staticRRODirs...)
 | |
| 	linkFlags = append(linkFlags, libFlags...)
 | |
| 	linkDeps = append(linkDeps, libDeps...)
 | |
| 	linkFlags = append(linkFlags, extraLinkFlags...)
 | |
| 	if a.isLibrary {
 | |
| 		linkFlags = append(linkFlags, "--static-lib")
 | |
| 	}
 | |
| 
 | |
| 	packageRes := android.PathForModuleOut(ctx, "package-res.apk")
 | |
| 	// the subdir "android" is required to be filtered by package names
 | |
| 	srcJar := android.PathForModuleGen(ctx, "android", "R.srcjar")
 | |
| 	proguardOptionsFile := android.PathForModuleGen(ctx, "proguard.options")
 | |
| 	rTxt := android.PathForModuleOut(ctx, "R.txt")
 | |
| 	// This file isn't used by Soong, but is generated for exporting
 | |
| 	extraPackages := android.PathForModuleOut(ctx, "extra_packages")
 | |
| 
 | |
| 	var compiledResDirs []android.Paths
 | |
| 	for _, dir := range resDirs {
 | |
| 		a.resourceFiles = append(a.resourceFiles, dir.files...)
 | |
| 		compiledResDirs = append(compiledResDirs, aapt2Compile(ctx, dir.dir, dir.files, compileFlags).Paths())
 | |
| 	}
 | |
| 
 | |
| 	for i, zip := range resZips {
 | |
| 		flata := android.PathForModuleOut(ctx, fmt.Sprintf("reszip.%d.flata", i))
 | |
| 		aapt2CompileZip(ctx, flata, zip, "", compileFlags)
 | |
| 		compiledResDirs = append(compiledResDirs, android.Paths{flata})
 | |
| 	}
 | |
| 
 | |
| 	var compiledRes, compiledOverlay android.Paths
 | |
| 
 | |
| 	compiledOverlay = append(compiledOverlay, transitiveStaticLibs...)
 | |
| 
 | |
| 	if len(transitiveStaticLibs) > 0 {
 | |
| 		// If we are using static android libraries, every source file becomes an overlay.
 | |
| 		// This is to emulate old AAPT behavior which simulated library support.
 | |
| 		for _, compiledResDir := range compiledResDirs {
 | |
| 			compiledOverlay = append(compiledOverlay, compiledResDir...)
 | |
| 		}
 | |
| 	} else if a.isLibrary {
 | |
| 		// Otherwise, for a static library we treat all the resources equally with no overlay.
 | |
| 		for _, compiledResDir := range compiledResDirs {
 | |
| 			compiledRes = append(compiledRes, compiledResDir...)
 | |
| 		}
 | |
| 	} else if len(compiledResDirs) > 0 {
 | |
| 		// Without static libraries, the first directory is our directory, which can then be
 | |
| 		// overlaid by the rest.
 | |
| 		compiledRes = append(compiledRes, compiledResDirs[0]...)
 | |
| 		for _, compiledResDir := range compiledResDirs[1:] {
 | |
| 			compiledOverlay = append(compiledOverlay, compiledResDir...)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	for _, dir := range overlayDirs {
 | |
| 		compiledOverlay = append(compiledOverlay, aapt2Compile(ctx, dir.dir, dir.files, compileFlags).Paths()...)
 | |
| 	}
 | |
| 
 | |
| 	var splitPackages android.WritablePaths
 | |
| 	var splits []split
 | |
| 
 | |
| 	for _, s := range a.splitNames {
 | |
| 		suffix := strings.Replace(s, ",", "_", -1)
 | |
| 		path := android.PathForModuleOut(ctx, "package_"+suffix+".apk")
 | |
| 		linkFlags = append(linkFlags, "--split", path.String()+":"+s)
 | |
| 		splitPackages = append(splitPackages, path)
 | |
| 		splits = append(splits, split{
 | |
| 			name:   s,
 | |
| 			suffix: suffix,
 | |
| 			path:   path,
 | |
| 		})
 | |
| 	}
 | |
| 
 | |
| 	aapt2Link(ctx, packageRes, srcJar, proguardOptionsFile, rTxt, extraPackages,
 | |
| 		linkFlags, linkDeps, compiledRes, compiledOverlay, assetPackages, splitPackages)
 | |
| 
 | |
| 	// Extract assets from the resource package output so that they can be used later in aapt2link
 | |
| 	// for modules that depend on this one.
 | |
| 	if android.PrefixInList(linkFlags, "-A ") || len(assetPackages) > 0 {
 | |
| 		assets := android.PathForModuleOut(ctx, "assets.zip")
 | |
| 		ctx.Build(pctx, android.BuildParams{
 | |
| 			Rule:        extractAssetsRule,
 | |
| 			Input:       packageRes,
 | |
| 			Output:      assets,
 | |
| 			Description: "extract assets from built resource file",
 | |
| 		})
 | |
| 		a.assetPackage = android.OptionalPathForPath(assets)
 | |
| 	}
 | |
| 
 | |
| 	a.aaptSrcJar = srcJar
 | |
| 	a.exportPackage = packageRes
 | |
| 	a.manifestPath = manifestPath
 | |
| 	a.proguardOptionsFile = proguardOptionsFile
 | |
| 	a.rroDirs = rroDirs
 | |
| 	a.extraAaptPackagesFile = extraPackages
 | |
| 	a.rTxt = rTxt
 | |
| 	a.splits = splits
 | |
| }
 | |
| 
 | |
| // aaptLibs collects libraries from dependencies and sdk_version and converts them into paths
 | |
| func aaptLibs(ctx android.ModuleContext, sdkContext android.SdkContext, classLoaderContexts dexpreopt.ClassLoaderContextMap) (
 | |
| 	transitiveStaticLibs, transitiveStaticLibManifests android.Paths, staticRRODirs []rroDir, assets, deps android.Paths, flags []string) {
 | |
| 
 | |
| 	var sharedLibs android.Paths
 | |
| 
 | |
| 	if classLoaderContexts == nil {
 | |
| 		// Not all callers need to compute class loader context, those who don't just pass nil.
 | |
| 		// Create a temporary class loader context here (it will be computed, but not used).
 | |
| 		classLoaderContexts = make(dexpreopt.ClassLoaderContextMap)
 | |
| 	}
 | |
| 
 | |
| 	sdkDep := decodeSdkDep(ctx, sdkContext)
 | |
| 	if sdkDep.useFiles {
 | |
| 		sharedLibs = append(sharedLibs, sdkDep.jars...)
 | |
| 	}
 | |
| 
 | |
| 	ctx.VisitDirectDeps(func(module android.Module) {
 | |
| 		depTag := ctx.OtherModuleDependencyTag(module)
 | |
| 
 | |
| 		var exportPackage android.Path
 | |
| 		aarDep, _ := module.(AndroidLibraryDependency)
 | |
| 		if aarDep != nil {
 | |
| 			exportPackage = aarDep.ExportPackage()
 | |
| 		}
 | |
| 
 | |
| 		switch depTag {
 | |
| 		case instrumentationForTag:
 | |
| 			// Nothing, instrumentationForTag is treated as libTag for javac but not for aapt2.
 | |
| 		case sdkLibTag, libTag:
 | |
| 			if exportPackage != nil {
 | |
| 				sharedLibs = append(sharedLibs, exportPackage)
 | |
| 			}
 | |
| 		case frameworkResTag:
 | |
| 			if exportPackage != nil {
 | |
| 				sharedLibs = append(sharedLibs, exportPackage)
 | |
| 			}
 | |
| 		case staticLibTag:
 | |
| 			if exportPackage != nil {
 | |
| 				transitiveStaticLibs = append(transitiveStaticLibs, aarDep.ExportedStaticPackages()...)
 | |
| 				transitiveStaticLibs = append(transitiveStaticLibs, exportPackage)
 | |
| 				transitiveStaticLibManifests = append(transitiveStaticLibManifests, aarDep.ExportedManifests()...)
 | |
| 				if aarDep.ExportedAssets().Valid() {
 | |
| 					assets = append(assets, aarDep.ExportedAssets().Path())
 | |
| 				}
 | |
| 
 | |
| 			outer:
 | |
| 				for _, d := range aarDep.ExportedRRODirs() {
 | |
| 					for _, e := range staticRRODirs {
 | |
| 						if d.path == e.path {
 | |
| 							continue outer
 | |
| 						}
 | |
| 					}
 | |
| 					staticRRODirs = append(staticRRODirs, d)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		addCLCFromDep(ctx, module, classLoaderContexts)
 | |
| 	})
 | |
| 
 | |
| 	deps = append(deps, sharedLibs...)
 | |
| 	deps = append(deps, transitiveStaticLibs...)
 | |
| 
 | |
| 	if len(transitiveStaticLibs) > 0 {
 | |
| 		flags = append(flags, "--auto-add-overlay")
 | |
| 	}
 | |
| 
 | |
| 	for _, sharedLib := range sharedLibs {
 | |
| 		flags = append(flags, "-I "+sharedLib.String())
 | |
| 	}
 | |
| 
 | |
| 	transitiveStaticLibs = android.FirstUniquePaths(transitiveStaticLibs)
 | |
| 	transitiveStaticLibManifests = android.FirstUniquePaths(transitiveStaticLibManifests)
 | |
| 
 | |
| 	return transitiveStaticLibs, transitiveStaticLibManifests, staticRRODirs, assets, deps, flags
 | |
| }
 | |
| 
 | |
| type AndroidLibrary struct {
 | |
| 	Library
 | |
| 	aapt
 | |
| 	android.BazelModuleBase
 | |
| 
 | |
| 	androidLibraryProperties androidLibraryProperties
 | |
| 
 | |
| 	aarFile android.WritablePath
 | |
| 
 | |
| 	exportedStaticPackages android.Paths
 | |
| }
 | |
| 
 | |
| var _ android.OutputFileProducer = (*AndroidLibrary)(nil)
 | |
| 
 | |
| // For OutputFileProducer interface
 | |
| func (a *AndroidLibrary) OutputFiles(tag string) (android.Paths, error) {
 | |
| 	switch tag {
 | |
| 	case ".aar":
 | |
| 		return []android.Path{a.aarFile}, nil
 | |
| 	default:
 | |
| 		return a.Library.OutputFiles(tag)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (a *AndroidLibrary) ExportedStaticPackages() android.Paths {
 | |
| 	return a.exportedStaticPackages
 | |
| }
 | |
| 
 | |
| var _ AndroidLibraryDependency = (*AndroidLibrary)(nil)
 | |
| 
 | |
| func (a *AndroidLibrary) DepsMutator(ctx android.BottomUpMutatorContext) {
 | |
| 	a.Module.deps(ctx)
 | |
| 	sdkDep := decodeSdkDep(ctx, android.SdkContext(a))
 | |
| 	if sdkDep.hasFrameworkLibs() {
 | |
| 		a.aapt.deps(ctx, sdkDep)
 | |
| 	}
 | |
| 	a.usesLibrary.deps(ctx, false)
 | |
| }
 | |
| 
 | |
| func (a *AndroidLibrary) GenerateAndroidBuildActions(ctx android.ModuleContext) {
 | |
| 	a.aapt.isLibrary = true
 | |
| 	a.classLoaderContexts = a.usesLibrary.classLoaderContextForUsesLibDeps(ctx)
 | |
| 	a.aapt.buildActions(ctx, android.SdkContext(a), a.classLoaderContexts, nil, false)
 | |
| 
 | |
| 	a.hideApexVariantFromMake = !ctx.Provider(android.ApexInfoProvider).(android.ApexInfo).IsForPlatform()
 | |
| 
 | |
| 	ctx.CheckbuildFile(a.proguardOptionsFile)
 | |
| 	ctx.CheckbuildFile(a.exportPackage)
 | |
| 	ctx.CheckbuildFile(a.aaptSrcJar)
 | |
| 
 | |
| 	// apps manifests are handled by aapt, don't let Module see them
 | |
| 	a.properties.Manifest = nil
 | |
| 
 | |
| 	a.linter.mergedManifest = a.aapt.mergedManifestFile
 | |
| 	a.linter.manifest = a.aapt.manifestPath
 | |
| 	a.linter.resources = a.aapt.resourceFiles
 | |
| 
 | |
| 	a.Module.extraProguardFlagFiles = append(a.Module.extraProguardFlagFiles,
 | |
| 		a.proguardOptionsFile)
 | |
| 
 | |
| 	a.Module.compile(ctx, a.aaptSrcJar)
 | |
| 
 | |
| 	a.aarFile = android.PathForModuleOut(ctx, ctx.ModuleName()+".aar")
 | |
| 	var res android.Paths
 | |
| 	if a.androidLibraryProperties.BuildAAR {
 | |
| 		BuildAAR(ctx, a.aarFile, a.outputFile, a.manifestPath, a.rTxt, res)
 | |
| 		ctx.CheckbuildFile(a.aarFile)
 | |
| 	}
 | |
| 
 | |
| 	a.exportedProguardFlagFiles = append(a.exportedProguardFlagFiles,
 | |
| 		android.PathsForModuleSrc(ctx, a.dexProperties.Optimize.Proguard_flags_files)...)
 | |
| 	ctx.VisitDirectDeps(func(m android.Module) {
 | |
| 		if ctx.OtherModuleDependencyTag(m) == staticLibTag {
 | |
| 			if lib, ok := m.(LibraryDependency); ok {
 | |
| 				a.exportedProguardFlagFiles = append(a.exportedProguardFlagFiles, lib.ExportedProguardFlagFiles()...)
 | |
| 			}
 | |
| 			if alib, ok := m.(AndroidLibraryDependency); ok {
 | |
| 				a.exportedStaticPackages = append(a.exportedStaticPackages, alib.ExportPackage())
 | |
| 				a.exportedStaticPackages = append(a.exportedStaticPackages, alib.ExportedStaticPackages()...)
 | |
| 			}
 | |
| 		}
 | |
| 	})
 | |
| 	a.exportedProguardFlagFiles = android.FirstUniquePaths(a.exportedProguardFlagFiles)
 | |
| 	a.exportedStaticPackages = android.FirstUniquePaths(a.exportedStaticPackages)
 | |
| 
 | |
| 	prebuiltJniPackages := android.Paths{}
 | |
| 	ctx.VisitDirectDeps(func(module android.Module) {
 | |
| 		if info, ok := ctx.OtherModuleProvider(module, JniPackageProvider).(JniPackageInfo); ok {
 | |
| 			prebuiltJniPackages = append(prebuiltJniPackages, info.JniPackages...)
 | |
| 		}
 | |
| 	})
 | |
| 	if len(prebuiltJniPackages) > 0 {
 | |
| 		ctx.SetProvider(JniPackageProvider, JniPackageInfo{
 | |
| 			JniPackages: prebuiltJniPackages,
 | |
| 		})
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // android_library builds and links sources into a `.jar` file for the device along with Android resources.
 | |
| //
 | |
| // An android_library has a single variant that produces a `.jar` file containing `.class` files that were
 | |
| // compiled against the device bootclasspath, along with a `package-res.apk` file containing Android resources compiled
 | |
| // with aapt2.  This module is not suitable for installing on a device, but can be used as a `static_libs` dependency of
 | |
| // an android_app module.
 | |
| func AndroidLibraryFactory() android.Module {
 | |
| 	module := &AndroidLibrary{}
 | |
| 
 | |
| 	module.Module.addHostAndDeviceProperties()
 | |
| 	module.AddProperties(
 | |
| 		&module.aaptProperties,
 | |
| 		&module.androidLibraryProperties)
 | |
| 
 | |
| 	module.androidLibraryProperties.BuildAAR = true
 | |
| 	module.Module.linter.library = true
 | |
| 
 | |
| 	android.InitApexModule(module)
 | |
| 	InitJavaModule(module, android.DeviceSupported)
 | |
| 	android.InitBazelModule(module)
 | |
| 	return module
 | |
| }
 | |
| 
 | |
| //
 | |
| // AAR (android library) prebuilts
 | |
| //
 | |
| 
 | |
| // Properties for android_library_import
 | |
| type AARImportProperties struct {
 | |
| 	// ARR (android library prebuilt) filepath. Exactly one ARR is required.
 | |
| 	Aars []string `android:"path"`
 | |
| 	// If not blank, set to the version of the sdk to compile against.
 | |
| 	// Defaults to private.
 | |
| 	// Values are of one of the following forms:
 | |
| 	// 1) numerical API level, "current", "none", or "core_platform"
 | |
| 	// 2) An SDK kind with an API level: "<sdk kind>_<API level>"
 | |
| 	// See build/soong/android/sdk_version.go for the complete and up to date list of SDK kinds.
 | |
| 	// If the SDK kind is empty, it will be set to public
 | |
| 	Sdk_version *string
 | |
| 	// If not blank, set the minimum version of the sdk that the compiled artifacts will run against.
 | |
| 	// Defaults to sdk_version if not set. See sdk_version for possible values.
 | |
| 	Min_sdk_version *string
 | |
| 	// List of java static libraries that the included ARR (android library prebuilts) has dependencies to.
 | |
| 	Static_libs []string
 | |
| 	// List of java libraries that the included ARR (android library prebuilts) has dependencies to.
 | |
| 	Libs []string
 | |
| 	// If set to true, run Jetifier against .aar file. Defaults to false.
 | |
| 	Jetifier *bool
 | |
| 	// If true, extract JNI libs from AAR archive. These libs will be accessible to android_app modules and
 | |
| 	// will be passed transitively through android_libraries to an android_app.
 | |
| 	//TODO(b/241138093) evaluate whether we can have this flag default to true for Bazel conversion
 | |
| 	Extract_jni *bool
 | |
| }
 | |
| 
 | |
| type AARImport struct {
 | |
| 	android.ModuleBase
 | |
| 	android.DefaultableModuleBase
 | |
| 	android.ApexModuleBase
 | |
| 	android.BazelModuleBase
 | |
| 	prebuilt android.Prebuilt
 | |
| 
 | |
| 	// Functionality common to Module and Import.
 | |
| 	embeddableInModuleAndImport
 | |
| 
 | |
| 	providesTransitiveHeaderJars
 | |
| 
 | |
| 	properties AARImportProperties
 | |
| 
 | |
| 	classpathFile         android.WritablePath
 | |
| 	proguardFlags         android.WritablePath
 | |
| 	exportPackage         android.WritablePath
 | |
| 	extraAaptPackagesFile android.WritablePath
 | |
| 	manifest              android.WritablePath
 | |
| 	assetsPackage         android.WritablePath
 | |
| 
 | |
| 	exportedStaticPackages android.Paths
 | |
| 
 | |
| 	hideApexVariantFromMake bool
 | |
| 
 | |
| 	aarPath     android.Path
 | |
| 	jniPackages android.Paths
 | |
| 
 | |
| 	sdkVersion    android.SdkSpec
 | |
| 	minSdkVersion android.ApiLevel
 | |
| }
 | |
| 
 | |
| var _ android.OutputFileProducer = (*AARImport)(nil)
 | |
| 
 | |
| // For OutputFileProducer interface
 | |
| func (a *AARImport) OutputFiles(tag string) (android.Paths, error) {
 | |
| 	switch tag {
 | |
| 	case ".aar":
 | |
| 		return []android.Path{a.aarPath}, nil
 | |
| 	case "":
 | |
| 		return []android.Path{a.classpathFile}, nil
 | |
| 	default:
 | |
| 		return nil, fmt.Errorf("unsupported module reference tag %q", tag)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (a *AARImport) SdkVersion(ctx android.EarlyModuleContext) android.SdkSpec {
 | |
| 	return android.SdkSpecFrom(ctx, String(a.properties.Sdk_version))
 | |
| }
 | |
| 
 | |
| func (a *AARImport) SystemModules() string {
 | |
| 	return ""
 | |
| }
 | |
| 
 | |
| func (a *AARImport) MinSdkVersion(ctx android.EarlyModuleContext) android.ApiLevel {
 | |
| 	if a.properties.Min_sdk_version != nil {
 | |
| 		return android.ApiLevelFrom(ctx, *a.properties.Min_sdk_version)
 | |
| 	}
 | |
| 	return a.SdkVersion(ctx).ApiLevel
 | |
| }
 | |
| 
 | |
| func (a *AARImport) ReplaceMaxSdkVersionPlaceholder(ctx android.EarlyModuleContext) android.ApiLevel {
 | |
| 	return android.SdkSpecFrom(ctx, "").ApiLevel
 | |
| }
 | |
| 
 | |
| func (a *AARImport) TargetSdkVersion(ctx android.EarlyModuleContext) android.ApiLevel {
 | |
| 	return a.SdkVersion(ctx).ApiLevel
 | |
| }
 | |
| 
 | |
| func (a *AARImport) javaVersion() string {
 | |
| 	return ""
 | |
| }
 | |
| 
 | |
| var _ AndroidLibraryDependency = (*AARImport)(nil)
 | |
| 
 | |
| func (a *AARImport) ExportPackage() android.Path {
 | |
| 	return a.exportPackage
 | |
| }
 | |
| 
 | |
| func (a *AARImport) ExportedProguardFlagFiles() android.Paths {
 | |
| 	return android.Paths{a.proguardFlags}
 | |
| }
 | |
| 
 | |
| func (a *AARImport) ExportedRRODirs() []rroDir {
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (a *AARImport) ExportedStaticPackages() android.Paths {
 | |
| 	return a.exportedStaticPackages
 | |
| }
 | |
| 
 | |
| func (a *AARImport) ExportedManifests() android.Paths {
 | |
| 	return android.Paths{a.manifest}
 | |
| }
 | |
| 
 | |
| func (a *AARImport) ExportedAssets() android.OptionalPath {
 | |
| 	return android.OptionalPathForPath(a.assetsPackage)
 | |
| }
 | |
| 
 | |
| // RRO enforcement is not available on aar_import since its RRO dirs are not
 | |
| // exported.
 | |
| func (a *AARImport) SetRROEnforcedForDependent(enforce bool) {
 | |
| }
 | |
| 
 | |
| // RRO enforcement is not available on aar_import since its RRO dirs are not
 | |
| // exported.
 | |
| func (a *AARImport) IsRROEnforced(ctx android.BaseModuleContext) bool {
 | |
| 	return false
 | |
| }
 | |
| 
 | |
| func (a *AARImport) Prebuilt() *android.Prebuilt {
 | |
| 	return &a.prebuilt
 | |
| }
 | |
| 
 | |
| func (a *AARImport) Name() string {
 | |
| 	return a.prebuilt.Name(a.ModuleBase.Name())
 | |
| }
 | |
| 
 | |
| func (a *AARImport) JacocoReportClassesFile() android.Path {
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (a *AARImport) DepsMutator(ctx android.BottomUpMutatorContext) {
 | |
| 	if !ctx.Config().AlwaysUsePrebuiltSdks() {
 | |
| 		sdkDep := decodeSdkDep(ctx, android.SdkContext(a))
 | |
| 		if sdkDep.useModule && sdkDep.frameworkResModule != "" {
 | |
| 			ctx.AddVariationDependencies(nil, frameworkResTag, sdkDep.frameworkResModule)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	ctx.AddVariationDependencies(nil, libTag, a.properties.Libs...)
 | |
| 	ctx.AddVariationDependencies(nil, staticLibTag, a.properties.Static_libs...)
 | |
| }
 | |
| 
 | |
| type JniPackageInfo struct {
 | |
| 	// List of zip files containing JNI libraries
 | |
| 	// Zip files should have directory structure jni/<arch>/*.so
 | |
| 	JniPackages android.Paths
 | |
| }
 | |
| 
 | |
| var JniPackageProvider = blueprint.NewProvider(JniPackageInfo{})
 | |
| 
 | |
| // Unzip an AAR and extract the JNI libs for $archString.
 | |
| var extractJNI = pctx.AndroidStaticRule("extractJNI",
 | |
| 	blueprint.RuleParams{
 | |
| 		Command: `rm -rf $out $outDir && touch $out && ` +
 | |
| 			`unzip -qoDD -d $outDir $in "jni/${archString}/*" && ` +
 | |
| 			`jni_files=$$(find $outDir/jni -type f) && ` +
 | |
| 			// print error message if there are no JNI libs for this arch
 | |
| 			`[ -n "$$jni_files" ] || (echo "ERROR: no JNI libs found for arch ${archString}" && exit 1) && ` +
 | |
| 			`${config.SoongZipCmd} -o $out -P 'lib/${archString}' ` +
 | |
| 			`-C $outDir/jni/${archString} $$(echo $$jni_files | xargs -n1 printf " -f %s")`,
 | |
| 		CommandDeps: []string{"${config.SoongZipCmd}"},
 | |
| 	},
 | |
| 	"outDir", "archString")
 | |
| 
 | |
| // Unzip an AAR into its constituent files and directories.  Any files in Outputs that don't exist in the AAR will be
 | |
| // touched to create an empty file. The res directory is not extracted, as it will be extracted in its own rule.
 | |
| var unzipAAR = pctx.AndroidStaticRule("unzipAAR",
 | |
| 	blueprint.RuleParams{
 | |
| 		Command: `rm -rf $outDir && mkdir -p $outDir && ` +
 | |
| 			`unzip -qoDD -d $outDir $in && rm -rf $outDir/res && touch $out && ` +
 | |
| 			`${config.Zip2ZipCmd} -i $in -o $assetsPackage 'assets/**/*' && ` +
 | |
| 			`${config.MergeZipsCmd} $combinedClassesJar $$(ls $outDir/classes.jar 2> /dev/null) $$(ls $outDir/libs/*.jar 2> /dev/null)`,
 | |
| 		CommandDeps: []string{"${config.MergeZipsCmd}", "${config.Zip2ZipCmd}"},
 | |
| 	},
 | |
| 	"outDir", "combinedClassesJar", "assetsPackage")
 | |
| 
 | |
| func (a *AARImport) GenerateAndroidBuildActions(ctx android.ModuleContext) {
 | |
| 	if len(a.properties.Aars) != 1 {
 | |
| 		ctx.PropertyErrorf("aars", "exactly one aar is required")
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	a.sdkVersion = a.SdkVersion(ctx)
 | |
| 	a.minSdkVersion = a.MinSdkVersion(ctx)
 | |
| 
 | |
| 	a.hideApexVariantFromMake = !ctx.Provider(android.ApexInfoProvider).(android.ApexInfo).IsForPlatform()
 | |
| 
 | |
| 	aarName := ctx.ModuleName() + ".aar"
 | |
| 	a.aarPath = android.PathForModuleSrc(ctx, a.properties.Aars[0])
 | |
| 
 | |
| 	if Bool(a.properties.Jetifier) {
 | |
| 		inputFile := a.aarPath
 | |
| 		a.aarPath = android.PathForModuleOut(ctx, "jetifier", aarName)
 | |
| 		TransformJetifier(ctx, a.aarPath.(android.WritablePath), inputFile)
 | |
| 	}
 | |
| 
 | |
| 	extractedAARDir := android.PathForModuleOut(ctx, "aar")
 | |
| 	a.classpathFile = extractedAARDir.Join(ctx, "classes-combined.jar")
 | |
| 	a.proguardFlags = extractedAARDir.Join(ctx, "proguard.txt")
 | |
| 	a.manifest = extractedAARDir.Join(ctx, "AndroidManifest.xml")
 | |
| 	a.assetsPackage = android.PathForModuleOut(ctx, "assets.zip")
 | |
| 
 | |
| 	ctx.Build(pctx, android.BuildParams{
 | |
| 		Rule:        unzipAAR,
 | |
| 		Input:       a.aarPath,
 | |
| 		Outputs:     android.WritablePaths{a.classpathFile, a.proguardFlags, a.manifest, a.assetsPackage},
 | |
| 		Description: "unzip AAR",
 | |
| 		Args: map[string]string{
 | |
| 			"outDir":             extractedAARDir.String(),
 | |
| 			"combinedClassesJar": a.classpathFile.String(),
 | |
| 			"assetsPackage":      a.assetsPackage.String(),
 | |
| 		},
 | |
| 	})
 | |
| 
 | |
| 	// Always set --pseudo-localize, it will be stripped out later for release
 | |
| 	// builds that don't want it.
 | |
| 	compileFlags := []string{"--pseudo-localize"}
 | |
| 	compiledResDir := android.PathForModuleOut(ctx, "flat-res")
 | |
| 	flata := compiledResDir.Join(ctx, "gen_res.flata")
 | |
| 	aapt2CompileZip(ctx, flata, a.aarPath, "res", compileFlags)
 | |
| 
 | |
| 	a.exportPackage = android.PathForModuleOut(ctx, "package-res.apk")
 | |
| 	// the subdir "android" is required to be filtered by package names
 | |
| 	srcJar := android.PathForModuleGen(ctx, "android", "R.srcjar")
 | |
| 	proguardOptionsFile := android.PathForModuleGen(ctx, "proguard.options")
 | |
| 	rTxt := android.PathForModuleOut(ctx, "R.txt")
 | |
| 	a.extraAaptPackagesFile = android.PathForModuleOut(ctx, "extra_packages")
 | |
| 
 | |
| 	var linkDeps android.Paths
 | |
| 
 | |
| 	linkFlags := []string{
 | |
| 		"--static-lib",
 | |
| 		"--no-static-lib-packages",
 | |
| 		"--auto-add-overlay",
 | |
| 	}
 | |
| 
 | |
| 	linkFlags = append(linkFlags, "--manifest "+a.manifest.String())
 | |
| 	linkDeps = append(linkDeps, a.manifest)
 | |
| 
 | |
| 	transitiveStaticLibs, staticLibManifests, staticRRODirs, transitiveAssets, libDeps, libFlags :=
 | |
| 		aaptLibs(ctx, android.SdkContext(a), nil)
 | |
| 
 | |
| 	_ = staticLibManifests
 | |
| 	_ = staticRRODirs
 | |
| 
 | |
| 	linkDeps = append(linkDeps, libDeps...)
 | |
| 	linkFlags = append(linkFlags, libFlags...)
 | |
| 
 | |
| 	overlayRes := append(android.Paths{flata}, transitiveStaticLibs...)
 | |
| 
 | |
| 	aapt2Link(ctx, a.exportPackage, srcJar, proguardOptionsFile, rTxt, a.extraAaptPackagesFile,
 | |
| 		linkFlags, linkDeps, nil, overlayRes, transitiveAssets, nil)
 | |
| 
 | |
| 	// Merge this import's assets with its dependencies' assets (if there are any).
 | |
| 	if len(transitiveAssets) > 0 {
 | |
| 		mergedAssets := android.PathForModuleOut(ctx, "merged-assets.zip")
 | |
| 		inputZips := append(android.Paths{a.assetsPackage}, transitiveAssets...)
 | |
| 		ctx.Build(pctx, android.BuildParams{
 | |
| 			Rule:        mergeAssetsRule,
 | |
| 			Inputs:      inputZips,
 | |
| 			Output:      mergedAssets,
 | |
| 			Description: "merge assets from dependencies and self",
 | |
| 		})
 | |
| 		a.assetsPackage = mergedAssets
 | |
| 	}
 | |
| 
 | |
| 	a.collectTransitiveHeaderJars(ctx)
 | |
| 	ctx.SetProvider(JavaInfoProvider, JavaInfo{
 | |
| 		HeaderJars:                     android.PathsIfNonNil(a.classpathFile),
 | |
| 		TransitiveLibsHeaderJars:       a.transitiveLibsHeaderJars,
 | |
| 		TransitiveStaticLibsHeaderJars: a.transitiveStaticLibsHeaderJars,
 | |
| 		ImplementationAndResourcesJars: android.PathsIfNonNil(a.classpathFile),
 | |
| 		ImplementationJars:             android.PathsIfNonNil(a.classpathFile),
 | |
| 	})
 | |
| 
 | |
| 	if proptools.Bool(a.properties.Extract_jni) {
 | |
| 		for _, t := range ctx.MultiTargets() {
 | |
| 			arch := t.Arch.Abi[0]
 | |
| 			path := android.PathForModuleOut(ctx, arch+"_jni.zip")
 | |
| 			a.jniPackages = append(a.jniPackages, path)
 | |
| 
 | |
| 			outDir := android.PathForModuleOut(ctx, "aarForJni")
 | |
| 			aarPath := android.PathForModuleSrc(ctx, a.properties.Aars[0])
 | |
| 			ctx.Build(pctx, android.BuildParams{
 | |
| 				Rule:        extractJNI,
 | |
| 				Input:       aarPath,
 | |
| 				Outputs:     android.WritablePaths{path},
 | |
| 				Description: "extract JNI from AAR",
 | |
| 				Args: map[string]string{
 | |
| 					"outDir":     outDir.String(),
 | |
| 					"archString": arch,
 | |
| 				},
 | |
| 			})
 | |
| 		}
 | |
| 
 | |
| 		ctx.SetProvider(JniPackageProvider, JniPackageInfo{
 | |
| 			JniPackages: a.jniPackages,
 | |
| 		})
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (a *AARImport) HeaderJars() android.Paths {
 | |
| 	return android.Paths{a.classpathFile}
 | |
| }
 | |
| 
 | |
| func (a *AARImport) ImplementationAndResourcesJars() android.Paths {
 | |
| 	return android.Paths{a.classpathFile}
 | |
| }
 | |
| 
 | |
| func (a *AARImport) DexJarBuildPath() android.Path {
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (a *AARImport) DexJarInstallPath() android.Path {
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (a *AARImport) ClassLoaderContexts() dexpreopt.ClassLoaderContextMap {
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| var _ android.ApexModule = (*AARImport)(nil)
 | |
| 
 | |
| // Implements android.ApexModule
 | |
| func (a *AARImport) DepIsInSameApex(ctx android.BaseModuleContext, dep android.Module) bool {
 | |
| 	return a.depIsInSameApex(ctx, dep)
 | |
| }
 | |
| 
 | |
| // Implements android.ApexModule
 | |
| func (g *AARImport) ShouldSupportSdkVersion(ctx android.BaseModuleContext,
 | |
| 	sdkVersion android.ApiLevel) error {
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| var _ android.PrebuiltInterface = (*AARImport)(nil)
 | |
| 
 | |
| // android_library_import imports an `.aar` file into the build graph as if it was built with android_library.
 | |
| //
 | |
| // This module is not suitable for installing on a device, but can be used as a `static_libs` dependency of
 | |
| // an android_app module.
 | |
| func AARImportFactory() android.Module {
 | |
| 	module := &AARImport{}
 | |
| 
 | |
| 	module.AddProperties(&module.properties)
 | |
| 
 | |
| 	android.InitPrebuiltModule(module, &module.properties.Aars)
 | |
| 	android.InitApexModule(module)
 | |
| 	InitJavaModuleMultiTargets(module, android.DeviceSupported)
 | |
| 	android.InitBazelModule(module)
 | |
| 	return module
 | |
| }
 | |
| 
 | |
| type bazelAapt struct {
 | |
| 	Manifest       bazel.Label
 | |
| 	Resource_files bazel.LabelListAttribute
 | |
| }
 | |
| 
 | |
| type bazelAndroidLibrary struct {
 | |
| 	*javaLibraryAttributes
 | |
| 	*bazelAapt
 | |
| }
 | |
| 
 | |
| type bazelAndroidLibraryImport struct {
 | |
| 	Aar         bazel.Label
 | |
| 	Deps        bazel.LabelListAttribute
 | |
| 	Exports     bazel.LabelListAttribute
 | |
| 	Sdk_version bazel.StringAttribute
 | |
| }
 | |
| 
 | |
| func (a *aapt) convertAaptAttrsWithBp2Build(ctx android.TopDownMutatorContext) *bazelAapt {
 | |
| 	manifest := proptools.StringDefault(a.aaptProperties.Manifest, "AndroidManifest.xml")
 | |
| 
 | |
| 	resourceFiles := bazel.LabelList{
 | |
| 		Includes: []bazel.Label{},
 | |
| 	}
 | |
| 	for _, dir := range android.PathsWithOptionalDefaultForModuleSrc(ctx, a.aaptProperties.Resource_dirs, "res") {
 | |
| 		files := android.RootToModuleRelativePaths(ctx, androidResourceGlob(ctx, dir))
 | |
| 		resourceFiles.Includes = append(resourceFiles.Includes, files...)
 | |
| 	}
 | |
| 	return &bazelAapt{
 | |
| 		android.BazelLabelForModuleSrcSingle(ctx, manifest),
 | |
| 		bazel.MakeLabelListAttribute(resourceFiles),
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (a *AARImport) ConvertWithBp2build(ctx android.TopDownMutatorContext) {
 | |
| 	aars := android.BazelLabelForModuleSrcExcludes(ctx, a.properties.Aars, []string{})
 | |
| 	exportableStaticLibs := []string{}
 | |
| 	// TODO(b/240716882): investigate and handle static_libs deps that are not imports. They are not supported for export by Bazel.
 | |
| 	for _, depName := range a.properties.Static_libs {
 | |
| 		if dep, ok := ctx.ModuleFromName(depName); ok {
 | |
| 			switch dep.(type) {
 | |
| 			case *AARImport, *Import:
 | |
| 				exportableStaticLibs = append(exportableStaticLibs, depName)
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	name := android.RemoveOptionalPrebuiltPrefix(a.Name())
 | |
| 	deps := android.BazelLabelForModuleDeps(ctx, android.LastUniqueStrings(android.CopyOf(append(a.properties.Static_libs, a.properties.Libs...))))
 | |
| 	exports := android.BazelLabelForModuleDeps(ctx, android.LastUniqueStrings(exportableStaticLibs))
 | |
| 
 | |
| 	ctx.CreateBazelTargetModule(
 | |
| 		bazel.BazelTargetModuleProperties{
 | |
| 			Rule_class:        "aar_import",
 | |
| 			Bzl_load_location: "//build/bazel/rules/android:rules.bzl",
 | |
| 		},
 | |
| 		android.CommonAttributes{Name: name},
 | |
| 		&bazelAndroidLibraryImport{
 | |
| 			Aar:         aars.Includes[0],
 | |
| 			Deps:        bazel.MakeLabelListAttribute(deps),
 | |
| 			Exports:     bazel.MakeLabelListAttribute(exports),
 | |
| 			Sdk_version: bazel.StringAttribute{Value: a.properties.Sdk_version},
 | |
| 		},
 | |
| 	)
 | |
| 
 | |
| 	neverlink := true
 | |
| 	ctx.CreateBazelTargetModule(
 | |
| 		AndroidLibraryBazelTargetModuleProperties(),
 | |
| 		android.CommonAttributes{Name: name + "-neverlink"},
 | |
| 		&bazelAndroidLibrary{
 | |
| 			javaLibraryAttributes: &javaLibraryAttributes{
 | |
| 				Neverlink: bazel.BoolAttribute{Value: &neverlink},
 | |
| 				Exports:   bazel.MakeSingleLabelListAttribute(bazel.Label{Label: ":" + name}),
 | |
| 				javaCommonAttributes: &javaCommonAttributes{
 | |
| 					Sdk_version: bazel.StringAttribute{Value: a.properties.Sdk_version},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 	)
 | |
| 
 | |
| }
 | |
| func AndroidLibraryBazelTargetModuleProperties() bazel.BazelTargetModuleProperties {
 | |
| 	return bazel.BazelTargetModuleProperties{
 | |
| 		Rule_class:        "android_library",
 | |
| 		Bzl_load_location: "//build/bazel/rules/android:rules.bzl",
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (a *AndroidLibrary) ConvertWithBp2build(ctx android.TopDownMutatorContext) {
 | |
| 	commonAttrs, bp2buildInfo := a.convertLibraryAttrsBp2Build(ctx)
 | |
| 	depLabels := bp2buildInfo.DepLabels
 | |
| 
 | |
| 	deps := depLabels.Deps
 | |
| 	if !commonAttrs.Srcs.IsEmpty() {
 | |
| 		deps.Append(depLabels.StaticDeps) // we should only append these if there are sources to use them
 | |
| 	} else if !depLabels.Deps.IsEmpty() {
 | |
| 		ctx.ModuleErrorf("Module has direct dependencies but no sources. Bazel will not allow this.")
 | |
| 	}
 | |
| 	name := a.Name()
 | |
| 	props := AndroidLibraryBazelTargetModuleProperties()
 | |
| 
 | |
| 	ctx.CreateBazelTargetModule(
 | |
| 		props,
 | |
| 		android.CommonAttributes{Name: name},
 | |
| 		&bazelAndroidLibrary{
 | |
| 			&javaLibraryAttributes{
 | |
| 				javaCommonAttributes: commonAttrs,
 | |
| 				Deps:                 deps,
 | |
| 				Exports:              depLabels.StaticDeps,
 | |
| 			},
 | |
| 			a.convertAaptAttrsWithBp2Build(ctx),
 | |
| 		},
 | |
| 	)
 | |
| 
 | |
| 	neverlink := true
 | |
| 	ctx.CreateBazelTargetModule(
 | |
| 		props,
 | |
| 		android.CommonAttributes{Name: name + "-neverlink"},
 | |
| 		&bazelAndroidLibrary{
 | |
| 			javaLibraryAttributes: &javaLibraryAttributes{
 | |
| 				Neverlink: bazel.BoolAttribute{Value: &neverlink},
 | |
| 				Exports:   bazel.MakeSingleLabelListAttribute(bazel.Label{Label: ":" + name}),
 | |
| 				javaCommonAttributes: &javaCommonAttributes{
 | |
| 					Sdk_version:  bazel.StringAttribute{Value: a.deviceProperties.Sdk_version},
 | |
| 					Java_version: bazel.StringAttribute{Value: a.properties.Java_version},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 	)
 | |
| }
 |