1// Copyright 2019 Google Inc. All rights reserved. 2// 3// Licensed under the Apache License, Version 2.0 (the "License"); 4// you may not use this file except in compliance with the License. 5// You may obtain a copy of the License at 6// 7// http://www.apache.org/licenses/LICENSE-2.0 8// 9// Unless required by applicable law or agreed to in writing, software 10// distributed under the License is distributed on an "AS IS" BASIS, 11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12// See the License for the specific language governing permissions and 13// limitations under the License. 14 15package java 16 17import ( 18 "fmt" 19 "path/filepath" 20 21 "android/soong/android" 22 "android/soong/java/config" 23 24 "github.com/google/blueprint/pathtools" 25) 26 27func init() { 28 android.RegisterParallelSingletonType("sdk", sdkSingletonFactory) 29 android.RegisterMakeVarsProvider(pctx, sdkMakeVars) 30} 31 32var sdkFrameworkAidlPathKey = android.NewOnceKey("sdkFrameworkAidlPathKey") 33var nonUpdatableFrameworkAidlPathKey = android.NewOnceKey("nonUpdatableFrameworkAidlPathKey") 34 35func UseApiFingerprint(ctx android.BaseModuleContext) (useApiFingerprint bool, fingerprintSdkVersion string, fingerprintDeps android.OutputPath) { 36 if ctx.Config().UnbundledBuild() && !ctx.Config().AlwaysUsePrebuiltSdks() { 37 apiFingerprintTrue := ctx.Config().IsEnvTrue("UNBUNDLED_BUILD_TARGET_SDK_WITH_API_FINGERPRINT") 38 dessertShaIsSet := ctx.Config().Getenv("UNBUNDLED_BUILD_TARGET_SDK_WITH_DESSERT_SHA") != "" 39 40 // Error when both UNBUNDLED_BUILD_TARGET_SDK_WITH_API_FINGERPRINT and UNBUNDLED_BUILD_TARGET_SDK_WITH_DESSERT_SHA are set 41 if apiFingerprintTrue && dessertShaIsSet { 42 ctx.ModuleErrorf("UNBUNDLED_BUILD_TARGET_SDK_WITH_API_FINGERPRINT=true cannot be set alongside with UNBUNDLED_BUILD_TARGET_SDK_WITH_DESSERT_SHA") 43 } 44 45 useApiFingerprint = apiFingerprintTrue || dessertShaIsSet 46 if apiFingerprintTrue { 47 fingerprintSdkVersion = ctx.Config().PlatformSdkCodename() + fmt.Sprintf(".$$(cat %s)", android.ApiFingerprintPath(ctx).String()) 48 fingerprintDeps = android.ApiFingerprintPath(ctx) 49 } 50 if dessertShaIsSet { 51 fingerprintSdkVersion = ctx.Config().Getenv("UNBUNDLED_BUILD_TARGET_SDK_WITH_DESSERT_SHA") 52 } 53 } 54 return useApiFingerprint, fingerprintSdkVersion, fingerprintDeps 55} 56 57func defaultJavaLanguageVersion(ctx android.EarlyModuleContext, s android.SdkSpec) javaVersion { 58 sdk, err := s.EffectiveVersion(ctx) 59 if err != nil { 60 ctx.PropertyErrorf("sdk_version", "%s", err) 61 } 62 if sdk.FinalOrFutureInt() <= 29 { 63 return JAVA_VERSION_8 64 } else if sdk.FinalOrFutureInt() <= 31 { 65 return JAVA_VERSION_9 66 } else if sdk.FinalOrFutureInt() <= 33 { 67 return JAVA_VERSION_11 68 } else if ctx.Config().TargetsJava21() { 69 // Temporary experimental flag to be able to try and build with 70 // java version 21 options. The flag, if used, just sets Java 71 // 21 as the default version, leaving any components that 72 // target an older version intact. 73 return JAVA_VERSION_21 74 } else { 75 return JAVA_VERSION_17 76 } 77} 78 79// systemModuleKind returns the kind of system modules to use for the supplied combination of sdk 80// kind and API level. 81func systemModuleKind(sdkKind android.SdkKind, apiLevel android.ApiLevel) android.SdkKind { 82 systemModuleKind := sdkKind 83 if apiLevel.LessThanOrEqualTo(android.LastWithoutModuleLibCoreSystemModules) { 84 // API levels less than or equal to 31 did not provide a core-for-system-modules.jar 85 // specifically for the module-lib API. So, always use the public system modules for them. 86 systemModuleKind = android.SdkPublic 87 } else if systemModuleKind == android.SdkCore { 88 // Core is by definition what is included in the system module for the public API so should 89 // just use its system modules. 90 systemModuleKind = android.SdkPublic 91 } else if systemModuleKind == android.SdkSystem || systemModuleKind == android.SdkTest || 92 systemModuleKind == android.SdkTestFrameworksCore { 93 // The core system and test APIs are currently the same as the public API so they should use 94 // its system modules. 95 systemModuleKind = android.SdkPublic 96 } else if systemModuleKind == android.SdkSystemServer { 97 // The core system server API is the same as the core module-lib API. 98 systemModuleKind = android.SdkModule 99 } 100 101 return systemModuleKind 102} 103 104func decodeSdkDep(ctx android.EarlyModuleContext, sdkContext android.SdkContext) sdkDep { 105 sdkVersion := sdkContext.SdkVersion(ctx) 106 if !sdkVersion.Valid() { 107 ctx.PropertyErrorf("sdk_version", "invalid version %q", sdkVersion.Raw) 108 return sdkDep{} 109 } 110 111 if ctx.DeviceSpecific() || ctx.SocSpecific() { 112 sdkVersion = sdkVersion.ForVendorPartition(ctx) 113 } 114 115 if !sdkVersion.ValidateSystemSdk(ctx) { 116 return sdkDep{} 117 } 118 119 if sdkVersion.UsePrebuilt(ctx) { 120 dir := filepath.Join("prebuilts", "sdk", sdkVersion.ApiLevel.String(), sdkVersion.Kind.String()) 121 jar := filepath.Join(dir, "android.jar") 122 // There's no aidl for other SDKs yet. 123 // TODO(77525052): Add aidl files for other SDKs too. 124 publicDir := filepath.Join("prebuilts", "sdk", sdkVersion.ApiLevel.String(), "public") 125 aidl := filepath.Join(publicDir, "framework.aidl") 126 jarPath := android.ExistentPathForSource(ctx, jar) 127 aidlPath := android.ExistentPathForSource(ctx, aidl) 128 lambdaStubsPath := android.PathForSource(ctx, config.SdkLambdaStubsPath) 129 130 if (!jarPath.Valid() || !aidlPath.Valid()) && ctx.Config().AllowMissingDependencies() { 131 return sdkDep{ 132 invalidVersion: true, 133 bootclasspath: []string{fmt.Sprintf("sdk_%s_%s_android", sdkVersion.Kind, sdkVersion.ApiLevel.String())}, 134 } 135 } 136 137 if !jarPath.Valid() { 138 ctx.PropertyErrorf("sdk_version", "invalid sdk version %q, %q does not exist", sdkVersion.Raw, jar) 139 return sdkDep{} 140 } 141 142 if !aidlPath.Valid() { 143 ctx.PropertyErrorf("sdk_version", "invalid sdk version %q, %q does not exist", sdkVersion.Raw, aidl) 144 return sdkDep{} 145 } 146 147 var systemModules string 148 if defaultJavaLanguageVersion(ctx, sdkVersion).usesJavaModules() { 149 systemModuleKind := systemModuleKind(sdkVersion.Kind, sdkVersion.ApiLevel) 150 systemModules = fmt.Sprintf("sdk_%s_%s_system_modules", systemModuleKind, sdkVersion.ApiLevel) 151 } 152 153 return sdkDep{ 154 useFiles: true, 155 jars: android.Paths{jarPath.Path(), lambdaStubsPath}, 156 aidl: android.OptionalPathForPath(aidlPath.Path()), 157 systemModules: systemModules, 158 } 159 } 160 161 toModule := func(module string, aidl android.Path) sdkDep { 162 // Select the kind of system modules needed for the sdk version. 163 systemModulesKind := systemModuleKind(sdkVersion.Kind, android.FutureApiLevel) 164 systemModules := fmt.Sprintf("core-%s-stubs-system-modules", systemModulesKind) 165 return sdkDep{ 166 useModule: true, 167 bootclasspath: []string{module, config.DefaultLambdaStubsLibrary}, 168 systemModules: systemModules, 169 java9Classpath: []string{module}, 170 frameworkResModule: "framework-res", 171 aidl: android.OptionalPathForPath(aidl), 172 } 173 } 174 175 switch sdkVersion.Kind { 176 case android.SdkPrivate: 177 return sdkDep{ 178 useModule: true, 179 systemModules: corePlatformSystemModules(ctx), 180 bootclasspath: corePlatformBootclasspathLibraries(ctx), 181 classpath: config.FrameworkLibraries, 182 frameworkResModule: "framework-res", 183 } 184 case android.SdkNone: 185 systemModules := sdkContext.SystemModules() 186 if systemModules == "" { 187 ctx.PropertyErrorf("sdk_version", 188 `system_modules is required to be set to a non-empty value when sdk_version is "none", did you mean sdk_version: "core_platform"?`) 189 } else if systemModules == "none" { 190 return sdkDep{ 191 noStandardLibs: true, 192 } 193 } 194 195 return sdkDep{ 196 useModule: true, 197 noStandardLibs: true, 198 systemModules: systemModules, 199 bootclasspath: []string{systemModules}, 200 } 201 case android.SdkCorePlatform: 202 return sdkDep{ 203 useModule: true, 204 systemModules: corePlatformSystemModules(ctx), 205 bootclasspath: corePlatformBootclasspathLibraries(ctx), 206 noFrameworksLibs: true, 207 } 208 case android.SdkPublic, android.SdkSystem, android.SdkTest, android.SdkTestFrameworksCore: 209 return toModule(sdkVersion.Kind.DefaultJavaLibraryName(), sdkFrameworkAidlPath(ctx)) 210 case android.SdkCore: 211 return sdkDep{ 212 useModule: true, 213 bootclasspath: []string{android.SdkCore.DefaultJavaLibraryName(), config.DefaultLambdaStubsLibrary}, 214 systemModules: "core-public-stubs-system-modules", 215 noFrameworksLibs: true, 216 } 217 case android.SdkModule: 218 // TODO(146757305): provide .apk and .aidl that have more APIs for modules 219 return toModule(sdkVersion.Kind.DefaultJavaLibraryName(), nonUpdatableFrameworkAidlPath(ctx)) 220 case android.SdkSystemServer: 221 // TODO(146757305): provide .apk and .aidl that have more APIs for modules 222 return toModule(sdkVersion.Kind.DefaultJavaLibraryName(), sdkFrameworkAidlPath(ctx)) 223 default: 224 panic(fmt.Errorf("invalid sdk %q", sdkVersion.Raw)) 225 } 226} 227 228func sdkSingletonFactory() android.Singleton { 229 return sdkSingleton{} 230} 231 232type sdkSingleton struct{} 233 234func (sdkSingleton) GenerateBuildActions(ctx android.SingletonContext) { 235 if ctx.Config().AlwaysUsePrebuiltSdks() { 236 return 237 } 238 239 createSdkFrameworkAidl(ctx) 240 createNonUpdatableFrameworkAidl(ctx) 241 createAPIFingerprint(ctx) 242} 243 244// Create framework.aidl by extracting anything that implements android.os.Parcelable from the SDK stubs modules. 245func createSdkFrameworkAidl(ctx android.SingletonContext) { 246 stubsModules := []string{ 247 android.SdkPublic.DefaultJavaLibraryName(), 248 android.SdkTest.DefaultJavaLibraryName(), 249 android.SdkSystem.DefaultJavaLibraryName(), 250 } 251 252 combinedAidl := sdkFrameworkAidlPath(ctx) 253 tempPath := tempPathForRestat(ctx, combinedAidl) 254 255 rule := createFrameworkAidl(stubsModules, tempPath, ctx) 256 257 commitChangeForRestat(rule, tempPath, combinedAidl) 258 259 rule.Build("framework_aidl", "generate framework.aidl") 260} 261 262// Creates a version of framework.aidl for the non-updatable part of the platform. 263func createNonUpdatableFrameworkAidl(ctx android.SingletonContext) { 264 stubsModules := []string{android.SdkModule.DefaultJavaLibraryName()} 265 266 combinedAidl := nonUpdatableFrameworkAidlPath(ctx) 267 tempPath := tempPathForRestat(ctx, combinedAidl) 268 269 rule := createFrameworkAidl(stubsModules, tempPath, ctx) 270 271 commitChangeForRestat(rule, tempPath, combinedAidl) 272 273 rule.Build("framework_non_updatable_aidl", "generate framework_non_updatable.aidl") 274} 275 276func createFrameworkAidl(stubsModules []string, path android.WritablePath, ctx android.SingletonContext) *android.RuleBuilder { 277 stubsJars := make([]android.Paths, len(stubsModules)) 278 279 ctx.VisitAllModules(func(module android.Module) { 280 // Collect dex jar paths for the modules listed above. 281 if j, ok := android.SingletonModuleProvider(ctx, module, JavaInfoProvider); ok { 282 name := ctx.ModuleName(module) 283 if i := android.IndexList(name, stubsModules); i != -1 { 284 stubsJars[i] = j.HeaderJars 285 } 286 } 287 }) 288 289 var missingDeps []string 290 291 for i := range stubsJars { 292 if stubsJars[i] == nil { 293 if ctx.Config().AllowMissingDependencies() { 294 missingDeps = append(missingDeps, stubsModules[i]) 295 } else { 296 ctx.Errorf("failed to find dex jar path for module %q", stubsModules[i]) 297 } 298 } 299 } 300 301 rule := android.NewRuleBuilder(pctx, ctx) 302 rule.MissingDeps(missingDeps) 303 304 var aidls android.Paths 305 for _, jars := range stubsJars { 306 for _, jar := range jars { 307 aidl := android.PathForOutput(ctx, "aidl", pathtools.ReplaceExtension(jar.Base(), "aidl")) 308 309 rule.Command(). 310 Text("rm -f").Output(aidl) 311 rule.Command(). 312 BuiltTool("sdkparcelables"). 313 Input(jar). 314 Output(aidl) 315 316 aidls = append(aidls, aidl) 317 } 318 } 319 320 rule.Command(). 321 Text("rm -f").Output(path) 322 rule.Command(). 323 Text("cat"). 324 Inputs(aidls). 325 Text("| sort -u >"). 326 Output(path) 327 328 return rule 329} 330 331func sdkFrameworkAidlPath(ctx android.PathContext) android.OutputPath { 332 return ctx.Config().Once(sdkFrameworkAidlPathKey, func() interface{} { 333 return android.PathForOutput(ctx, "framework.aidl") 334 }).(android.OutputPath) 335} 336 337func nonUpdatableFrameworkAidlPath(ctx android.PathContext) android.OutputPath { 338 return ctx.Config().Once(nonUpdatableFrameworkAidlPathKey, func() interface{} { 339 return android.PathForOutput(ctx, "framework_non_updatable.aidl") 340 }).(android.OutputPath) 341} 342 343// Create api_fingerprint.txt 344func createAPIFingerprint(ctx android.SingletonContext) { 345 out := android.ApiFingerprintPath(ctx) 346 347 rule := android.NewRuleBuilder(pctx, ctx) 348 349 rule.Command(). 350 Text("rm -f").Output(out) 351 cmd := rule.Command() 352 353 if ctx.Config().PlatformSdkCodename() == "REL" { 354 cmd.Text("echo REL >").Output(out) 355 } else if ctx.Config().FrameworksBaseDirExists(ctx) && !ctx.Config().AlwaysUsePrebuiltSdks() { 356 cmd.Text("cat") 357 apiTxtFileModules := []string{ 358 "api_fingerprint", 359 } 360 count := 0 361 ctx.VisitAllModules(func(module android.Module) { 362 name := ctx.ModuleName(module) 363 if android.InList(name, apiTxtFileModules) { 364 cmd.Inputs(android.OutputFilesForModule(ctx, module, "")) 365 count++ 366 } 367 }) 368 if count != len(apiTxtFileModules) { 369 ctx.Errorf("Could not find expected API module %v, found %d\n", apiTxtFileModules, count) 370 return 371 } 372 cmd.Text(">"). 373 Output(out) 374 } else { 375 // Unbundled build 376 // TODO: use a prebuilt api_fingerprint.txt from prebuilts/sdk/current.txt once we have one 377 cmd.Text("echo"). 378 Flag(ctx.Config().PlatformPreviewSdkVersion()). 379 Text(">"). 380 Output(out) 381 } 382 383 rule.Build("api_fingerprint", "generate api_fingerprint.txt") 384} 385 386func sdkMakeVars(ctx android.MakeVarsContext) { 387 if ctx.Config().AlwaysUsePrebuiltSdks() { 388 return 389 } 390 391 ctx.Strict("FRAMEWORK_AIDL", sdkFrameworkAidlPath(ctx).String()) 392 ctx.Strict("API_FINGERPRINT", android.ApiFingerprintPath(ctx).String()) 393} 394