1// Copyright 2015 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 17// This file contains the module implementations for android_app, android_test, and some more 18// related module types, including their override variants. 19 20import ( 21 "path/filepath" 22 "sort" 23 "strings" 24 25 "github.com/google/blueprint" 26 "github.com/google/blueprint/proptools" 27 28 "android/soong/android" 29 "android/soong/cc" 30 "android/soong/dexpreopt" 31 "android/soong/tradefed" 32) 33 34func init() { 35 RegisterAppBuildComponents(android.InitRegistrationContext) 36} 37 38func RegisterAppBuildComponents(ctx android.RegistrationContext) { 39 ctx.RegisterModuleType("android_app", AndroidAppFactory) 40 ctx.RegisterModuleType("android_test", AndroidTestFactory) 41 ctx.RegisterModuleType("android_test_helper_app", AndroidTestHelperAppFactory) 42 ctx.RegisterModuleType("android_app_certificate", AndroidAppCertificateFactory) 43 ctx.RegisterModuleType("override_android_app", OverrideAndroidAppModuleFactory) 44 ctx.RegisterModuleType("override_android_test", OverrideAndroidTestModuleFactory) 45} 46 47// AndroidManifest.xml merging 48// package splits 49 50type appProperties struct { 51 // Names of extra android_app_certificate modules to sign the apk with in the form ":module". 52 Additional_certificates []string 53 54 // If set, create package-export.apk, which other packages can 55 // use to get PRODUCT-agnostic resource data like IDs and type definitions. 56 Export_package_resources *bool 57 58 // Specifies that this app should be installed to the priv-app directory, 59 // where the system will grant it additional privileges not available to 60 // normal apps. 61 Privileged *bool 62 63 // list of resource labels to generate individual resource packages 64 Package_splits []string 65 66 // Names of modules to be overridden. Listed modules can only be other binaries 67 // (in Make or Soong). 68 // This does not completely prevent installation of the overridden binaries, but if both 69 // binaries would be installed by default (in PRODUCT_PACKAGES) the other binary will be removed 70 // from PRODUCT_PACKAGES. 71 Overrides []string 72 73 // list of native libraries that will be provided in or alongside the resulting jar 74 Jni_libs []string `android:"arch_variant"` 75 76 // if true, use JNI libraries that link against platform APIs even if this module sets 77 // sdk_version. 78 Jni_uses_platform_apis *bool 79 80 // if true, use JNI libraries that link against SDK APIs even if this module does not set 81 // sdk_version. 82 Jni_uses_sdk_apis *bool 83 84 // STL library to use for JNI libraries. 85 Stl *string `android:"arch_variant"` 86 87 // Store native libraries uncompressed in the APK and set the android:extractNativeLibs="false" manifest 88 // flag so that they are used from inside the APK at runtime. Defaults to true for android_test modules unless 89 // sdk_version or min_sdk_version is set to a version that doesn't support it (<23), defaults to true for 90 // android_app modules that are embedded to APEXes, defaults to false for other module types where the native 91 // libraries are generally preinstalled outside the APK. 92 Use_embedded_native_libs *bool 93 94 // Store dex files uncompressed in the APK and set the android:useEmbeddedDex="true" manifest attribute so that 95 // they are used from inside the APK at runtime. 96 Use_embedded_dex *bool 97 98 // Forces native libraries to always be packaged into the APK, 99 // Use_embedded_native_libs still selects whether they are stored uncompressed and aligned or compressed. 100 // True for android_test* modules. 101 AlwaysPackageNativeLibs bool `blueprint:"mutated"` 102 103 // If set, find and merge all NOTICE files that this module and its dependencies have and store 104 // it in the APK as an asset. 105 Embed_notices *bool 106 107 // cc.Coverage related properties 108 PreventInstall bool `blueprint:"mutated"` 109 HideFromMake bool `blueprint:"mutated"` 110 IsCoverageVariant bool `blueprint:"mutated"` 111 112 // Whether this app is considered mainline updatable or not. When set to true, this will enforce 113 // additional rules to make sure an app can safely be updated. Default is false. 114 // Prefer using other specific properties if build behaviour must be changed; avoid using this 115 // flag for anything but neverallow rules (unless the behaviour change is invisible to owners). 116 Updatable *bool 117} 118 119// android_app properties that can be overridden by override_android_app 120type overridableAppProperties struct { 121 // The name of a certificate in the default certificate directory, blank to use the default product certificate, 122 // or an android_app_certificate module name in the form ":module". 123 Certificate *string 124 125 // Name of the signing certificate lineage file or filegroup module. 126 Lineage *string `android:"path"` 127 128 // the package name of this app. The package name in the manifest file is used if one was not given. 129 Package_name *string 130 131 // the logging parent of this app. 132 Logging_parent *string 133 134 // Whether to rename the package in resources to the override name rather than the base name. Defaults to true. 135 Rename_resources_package *bool 136} 137 138type AndroidApp struct { 139 Library 140 aapt 141 android.OverridableModuleBase 142 143 certificate Certificate 144 145 appProperties appProperties 146 147 overridableAppProperties overridableAppProperties 148 149 jniLibs []jniLib 150 installPathForJNISymbols android.Path 151 embeddedJniLibs bool 152 jniCoverageOutputs android.Paths 153 154 bundleFile android.Path 155 156 // the install APK name is normally the same as the module name, but can be overridden with PRODUCT_PACKAGE_NAME_OVERRIDES. 157 installApkName string 158 159 installDir android.InstallPath 160 161 onDeviceDir string 162 163 additionalAaptFlags []string 164 165 noticeOutputs android.NoticeOutputs 166 167 overriddenManifestPackageName string 168 169 android.ApexBundleDepsInfo 170} 171 172func (a *AndroidApp) IsInstallable() bool { 173 return Bool(a.properties.Installable) 174} 175 176func (a *AndroidApp) ExportedProguardFlagFiles() android.Paths { 177 return nil 178} 179 180func (a *AndroidApp) ExportedStaticPackages() android.Paths { 181 return nil 182} 183 184func (a *AndroidApp) OutputFile() android.Path { 185 return a.outputFile 186} 187 188func (a *AndroidApp) Certificate() Certificate { 189 return a.certificate 190} 191 192func (a *AndroidApp) JniCoverageOutputs() android.Paths { 193 return a.jniCoverageOutputs 194} 195 196var _ AndroidLibraryDependency = (*AndroidApp)(nil) 197 198type Certificate struct { 199 Pem, Key android.Path 200 presigned bool 201} 202 203var PresignedCertificate = Certificate{presigned: true} 204 205func (c Certificate) AndroidMkString() string { 206 if c.presigned { 207 return "PRESIGNED" 208 } else { 209 return c.Pem.String() 210 } 211} 212 213func (a *AndroidApp) DepsMutator(ctx android.BottomUpMutatorContext) { 214 a.Module.deps(ctx) 215 216 if String(a.appProperties.Stl) == "c++_shared" && !a.SdkVersion(ctx).Specified() { 217 ctx.PropertyErrorf("stl", "sdk_version must be set in order to use c++_shared") 218 } 219 220 sdkDep := decodeSdkDep(ctx, android.SdkContext(a)) 221 if sdkDep.hasFrameworkLibs() { 222 a.aapt.deps(ctx, sdkDep) 223 } 224 225 usesSDK := a.SdkVersion(ctx).Specified() && a.SdkVersion(ctx).Kind != android.SdkCorePlatform 226 227 if usesSDK && Bool(a.appProperties.Jni_uses_sdk_apis) { 228 ctx.PropertyErrorf("jni_uses_sdk_apis", 229 "can only be set for modules that do not set sdk_version") 230 } else if !usesSDK && Bool(a.appProperties.Jni_uses_platform_apis) { 231 ctx.PropertyErrorf("jni_uses_platform_apis", 232 "can only be set for modules that set sdk_version") 233 } 234 235 for _, jniTarget := range ctx.MultiTargets() { 236 variation := append(jniTarget.Variations(), 237 blueprint.Variation{Mutator: "link", Variation: "shared"}) 238 239 // If the app builds against an Android SDK use the SDK variant of JNI dependencies 240 // unless jni_uses_platform_apis is set. 241 // Don't require the SDK variant for apps that are shipped on vendor, etc., as they already 242 // have stable APIs through the VNDK. 243 if (usesSDK && !a.RequiresStableAPIs(ctx) && 244 !Bool(a.appProperties.Jni_uses_platform_apis)) || 245 Bool(a.appProperties.Jni_uses_sdk_apis) { 246 variation = append(variation, blueprint.Variation{Mutator: "sdk", Variation: "sdk"}) 247 } 248 ctx.AddFarVariationDependencies(variation, jniLibTag, a.appProperties.Jni_libs...) 249 } 250 251 a.usesLibrary.deps(ctx, sdkDep.hasFrameworkLibs()) 252} 253 254func (a *AndroidApp) OverridablePropertiesDepsMutator(ctx android.BottomUpMutatorContext) { 255 cert := android.SrcIsModule(a.getCertString(ctx)) 256 if cert != "" { 257 ctx.AddDependency(ctx.Module(), certificateTag, cert) 258 } 259 260 for _, cert := range a.appProperties.Additional_certificates { 261 cert = android.SrcIsModule(cert) 262 if cert != "" { 263 ctx.AddDependency(ctx.Module(), certificateTag, cert) 264 } else { 265 ctx.PropertyErrorf("additional_certificates", 266 `must be names of android_app_certificate modules in the form ":module"`) 267 } 268 } 269} 270 271func (a *AndroidTestHelperApp) GenerateAndroidBuildActions(ctx android.ModuleContext) { 272 a.generateAndroidBuildActions(ctx) 273} 274 275func (a *AndroidApp) GenerateAndroidBuildActions(ctx android.ModuleContext) { 276 a.checkAppSdkVersions(ctx) 277 a.generateAndroidBuildActions(ctx) 278} 279 280func (a *AndroidApp) checkAppSdkVersions(ctx android.ModuleContext) { 281 if a.Updatable() { 282 if !a.SdkVersion(ctx).Stable() { 283 ctx.PropertyErrorf("sdk_version", "Updatable apps must use stable SDKs, found %v", a.SdkVersion(ctx)) 284 } 285 if String(a.deviceProperties.Min_sdk_version) == "" { 286 ctx.PropertyErrorf("updatable", "updatable apps must set min_sdk_version.") 287 } 288 289 if minSdkVersion, err := a.MinSdkVersion(ctx).EffectiveVersion(ctx); err == nil { 290 a.checkJniLibsSdkVersion(ctx, minSdkVersion) 291 android.CheckMinSdkVersion(a, ctx, minSdkVersion) 292 } else { 293 ctx.PropertyErrorf("min_sdk_version", "%s", err.Error()) 294 } 295 } 296 297 a.checkPlatformAPI(ctx) 298 a.checkSdkVersions(ctx) 299} 300 301// If an updatable APK sets min_sdk_version, min_sdk_vesion of JNI libs should match with it. 302// This check is enforced for "updatable" APKs (including APK-in-APEX). 303// b/155209650: until min_sdk_version is properly supported, use sdk_version instead. 304// because, sdk_version is overridden by min_sdk_version (if set as smaller) 305// and sdkLinkType is checked with dependencies so we can be sure that the whole dependency tree 306// will meet the requirements. 307func (a *AndroidApp) checkJniLibsSdkVersion(ctx android.ModuleContext, minSdkVersion android.ApiLevel) { 308 // It's enough to check direct JNI deps' sdk_version because all transitive deps from JNI deps are checked in cc.checkLinkType() 309 ctx.VisitDirectDeps(func(m android.Module) { 310 if !IsJniDepTag(ctx.OtherModuleDependencyTag(m)) { 311 return 312 } 313 dep, _ := m.(*cc.Module) 314 // The domain of cc.sdk_version is "current" and <number> 315 // We can rely on android.SdkSpec to convert it to <number> so that "current" is 316 // handled properly regardless of sdk finalization. 317 jniSdkVersion, err := android.SdkSpecFrom(ctx, dep.SdkVersion()).EffectiveVersion(ctx) 318 if err != nil || minSdkVersion.LessThan(jniSdkVersion) { 319 ctx.OtherModuleErrorf(dep, "sdk_version(%v) is higher than min_sdk_version(%v) of the containing android_app(%v)", 320 dep.SdkVersion(), minSdkVersion, ctx.ModuleName()) 321 return 322 } 323 324 }) 325} 326 327// Returns true if the native libraries should be stored in the APK uncompressed and the 328// extractNativeLibs application flag should be set to false in the manifest. 329func (a *AndroidApp) useEmbeddedNativeLibs(ctx android.ModuleContext) bool { 330 minSdkVersion, err := a.MinSdkVersion(ctx).EffectiveVersion(ctx) 331 if err != nil { 332 ctx.PropertyErrorf("min_sdk_version", "invalid value %q: %s", a.MinSdkVersion(ctx), err) 333 } 334 335 apexInfo := ctx.Provider(android.ApexInfoProvider).(android.ApexInfo) 336 return (minSdkVersion.FinalOrFutureInt() >= 23 && Bool(a.appProperties.Use_embedded_native_libs)) || 337 !apexInfo.IsForPlatform() 338} 339 340// Returns whether this module should have the dex file stored uncompressed in the APK. 341func (a *AndroidApp) shouldUncompressDex(ctx android.ModuleContext) bool { 342 if Bool(a.appProperties.Use_embedded_dex) { 343 return true 344 } 345 346 // Uncompress dex in APKs of privileged apps (even for unbundled builds, they may 347 // be preinstalled as prebuilts). 348 if ctx.Config().UncompressPrivAppDex() && a.Privileged() { 349 return true 350 } 351 352 if ctx.Config().UnbundledBuild() { 353 return false 354 } 355 356 return shouldUncompressDex(ctx, &a.dexpreopter) 357} 358 359func (a *AndroidApp) shouldEmbedJnis(ctx android.BaseModuleContext) bool { 360 apexInfo := ctx.Provider(android.ApexInfoProvider).(android.ApexInfo) 361 return ctx.Config().UnbundledBuild() || Bool(a.appProperties.Use_embedded_native_libs) || 362 !apexInfo.IsForPlatform() || a.appProperties.AlwaysPackageNativeLibs 363} 364 365func generateAaptRenamePackageFlags(packageName string, renameResourcesPackage bool) []string { 366 aaptFlags := []string{"--rename-manifest-package " + packageName} 367 if renameResourcesPackage { 368 // Required to rename the package name in the resources table. 369 aaptFlags = append(aaptFlags, "--rename-resources-package "+packageName) 370 } 371 return aaptFlags 372} 373 374func (a *AndroidApp) OverriddenManifestPackageName() string { 375 return a.overriddenManifestPackageName 376} 377 378func (a *AndroidApp) renameResourcesPackage() bool { 379 return proptools.BoolDefault(a.overridableAppProperties.Rename_resources_package, true) 380} 381 382func (a *AndroidApp) aaptBuildActions(ctx android.ModuleContext) { 383 usePlatformAPI := proptools.Bool(a.Module.deviceProperties.Platform_apis) 384 if ctx.Module().(android.SdkContext).SdkVersion(ctx).Kind == android.SdkModule { 385 usePlatformAPI = true 386 } 387 a.aapt.usesNonSdkApis = usePlatformAPI 388 389 // Ask manifest_fixer to add or update the application element indicating this app has no code. 390 a.aapt.hasNoCode = !a.hasCode(ctx) 391 392 aaptLinkFlags := []string{} 393 394 // Add TARGET_AAPT_CHARACTERISTICS values to AAPT link flags if they exist and --product flags were not provided. 395 hasProduct := android.PrefixInList(a.aaptProperties.Aaptflags, "--product") 396 if !hasProduct && len(ctx.Config().ProductAAPTCharacteristics()) > 0 { 397 aaptLinkFlags = append(aaptLinkFlags, "--product", ctx.Config().ProductAAPTCharacteristics()) 398 } 399 400 if !Bool(a.aaptProperties.Aapt_include_all_resources) { 401 // Product AAPT config 402 for _, aaptConfig := range ctx.Config().ProductAAPTConfig() { 403 aaptLinkFlags = append(aaptLinkFlags, "-c", aaptConfig) 404 } 405 406 // Product AAPT preferred config 407 if len(ctx.Config().ProductAAPTPreferredConfig()) > 0 { 408 aaptLinkFlags = append(aaptLinkFlags, "--preferred-density", ctx.Config().ProductAAPTPreferredConfig()) 409 } 410 } 411 412 manifestPackageName, overridden := ctx.DeviceConfig().OverrideManifestPackageNameFor(ctx.ModuleName()) 413 if overridden || a.overridableAppProperties.Package_name != nil { 414 // The product override variable has a priority over the package_name property. 415 if !overridden { 416 manifestPackageName = *a.overridableAppProperties.Package_name 417 } 418 aaptLinkFlags = append(aaptLinkFlags, generateAaptRenamePackageFlags(manifestPackageName, a.renameResourcesPackage())...) 419 a.overriddenManifestPackageName = manifestPackageName 420 } 421 422 aaptLinkFlags = append(aaptLinkFlags, a.additionalAaptFlags...) 423 424 a.aapt.splitNames = a.appProperties.Package_splits 425 a.aapt.LoggingParent = String(a.overridableAppProperties.Logging_parent) 426 a.aapt.buildActions(ctx, android.SdkContext(a), a.classLoaderContexts, aaptLinkFlags...) 427 428 // apps manifests are handled by aapt, don't let Module see them 429 a.properties.Manifest = nil 430} 431 432func (a *AndroidApp) proguardBuildActions(ctx android.ModuleContext) { 433 var staticLibProguardFlagFiles android.Paths 434 ctx.VisitDirectDeps(func(m android.Module) { 435 if lib, ok := m.(AndroidLibraryDependency); ok && ctx.OtherModuleDependencyTag(m) == staticLibTag { 436 staticLibProguardFlagFiles = append(staticLibProguardFlagFiles, lib.ExportedProguardFlagFiles()...) 437 } 438 }) 439 440 staticLibProguardFlagFiles = android.FirstUniquePaths(staticLibProguardFlagFiles) 441 442 a.Module.extraProguardFlagFiles = append(a.Module.extraProguardFlagFiles, staticLibProguardFlagFiles...) 443 a.Module.extraProguardFlagFiles = append(a.Module.extraProguardFlagFiles, a.proguardOptionsFile) 444} 445 446func (a *AndroidApp) installPath(ctx android.ModuleContext) android.InstallPath { 447 var installDir string 448 if ctx.ModuleName() == "framework-res" { 449 // framework-res.apk is installed as system/framework/framework-res.apk 450 installDir = "framework" 451 } else if a.Privileged() { 452 installDir = filepath.Join("priv-app", a.installApkName) 453 } else { 454 installDir = filepath.Join("app", a.installApkName) 455 } 456 457 return android.PathForModuleInstall(ctx, installDir, a.installApkName+".apk") 458} 459 460func (a *AndroidApp) dexBuildActions(ctx android.ModuleContext) android.Path { 461 a.dexpreopter.installPath = a.installPath(ctx) 462 a.dexpreopter.isApp = true 463 if a.dexProperties.Uncompress_dex == nil { 464 // If the value was not force-set by the user, use reasonable default based on the module. 465 a.dexProperties.Uncompress_dex = proptools.BoolPtr(a.shouldUncompressDex(ctx)) 466 } 467 a.dexpreopter.uncompressedDex = *a.dexProperties.Uncompress_dex 468 a.dexpreopter.enforceUsesLibs = a.usesLibrary.enforceUsesLibraries() 469 a.dexpreopter.classLoaderContexts = a.classLoaderContexts 470 a.dexpreopter.manifestFile = a.mergedManifestFile 471 472 if ctx.ModuleName() != "framework-res" { 473 a.Module.compile(ctx, a.aaptSrcJar) 474 } 475 476 return a.dexJarFile 477} 478 479func (a *AndroidApp) jniBuildActions(jniLibs []jniLib, ctx android.ModuleContext) android.WritablePath { 480 var jniJarFile android.WritablePath 481 if len(jniLibs) > 0 { 482 a.jniLibs = jniLibs 483 if a.shouldEmbedJnis(ctx) { 484 jniJarFile = android.PathForModuleOut(ctx, "jnilibs.zip") 485 a.installPathForJNISymbols = a.installPath(ctx).ToMakePath() 486 TransformJniLibsToJar(ctx, jniJarFile, jniLibs, a.useEmbeddedNativeLibs(ctx)) 487 for _, jni := range jniLibs { 488 if jni.coverageFile.Valid() { 489 // Only collect coverage for the first target arch if this is a multilib target. 490 // TODO(jungjw): Ideally, we want to collect both reports, but that would cause coverage 491 // data file path collisions since the current coverage file path format doesn't contain 492 // arch-related strings. This is fine for now though; the code coverage team doesn't use 493 // multi-arch targets such as test_suite_* for coverage collections yet. 494 // 495 // Work with the team to come up with a new format that handles multilib modules properly 496 // and change this. 497 if len(ctx.Config().Targets[android.Android]) == 1 || 498 ctx.Config().AndroidFirstDeviceTarget.Arch.ArchType == jni.target.Arch.ArchType { 499 a.jniCoverageOutputs = append(a.jniCoverageOutputs, jni.coverageFile.Path()) 500 } 501 } 502 } 503 a.embeddedJniLibs = true 504 } 505 } 506 return jniJarFile 507} 508 509func (a *AndroidApp) JNISymbolsInstalls(installPath string) android.RuleBuilderInstalls { 510 var jniSymbols android.RuleBuilderInstalls 511 for _, jniLib := range a.jniLibs { 512 if jniLib.unstrippedFile != nil { 513 jniSymbols = append(jniSymbols, android.RuleBuilderInstall{ 514 From: jniLib.unstrippedFile, 515 To: filepath.Join(installPath, targetToJniDir(jniLib.target), jniLib.unstrippedFile.Base()), 516 }) 517 } 518 } 519 return jniSymbols 520} 521 522func (a *AndroidApp) noticeBuildActions(ctx android.ModuleContext) { 523 // Collect NOTICE files from all dependencies. 524 seenModules := make(map[android.Module]bool) 525 noticePathSet := make(map[android.Path]bool) 526 527 ctx.WalkDeps(func(child android.Module, parent android.Module) bool { 528 // Have we already seen this? 529 if _, ok := seenModules[child]; ok { 530 return false 531 } 532 seenModules[child] = true 533 534 // Skip host modules. 535 if child.Target().Os.Class == android.Host { 536 return false 537 } 538 539 paths := child.(android.Module).NoticeFiles() 540 if len(paths) > 0 { 541 for _, path := range paths { 542 noticePathSet[path] = true 543 } 544 } 545 return true 546 }) 547 548 // If the app has one, add it too. 549 if len(a.NoticeFiles()) > 0 { 550 for _, path := range a.NoticeFiles() { 551 noticePathSet[path] = true 552 } 553 } 554 555 if len(noticePathSet) == 0 { 556 return 557 } 558 var noticePaths []android.Path 559 for path := range noticePathSet { 560 noticePaths = append(noticePaths, path) 561 } 562 sort.Slice(noticePaths, func(i, j int) bool { 563 return noticePaths[i].String() < noticePaths[j].String() 564 }) 565 566 a.noticeOutputs = android.BuildNoticeOutput(ctx, a.installDir, a.installApkName+".apk", noticePaths) 567} 568 569// Reads and prepends a main cert from the default cert dir if it hasn't been set already, i.e. it 570// isn't a cert module reference. Also checks and enforces system cert restriction if applicable. 571func processMainCert(m android.ModuleBase, certPropValue string, certificates []Certificate, ctx android.ModuleContext) []Certificate { 572 if android.SrcIsModule(certPropValue) == "" { 573 var mainCert Certificate 574 if certPropValue != "" { 575 defaultDir := ctx.Config().DefaultAppCertificateDir(ctx) 576 mainCert = Certificate{ 577 Pem: defaultDir.Join(ctx, certPropValue+".x509.pem"), 578 Key: defaultDir.Join(ctx, certPropValue+".pk8"), 579 } 580 } else { 581 pem, key := ctx.Config().DefaultAppCertificate(ctx) 582 mainCert = Certificate{ 583 Pem: pem, 584 Key: key, 585 } 586 } 587 certificates = append([]Certificate{mainCert}, certificates...) 588 } 589 590 if !m.Platform() { 591 certPath := certificates[0].Pem.String() 592 systemCertPath := ctx.Config().DefaultAppCertificateDir(ctx).String() 593 if strings.HasPrefix(certPath, systemCertPath) { 594 enforceSystemCert := ctx.Config().EnforceSystemCertificate() 595 allowed := ctx.Config().EnforceSystemCertificateAllowList() 596 597 if enforceSystemCert && !inList(m.Name(), allowed) { 598 ctx.PropertyErrorf("certificate", "The module in product partition cannot be signed with certificate in system.") 599 } 600 } 601 } 602 603 return certificates 604} 605 606func (a *AndroidApp) InstallApkName() string { 607 return a.installApkName 608} 609 610func (a *AndroidApp) generateAndroidBuildActions(ctx android.ModuleContext) { 611 var apkDeps android.Paths 612 613 if !ctx.Provider(android.ApexInfoProvider).(android.ApexInfo).IsForPlatform() { 614 a.hideApexVariantFromMake = true 615 } 616 617 a.aapt.useEmbeddedNativeLibs = a.useEmbeddedNativeLibs(ctx) 618 a.aapt.useEmbeddedDex = Bool(a.appProperties.Use_embedded_dex) 619 620 // Check if the install APK name needs to be overridden. 621 a.installApkName = ctx.DeviceConfig().OverridePackageNameFor(a.Name()) 622 623 if ctx.ModuleName() == "framework-res" { 624 // framework-res.apk is installed as system/framework/framework-res.apk 625 a.installDir = android.PathForModuleInstall(ctx, "framework") 626 } else if a.Privileged() { 627 a.installDir = android.PathForModuleInstall(ctx, "priv-app", a.installApkName) 628 } else if ctx.InstallInTestcases() { 629 a.installDir = android.PathForModuleInstall(ctx, a.installApkName, ctx.DeviceConfig().DeviceArch()) 630 } else { 631 a.installDir = android.PathForModuleInstall(ctx, "app", a.installApkName) 632 } 633 a.onDeviceDir = android.InstallPathToOnDevicePath(ctx, a.installDir) 634 635 a.noticeBuildActions(ctx) 636 if Bool(a.appProperties.Embed_notices) || ctx.Config().IsEnvTrue("ALWAYS_EMBED_NOTICES") { 637 a.aapt.noticeFile = a.noticeOutputs.HtmlGzOutput 638 } 639 640 a.classLoaderContexts = a.usesLibrary.classLoaderContextForUsesLibDeps(ctx) 641 642 // Process all building blocks, from AAPT to certificates. 643 a.aaptBuildActions(ctx) 644 645 // The decision to enforce <uses-library> checks is made before adding implicit SDK libraries. 646 a.usesLibrary.freezeEnforceUsesLibraries() 647 648 // Add implicit SDK libraries to <uses-library> list. 649 for _, usesLib := range a.classLoaderContexts.UsesLibs() { 650 a.usesLibrary.addLib(usesLib, inList(usesLib, dexpreopt.OptionalCompatUsesLibs)) 651 } 652 653 // Check that the <uses-library> list is coherent with the manifest. 654 if a.usesLibrary.enforceUsesLibraries() { 655 manifestCheckFile := a.usesLibrary.verifyUsesLibrariesManifest(ctx, a.mergedManifestFile) 656 apkDeps = append(apkDeps, manifestCheckFile) 657 } 658 659 a.proguardBuildActions(ctx) 660 661 a.linter.mergedManifest = a.aapt.mergedManifestFile 662 a.linter.manifest = a.aapt.manifestPath 663 a.linter.resources = a.aapt.resourceFiles 664 a.linter.buildModuleReportZip = ctx.Config().UnbundledBuildApps() 665 666 dexJarFile := a.dexBuildActions(ctx) 667 668 jniLibs, certificateDeps := collectAppDeps(ctx, a, a.shouldEmbedJnis(ctx), !Bool(a.appProperties.Jni_uses_platform_apis)) 669 jniJarFile := a.jniBuildActions(jniLibs, ctx) 670 671 if ctx.Failed() { 672 return 673 } 674 675 certificates := processMainCert(a.ModuleBase, a.getCertString(ctx), certificateDeps, ctx) 676 a.certificate = certificates[0] 677 678 // Build a final signed app package. 679 packageFile := android.PathForModuleOut(ctx, a.installApkName+".apk") 680 v4SigningRequested := Bool(a.Module.deviceProperties.V4_signature) 681 var v4SignatureFile android.WritablePath = nil 682 if v4SigningRequested { 683 v4SignatureFile = android.PathForModuleOut(ctx, a.installApkName+".apk.idsig") 684 } 685 var lineageFile android.Path 686 if lineage := String(a.overridableAppProperties.Lineage); lineage != "" { 687 lineageFile = android.PathForModuleSrc(ctx, lineage) 688 } 689 CreateAndSignAppPackage(ctx, packageFile, a.exportPackage, jniJarFile, dexJarFile, certificates, apkDeps, v4SignatureFile, lineageFile) 690 a.outputFile = packageFile 691 if v4SigningRequested { 692 a.extraOutputFiles = append(a.extraOutputFiles, v4SignatureFile) 693 } 694 695 for _, split := range a.aapt.splits { 696 // Sign the split APKs 697 packageFile := android.PathForModuleOut(ctx, a.installApkName+"_"+split.suffix+".apk") 698 if v4SigningRequested { 699 v4SignatureFile = android.PathForModuleOut(ctx, a.installApkName+"_"+split.suffix+".apk.idsig") 700 } 701 CreateAndSignAppPackage(ctx, packageFile, split.path, nil, nil, certificates, apkDeps, v4SignatureFile, lineageFile) 702 a.extraOutputFiles = append(a.extraOutputFiles, packageFile) 703 if v4SigningRequested { 704 a.extraOutputFiles = append(a.extraOutputFiles, v4SignatureFile) 705 } 706 } 707 708 // Build an app bundle. 709 bundleFile := android.PathForModuleOut(ctx, "base.zip") 710 BuildBundleModule(ctx, bundleFile, a.exportPackage, jniJarFile, dexJarFile) 711 a.bundleFile = bundleFile 712 713 apexInfo := ctx.Provider(android.ApexInfoProvider).(android.ApexInfo) 714 715 // Install the app package. 716 if (Bool(a.Module.properties.Installable) || ctx.Host()) && apexInfo.IsForPlatform() { 717 ctx.InstallFile(a.installDir, a.outputFile.Base(), a.outputFile) 718 for _, extra := range a.extraOutputFiles { 719 ctx.InstallFile(a.installDir, extra.Base(), extra) 720 } 721 } 722 723 a.buildAppDependencyInfo(ctx) 724} 725 726type appDepsInterface interface { 727 SdkVersion(ctx android.EarlyModuleContext) android.SdkSpec 728 MinSdkVersion(ctx android.EarlyModuleContext) android.SdkSpec 729 RequiresStableAPIs(ctx android.BaseModuleContext) bool 730} 731 732func collectAppDeps(ctx android.ModuleContext, app appDepsInterface, 733 shouldCollectRecursiveNativeDeps bool, 734 checkNativeSdkVersion bool) ([]jniLib, []Certificate) { 735 736 var jniLibs []jniLib 737 var certificates []Certificate 738 seenModulePaths := make(map[string]bool) 739 740 if checkNativeSdkVersion { 741 checkNativeSdkVersion = app.SdkVersion(ctx).Specified() && 742 app.SdkVersion(ctx).Kind != android.SdkCorePlatform && !app.RequiresStableAPIs(ctx) 743 } 744 745 ctx.WalkDeps(func(module android.Module, parent android.Module) bool { 746 otherName := ctx.OtherModuleName(module) 747 tag := ctx.OtherModuleDependencyTag(module) 748 749 if IsJniDepTag(tag) || cc.IsSharedDepTag(tag) { 750 if dep, ok := module.(*cc.Module); ok { 751 if dep.IsNdk(ctx.Config()) || dep.IsStubs() { 752 return false 753 } 754 755 lib := dep.OutputFile() 756 path := lib.Path() 757 if seenModulePaths[path.String()] { 758 return false 759 } 760 seenModulePaths[path.String()] = true 761 762 if checkNativeSdkVersion && dep.SdkVersion() == "" { 763 ctx.PropertyErrorf("jni_libs", "JNI dependency %q uses platform APIs, but this module does not", 764 otherName) 765 } 766 767 if lib.Valid() { 768 jniLibs = append(jniLibs, jniLib{ 769 name: ctx.OtherModuleName(module), 770 path: path, 771 target: module.Target(), 772 coverageFile: dep.CoverageOutputFile(), 773 unstrippedFile: dep.UnstrippedOutputFile(), 774 }) 775 } else { 776 ctx.ModuleErrorf("dependency %q missing output file", otherName) 777 } 778 } else { 779 ctx.ModuleErrorf("jni_libs dependency %q must be a cc library", otherName) 780 } 781 782 return shouldCollectRecursiveNativeDeps 783 } 784 785 if tag == certificateTag { 786 if dep, ok := module.(*AndroidAppCertificate); ok { 787 certificates = append(certificates, dep.Certificate) 788 } else { 789 ctx.ModuleErrorf("certificate dependency %q must be an android_app_certificate module", otherName) 790 } 791 } 792 793 return false 794 }) 795 796 return jniLibs, certificates 797} 798 799func (a *AndroidApp) WalkPayloadDeps(ctx android.ModuleContext, do android.PayloadDepsCallback) { 800 ctx.WalkDeps(func(child, parent android.Module) bool { 801 isExternal := !a.DepIsInSameApex(ctx, child) 802 if am, ok := child.(android.ApexModule); ok { 803 if !do(ctx, parent, am, isExternal) { 804 return false 805 } 806 } 807 return !isExternal 808 }) 809} 810 811func (a *AndroidApp) buildAppDependencyInfo(ctx android.ModuleContext) { 812 if ctx.Host() { 813 return 814 } 815 816 depsInfo := android.DepNameToDepInfoMap{} 817 a.WalkPayloadDeps(ctx, func(ctx android.ModuleContext, from blueprint.Module, to android.ApexModule, externalDep bool) bool { 818 depName := to.Name() 819 820 // Skip dependencies that are only available to APEXes; they are developed with updatability 821 // in mind and don't need manual approval. 822 if to.(android.ApexModule).NotAvailableForPlatform() { 823 return true 824 } 825 826 if info, exist := depsInfo[depName]; exist { 827 info.From = append(info.From, from.Name()) 828 info.IsExternal = info.IsExternal && externalDep 829 depsInfo[depName] = info 830 } else { 831 toMinSdkVersion := "(no version)" 832 if m, ok := to.(interface { 833 MinSdkVersion(ctx android.EarlyModuleContext) android.SdkSpec 834 }); ok { 835 if v := m.MinSdkVersion(ctx); !v.ApiLevel.IsNone() { 836 toMinSdkVersion = v.ApiLevel.String() 837 } 838 } else if m, ok := to.(interface{ MinSdkVersion() string }); ok { 839 // TODO(b/175678607) eliminate the use of MinSdkVersion returning 840 // string 841 if v := m.MinSdkVersion(); v != "" { 842 toMinSdkVersion = v 843 } 844 } 845 depsInfo[depName] = android.ApexModuleDepInfo{ 846 To: depName, 847 From: []string{from.Name()}, 848 IsExternal: externalDep, 849 MinSdkVersion: toMinSdkVersion, 850 } 851 } 852 return true 853 }) 854 855 a.ApexBundleDepsInfo.BuildDepsInfoLists(ctx, a.MinSdkVersion(ctx).String(), depsInfo) 856} 857 858func (a *AndroidApp) Updatable() bool { 859 return Bool(a.appProperties.Updatable) 860} 861 862func (a *AndroidApp) getCertString(ctx android.BaseModuleContext) string { 863 certificate, overridden := ctx.DeviceConfig().OverrideCertificateFor(ctx.ModuleName()) 864 if overridden { 865 return ":" + certificate 866 } 867 return String(a.overridableAppProperties.Certificate) 868} 869 870func (a *AndroidApp) DepIsInSameApex(ctx android.BaseModuleContext, dep android.Module) bool { 871 if IsJniDepTag(ctx.OtherModuleDependencyTag(dep)) { 872 return true 873 } 874 return a.Library.DepIsInSameApex(ctx, dep) 875} 876 877// For OutputFileProducer interface 878func (a *AndroidApp) OutputFiles(tag string) (android.Paths, error) { 879 switch tag { 880 case ".aapt.srcjar": 881 return []android.Path{a.aaptSrcJar}, nil 882 case ".export-package.apk": 883 return []android.Path{a.exportPackage}, nil 884 } 885 return a.Library.OutputFiles(tag) 886} 887 888func (a *AndroidApp) Privileged() bool { 889 return Bool(a.appProperties.Privileged) 890} 891 892func (a *AndroidApp) IsNativeCoverageNeeded(ctx android.BaseModuleContext) bool { 893 return ctx.Device() && ctx.DeviceConfig().NativeCoverageEnabled() 894} 895 896func (a *AndroidApp) SetPreventInstall() { 897 a.appProperties.PreventInstall = true 898} 899 900func (a *AndroidApp) HideFromMake() { 901 a.appProperties.HideFromMake = true 902} 903 904func (a *AndroidApp) MarkAsCoverageVariant(coverage bool) { 905 a.appProperties.IsCoverageVariant = coverage 906} 907 908func (a *AndroidApp) EnableCoverageIfNeeded() {} 909 910var _ cc.Coverage = (*AndroidApp)(nil) 911 912// android_app compiles sources and Android resources into an Android application package `.apk` file. 913func AndroidAppFactory() android.Module { 914 module := &AndroidApp{} 915 916 module.Module.dexProperties.Optimize.EnabledByDefault = true 917 module.Module.dexProperties.Optimize.Shrink = proptools.BoolPtr(true) 918 919 module.Module.properties.Instrument = true 920 module.Module.properties.Installable = proptools.BoolPtr(true) 921 922 module.addHostAndDeviceProperties() 923 module.AddProperties( 924 &module.aaptProperties, 925 &module.appProperties, 926 &module.overridableAppProperties) 927 928 module.usesLibrary.enforce = true 929 930 android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibCommon) 931 android.InitDefaultableModule(module) 932 android.InitOverridableModule(module, &module.appProperties.Overrides) 933 android.InitApexModule(module) 934 935 return module 936} 937 938type appTestProperties struct { 939 // The name of the android_app module that the tests will run against. 940 Instrumentation_for *string 941 942 // if specified, the instrumentation target package name in the manifest is overwritten by it. 943 Instrumentation_target_package *string 944} 945 946type AndroidTest struct { 947 AndroidApp 948 949 appTestProperties appTestProperties 950 951 testProperties testProperties 952 953 testConfig android.Path 954 extraTestConfigs android.Paths 955 data android.Paths 956} 957 958func (a *AndroidTest) InstallInTestcases() bool { 959 return true 960} 961 962func (a *AndroidTest) GenerateAndroidBuildActions(ctx android.ModuleContext) { 963 var configs []tradefed.Config 964 if a.appTestProperties.Instrumentation_target_package != nil { 965 a.additionalAaptFlags = append(a.additionalAaptFlags, 966 "--rename-instrumentation-target-package "+*a.appTestProperties.Instrumentation_target_package) 967 } else if a.appTestProperties.Instrumentation_for != nil { 968 // Check if the instrumentation target package is overridden. 969 manifestPackageName, overridden := ctx.DeviceConfig().OverrideManifestPackageNameFor(*a.appTestProperties.Instrumentation_for) 970 if overridden { 971 a.additionalAaptFlags = append(a.additionalAaptFlags, "--rename-instrumentation-target-package "+manifestPackageName) 972 } 973 } 974 a.generateAndroidBuildActions(ctx) 975 976 for _, module := range a.testProperties.Test_mainline_modules { 977 configs = append(configs, tradefed.Option{Name: "config-descriptor:metadata", Key: "mainline-param", Value: module}) 978 } 979 980 testConfig := tradefed.AutoGenInstrumentationTestConfig(ctx, a.testProperties.Test_config, 981 a.testProperties.Test_config_template, a.manifestPath, a.testProperties.Test_suites, a.testProperties.Auto_gen_config, configs) 982 a.testConfig = a.FixTestConfig(ctx, testConfig) 983 a.extraTestConfigs = android.PathsForModuleSrc(ctx, a.testProperties.Test_options.Extra_test_configs) 984 a.data = android.PathsForModuleSrc(ctx, a.testProperties.Data) 985} 986 987func (a *AndroidTest) FixTestConfig(ctx android.ModuleContext, testConfig android.Path) android.Path { 988 if testConfig == nil { 989 return nil 990 } 991 992 fixedConfig := android.PathForModuleOut(ctx, "test_config_fixer", "AndroidTest.xml") 993 rule := android.NewRuleBuilder(pctx, ctx) 994 command := rule.Command().BuiltTool("test_config_fixer").Input(testConfig).Output(fixedConfig) 995 fixNeeded := false 996 997 if ctx.ModuleName() != a.installApkName { 998 fixNeeded = true 999 command.FlagWithArg("--test-file-name ", a.installApkName+".apk") 1000 } 1001 1002 if a.overridableAppProperties.Package_name != nil { 1003 fixNeeded = true 1004 command.FlagWithInput("--manifest ", a.manifestPath). 1005 FlagWithArg("--package-name ", *a.overridableAppProperties.Package_name) 1006 } 1007 1008 if fixNeeded { 1009 rule.Build("fix_test_config", "fix test config") 1010 return fixedConfig 1011 } 1012 return testConfig 1013} 1014 1015func (a *AndroidTest) DepsMutator(ctx android.BottomUpMutatorContext) { 1016 a.AndroidApp.DepsMutator(ctx) 1017} 1018 1019func (a *AndroidTest) OverridablePropertiesDepsMutator(ctx android.BottomUpMutatorContext) { 1020 a.AndroidApp.OverridablePropertiesDepsMutator(ctx) 1021 if a.appTestProperties.Instrumentation_for != nil { 1022 // The android_app dependency listed in instrumentation_for needs to be added to the classpath for javac, 1023 // but not added to the aapt2 link includes like a normal android_app or android_library dependency, so 1024 // use instrumentationForTag instead of libTag. 1025 ctx.AddVariationDependencies(nil, instrumentationForTag, String(a.appTestProperties.Instrumentation_for)) 1026 } 1027} 1028 1029// android_test compiles test sources and Android resources into an Android application package `.apk` file and 1030// creates an `AndroidTest.xml` file to allow running the test with `atest` or a `TEST_MAPPING` file. 1031func AndroidTestFactory() android.Module { 1032 module := &AndroidTest{} 1033 1034 module.Module.dexProperties.Optimize.EnabledByDefault = true 1035 1036 module.Module.properties.Instrument = true 1037 module.Module.properties.Installable = proptools.BoolPtr(true) 1038 module.appProperties.Use_embedded_native_libs = proptools.BoolPtr(true) 1039 module.appProperties.AlwaysPackageNativeLibs = true 1040 module.Module.dexpreopter.isTest = true 1041 module.Module.linter.test = true 1042 1043 module.addHostAndDeviceProperties() 1044 module.AddProperties( 1045 &module.aaptProperties, 1046 &module.appProperties, 1047 &module.appTestProperties, 1048 &module.overridableAppProperties, 1049 &module.testProperties) 1050 1051 android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibCommon) 1052 android.InitDefaultableModule(module) 1053 android.InitOverridableModule(module, &module.appProperties.Overrides) 1054 return module 1055} 1056 1057type appTestHelperAppProperties struct { 1058 // list of compatibility suites (for example "cts", "vts") that the module should be 1059 // installed into. 1060 Test_suites []string `android:"arch_variant"` 1061 1062 // Flag to indicate whether or not to create test config automatically. If AndroidTest.xml 1063 // doesn't exist next to the Android.bp, this attribute doesn't need to be set to true 1064 // explicitly. 1065 Auto_gen_config *bool 1066} 1067 1068type AndroidTestHelperApp struct { 1069 AndroidApp 1070 1071 appTestHelperAppProperties appTestHelperAppProperties 1072} 1073 1074func (a *AndroidTestHelperApp) InstallInTestcases() bool { 1075 return true 1076} 1077 1078// android_test_helper_app compiles sources and Android resources into an Android application package `.apk` file that 1079// will be used by tests, but does not produce an `AndroidTest.xml` file so the module will not be run directly as a 1080// test. 1081func AndroidTestHelperAppFactory() android.Module { 1082 module := &AndroidTestHelperApp{} 1083 1084 module.Module.dexProperties.Optimize.EnabledByDefault = true 1085 1086 module.Module.properties.Installable = proptools.BoolPtr(true) 1087 module.appProperties.Use_embedded_native_libs = proptools.BoolPtr(true) 1088 module.appProperties.AlwaysPackageNativeLibs = true 1089 module.Module.dexpreopter.isTest = true 1090 module.Module.linter.test = true 1091 1092 module.addHostAndDeviceProperties() 1093 module.AddProperties( 1094 &module.aaptProperties, 1095 &module.appProperties, 1096 &module.appTestHelperAppProperties, 1097 &module.overridableAppProperties) 1098 1099 android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibCommon) 1100 android.InitDefaultableModule(module) 1101 android.InitApexModule(module) 1102 return module 1103} 1104 1105type AndroidAppCertificate struct { 1106 android.ModuleBase 1107 properties AndroidAppCertificateProperties 1108 Certificate Certificate 1109} 1110 1111type AndroidAppCertificateProperties struct { 1112 // Name of the certificate files. Extensions .x509.pem and .pk8 will be added to the name. 1113 Certificate *string 1114} 1115 1116// android_app_certificate modules can be referenced by the certificates property of android_app modules to select 1117// the signing key. 1118func AndroidAppCertificateFactory() android.Module { 1119 module := &AndroidAppCertificate{} 1120 module.AddProperties(&module.properties) 1121 android.InitAndroidModule(module) 1122 return module 1123} 1124 1125func (c *AndroidAppCertificate) GenerateAndroidBuildActions(ctx android.ModuleContext) { 1126 cert := String(c.properties.Certificate) 1127 c.Certificate = Certificate{ 1128 Pem: android.PathForModuleSrc(ctx, cert+".x509.pem"), 1129 Key: android.PathForModuleSrc(ctx, cert+".pk8"), 1130 } 1131} 1132 1133type OverrideAndroidApp struct { 1134 android.ModuleBase 1135 android.OverrideModuleBase 1136} 1137 1138func (i *OverrideAndroidApp) GenerateAndroidBuildActions(_ android.ModuleContext) { 1139 // All the overrides happen in the base module. 1140 // TODO(jungjw): Check the base module type. 1141} 1142 1143// override_android_app is used to create an android_app module based on another android_app by overriding 1144// some of its properties. 1145func OverrideAndroidAppModuleFactory() android.Module { 1146 m := &OverrideAndroidApp{} 1147 m.AddProperties(&overridableAppProperties{}) 1148 1149 android.InitAndroidMultiTargetsArchModule(m, android.DeviceSupported, android.MultilibCommon) 1150 android.InitOverrideModule(m) 1151 return m 1152} 1153 1154type OverrideAndroidTest struct { 1155 android.ModuleBase 1156 android.OverrideModuleBase 1157} 1158 1159func (i *OverrideAndroidTest) GenerateAndroidBuildActions(_ android.ModuleContext) { 1160 // All the overrides happen in the base module. 1161 // TODO(jungjw): Check the base module type. 1162} 1163 1164// override_android_test is used to create an android_app module based on another android_test by overriding 1165// some of its properties. 1166func OverrideAndroidTestModuleFactory() android.Module { 1167 m := &OverrideAndroidTest{} 1168 m.AddProperties(&overridableAppProperties{}) 1169 m.AddProperties(&appTestProperties{}) 1170 1171 android.InitAndroidMultiTargetsArchModule(m, android.DeviceSupported, android.MultilibCommon) 1172 android.InitOverrideModule(m) 1173 return m 1174} 1175 1176type UsesLibraryProperties struct { 1177 // A list of shared library modules that will be listed in uses-library tags in the AndroidManifest.xml file. 1178 Uses_libs []string 1179 1180 // A list of shared library modules that will be listed in uses-library tags in the AndroidManifest.xml file with 1181 // required=false. 1182 Optional_uses_libs []string 1183 1184 // If true, the list of uses_libs and optional_uses_libs modules must match the AndroidManifest.xml file. Defaults 1185 // to true if either uses_libs or optional_uses_libs is set. Will unconditionally default to true in the future. 1186 Enforce_uses_libs *bool 1187 1188 // Optional name of the <uses-library> provided by this module. This is needed for non-SDK 1189 // libraries, because SDK ones are automatically picked up by Soong. The <uses-library> name 1190 // normally is the same as the module name, but there are exceptions. 1191 Provides_uses_lib *string 1192} 1193 1194// usesLibrary provides properties and helper functions for AndroidApp and AndroidAppImport to verify that the 1195// <uses-library> tags that end up in the manifest of an APK match the ones known to the build system through the 1196// uses_libs and optional_uses_libs properties. The build system's values are used by dexpreopt to preopt apps 1197// with knowledge of their shared libraries. 1198type usesLibrary struct { 1199 usesLibraryProperties UsesLibraryProperties 1200 1201 // Whether to enforce verify_uses_library check. 1202 enforce bool 1203} 1204 1205func (u *usesLibrary) addLib(lib string, optional bool) { 1206 if !android.InList(lib, u.usesLibraryProperties.Uses_libs) && !android.InList(lib, u.usesLibraryProperties.Optional_uses_libs) { 1207 if optional { 1208 u.usesLibraryProperties.Optional_uses_libs = append(u.usesLibraryProperties.Optional_uses_libs, lib) 1209 } else { 1210 u.usesLibraryProperties.Uses_libs = append(u.usesLibraryProperties.Uses_libs, lib) 1211 } 1212 } 1213} 1214 1215func (u *usesLibrary) deps(ctx android.BottomUpMutatorContext, hasFrameworkLibs bool) { 1216 if !ctx.Config().UnbundledBuild() { 1217 ctx.AddVariationDependencies(nil, usesLibTag, u.usesLibraryProperties.Uses_libs...) 1218 ctx.AddVariationDependencies(nil, usesLibTag, u.presentOptionalUsesLibs(ctx)...) 1219 // Only add these extra dependencies if the module depends on framework libs. This avoids 1220 // creating a cyclic dependency: 1221 // e.g. framework-res -> org.apache.http.legacy -> ... -> framework-res. 1222 if hasFrameworkLibs { 1223 // Dexpreopt needs paths to the dex jars of these libraries in order to construct 1224 // class loader context for dex2oat. Add them as a dependency with a special tag. 1225 ctx.AddVariationDependencies(nil, usesLibCompat29Tag, dexpreopt.CompatUsesLibs29...) 1226 ctx.AddVariationDependencies(nil, usesLibCompat28Tag, dexpreopt.OptionalCompatUsesLibs28...) 1227 ctx.AddVariationDependencies(nil, usesLibCompat30Tag, dexpreopt.OptionalCompatUsesLibs30...) 1228 } 1229 } 1230} 1231 1232// presentOptionalUsesLibs returns optional_uses_libs after filtering out MissingUsesLibraries, which don't exist in the 1233// build. 1234func (u *usesLibrary) presentOptionalUsesLibs(ctx android.BaseModuleContext) []string { 1235 optionalUsesLibs, _ := android.FilterList(u.usesLibraryProperties.Optional_uses_libs, ctx.Config().MissingUsesLibraries()) 1236 return optionalUsesLibs 1237} 1238 1239// Helper function to replace string in a list. 1240func replaceInList(list []string, oldstr, newstr string) { 1241 for i, str := range list { 1242 if str == oldstr { 1243 list[i] = newstr 1244 } 1245 } 1246} 1247 1248// Returns a map of module names of shared library dependencies to the paths 1249// to their dex jars on host and on device. 1250func (u *usesLibrary) classLoaderContextForUsesLibDeps(ctx android.ModuleContext) dexpreopt.ClassLoaderContextMap { 1251 clcMap := make(dexpreopt.ClassLoaderContextMap) 1252 1253 if !ctx.Config().UnbundledBuild() { 1254 ctx.VisitDirectDeps(func(m android.Module) { 1255 if tag, ok := ctx.OtherModuleDependencyTag(m).(usesLibraryDependencyTag); ok { 1256 dep := ctx.OtherModuleName(m) 1257 if lib, ok := m.(UsesLibraryDependency); ok { 1258 libName := dep 1259 if ulib, ok := m.(ProvidesUsesLib); ok && ulib.ProvidesUsesLib() != nil { 1260 libName = *ulib.ProvidesUsesLib() 1261 // Replace module name with library name in `uses_libs`/`optional_uses_libs` 1262 // in order to pass verify_uses_libraries check (which compares these 1263 // properties against library names written in the manifest). 1264 replaceInList(u.usesLibraryProperties.Uses_libs, dep, libName) 1265 replaceInList(u.usesLibraryProperties.Optional_uses_libs, dep, libName) 1266 } 1267 clcMap.AddContext(ctx, tag.sdkVersion, libName, 1268 lib.DexJarBuildPath(), lib.DexJarInstallPath(), lib.ClassLoaderContexts()) 1269 } else if ctx.Config().AllowMissingDependencies() { 1270 ctx.AddMissingDependencies([]string{dep}) 1271 } else { 1272 ctx.ModuleErrorf("module %q in uses_libs or optional_uses_libs must be a java library", dep) 1273 } 1274 } 1275 }) 1276 } 1277 1278 return clcMap 1279} 1280 1281// enforceUsesLibraries returns true of <uses-library> tags should be checked against uses_libs and optional_uses_libs 1282// properties. Defaults to true if either of uses_libs or optional_uses_libs is specified. Will default to true 1283// unconditionally in the future. 1284func (u *usesLibrary) enforceUsesLibraries() bool { 1285 defaultEnforceUsesLibs := len(u.usesLibraryProperties.Uses_libs) > 0 || 1286 len(u.usesLibraryProperties.Optional_uses_libs) > 0 1287 return BoolDefault(u.usesLibraryProperties.Enforce_uses_libs, u.enforce || defaultEnforceUsesLibs) 1288} 1289 1290// Freeze the value of `enforce_uses_libs` based on the current values of `uses_libs` and `optional_uses_libs`. 1291func (u *usesLibrary) freezeEnforceUsesLibraries() { 1292 enforce := u.enforceUsesLibraries() 1293 u.usesLibraryProperties.Enforce_uses_libs = &enforce 1294} 1295 1296// verifyUsesLibraries checks the <uses-library> tags in the manifest against the ones specified 1297// in the `uses_libs`/`optional_uses_libs` properties. The input can be either an XML manifest, or 1298// an APK with the manifest embedded in it (manifest_check will know which one it is by the file 1299// extension: APKs are supposed to end with '.apk'). 1300func (u *usesLibrary) verifyUsesLibraries(ctx android.ModuleContext, inputFile android.Path, 1301 outputFile android.WritablePath) android.Path { 1302 1303 statusFile := dexpreopt.UsesLibrariesStatusFile(ctx) 1304 1305 // Disable verify_uses_libraries check if dexpreopt is globally disabled. Without dexpreopt the 1306 // check is not necessary, and although it is good to have, it is difficult to maintain on 1307 // non-linux build platforms where dexpreopt is generally disabled (the check may fail due to 1308 // various unrelated reasons, such as a failure to get manifest from an APK). 1309 global := dexpreopt.GetGlobalConfig(ctx) 1310 if global.DisablePreopt || global.OnlyPreoptBootImageAndSystemServer { 1311 return inputFile 1312 } 1313 1314 rule := android.NewRuleBuilder(pctx, ctx) 1315 cmd := rule.Command().BuiltTool("manifest_check"). 1316 Flag("--enforce-uses-libraries"). 1317 Input(inputFile). 1318 FlagWithOutput("--enforce-uses-libraries-status ", statusFile). 1319 FlagWithInput("--aapt ", ctx.Config().HostToolPath(ctx, "aapt")) 1320 1321 if outputFile != nil { 1322 cmd.FlagWithOutput("-o ", outputFile) 1323 } 1324 1325 if dexpreopt.GetGlobalConfig(ctx).RelaxUsesLibraryCheck { 1326 cmd.Flag("--enforce-uses-libraries-relax") 1327 } 1328 1329 for _, lib := range u.usesLibraryProperties.Uses_libs { 1330 cmd.FlagWithArg("--uses-library ", lib) 1331 } 1332 1333 for _, lib := range u.usesLibraryProperties.Optional_uses_libs { 1334 cmd.FlagWithArg("--optional-uses-library ", lib) 1335 } 1336 1337 rule.Build("verify_uses_libraries", "verify <uses-library>") 1338 return outputFile 1339} 1340 1341// verifyUsesLibrariesManifest checks the <uses-library> tags in an AndroidManifest.xml against 1342// the build system and returns the path to a copy of the manifest. 1343func (u *usesLibrary) verifyUsesLibrariesManifest(ctx android.ModuleContext, manifest android.Path) android.Path { 1344 outputFile := android.PathForModuleOut(ctx, "manifest_check", "AndroidManifest.xml") 1345 return u.verifyUsesLibraries(ctx, manifest, outputFile) 1346} 1347 1348// verifyUsesLibrariesAPK checks the <uses-library> tags in the manifest of an APK against the build 1349// system and returns the path to a copy of the APK. 1350func (u *usesLibrary) verifyUsesLibrariesAPK(ctx android.ModuleContext, apk android.Path) android.Path { 1351 u.verifyUsesLibraries(ctx, apk, nil) // for APKs manifest_check does not write output file 1352 outputFile := android.PathForModuleOut(ctx, "verify_uses_libraries", apk.Base()) 1353 return outputFile 1354} 1355