These hardcoded refs will need to be updated when we start using .txt stub equivalent (single-tree/multi-tree). Instead of strewing this logic all over the codebase, create a helper function that contains the replacement logic. All other places should call this helper function instead of calculating the name of .txt equivalent soong module by itself. (Will do a similar cleanup in build/make) Test: no change in ninja file Change-Id: I6bf999eb4aeaba6ac2a44b9016bae4ec8c79ce19
		
			
				
	
	
		
			425 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			425 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2019 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"
 | |
| 	"sort"
 | |
| 	"strconv"
 | |
| 
 | |
| 	"android/soong/android"
 | |
| 	"android/soong/java/config"
 | |
| 
 | |
| 	"github.com/google/blueprint/pathtools"
 | |
| )
 | |
| 
 | |
| func init() {
 | |
| 	android.RegisterPreSingletonType("sdk_versions", sdkPreSingletonFactory)
 | |
| 	android.RegisterSingletonType("sdk", sdkSingletonFactory)
 | |
| 	android.RegisterMakeVarsProvider(pctx, sdkMakeVars)
 | |
| }
 | |
| 
 | |
| var sdkVersionsKey = android.NewOnceKey("sdkVersionsKey")
 | |
| var sdkFrameworkAidlPathKey = android.NewOnceKey("sdkFrameworkAidlPathKey")
 | |
| var nonUpdatableFrameworkAidlPathKey = android.NewOnceKey("nonUpdatableFrameworkAidlPathKey")
 | |
| var apiFingerprintPathKey = android.NewOnceKey("apiFingerprintPathKey")
 | |
| 
 | |
| func UseApiFingerprint(ctx android.BaseModuleContext) bool {
 | |
| 	if ctx.Config().UnbundledBuild() &&
 | |
| 		!ctx.Config().AlwaysUsePrebuiltSdks() &&
 | |
| 		ctx.Config().IsEnvTrue("UNBUNDLED_BUILD_TARGET_SDK_WITH_API_FINGERPRINT") {
 | |
| 		return true
 | |
| 	}
 | |
| 	return false
 | |
| }
 | |
| 
 | |
| func defaultJavaLanguageVersion(ctx android.EarlyModuleContext, s android.SdkSpec) javaVersion {
 | |
| 	sdk, err := s.EffectiveVersion(ctx)
 | |
| 	if err != nil {
 | |
| 		ctx.PropertyErrorf("sdk_version", "%s", err)
 | |
| 	}
 | |
| 	if sdk.FinalOrFutureInt() <= 23 {
 | |
| 		return JAVA_VERSION_7
 | |
| 	} else if sdk.FinalOrFutureInt() <= 29 {
 | |
| 		return JAVA_VERSION_8
 | |
| 	} else if sdk.FinalOrFutureInt() <= 31 {
 | |
| 		return JAVA_VERSION_9
 | |
| 	} else if sdk.FinalOrFutureInt() <= 33 {
 | |
| 		return JAVA_VERSION_11
 | |
| 	} else {
 | |
| 		return JAVA_VERSION_17
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // systemModuleKind returns the kind of system modules to use for the supplied combination of sdk
 | |
| // kind and API level.
 | |
| func systemModuleKind(sdkKind android.SdkKind, apiLevel android.ApiLevel) android.SdkKind {
 | |
| 	systemModuleKind := sdkKind
 | |
| 	if apiLevel.LessThanOrEqualTo(android.LastWithoutModuleLibCoreSystemModules) {
 | |
| 		// API levels less than or equal to 31 did not provide a core-for-system-modules.jar
 | |
| 		// specifically for the module-lib API. So, always use the public system modules for them.
 | |
| 		systemModuleKind = android.SdkPublic
 | |
| 	} else if systemModuleKind == android.SdkCore {
 | |
| 		// Core is by definition what is included in the system module for the public API so should
 | |
| 		// just use its system modules.
 | |
| 		systemModuleKind = android.SdkPublic
 | |
| 	} else if systemModuleKind == android.SdkSystem || systemModuleKind == android.SdkTest {
 | |
| 		// The core system and test APIs are currently the same as the public API so they should use
 | |
| 		// its system modules.
 | |
| 		systemModuleKind = android.SdkPublic
 | |
| 	} else if systemModuleKind == android.SdkSystemServer {
 | |
| 		// The core system server API is the same as the core module-lib API.
 | |
| 		systemModuleKind = android.SdkModule
 | |
| 	}
 | |
| 
 | |
| 	return systemModuleKind
 | |
| }
 | |
| 
 | |
| func decodeSdkDep(ctx android.EarlyModuleContext, sdkContext android.SdkContext) sdkDep {
 | |
| 	sdkVersion := sdkContext.SdkVersion(ctx)
 | |
| 	if !sdkVersion.Valid() {
 | |
| 		ctx.PropertyErrorf("sdk_version", "invalid version %q", sdkVersion.Raw)
 | |
| 		return sdkDep{}
 | |
| 	}
 | |
| 
 | |
| 	if ctx.DeviceSpecific() || ctx.SocSpecific() {
 | |
| 		sdkVersion = sdkVersion.ForVendorPartition(ctx)
 | |
| 	}
 | |
| 
 | |
| 	if !sdkVersion.ValidateSystemSdk(ctx) {
 | |
| 		return sdkDep{}
 | |
| 	}
 | |
| 
 | |
| 	if sdkVersion.UsePrebuilt(ctx) {
 | |
| 		dir := filepath.Join("prebuilts", "sdk", sdkVersion.ApiLevel.String(), sdkVersion.Kind.String())
 | |
| 		jar := filepath.Join(dir, "android.jar")
 | |
| 		// There's no aidl for other SDKs yet.
 | |
| 		// TODO(77525052): Add aidl files for other SDKs too.
 | |
| 		publicDir := filepath.Join("prebuilts", "sdk", sdkVersion.ApiLevel.String(), "public")
 | |
| 		aidl := filepath.Join(publicDir, "framework.aidl")
 | |
| 		jarPath := android.ExistentPathForSource(ctx, jar)
 | |
| 		aidlPath := android.ExistentPathForSource(ctx, aidl)
 | |
| 		lambdaStubsPath := android.PathForSource(ctx, config.SdkLambdaStubsPath)
 | |
| 
 | |
| 		if (!jarPath.Valid() || !aidlPath.Valid()) && ctx.Config().AllowMissingDependencies() {
 | |
| 			return sdkDep{
 | |
| 				invalidVersion: true,
 | |
| 				bootclasspath:  []string{fmt.Sprintf("sdk_%s_%s_android", sdkVersion.Kind, sdkVersion.ApiLevel.String())},
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if !jarPath.Valid() {
 | |
| 			ctx.PropertyErrorf("sdk_version", "invalid sdk version %q, %q does not exist", sdkVersion.Raw, jar)
 | |
| 			return sdkDep{}
 | |
| 		}
 | |
| 
 | |
| 		if !aidlPath.Valid() {
 | |
| 			ctx.PropertyErrorf("sdk_version", "invalid sdk version %q, %q does not exist", sdkVersion.Raw, aidl)
 | |
| 			return sdkDep{}
 | |
| 		}
 | |
| 
 | |
| 		var systemModules string
 | |
| 		if defaultJavaLanguageVersion(ctx, sdkVersion).usesJavaModules() {
 | |
| 			systemModuleKind := systemModuleKind(sdkVersion.Kind, sdkVersion.ApiLevel)
 | |
| 			systemModules = fmt.Sprintf("sdk_%s_%s_system_modules", systemModuleKind, sdkVersion.ApiLevel)
 | |
| 		}
 | |
| 
 | |
| 		return sdkDep{
 | |
| 			useFiles:      true,
 | |
| 			jars:          android.Paths{jarPath.Path(), lambdaStubsPath},
 | |
| 			aidl:          android.OptionalPathForPath(aidlPath.Path()),
 | |
| 			systemModules: systemModules,
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	toModule := func(module string, aidl android.Path) sdkDep {
 | |
| 		// Select the kind of system modules needed for the sdk version.
 | |
| 		systemModulesKind := systemModuleKind(sdkVersion.Kind, android.FutureApiLevel)
 | |
| 		return sdkDep{
 | |
| 			useModule:          true,
 | |
| 			bootclasspath:      []string{module, config.DefaultLambdaStubsLibrary},
 | |
| 			systemModules:      fmt.Sprintf("core-%s-stubs-system-modules", systemModulesKind),
 | |
| 			java9Classpath:     []string{module},
 | |
| 			frameworkResModule: "framework-res",
 | |
| 			aidl:               android.OptionalPathForPath(aidl),
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	switch sdkVersion.Kind {
 | |
| 	case android.SdkPrivate:
 | |
| 		return sdkDep{
 | |
| 			useModule:          true,
 | |
| 			systemModules:      corePlatformSystemModules(ctx),
 | |
| 			bootclasspath:      corePlatformBootclasspathLibraries(ctx),
 | |
| 			classpath:          config.FrameworkLibraries,
 | |
| 			frameworkResModule: "framework-res",
 | |
| 		}
 | |
| 	case android.SdkNone:
 | |
| 		systemModules := sdkContext.SystemModules()
 | |
| 		if systemModules == "" {
 | |
| 			ctx.PropertyErrorf("sdk_version",
 | |
| 				`system_modules is required to be set to a non-empty value when sdk_version is "none", did you mean sdk_version: "core_platform"?`)
 | |
| 		} else if systemModules == "none" {
 | |
| 			return sdkDep{
 | |
| 				noStandardLibs: true,
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		return sdkDep{
 | |
| 			useModule:      true,
 | |
| 			noStandardLibs: true,
 | |
| 			systemModules:  systemModules,
 | |
| 			bootclasspath:  []string{systemModules},
 | |
| 		}
 | |
| 	case android.SdkCorePlatform:
 | |
| 		return sdkDep{
 | |
| 			useModule:        true,
 | |
| 			systemModules:    corePlatformSystemModules(ctx),
 | |
| 			bootclasspath:    corePlatformBootclasspathLibraries(ctx),
 | |
| 			noFrameworksLibs: true,
 | |
| 		}
 | |
| 	case android.SdkPublic, android.SdkSystem, android.SdkTest:
 | |
| 		return toModule(sdkVersion.Kind.JavaLibraryName(ctx.Config()), sdkFrameworkAidlPath(ctx))
 | |
| 	case android.SdkCore:
 | |
| 		return sdkDep{
 | |
| 			useModule:        true,
 | |
| 			bootclasspath:    []string{"core.current.stubs", config.DefaultLambdaStubsLibrary},
 | |
| 			systemModules:    "core-public-stubs-system-modules",
 | |
| 			noFrameworksLibs: true,
 | |
| 		}
 | |
| 	case android.SdkModule:
 | |
| 		// TODO(146757305): provide .apk and .aidl that have more APIs for modules
 | |
| 		return toModule(sdkVersion.Kind.JavaLibraryName(ctx.Config()), nonUpdatableFrameworkAidlPath(ctx))
 | |
| 	case android.SdkSystemServer:
 | |
| 		// TODO(146757305): provide .apk and .aidl that have more APIs for modules
 | |
| 		return toModule(sdkVersion.Kind.JavaLibraryName(ctx.Config()), sdkFrameworkAidlPath(ctx))
 | |
| 	default:
 | |
| 		panic(fmt.Errorf("invalid sdk %q", sdkVersion.Raw))
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func sdkPreSingletonFactory() android.Singleton {
 | |
| 	return sdkPreSingleton{}
 | |
| }
 | |
| 
 | |
| type sdkPreSingleton struct{}
 | |
| 
 | |
| func (sdkPreSingleton) GenerateBuildActions(ctx android.SingletonContext) {
 | |
| 	sdkJars, err := ctx.GlobWithDeps("prebuilts/sdk/*/public/android.jar", nil)
 | |
| 	if err != nil {
 | |
| 		ctx.Errorf("failed to glob prebuilts/sdk/*/public/android.jar: %s", err.Error())
 | |
| 	}
 | |
| 
 | |
| 	var sdkVersions []int
 | |
| 	for _, sdkJar := range sdkJars {
 | |
| 		dir := filepath.Base(filepath.Dir(filepath.Dir(sdkJar)))
 | |
| 		v, err := strconv.Atoi(dir)
 | |
| 		if scerr, ok := err.(*strconv.NumError); ok && scerr.Err == strconv.ErrSyntax {
 | |
| 			continue
 | |
| 		} else if err != nil {
 | |
| 			ctx.Errorf("invalid sdk jar %q, %s, %v", sdkJar, err.Error())
 | |
| 		}
 | |
| 		sdkVersions = append(sdkVersions, v)
 | |
| 	}
 | |
| 
 | |
| 	sort.Ints(sdkVersions)
 | |
| 
 | |
| 	ctx.Config().Once(sdkVersionsKey, func() interface{} { return sdkVersions })
 | |
| }
 | |
| 
 | |
| func LatestSdkVersionInt(ctx android.EarlyModuleContext) int {
 | |
| 	sdkVersions := ctx.Config().Get(sdkVersionsKey).([]int)
 | |
| 	latestSdkVersion := 0
 | |
| 	if len(sdkVersions) > 0 {
 | |
| 		latestSdkVersion = sdkVersions[len(sdkVersions)-1]
 | |
| 	}
 | |
| 	return latestSdkVersion
 | |
| }
 | |
| 
 | |
| func sdkSingletonFactory() android.Singleton {
 | |
| 	return sdkSingleton{}
 | |
| }
 | |
| 
 | |
| type sdkSingleton struct{}
 | |
| 
 | |
| func (sdkSingleton) GenerateBuildActions(ctx android.SingletonContext) {
 | |
| 	if ctx.Config().AlwaysUsePrebuiltSdks() {
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	createSdkFrameworkAidl(ctx)
 | |
| 	createNonUpdatableFrameworkAidl(ctx)
 | |
| 	createAPIFingerprint(ctx)
 | |
| }
 | |
| 
 | |
| // Create framework.aidl by extracting anything that implements android.os.Parcelable from the SDK stubs modules.
 | |
| func createSdkFrameworkAidl(ctx android.SingletonContext) {
 | |
| 	stubsModules := []string{
 | |
| 		android.SdkPublic.JavaLibraryName(ctx.Config()),
 | |
| 		android.SdkTest.JavaLibraryName(ctx.Config()),
 | |
| 		android.SdkSystem.JavaLibraryName(ctx.Config()),
 | |
| 	}
 | |
| 
 | |
| 	combinedAidl := sdkFrameworkAidlPath(ctx)
 | |
| 	tempPath := tempPathForRestat(ctx, combinedAidl)
 | |
| 
 | |
| 	rule := createFrameworkAidl(stubsModules, tempPath, ctx)
 | |
| 
 | |
| 	commitChangeForRestat(rule, tempPath, combinedAidl)
 | |
| 
 | |
| 	rule.Build("framework_aidl", "generate framework.aidl")
 | |
| }
 | |
| 
 | |
| // Creates a version of framework.aidl for the non-updatable part of the platform.
 | |
| func createNonUpdatableFrameworkAidl(ctx android.SingletonContext) {
 | |
| 	stubsModules := []string{android.SdkModule.JavaLibraryName(ctx.Config())}
 | |
| 
 | |
| 	combinedAidl := nonUpdatableFrameworkAidlPath(ctx)
 | |
| 	tempPath := tempPathForRestat(ctx, combinedAidl)
 | |
| 
 | |
| 	rule := createFrameworkAidl(stubsModules, tempPath, ctx)
 | |
| 
 | |
| 	commitChangeForRestat(rule, tempPath, combinedAidl)
 | |
| 
 | |
| 	rule.Build("framework_non_updatable_aidl", "generate framework_non_updatable.aidl")
 | |
| }
 | |
| 
 | |
| func createFrameworkAidl(stubsModules []string, path android.WritablePath, ctx android.SingletonContext) *android.RuleBuilder {
 | |
| 	stubsJars := make([]android.Paths, len(stubsModules))
 | |
| 
 | |
| 	ctx.VisitAllModules(func(module android.Module) {
 | |
| 		// Collect dex jar paths for the modules listed above.
 | |
| 		if ctx.ModuleHasProvider(module, JavaInfoProvider) {
 | |
| 			j := ctx.ModuleProvider(module, JavaInfoProvider).(JavaInfo)
 | |
| 			name := ctx.ModuleName(module)
 | |
| 			if i := android.IndexList(name, stubsModules); i != -1 {
 | |
| 				stubsJars[i] = j.HeaderJars
 | |
| 			}
 | |
| 		}
 | |
| 	})
 | |
| 
 | |
| 	var missingDeps []string
 | |
| 
 | |
| 	for i := range stubsJars {
 | |
| 		if stubsJars[i] == nil {
 | |
| 			if ctx.Config().AllowMissingDependencies() {
 | |
| 				missingDeps = append(missingDeps, stubsModules[i])
 | |
| 			} else {
 | |
| 				ctx.Errorf("failed to find dex jar path for module %q", stubsModules[i])
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	rule := android.NewRuleBuilder(pctx, ctx)
 | |
| 	rule.MissingDeps(missingDeps)
 | |
| 
 | |
| 	var aidls android.Paths
 | |
| 	for _, jars := range stubsJars {
 | |
| 		for _, jar := range jars {
 | |
| 			aidl := android.PathForOutput(ctx, "aidl", pathtools.ReplaceExtension(jar.Base(), "aidl"))
 | |
| 
 | |
| 			rule.Command().
 | |
| 				Text("rm -f").Output(aidl)
 | |
| 			rule.Command().
 | |
| 				BuiltTool("sdkparcelables").
 | |
| 				Input(jar).
 | |
| 				Output(aidl)
 | |
| 
 | |
| 			aidls = append(aidls, aidl)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	rule.Command().
 | |
| 		Text("rm -f").Output(path)
 | |
| 	rule.Command().
 | |
| 		Text("cat").
 | |
| 		Inputs(aidls).
 | |
| 		Text("| sort -u >").
 | |
| 		Output(path)
 | |
| 
 | |
| 	return rule
 | |
| }
 | |
| 
 | |
| func sdkFrameworkAidlPath(ctx android.PathContext) android.OutputPath {
 | |
| 	return ctx.Config().Once(sdkFrameworkAidlPathKey, func() interface{} {
 | |
| 		return android.PathForOutput(ctx, "framework.aidl")
 | |
| 	}).(android.OutputPath)
 | |
| }
 | |
| 
 | |
| func nonUpdatableFrameworkAidlPath(ctx android.PathContext) android.OutputPath {
 | |
| 	return ctx.Config().Once(nonUpdatableFrameworkAidlPathKey, func() interface{} {
 | |
| 		return android.PathForOutput(ctx, "framework_non_updatable.aidl")
 | |
| 	}).(android.OutputPath)
 | |
| }
 | |
| 
 | |
| // Create api_fingerprint.txt
 | |
| func createAPIFingerprint(ctx android.SingletonContext) {
 | |
| 	out := ApiFingerprintPath(ctx)
 | |
| 
 | |
| 	rule := android.NewRuleBuilder(pctx, ctx)
 | |
| 
 | |
| 	rule.Command().
 | |
| 		Text("rm -f").Output(out)
 | |
| 	cmd := rule.Command()
 | |
| 
 | |
| 	if ctx.Config().PlatformSdkCodename() == "REL" {
 | |
| 		cmd.Text("echo REL >").Output(out)
 | |
| 	} else if ctx.Config().FrameworksBaseDirExists(ctx) && !ctx.Config().AlwaysUsePrebuiltSdks() {
 | |
| 		cmd.Text("cat")
 | |
| 		apiTxtFileModules := []string{
 | |
| 			"api_fingerprint",
 | |
| 		}
 | |
| 		count := 0
 | |
| 		ctx.VisitAllModules(func(module android.Module) {
 | |
| 			name := ctx.ModuleName(module)
 | |
| 			if android.InList(name, apiTxtFileModules) {
 | |
| 				cmd.Inputs(android.OutputFilesForModule(ctx, module, ""))
 | |
| 				count++
 | |
| 			}
 | |
| 		})
 | |
| 		if count != len(apiTxtFileModules) {
 | |
| 			ctx.Errorf("Could not find expected API module %v, found %d\n", apiTxtFileModules, count)
 | |
| 			return
 | |
| 		}
 | |
| 		cmd.Text(">").
 | |
| 			Output(out)
 | |
| 	} else {
 | |
| 		// Unbundled build
 | |
| 		// TODO: use a prebuilt api_fingerprint.txt from prebuilts/sdk/current.txt once we have one
 | |
| 		cmd.Text("echo").
 | |
| 			Flag(ctx.Config().PlatformPreviewSdkVersion()).
 | |
| 			Text(">").
 | |
| 			Output(out)
 | |
| 	}
 | |
| 
 | |
| 	rule.Build("api_fingerprint", "generate api_fingerprint.txt")
 | |
| }
 | |
| 
 | |
| func ApiFingerprintPath(ctx android.PathContext) android.OutputPath {
 | |
| 	return ctx.Config().Once(apiFingerprintPathKey, func() interface{} {
 | |
| 		return android.PathForOutput(ctx, "api_fingerprint.txt")
 | |
| 	}).(android.OutputPath)
 | |
| }
 | |
| 
 | |
| func sdkMakeVars(ctx android.MakeVarsContext) {
 | |
| 	if ctx.Config().AlwaysUsePrebuiltSdks() {
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	ctx.Strict("FRAMEWORK_AIDL", sdkFrameworkAidlPath(ctx).String())
 | |
| 	ctx.Strict("API_FINGERPRINT", ApiFingerprintPath(ctx).String())
 | |
| }
 |