blob: 1e5c95a7f2a9bb118fbb8827375495b9550dfe21 [file] [log] [blame]
Colin Crossfabb6082018-02-20 17:22:23 -08001// Copyright 2018 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 (
Cole Faust6265120c2024-09-05 16:27:56 -070018 "crypto/sha256"
Colin Crossa592e3e2019-02-19 16:59:53 -080019 "fmt"
Jaewoong Jung5b425e22019-06-17 17:40:56 -070020 "path/filepath"
Colin Cross312634e2023-11-21 15:13:56 -080021 "slices"
Colin Crossa97c5d32018-03-28 14:58:31 -070022 "strings"
Colin Crossfabb6082018-02-20 17:22:23 -080023
Jaewoong Jung9befb0c2020-01-18 10:33:43 -080024 "android/soong/android"
Ulya Trafimovich31e444e2020-08-14 17:32:16 +010025 "android/soong/dexpreopt"
Jihoon Kangfe914ed2024-02-12 22:49:21 +000026
Colin Crossfabb6082018-02-20 17:22:23 -080027 "github.com/google/blueprint"
Colin Crossa14fb6a2024-10-23 16:57:06 -070028 "github.com/google/blueprint/depset"
Colin Crossa97c5d32018-03-28 14:58:31 -070029 "github.com/google/blueprint/proptools"
Colin Crossfabb6082018-02-20 17:22:23 -080030)
31
Colin Crossa97c5d32018-03-28 14:58:31 -070032type AndroidLibraryDependency interface {
Colin Crossa97c5d32018-03-28 14:58:31 -070033 ExportPackage() android.Path
Colin Crossa14fb6a2024-10-23 16:57:06 -070034 ResourcesNodeDepSet() depset.DepSet[*resourcesNode]
35 RRODirsDepSet() depset.DepSet[rroDir]
36 ManifestsDepSet() depset.DepSet[android.Path]
Jaewoong Jungc779cd42020-10-06 18:56:10 -070037 SetRROEnforcedForDependent(enforce bool)
38 IsRROEnforced(ctx android.BaseModuleContext) bool
Colin Crossa97c5d32018-03-28 14:58:31 -070039}
40
41func init() {
Paul Duffinf9b1da02019-12-18 19:51:55 +000042 RegisterAARBuildComponents(android.InitRegistrationContext)
43}
44
45func RegisterAARBuildComponents(ctx android.RegistrationContext) {
46 ctx.RegisterModuleType("android_library_import", AARImportFactory)
47 ctx.RegisterModuleType("android_library", AndroidLibraryFactory)
Paul Duffin04ba70d2021-03-22 13:56:43 +000048 ctx.PostDepsMutators(func(ctx android.RegisterMutatorsContext) {
Colin Crossc4441622024-09-18 14:55:49 -070049 ctx.Transition("propagate_rro_enforcement", &propagateRROEnforcementTransitionMutator{})
Paul Duffin04ba70d2021-03-22 13:56:43 +000050 })
Colin Crossa97c5d32018-03-28 14:58:31 -070051}
52
53//
54// AAR (android library)
55//
56
57type androidLibraryProperties struct {
58 BuildAAR bool `blueprint:"mutated"`
59}
60
61type aaptProperties struct {
62 // flags passed to aapt when creating the apk
63 Aaptflags []string
64
Dan Willemsen72be5902018-10-24 20:24:57 -070065 // include all resource configurations, not just the product-configured
66 // ones.
67 Aapt_include_all_resources *bool
68
Jiakai Zhangba82e282023-10-13 18:08:59 +010069 // list of files to use as assets.
70 Assets []string `android:"path"`
71
Colin Crossa97c5d32018-03-28 14:58:31 -070072 // list of directories relative to the Blueprints file containing assets.
Colin Cross0ddae7f2019-02-07 15:30:01 -080073 // Defaults to ["assets"] if a directory called assets exists. Set to []
74 // to disable the default.
Colin Crossa97c5d32018-03-28 14:58:31 -070075 Asset_dirs []string
76
77 // list of directories relative to the Blueprints file containing
Colin Cross0ddae7f2019-02-07 15:30:01 -080078 // Android resources. Defaults to ["res"] if a directory called res exists.
79 // Set to [] to disable the default.
Cole Faust5b989772024-11-06 11:03:16 -080080 Resource_dirs proptools.Configurable[[]string] `android:"path"`
Colin Crossa97c5d32018-03-28 14:58:31 -070081
Colin Crossa592e3e2019-02-19 16:59:53 -080082 // list of zip files containing Android resources.
Colin Cross27b922f2019-03-04 22:35:41 -080083 Resource_zips []string `android:"path"`
Colin Crossa592e3e2019-02-19 16:59:53 -080084
Colin Crossa97c5d32018-03-28 14:58:31 -070085 // path to AndroidManifest.xml. If unset, defaults to "AndroidManifest.xml".
Colin Cross27b922f2019-03-04 22:35:41 -080086 Manifest *string `android:"path"`
changho.shinb5432b72019-08-08 18:37:17 +090087
88 // paths to additional manifest files to merge with main manifest.
89 Additional_manifests []string `android:"path"`
Sasha Smundak541056c2019-10-28 15:50:06 -070090
91 // do not include AndroidManifest from dependent libraries
92 Dont_merge_manifests *bool
Jaewoong Jungc779cd42020-10-06 18:56:10 -070093
Colin Cross4eae06d2023-06-20 22:40:02 -070094 // If use_resource_processor is set, use Bazel's resource processor instead of aapt2 to generate R.class files.
95 // The resource processor produces more optimal R.class files that only list resources in the package of the
96 // library that provided them, as opposed to aapt2 which produces R.java files for every package containing
97 // every resource. Using the resource processor can provide significant build time speedups, but requires
98 // fixing the module to use the correct package to reference each resource, and to avoid having any other
99 // libraries in the tree that use the same package name. Defaults to false, but will default to true in the
100 // future.
101 Use_resource_processor *bool
102
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700103 // true if RRO is enforced for any of the dependent modules
104 RROEnforcedForDependent bool `blueprint:"mutated"`
Inseob Kim34dc4cd2023-11-07 13:37:14 +0900105
106 // Filter only specified product and ignore other products
107 Filter_product *string `blueprint:"mutated"`
Jihoon Kang9049c272024-03-19 21:57:36 +0000108
109 // Names of aconfig_declarations modules that specify aconfig flags that the module depends on.
110 Flags_packages []string
Colin Crossa97c5d32018-03-28 14:58:31 -0700111}
112
113type aapt struct {
Colin Cross312634e2023-11-21 15:13:56 -0800114 aaptSrcJar android.Path
115 transitiveAaptRJars android.Paths
116 transitiveAaptResourcePackagesFile android.Path
117 exportPackage android.Path
118 manifestPath android.Path
119 proguardOptionsFile android.Path
120 rTxt android.Path
121 rJar android.Path
122 extraAaptPackagesFile android.Path
123 mergedManifestFile android.Path
124 noticeFile android.OptionalPath
125 assetPackage android.OptionalPath
126 isLibrary bool
127 defaultManifestVersion string
128 useEmbeddedNativeLibs bool
129 useEmbeddedDex bool
130 usesNonSdkApis bool
131 hasNoCode bool
132 LoggingParent string
133 resourceFiles android.Paths
Colin Crossa97c5d32018-03-28 14:58:31 -0700134
Colin Crosse560c4a2019-03-19 16:03:11 -0700135 splitNames []string
136 splits []split
137
Colin Crossa97c5d32018-03-28 14:58:31 -0700138 aaptProperties aaptProperties
Colin Crossab8d1382023-07-14 17:23:41 +0000139
Colin Crossa14fb6a2024-10-23 16:57:06 -0700140 resourcesNodesDepSet depset.DepSet[*resourcesNode]
141 rroDirsDepSet depset.DepSet[rroDir]
142 manifestsDepSet depset.DepSet[android.Path]
Alix96ea88452023-08-31 15:48:23 +0000143
144 manifestValues struct {
145 applicationId string
146 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700147}
148
Colin Crosse560c4a2019-03-19 16:03:11 -0700149type split struct {
150 name string
151 suffix string
152 path android.Path
153}
154
Colin Crossc4441622024-09-18 14:55:49 -0700155// Propagate RRO enforcement flag to static lib dependencies transitively. If EnforceRROGlobally is set then
156// all modules will use the "" variant. If specific modules have RRO enforced, then modules (usually apps) with
157// RRO enabled will use the "" variation for themselves, but use the "rro" variant of direct and transitive static
158// android_library dependencies.
159type propagateRROEnforcementTransitionMutator struct{}
160
161func (p propagateRROEnforcementTransitionMutator) Split(ctx android.BaseModuleContext) []string {
162 // Never split modules, apps with or without RRO enabled use the "" variant, static android_library dependencies
163 // will use create the "rro" variant from incoming tranisitons.
164 return []string{""}
165}
166
167func (p propagateRROEnforcementTransitionMutator) OutgoingTransition(ctx android.OutgoingTransitionContext, sourceVariation string) string {
168 // Non-static dependencies are not involved in RRO and always use the empty variant.
169 if ctx.DepTag() != staticLibTag {
170 return ""
171 }
172
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700173 m := ctx.Module()
Colin Crossc4441622024-09-18 14:55:49 -0700174 if _, ok := m.(AndroidLibraryDependency); ok {
175 // If RRO is enforced globally don't bother using "rro" variants, the empty variant will have RRO enabled.
176 if ctx.Config().EnforceRROGlobally() {
177 return ""
178 }
179
180 // If RRO is enabled for this module use the "rro" variants of static dependencies. IncomingTransition will
181 // rewrite this back to "" if the dependency is not an android_library.
182 if ctx.Config().EnforceRROForModule(ctx.Module().Name()) {
183 return "rro"
184 }
185 }
186
187 return sourceVariation
188}
189
190func (p propagateRROEnforcementTransitionMutator) IncomingTransition(ctx android.IncomingTransitionContext, incomingVariation string) string {
191 // Propagate the "rro" variant to android_library modules, but use the empty variant for everything else.
192 if incomingVariation == "rro" {
193 m := ctx.Module()
194 if _, ok := m.(AndroidLibraryDependency); ok {
195 return "rro"
196 }
197 return ""
198 }
199
200 return ""
201}
202
203func (p propagateRROEnforcementTransitionMutator) Mutate(ctx android.BottomUpMutatorContext, variation string) {
204 m := ctx.Module()
205 if d, ok := m.(AndroidLibraryDependency); ok {
206 if variation == "rro" {
207 // This is the "rro" variant of a module that has both variants, mark this one as RRO enabled and
208 // hide it from make to avoid collisions with the non-RRO empty variant.
209 d.SetRROEnforcedForDependent(true)
210 m.HideFromMake()
211 } else if ctx.Config().EnforceRROGlobally() {
212 // RRO is enabled globally, mark it enabled for this module, but there is only one variant so no
213 // need to hide it from make.
214 d.SetRROEnforcedForDependent(true)
215 }
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700216 }
217}
218
Colin Cross8f1b0332024-01-25 13:39:06 -0800219func (a *aapt) useResourceProcessorBusyBox(ctx android.BaseModuleContext) bool {
Colin Crosseee4ab12024-03-27 11:54:10 -0700220 return BoolDefault(a.aaptProperties.Use_resource_processor, ctx.Config().UseResourceProcessorByDefault()) &&
221 // TODO(b/331641946): remove this when ResourceProcessorBusyBox supports generating shared libraries.
Spandan Dasa62d9562024-08-06 00:13:31 +0000222 !slices.Contains(a.aaptProperties.Aaptflags, "--shared-lib") &&
223 // Use the legacy resource processor in kythe builds.
224 // The legacy resource processor creates an R.srcjar, which kythe can use for generating crossrefs.
225 // TODO(b/354854007): Re-enable BusyBox in kythe builds
226 !ctx.Config().EmitXrefRules()
Colin Cross4eae06d2023-06-20 22:40:02 -0700227}
228
Inseob Kim34dc4cd2023-11-07 13:37:14 +0900229func (a *aapt) filterProduct() string {
230 return String(a.aaptProperties.Filter_product)
231}
232
Colin Crossa97c5d32018-03-28 14:58:31 -0700233func (a *aapt) ExportPackage() android.Path {
234 return a.exportPackage
235}
Colin Crossa14fb6a2024-10-23 16:57:06 -0700236func (a *aapt) ResourcesNodeDepSet() depset.DepSet[*resourcesNode] {
Colin Crossab8d1382023-07-14 17:23:41 +0000237 return a.resourcesNodesDepSet
Colin Crossc1c37552019-01-31 11:42:41 -0800238}
239
Colin Crossa14fb6a2024-10-23 16:57:06 -0700240func (a *aapt) RRODirsDepSet() depset.DepSet[rroDir] {
Colin Crossab8d1382023-07-14 17:23:41 +0000241 return a.rroDirsDepSet
Colin Crossc1c37552019-01-31 11:42:41 -0800242}
243
Colin Crossa14fb6a2024-10-23 16:57:06 -0700244func (a *aapt) ManifestsDepSet() depset.DepSet[android.Path] {
Colin Crossab8d1382023-07-14 17:23:41 +0000245 return a.manifestsDepSet
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800246}
247
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700248func (a *aapt) SetRROEnforcedForDependent(enforce bool) {
249 a.aaptProperties.RROEnforcedForDependent = enforce
250}
251
252func (a *aapt) IsRROEnforced(ctx android.BaseModuleContext) bool {
253 // True if RRO is enforced for this module or...
254 return ctx.Config().EnforceRROForModule(ctx.ModuleName()) ||
Jeongik Chacee5ba92021-02-19 12:11:51 +0900255 // if RRO is enforced for any of its dependents.
256 a.aaptProperties.RROEnforcedForDependent
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700257}
258
Jiyong Parkf1691d22021-03-29 20:11:58 +0900259func (a *aapt) aapt2Flags(ctx android.ModuleContext, sdkContext android.SdkContext,
Colin Crossa0ba2f52019-06-22 12:59:27 -0700260 manifestPath android.Path) (compileFlags, linkFlags []string, linkDeps android.Paths,
261 resDirs, overlayDirs []globbedResourceDir, rroDirs []rroDir, resZips android.Paths) {
Colin Crossa97c5d32018-03-28 14:58:31 -0700262
Jaewoong Jung3aff5782020-02-11 07:54:35 -0800263 hasVersionCode := android.PrefixInList(a.aaptProperties.Aaptflags, "--version-code")
264 hasVersionName := android.PrefixInList(a.aaptProperties.Aaptflags, "--version-name")
Colin Crossa97c5d32018-03-28 14:58:31 -0700265
Colin Crossa97c5d32018-03-28 14:58:31 -0700266 // Flags specified in Android.bp
267 linkFlags = append(linkFlags, a.aaptProperties.Aaptflags...)
268
Eric Miao40eab202023-03-30 16:57:17 +0000269 linkFlags = append(linkFlags, "--enable-compact-entries")
Colin Crossa97c5d32018-03-28 14:58:31 -0700270
271 // Find implicit or explicit asset and resource dirs
Jiakai Zhangba82e282023-10-13 18:08:59 +0100272 assets := android.PathsRelativeToModuleSourceDir(android.SourceInput{
273 Context: ctx,
274 Paths: a.aaptProperties.Assets,
275 IncludeDirs: false,
276 })
Colin Crossa97c5d32018-03-28 14:58:31 -0700277 assetDirs := android.PathsWithOptionalDefaultForModuleSrc(ctx, a.aaptProperties.Asset_dirs, "assets")
Cole Faust5b989772024-11-06 11:03:16 -0800278 resourceDirs := android.PathsWithOptionalDefaultForModuleSrc(ctx, a.aaptProperties.Resource_dirs.GetOrDefault(ctx, nil), "res")
Colin Cross8a497952019-03-05 22:25:09 -0800279 resourceZips := android.PathsForModuleSrc(ctx, a.aaptProperties.Resource_zips)
Colin Crossa97c5d32018-03-28 14:58:31 -0700280
Colin Crossa97c5d32018-03-28 14:58:31 -0700281 // Glob directories into lists of paths
282 for _, dir := range resourceDirs {
283 resDirs = append(resDirs, globbedResourceDir{
284 dir: dir,
285 files: androidResourceGlob(ctx, dir),
286 })
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700287 resOverlayDirs, resRRODirs := overlayResourceGlob(ctx, a, dir)
Colin Crossa97c5d32018-03-28 14:58:31 -0700288 overlayDirs = append(overlayDirs, resOverlayDirs...)
289 rroDirs = append(rroDirs, resRRODirs...)
290 }
291
Cole Faust6265120c2024-09-05 16:27:56 -0700292 assetDirsHasher := sha256.New()
Colin Crossc20dc852020-11-10 12:27:45 -0800293 var assetDeps android.Paths
Cole Faust6265120c2024-09-05 16:27:56 -0700294 for _, dir := range assetDirs {
Colin Crossc20dc852020-11-10 12:27:45 -0800295 // Add a dependency on every file in the asset directory. This ensures the aapt2
296 // rule will be rerun if one of the files in the asset directory is modified.
Cole Faust6265120c2024-09-05 16:27:56 -0700297 dirContents := androidResourceGlob(ctx, dir)
298 assetDeps = append(assetDeps, dirContents...)
Colin Crossc20dc852020-11-10 12:27:45 -0800299
Cole Faust6265120c2024-09-05 16:27:56 -0700300 // Add a hash of all the files in the asset directory to the command line.
Colin Crossc20dc852020-11-10 12:27:45 -0800301 // This ensures the aapt2 rule will be run if a file is removed from the asset directory,
302 // or a file is added whose timestamp is older than the output of aapt2.
Cole Faust6265120c2024-09-05 16:27:56 -0700303 for _, path := range dirContents.Strings() {
304 assetDirsHasher.Write([]byte(path))
305 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700306 }
307
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700308 assetDirStrings := assetDirs.Strings()
309 if a.noticeFile.Valid() {
310 assetDirStrings = append(assetDirStrings, filepath.Dir(a.noticeFile.Path().String()))
Colin Crossc20dc852020-11-10 12:27:45 -0800311 assetDeps = append(assetDeps, a.noticeFile.Path())
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700312 }
Jiakai Zhangba82e282023-10-13 18:08:59 +0100313 if len(assets) > 0 {
314 // aapt2 doesn't support adding individual asset files. Create a temp directory to hold asset
315 // files and pass it to aapt2.
316 tmpAssetDir := android.PathForModuleOut(ctx, "tmp_asset_dir")
317
318 rule := android.NewRuleBuilder(pctx, ctx)
319 rule.Command().
320 Text("rm -rf").Text(tmpAssetDir.String()).
321 Text("&&").
322 Text("mkdir -p").Text(tmpAssetDir.String())
323
324 for _, asset := range assets {
325 output := tmpAssetDir.Join(ctx, asset.Rel())
326 assetDeps = append(assetDeps, output)
327 rule.Command().Text("mkdir -p").Text(filepath.Dir(output.String()))
328 rule.Command().Text("cp").Input(asset).Output(output)
329 }
330
331 rule.Build("tmp_asset_dir", "tmp_asset_dir")
332
333 assetDirStrings = append(assetDirStrings, tmpAssetDir.String())
334 }
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700335
Colin Crossa97c5d32018-03-28 14:58:31 -0700336 linkFlags = append(linkFlags, "--manifest "+manifestPath.String())
337 linkDeps = append(linkDeps, manifestPath)
338
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700339 linkFlags = append(linkFlags, android.JoinWithPrefix(assetDirStrings, "-A "))
Cole Faust6265120c2024-09-05 16:27:56 -0700340 linkFlags = append(linkFlags, fmt.Sprintf("$$(: %x)", assetDirsHasher.Sum(nil)))
Colin Crossc20dc852020-11-10 12:27:45 -0800341 linkDeps = append(linkDeps, assetDeps...)
Colin Crossa97c5d32018-03-28 14:58:31 -0700342
Spandan Das50885c02023-02-23 21:31:33 +0000343 // Returns the effective version for {min|target}_sdk_version
Spandan Das8c9ae7e2023-03-03 21:20:36 +0000344 effectiveVersionString := func(sdkVersion android.SdkSpec, minSdkVersion android.ApiLevel) string {
Spandan Das50885c02023-02-23 21:31:33 +0000345 // If {min|target}_sdk_version is current, use sdk_version to determine the effective level
346 // This is necessary for vendor modules.
347 // The effective version does not _only_ depend on {min|target}_sdk_version(level),
348 // but also on the sdk_version (kind+level)
Spandan Das8c9ae7e2023-03-03 21:20:36 +0000349 if minSdkVersion.IsCurrent() {
Spandan Das50885c02023-02-23 21:31:33 +0000350 ret, err := sdkVersion.EffectiveVersionString(ctx)
351 if err != nil {
352 ctx.ModuleErrorf("invalid sdk_version: %s", err)
353 }
354 return ret
355 }
356 ret, err := minSdkVersion.EffectiveVersionString(ctx)
357 if err != nil {
358 ctx.ModuleErrorf("invalid min_sdk_version: %s", err)
359 }
360 return ret
Jiyong Park6a927c42020-01-21 02:03:43 +0900361 }
Spandan Das50885c02023-02-23 21:31:33 +0000362 // SDK version flags
363 sdkVersion := sdkContext.SdkVersion(ctx)
364 minSdkVersion := effectiveVersionString(sdkVersion, sdkContext.MinSdkVersion(ctx))
Colin Crossa97c5d32018-03-28 14:58:31 -0700365
Colin Cross83bb3162018-06-25 15:48:06 -0700366 linkFlags = append(linkFlags, "--min-sdk-version "+minSdkVersion)
Spandan Das6450b552023-02-23 19:27:07 +0000367 // Use minSdkVersion for target-sdk-version, even if `target_sdk_version` is set
368 // This behavior has been copied from Make.
Colin Cross83bb3162018-06-25 15:48:06 -0700369 linkFlags = append(linkFlags, "--target-sdk-version "+minSdkVersion)
Colin Crossa97c5d32018-03-28 14:58:31 -0700370
Colin Crossa97c5d32018-03-28 14:58:31 -0700371 // Version code
372 if !hasVersionCode {
Dan Albert4f378d72020-07-23 17:32:15 -0700373 linkFlags = append(linkFlags, "--version-code", ctx.Config().PlatformSdkVersion().String())
Colin Crossa97c5d32018-03-28 14:58:31 -0700374 }
375
376 if !hasVersionName {
Colin Cross402d5e02018-04-25 14:54:06 -0700377 var versionName string
378 if ctx.ModuleName() == "framework-res" {
379 // Some builds set AppsDefaultVersionName() to include the build number ("O-123456"). aapt2 copies the
380 // version name of framework-res into app manifests as compileSdkVersionCodename, which confuses things
Colin Crossbfd347d2018-05-09 11:11:35 -0700381 // if it contains the build number. Use the PlatformVersionName instead.
382 versionName = ctx.Config().PlatformVersionName()
Colin Cross402d5e02018-04-25 14:54:06 -0700383 } else {
384 versionName = ctx.Config().AppsDefaultVersionName()
385 }
Colin Cross0b9f31f2019-02-28 11:00:01 -0800386 versionName = proptools.NinjaEscape(versionName)
Colin Crossa97c5d32018-03-28 14:58:31 -0700387 linkFlags = append(linkFlags, "--version-name ", versionName)
388 }
389
Colin Crossa0ba2f52019-06-22 12:59:27 -0700390 linkFlags, compileFlags = android.FilterList(linkFlags, []string{"--legacy"})
391
392 // Always set --pseudo-localize, it will be stripped out later for release
393 // builds that don't want it.
394 compileFlags = append(compileFlags, "--pseudo-localize")
395
396 return compileFlags, linkFlags, linkDeps, resDirs, overlayDirs, rroDirs, resourceZips
Colin Crossa97c5d32018-03-28 14:58:31 -0700397}
398
Paul Duffin250e6192019-06-07 10:44:37 +0100399func (a *aapt) deps(ctx android.BottomUpMutatorContext, sdkDep sdkDep) {
Colin Cross42308aa2018-11-14 21:44:17 -0800400 if sdkDep.frameworkResModule != "" {
401 ctx.AddVariationDependencies(nil, frameworkResTag, sdkDep.frameworkResModule)
Colin Crossa97c5d32018-03-28 14:58:31 -0700402 }
403}
404
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800405var extractAssetsRule = pctx.AndroidStaticRule("extractAssets",
406 blueprint.RuleParams{
407 Command: `${config.Zip2ZipCmd} -i ${in} -o ${out} "assets/**/*"`,
408 CommandDeps: []string{"${config.Zip2ZipCmd}"},
409 })
410
Alixf7a10272023-09-27 16:47:56 +0000411type aaptBuildActionOptions struct {
412 sdkContext android.SdkContext
413 classLoaderContexts dexpreopt.ClassLoaderContextMap
414 excludedLibs []string
415 enforceDefaultTargetSdkVersion bool
Rico Winda2fa2632024-03-13 13:09:17 +0100416 forceNonFinalResourceIDs bool
Alixf7a10272023-09-27 16:47:56 +0000417 extraLinkFlags []string
Jihoon Kang84b25892023-12-01 22:01:06 +0000418 aconfigTextFiles android.Paths
Jiakai Zhang36937082024-04-15 11:15:50 +0000419 usesLibrary *usesLibrary
Spandan Dase2f98da2024-11-18 19:22:39 +0000420 // If rroDirs is provided, it will be used to generate package-res.apk
421 rroDirs *android.Paths
422 // If manifestForAapt is not nil, it will be used for aapt instead of the default source manifest.
423 manifestForAapt android.Path
424}
425
426func filterRRO(rroDirsDepSet depset.DepSet[rroDir], filter overlayType) android.Paths {
427 var paths android.Paths
428 seen := make(map[android.Path]bool)
429 for _, d := range rroDirsDepSet.ToList() {
430 if d.overlayType == filter {
431 if seen[d.path] {
432 continue
433 }
434 seen[d.path] = true
435 paths = append(paths, d.path)
436 }
437 }
438 return paths
Alixf7a10272023-09-27 16:47:56 +0000439}
440
441func (a *aapt) buildActions(ctx android.ModuleContext, opts aaptBuildActionOptions) {
Colin Cross5446e882019-05-22 10:46:27 -0700442
Colin Cross8676c8c2023-10-12 15:58:57 -0700443 staticResourcesNodesDepSet, sharedResourcesNodesDepSet, staticRRODirsDepSet, staticManifestsDepSet, sharedExportPackages, libFlags :=
Jiakai Zhang36937082024-04-15 11:15:50 +0000444 aaptLibs(ctx, opts.sdkContext, opts.classLoaderContexts, opts.usesLibrary)
Ulya Trafimovich31e444e2020-08-14 17:32:16 +0100445
Paul Duffin06530572022-02-03 17:54:15 +0000446 // Exclude any libraries from the supplied list.
Alixf7a10272023-09-27 16:47:56 +0000447 opts.classLoaderContexts = opts.classLoaderContexts.ExcludeLibs(opts.excludedLibs)
Paul Duffin06530572022-02-03 17:54:15 +0000448
Colin Cross31656952018-05-24 16:11:20 -0700449 // App manifest file
Spandan Dase2f98da2024-11-18 19:22:39 +0000450 var manifestFilePath android.Path
451 if opts.manifestForAapt != nil {
452 manifestFilePath = opts.manifestForAapt
453 } else {
454 manifestFile := proptools.StringDefault(a.aaptProperties.Manifest, "AndroidManifest.xml")
455 manifestFilePath = android.PathForModuleSrc(ctx, manifestFile)
456 }
Colin Cross31656952018-05-24 16:11:20 -0700457
Spandan Dase2f98da2024-11-18 19:22:39 +0000458 manifestPath := ManifestFixer(ctx, manifestFilePath, ManifestFixerParams{
Alixf7a10272023-09-27 16:47:56 +0000459 SdkContext: opts.sdkContext,
460 ClassLoaderContexts: opts.classLoaderContexts,
Harshit Mahajan5b8b7302022-06-10 11:24:05 +0000461 IsLibrary: a.isLibrary,
462 DefaultManifestVersion: a.defaultManifestVersion,
463 UseEmbeddedNativeLibs: a.useEmbeddedNativeLibs,
464 UsesNonSdkApis: a.usesNonSdkApis,
465 UseEmbeddedDex: a.useEmbeddedDex,
466 HasNoCode: a.hasNoCode,
467 LoggingParent: a.LoggingParent,
Alixf7a10272023-09-27 16:47:56 +0000468 EnforceDefaultTargetSdkVersion: opts.enforceDefaultTargetSdkVersion,
Gurpreet Singh75d65f32022-01-24 17:44:05 +0000469 })
Colin Cross90c25c62019-04-19 16:22:57 -0700470
Colin Crossab8d1382023-07-14 17:23:41 +0000471 staticDeps := transitiveAarDeps(staticResourcesNodesDepSet.ToList())
Colin Cross8676c8c2023-10-12 15:58:57 -0700472 sharedDeps := transitiveAarDeps(sharedResourcesNodesDepSet.ToList())
Colin Crossab8d1382023-07-14 17:23:41 +0000473
Luca Stefanifd898822019-09-10 22:13:31 +0200474 // Add additional manifest files to transitive manifests.
475 additionalManifests := android.PathsForModuleSrc(ctx, a.aaptProperties.Additional_manifests)
Colin Crossab8d1382023-07-14 17:23:41 +0000476 transitiveManifestPaths := append(android.Paths{manifestPath}, additionalManifests...)
Colin Crossab8d1382023-07-14 17:23:41 +0000477 transitiveManifestPaths = append(transitiveManifestPaths, staticManifestsDepSet.ToList()...)
Colin Cross90c25c62019-04-19 16:22:57 -0700478
Colin Crossab8d1382023-07-14 17:23:41 +0000479 if len(transitiveManifestPaths) > 1 && !Bool(a.aaptProperties.Dont_merge_manifests) {
Alixf7a10272023-09-27 16:47:56 +0000480 manifestMergerParams := ManifestMergerParams{
481 staticLibManifests: transitiveManifestPaths[1:],
Alix96ea88452023-08-31 15:48:23 +0000482 isLibrary: a.isLibrary,
483 packageName: a.manifestValues.applicationId,
484 }
Alixf7a10272023-09-27 16:47:56 +0000485 a.mergedManifestFile = manifestMerger(ctx, transitiveManifestPaths[0], manifestMergerParams)
Colin Crossa6182ab2024-08-21 10:47:44 -0700486 ctx.CheckbuildFile(a.mergedManifestFile)
Colin Cross90c25c62019-04-19 16:22:57 -0700487 if !a.isLibrary {
488 // Only use the merged manifest for applications. For libraries, the transitive closure of manifests
489 // will be propagated to the final application and merged there. The merged manifest for libraries is
490 // only passed to Make, which can't handle transitive dependencies.
491 manifestPath = a.mergedManifestFile
492 }
493 } else {
494 a.mergedManifestFile = manifestPath
495 }
Colin Cross31656952018-05-24 16:11:20 -0700496
Alixf7a10272023-09-27 16:47:56 +0000497 compileFlags, linkFlags, linkDeps, resDirs, overlayDirs, rroDirs, resZips := a.aapt2Flags(ctx, opts.sdkContext, manifestPath)
Colin Cross31656952018-05-24 16:11:20 -0700498
Spandan Dase2f98da2024-11-18 19:22:39 +0000499 a.rroDirsDepSet = depset.NewBuilder[rroDir](depset.TOPOLOGICAL).
500 Direct(rroDirs...).
501 Transitive(staticRRODirsDepSet).Build()
502
Colin Cross31656952018-05-24 16:11:20 -0700503 linkFlags = append(linkFlags, libFlags...)
Colin Cross8676c8c2023-10-12 15:58:57 -0700504 linkDeps = append(linkDeps, sharedExportPackages...)
Colin Crossab8d1382023-07-14 17:23:41 +0000505 linkDeps = append(linkDeps, staticDeps.resPackages()...)
Alixf7a10272023-09-27 16:47:56 +0000506 linkFlags = append(linkFlags, opts.extraLinkFlags...)
Colin Cross1b6a3cf2018-07-24 14:51:30 -0700507 if a.isLibrary {
508 linkFlags = append(linkFlags, "--static-lib")
509 }
Rico Wind7152e822024-04-19 08:14:15 +0200510 if opts.forceNonFinalResourceIDs {
511 linkFlags = append(linkFlags, "--non-final-ids")
512 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700513
Colin Cross7c4dc5d2024-02-13 14:29:45 -0800514 linkFlags = append(linkFlags, "--no-static-lib-packages")
Spandan Das457e5062024-10-05 01:00:34 +0000515 if a.isLibrary {
516 // Pass --merge-only to skip resource references validation until the final
517 // app link step when when all static libraries are present.
Colin Cross4eae06d2023-06-20 22:40:02 -0700518 linkFlags = append(linkFlags, "--merge-only")
Colin Cross4eae06d2023-06-20 22:40:02 -0700519 }
520
Colin Crossa97c5d32018-03-28 14:58:31 -0700521 packageRes := android.PathForModuleOut(ctx, "package-res.apk")
Colin Crossa97c5d32018-03-28 14:58:31 -0700522 proguardOptionsFile := android.PathForModuleGen(ctx, "proguard.options")
523 rTxt := android.PathForModuleOut(ctx, "R.txt")
Colin Cross66f78822018-05-02 12:58:28 -0700524 // This file isn't used by Soong, but is generated for exporting
525 extraPackages := android.PathForModuleOut(ctx, "extra_packages")
Colin Cross4eae06d2023-06-20 22:40:02 -0700526 var transitiveRJars android.Paths
Colin Crossf3b7bad2023-08-02 15:49:00 -0700527 var srcJar android.WritablePath
Colin Crossa97c5d32018-03-28 14:58:31 -0700528
Colin Cross4aaa84a2018-08-21 15:14:37 -0700529 var compiledResDirs []android.Paths
Colin Crossa97c5d32018-03-28 14:58:31 -0700530 for _, dir := range resDirs {
Colin Cross014489c2020-06-02 20:09:13 -0700531 a.resourceFiles = append(a.resourceFiles, dir.files...)
Jihoon Kang98ea8362024-07-16 18:20:03 +0000532 compiledResDirs = append(compiledResDirs, aapt2Compile(ctx, dir.dir, dir.files,
533 compileFlags, a.filterProduct(), opts.aconfigTextFiles).Paths())
Colin Crossa97c5d32018-03-28 14:58:31 -0700534 }
Colin Cross4aaa84a2018-08-21 15:14:37 -0700535
Colin Crossa592e3e2019-02-19 16:59:53 -0800536 for i, zip := range resZips {
537 flata := android.PathForModuleOut(ctx, fmt.Sprintf("reszip.%d.flata", i))
Colin Crossa0ba2f52019-06-22 12:59:27 -0700538 aapt2CompileZip(ctx, flata, zip, "", compileFlags)
Colin Crossa592e3e2019-02-19 16:59:53 -0800539 compiledResDirs = append(compiledResDirs, android.Paths{flata})
540 }
541
Colin Cross4aaa84a2018-08-21 15:14:37 -0700542 var compiledRes, compiledOverlay android.Paths
543
Colin Crossab8d1382023-07-14 17:23:41 +0000544 // AAPT2 overlays are in lowest to highest priority order, reverse the topological order
545 // of transitiveStaticLibs.
546 transitiveStaticLibs := android.ReversePaths(staticDeps.resPackages())
547
Colin Cross8f1b0332024-01-25 13:39:06 -0800548 if a.isLibrary && a.useResourceProcessorBusyBox(ctx) {
Colin Cross4eae06d2023-06-20 22:40:02 -0700549 // When building an android_library with ResourceProcessorBusyBox enabled treat static library dependencies
550 // as imports. The resources from dependencies will not be merged into this module's package-res.apk, and
551 // instead modules depending on this module will reference package-res.apk from all transitive static
552 // dependencies.
Colin Cross1d3f5902024-03-05 11:51:54 -0800553 for _, sharedDep := range sharedDeps {
554 if sharedDep.usedResourceProcessor {
555 transitiveRJars = append(transitiveRJars, sharedDep.rJar)
556 }
557 }
Colin Cross4eae06d2023-06-20 22:40:02 -0700558 for _, staticDep := range staticDeps {
559 linkDeps = append(linkDeps, staticDep.resPackage)
560 linkFlags = append(linkFlags, "-I "+staticDep.resPackage.String())
561 if staticDep.usedResourceProcessor {
562 transitiveRJars = append(transitiveRJars, staticDep.rJar)
563 }
564 }
565 } else {
566 // When building an app or building a library without ResourceProcessorBusyBox enabled all static
567 // dependencies are compiled into this module's package-res.apk as overlays.
568 compiledOverlay = append(compiledOverlay, transitiveStaticLibs...)
569 }
Colin Cross4aaa84a2018-08-21 15:14:37 -0700570
Colin Crossbec85302019-02-13 13:15:46 -0800571 if len(transitiveStaticLibs) > 0 {
Colin Cross4aaa84a2018-08-21 15:14:37 -0700572 // If we are using static android libraries, every source file becomes an overlay.
573 // This is to emulate old AAPT behavior which simulated library support.
574 for _, compiledResDir := range compiledResDirs {
575 compiledOverlay = append(compiledOverlay, compiledResDir...)
576 }
Colin Crossbec85302019-02-13 13:15:46 -0800577 } else if a.isLibrary {
578 // Otherwise, for a static library we treat all the resources equally with no overlay.
579 for _, compiledResDir := range compiledResDirs {
580 compiledRes = append(compiledRes, compiledResDir...)
581 }
Colin Cross4aaa84a2018-08-21 15:14:37 -0700582 } else if len(compiledResDirs) > 0 {
583 // Without static libraries, the first directory is our directory, which can then be
584 // overlaid by the rest.
585 compiledRes = append(compiledRes, compiledResDirs[0]...)
586 for _, compiledResDir := range compiledResDirs[1:] {
587 compiledOverlay = append(compiledOverlay, compiledResDir...)
588 }
589 }
590
Spandan Dase2f98da2024-11-18 19:22:39 +0000591 var compiledRro, compiledRroOverlay android.Paths
592 if opts.rroDirs != nil {
593 compiledRro, compiledRroOverlay = a.compileResInDir(ctx, *opts.rroDirs, compileFlags, opts.aconfigTextFiles)
Spandan Dasde588a32024-12-03 22:52:24 +0000594 } else {
595 // RRO enforcement is done based on module name. Compile the overlayDirs only if rroDirs is nil.
596 // This ensures that the autogenerated RROs do not compile the overlay dirs twice.
597 for _, dir := range overlayDirs {
598 compiledOverlay = append(compiledOverlay, aapt2Compile(ctx, dir.dir, dir.files,
599 compileFlags, a.filterProduct(), opts.aconfigTextFiles).Paths()...)
600 }
Spandan Dase2f98da2024-11-18 19:22:39 +0000601 }
602
Colin Crosse560c4a2019-03-19 16:03:11 -0700603 var splitPackages android.WritablePaths
604 var splits []split
605
606 for _, s := range a.splitNames {
607 suffix := strings.Replace(s, ",", "_", -1)
608 path := android.PathForModuleOut(ctx, "package_"+suffix+".apk")
609 linkFlags = append(linkFlags, "--split", path.String()+":"+s)
610 splitPackages = append(splitPackages, path)
611 splits = append(splits, split{
612 name: s,
613 suffix: suffix,
614 path: path,
615 })
616 }
617
Colin Cross8f1b0332024-01-25 13:39:06 -0800618 if !a.useResourceProcessorBusyBox(ctx) {
Colin Crossf3b7bad2023-08-02 15:49:00 -0700619 // the subdir "android" is required to be filtered by package names
620 srcJar = android.PathForModuleGen(ctx, "android", "R.srcjar")
621 }
622
Colin Crossab8d1382023-07-14 17:23:41 +0000623 // No need to specify assets from dependencies to aapt2Link for libraries, all transitive assets will be
624 // provided to the final app aapt2Link step.
625 var transitiveAssets android.Paths
626 if !a.isLibrary {
627 transitiveAssets = android.ReverseSliceInPlace(staticDeps.assets())
628 }
Spandan Dase2f98da2024-11-18 19:22:39 +0000629 if opts.rroDirs == nil { // link resources and overlay
630 aapt2Link(ctx, packageRes, srcJar, proguardOptionsFile, rTxt,
631 linkFlags, linkDeps, compiledRes, compiledOverlay, transitiveAssets, splitPackages,
632 opts.aconfigTextFiles)
633 ctx.CheckbuildFile(packageRes)
634 } else { // link autogenerated rro
635 if len(compiledRro) == 0 {
636 return
637 }
638 aapt2Link(ctx, packageRes, srcJar, proguardOptionsFile, rTxt,
639 linkFlags, linkDeps, compiledRro, compiledRroOverlay, nil, nil,
640 opts.aconfigTextFiles)
641 ctx.CheckbuildFile(packageRes)
642 }
Colin Crossa6182ab2024-08-21 10:47:44 -0700643
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800644 // Extract assets from the resource package output so that they can be used later in aapt2link
645 // for modules that depend on this one.
Colin Crossab8d1382023-07-14 17:23:41 +0000646 if android.PrefixInList(linkFlags, "-A ") {
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800647 assets := android.PathForModuleOut(ctx, "assets.zip")
648 ctx.Build(pctx, android.BuildParams{
649 Rule: extractAssetsRule,
650 Input: packageRes,
651 Output: assets,
652 Description: "extract assets from built resource file",
653 })
654 a.assetPackage = android.OptionalPathForPath(assets)
655 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700656
Colin Cross8f1b0332024-01-25 13:39:06 -0800657 if a.useResourceProcessorBusyBox(ctx) {
Colin Cross4eae06d2023-06-20 22:40:02 -0700658 rJar := android.PathForModuleOut(ctx, "busybox/R.jar")
Rico Winda2fa2632024-03-13 13:09:17 +0100659 resourceProcessorBusyBoxGenerateBinaryR(ctx, rTxt, a.mergedManifestFile, rJar, staticDeps, a.isLibrary, a.aaptProperties.Aaptflags,
660 opts.forceNonFinalResourceIDs)
Colin Crossf3b7bad2023-08-02 15:49:00 -0700661 aapt2ExtractExtraPackages(ctx, extraPackages, rJar)
Colin Cross4eae06d2023-06-20 22:40:02 -0700662 transitiveRJars = append(transitiveRJars, rJar)
663 a.rJar = rJar
Colin Crossf3b7bad2023-08-02 15:49:00 -0700664 } else {
665 aapt2ExtractExtraPackages(ctx, extraPackages, srcJar)
Colin Cross4eae06d2023-06-20 22:40:02 -0700666 }
667
Colin Cross312634e2023-11-21 15:13:56 -0800668 transitiveAaptResourcePackages := staticDeps.resPackages().Strings()
669 transitiveAaptResourcePackages = slices.DeleteFunc(transitiveAaptResourcePackages, func(p string) bool {
670 return p == packageRes.String()
671 })
672 transitiveAaptResourcePackagesFile := android.PathForModuleOut(ctx, "transitive-res-packages")
673 android.WriteFileRule(ctx, transitiveAaptResourcePackagesFile, strings.Join(transitiveAaptResourcePackages, "\n"))
674
Colin Cross1d3f5902024-03-05 11:51:54 -0800675 // Reverse the list of R.jar files so that the current module comes first, and direct dependencies come before
676 // transitive dependencies.
677 transitiveRJars = android.ReversePaths(transitiveRJars)
678
Colin Crossa97c5d32018-03-28 14:58:31 -0700679 a.aaptSrcJar = srcJar
Colin Cross4eae06d2023-06-20 22:40:02 -0700680 a.transitiveAaptRJars = transitiveRJars
Colin Cross312634e2023-11-21 15:13:56 -0800681 a.transitiveAaptResourcePackagesFile = transitiveAaptResourcePackagesFile
Colin Crossa97c5d32018-03-28 14:58:31 -0700682 a.exportPackage = packageRes
683 a.manifestPath = manifestPath
684 a.proguardOptionsFile = proguardOptionsFile
Colin Cross66f78822018-05-02 12:58:28 -0700685 a.extraAaptPackagesFile = extraPackages
Colin Crossa97c5d32018-03-28 14:58:31 -0700686 a.rTxt = rTxt
Colin Crosse560c4a2019-03-19 16:03:11 -0700687 a.splits = splits
Colin Crossa14fb6a2024-10-23 16:57:06 -0700688 a.resourcesNodesDepSet = depset.NewBuilder[*resourcesNode](depset.TOPOLOGICAL).
Colin Crossab8d1382023-07-14 17:23:41 +0000689 Direct(&resourcesNode{
690 resPackage: a.exportPackage,
691 manifest: a.manifestPath,
692 additionalManifests: additionalManifests,
Colin Cross4eae06d2023-06-20 22:40:02 -0700693 rTxt: a.rTxt,
694 rJar: a.rJar,
Colin Crossab8d1382023-07-14 17:23:41 +0000695 assets: a.assetPackage,
Colin Cross4eae06d2023-06-20 22:40:02 -0700696
Colin Cross8f1b0332024-01-25 13:39:06 -0800697 usedResourceProcessor: a.useResourceProcessorBusyBox(ctx),
Colin Crossab8d1382023-07-14 17:23:41 +0000698 }).
699 Transitive(staticResourcesNodesDepSet).Build()
Colin Crossa14fb6a2024-10-23 16:57:06 -0700700 a.manifestsDepSet = depset.NewBuilder[android.Path](depset.TOPOLOGICAL).
Colin Crossab8d1382023-07-14 17:23:41 +0000701 Direct(a.manifestPath).
702 DirectSlice(additionalManifests).
703 Transitive(staticManifestsDepSet).Build()
704}
705
Spandan Dase2f98da2024-11-18 19:22:39 +0000706// comileResInDir finds the resource files in dirs by globbing and then compiles them using aapt2
707// returns the file paths of compiled resources
708// dirs[0] is used as compileRes
709// dirs[1:] is used as compileOverlay
710func (a *aapt) compileResInDir(ctx android.ModuleContext, dirs android.Paths, compileFlags []string, aconfig android.Paths) (android.Paths, android.Paths) {
711 filesInDir := func(dir android.Path) android.Paths {
712 files, err := ctx.GlobWithDeps(filepath.Join(dir.String(), "**/*"), androidResourceIgnoreFilenames)
713 if err != nil {
714 ctx.ModuleErrorf("failed to glob overlay resource dir %q: %s", dir, err.Error())
715 return nil
716 }
717 var filePaths android.Paths
718 for _, file := range files {
719 if strings.HasSuffix(file, "/") {
720 continue // ignore directories
721 }
722 filePaths = append(filePaths, android.PathForSource(ctx, file))
723 }
724 return filePaths
725 }
726
727 var compiledRes, compiledOverlay android.Paths
728 if len(dirs) == 0 {
729 return nil, nil
730 }
731 compiledRes = append(compiledRes, aapt2Compile(ctx, dirs[0], filesInDir(dirs[0]), compileFlags, a.filterProduct(), aconfig).Paths()...)
732 if len(dirs) > 0 {
733 for _, dir := range dirs[1:] {
734 compiledOverlay = append(compiledOverlay, aapt2Compile(ctx, dir, filesInDir(dir), compileFlags, a.filterProduct(), aconfig).Paths()...)
735 }
736 }
737 return compiledRes, compiledOverlay
738}
739
Colin Cross4eae06d2023-06-20 22:40:02 -0700740var resourceProcessorBusyBox = pctx.AndroidStaticRule("resourceProcessorBusyBox",
741 blueprint.RuleParams{
742 Command: "${config.JavaCmd} -cp ${config.ResourceProcessorBusyBox} " +
743 "com.google.devtools.build.android.ResourceProcessorBusyBox --tool=GENERATE_BINARY_R -- @${out}.args && " +
744 "if cmp -s ${out}.tmp ${out} ; then rm ${out}.tmp ; else mv ${out}.tmp ${out}; fi",
745 CommandDeps: []string{"${config.ResourceProcessorBusyBox}"},
746 Rspfile: "${out}.args",
747 RspfileContent: "--primaryRTxt ${rTxt} --primaryManifest ${manifest} --classJarOutput ${out}.tmp ${args}",
748 Restat: true,
749 }, "rTxt", "manifest", "args")
750
751// resourceProcessorBusyBoxGenerateBinaryR converts the R.txt file produced by aapt2 into R.class files
752// using Bazel's ResourceProcessorBusyBox tool, which is faster than compiling the R.java files and
753// supports producing classes for static dependencies that only include resources from that dependency.
754func resourceProcessorBusyBoxGenerateBinaryR(ctx android.ModuleContext, rTxt, manifest android.Path,
Rico Winda2fa2632024-03-13 13:09:17 +0100755 rJar android.WritablePath, transitiveDeps transitiveAarDeps, isLibrary bool, aaptFlags []string,
756 forceNonFinalIds bool) {
Colin Cross4eae06d2023-06-20 22:40:02 -0700757
758 var args []string
759 var deps android.Paths
760
761 if !isLibrary {
762 // When compiling an app, pass all R.txt and AndroidManifest.xml from transitive static library dependencies
763 // to ResourceProcessorBusyBox so that it can regenerate R.class files with the final resource IDs for each
764 // package.
765 args, deps = transitiveDeps.resourceProcessorDeps()
Rico Winda2fa2632024-03-13 13:09:17 +0100766 if forceNonFinalIds {
767 args = append(args, "--finalFields=false")
768 }
Colin Cross4eae06d2023-06-20 22:40:02 -0700769 } else {
770 // When compiling a library don't pass any dependencies as it only needs to generate an R.class file for this
771 // library. Pass --finalFields=false so that the R.class file contains non-final fields so they don't get
772 // inlined into the library before the final IDs are assigned during app compilation.
773 args = append(args, "--finalFields=false")
774 }
775
Colin Crossd3f7d1a2024-01-03 19:42:25 -0800776 for i, arg := range aaptFlags {
777 const AAPT_CUSTOM_PACKAGE = "--custom-package"
778 if strings.HasPrefix(arg, AAPT_CUSTOM_PACKAGE) {
779 pkg := strings.TrimSpace(strings.TrimPrefix(arg, AAPT_CUSTOM_PACKAGE))
780 if pkg == "" && i+1 < len(aaptFlags) {
781 pkg = aaptFlags[i+1]
782 }
783 args = append(args, "--packageForR "+pkg)
784 }
785 }
786
Colin Cross4eae06d2023-06-20 22:40:02 -0700787 deps = append(deps, rTxt, manifest)
788
789 ctx.Build(pctx, android.BuildParams{
790 Rule: resourceProcessorBusyBox,
791 Output: rJar,
792 Implicits: deps,
793 Description: "ResourceProcessorBusyBox",
794 Args: map[string]string{
795 "rTxt": rTxt.String(),
796 "manifest": manifest.String(),
797 "args": strings.Join(args, " "),
798 },
799 })
800}
801
Colin Crossab8d1382023-07-14 17:23:41 +0000802type resourcesNode struct {
803 resPackage android.Path
804 manifest android.Path
805 additionalManifests android.Paths
Colin Cross4eae06d2023-06-20 22:40:02 -0700806 rTxt android.Path
807 rJar android.Path
Colin Crossab8d1382023-07-14 17:23:41 +0000808 assets android.OptionalPath
Colin Cross4eae06d2023-06-20 22:40:02 -0700809
810 usedResourceProcessor bool
Colin Crossab8d1382023-07-14 17:23:41 +0000811}
812
813type transitiveAarDeps []*resourcesNode
814
815func (t transitiveAarDeps) resPackages() android.Paths {
Colin Cross4eae06d2023-06-20 22:40:02 -0700816 paths := make(android.Paths, 0, len(t))
Colin Crossab8d1382023-07-14 17:23:41 +0000817 for _, dep := range t {
818 paths = append(paths, dep.resPackage)
819 }
Colin Cross4eae06d2023-06-20 22:40:02 -0700820 return paths
Colin Crossab8d1382023-07-14 17:23:41 +0000821}
822
823func (t transitiveAarDeps) manifests() android.Paths {
Colin Cross4eae06d2023-06-20 22:40:02 -0700824 paths := make(android.Paths, 0, len(t))
Colin Crossab8d1382023-07-14 17:23:41 +0000825 for _, dep := range t {
826 paths = append(paths, dep.manifest)
827 paths = append(paths, dep.additionalManifests...)
828 }
Colin Cross4eae06d2023-06-20 22:40:02 -0700829 return paths
830}
831
832func (t transitiveAarDeps) resourceProcessorDeps() (args []string, deps android.Paths) {
833 for _, dep := range t {
834 args = append(args, "--library="+dep.rTxt.String()+","+dep.manifest.String())
835 deps = append(deps, dep.rTxt, dep.manifest)
836 }
837 return args, deps
Colin Crossab8d1382023-07-14 17:23:41 +0000838}
839
840func (t transitiveAarDeps) assets() android.Paths {
Colin Cross4eae06d2023-06-20 22:40:02 -0700841 paths := make(android.Paths, 0, len(t))
Colin Crossab8d1382023-07-14 17:23:41 +0000842 for _, dep := range t {
843 if dep.assets.Valid() {
844 paths = append(paths, dep.assets.Path())
845 }
846 }
847 return paths
Colin Crossa97c5d32018-03-28 14:58:31 -0700848}
849
850// aaptLibs collects libraries from dependencies and sdk_version and converts them into paths
Jiakai Zhang36937082024-04-15 11:15:50 +0000851func aaptLibs(ctx android.ModuleContext, sdkContext android.SdkContext,
852 classLoaderContexts dexpreopt.ClassLoaderContextMap, usesLibrary *usesLibrary) (
Colin Crossa14fb6a2024-10-23 16:57:06 -0700853 staticResourcesNodes, sharedResourcesNodes depset.DepSet[*resourcesNode], staticRRODirs depset.DepSet[rroDir],
854 staticManifests depset.DepSet[android.Path], sharedLibs android.Paths, flags []string) {
Colin Crossa97c5d32018-03-28 14:58:31 -0700855
Ulya Trafimovichb23d28c2020-10-08 12:53:58 +0100856 if classLoaderContexts == nil {
Ulya Trafimovich18554242020-11-03 15:55:11 +0000857 // Not all callers need to compute class loader context, those who don't just pass nil.
858 // Create a temporary class loader context here (it will be computed, but not used).
Ulya Trafimovichb23d28c2020-10-08 12:53:58 +0100859 classLoaderContexts = make(dexpreopt.ClassLoaderContextMap)
Ulya Trafimovich18554242020-11-03 15:55:11 +0000860 }
861
Colin Cross83bb3162018-06-25 15:48:06 -0700862 sdkDep := decodeSdkDep(ctx, sdkContext)
Colin Crossa97c5d32018-03-28 14:58:31 -0700863 if sdkDep.useFiles {
Colin Cross86a60ae2018-05-29 14:44:55 -0700864 sharedLibs = append(sharedLibs, sdkDep.jars...)
Colin Crossa97c5d32018-03-28 14:58:31 -0700865 }
866
Colin Crossa14fb6a2024-10-23 16:57:06 -0700867 var staticResourcesNodeDepSets []depset.DepSet[*resourcesNode]
868 var sharedResourcesNodeDepSets []depset.DepSet[*resourcesNode]
869 rroDirsDepSetBuilder := depset.NewBuilder[rroDir](depset.TOPOLOGICAL)
870 manifestsDepSetBuilder := depset.NewBuilder[android.Path](depset.TOPOLOGICAL)
Colin Crossab8d1382023-07-14 17:23:41 +0000871
Colin Crossa97c5d32018-03-28 14:58:31 -0700872 ctx.VisitDirectDeps(func(module android.Module) {
Ulya Trafimovich65b03192020-12-03 16:50:22 +0000873 depTag := ctx.OtherModuleDependencyTag(module)
Ulya Trafimovich18554242020-11-03 15:55:11 +0000874
Colin Crossa97c5d32018-03-28 14:58:31 -0700875 var exportPackage android.Path
Colin Cross66f78822018-05-02 12:58:28 -0700876 aarDep, _ := module.(AndroidLibraryDependency)
877 if aarDep != nil {
Colin Crossa97c5d32018-03-28 14:58:31 -0700878 exportPackage = aarDep.ExportPackage()
879 }
880
Ulya Trafimovich65b03192020-12-03 16:50:22 +0000881 switch depTag {
Colin Cross4b964c02018-10-15 16:18:06 -0700882 case instrumentationForTag:
883 // Nothing, instrumentationForTag is treated as libTag for javac but not for aapt2.
Spandan Dase2f98da2024-11-18 19:22:39 +0000884 case sdkLibTag, libTag, rroDepTag:
Colin Cross5446e882019-05-22 10:46:27 -0700885 if exportPackage != nil {
Colin Cross8676c8c2023-10-12 15:58:57 -0700886 sharedResourcesNodeDepSets = append(sharedResourcesNodeDepSets, aarDep.ResourcesNodeDepSet())
Colin Cross5446e882019-05-22 10:46:27 -0700887 sharedLibs = append(sharedLibs, exportPackage)
888 }
Colin Cross5446e882019-05-22 10:46:27 -0700889 case frameworkResTag:
Colin Crossa97c5d32018-03-28 14:58:31 -0700890 if exportPackage != nil {
891 sharedLibs = append(sharedLibs, exportPackage)
892 }
893 case staticLibTag:
894 if exportPackage != nil {
Colin Cross8676c8c2023-10-12 15:58:57 -0700895 staticResourcesNodeDepSets = append(staticResourcesNodeDepSets, aarDep.ResourcesNodeDepSet())
Colin Crossab8d1382023-07-14 17:23:41 +0000896 rroDirsDepSetBuilder.Transitive(aarDep.RRODirsDepSet())
897 manifestsDepSetBuilder.Transitive(aarDep.ManifestsDepSet())
Colin Crossa97c5d32018-03-28 14:58:31 -0700898 }
899 }
Ulya Trafimovich18554242020-11-03 15:55:11 +0000900
Ulya Trafimovich88bb6f62020-12-16 16:16:11 +0000901 addCLCFromDep(ctx, module, classLoaderContexts)
Jiakai Zhang36937082024-04-15 11:15:50 +0000902 if usesLibrary != nil {
903 addMissingOptionalUsesLibsFromDep(ctx, module, usesLibrary)
904 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700905 })
906
Colin Crossab8d1382023-07-14 17:23:41 +0000907 // AAPT2 overlays are in lowest to highest priority order, the topological order will be reversed later.
908 // Reverse the dependency order now going into the depset so that it comes out in order after the second
909 // reverse later.
910 // NOTE: this is legacy and probably incorrect behavior, for most other cases (e.g. conflicting classes in
911 // dependencies) the highest priority dependency is listed first, but for resources the highest priority
Colin Cross9055e212024-03-23 04:43:41 +0000912 // dependency has to be listed last. This is also inconsistent with the way manifests from the same
913 // transitive dependencies are merged.
Colin Crossa14fb6a2024-10-23 16:57:06 -0700914 staticResourcesNodes = depset.New(depset.TOPOLOGICAL, nil,
Colin Cross8676c8c2023-10-12 15:58:57 -0700915 android.ReverseSliceInPlace(staticResourcesNodeDepSets))
Colin Crossa14fb6a2024-10-23 16:57:06 -0700916 sharedResourcesNodes = depset.New(depset.TOPOLOGICAL, nil,
Colin Cross8676c8c2023-10-12 15:58:57 -0700917 android.ReverseSliceInPlace(sharedResourcesNodeDepSets))
Colin Crossa97c5d32018-03-28 14:58:31 -0700918
Colin Crossab8d1382023-07-14 17:23:41 +0000919 staticRRODirs = rroDirsDepSetBuilder.Build()
920 staticManifests = manifestsDepSetBuilder.Build()
921
922 if len(staticResourcesNodes.ToList()) > 0 {
Colin Crossa97c5d32018-03-28 14:58:31 -0700923 flags = append(flags, "--auto-add-overlay")
924 }
925
926 for _, sharedLib := range sharedLibs {
927 flags = append(flags, "-I "+sharedLib.String())
928 }
929
Colin Cross8676c8c2023-10-12 15:58:57 -0700930 return staticResourcesNodes, sharedResourcesNodes, staticRRODirs, staticManifests, sharedLibs, flags
Colin Crossa97c5d32018-03-28 14:58:31 -0700931}
932
933type AndroidLibrary struct {
934 Library
935 aapt
936
937 androidLibraryProperties androidLibraryProperties
938
939 aarFile android.WritablePath
Colin Cross89c31582018-04-30 15:55:11 -0700940}
941
Colin Crossa97c5d32018-03-28 14:58:31 -0700942var _ AndroidLibraryDependency = (*AndroidLibrary)(nil)
943
944func (a *AndroidLibrary) DepsMutator(ctx android.BottomUpMutatorContext) {
Jiakai Zhangf98da192024-04-15 11:15:41 +0000945 a.usesLibrary.deps(ctx, false)
Colin Crossa97c5d32018-03-28 14:58:31 -0700946 a.Module.deps(ctx)
Jiyong Parkf1691d22021-03-29 20:11:58 +0900947 sdkDep := decodeSdkDep(ctx, android.SdkContext(a))
Paul Duffin250e6192019-06-07 10:44:37 +0100948 if sdkDep.hasFrameworkLibs() {
949 a.aapt.deps(ctx, sdkDep)
Colin Crossa97c5d32018-03-28 14:58:31 -0700950 }
Jihoon Kang9049c272024-03-19 21:57:36 +0000951
952 for _, aconfig_declaration := range a.aaptProperties.Flags_packages {
953 ctx.AddDependency(ctx.Module(), aconfigDeclarationTag, aconfig_declaration)
954 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700955}
956
957func (a *AndroidLibrary) GenerateAndroidBuildActions(ctx android.ModuleContext) {
Colin Crosse4246ab2019-02-05 21:55:21 -0800958 a.aapt.isLibrary = true
Ulya Trafimovich42c7f0d2021-08-17 16:20:29 +0100959 a.classLoaderContexts = a.usesLibrary.classLoaderContextForUsesLibDeps(ctx)
Spandan Das0727ba72024-02-13 16:37:43 +0000960 if a.usesLibrary.shouldDisableDexpreopt {
961 a.dexpreopter.disableDexpreopt()
962 }
Jihoon Kang9aef7772024-06-14 23:45:06 +0000963 aconfigTextFilePaths := getAconfigFilePaths(ctx)
Alixf7a10272023-09-27 16:47:56 +0000964 a.aapt.buildActions(ctx,
965 aaptBuildActionOptions{
966 sdkContext: android.SdkContext(a),
967 classLoaderContexts: a.classLoaderContexts,
968 enforceDefaultTargetSdkVersion: false,
Jihoon Kang9aef7772024-06-14 23:45:06 +0000969 aconfigTextFiles: aconfigTextFilePaths,
Jiakai Zhang36937082024-04-15 11:15:50 +0000970 usesLibrary: &a.usesLibrary,
Alixf7a10272023-09-27 16:47:56 +0000971 },
972 )
Colin Crossa97c5d32018-03-28 14:58:31 -0700973
Colin Crossff694a82023-12-13 15:54:49 -0800974 apexInfo, _ := android.ModuleProvider(ctx, android.ApexInfoProvider)
975 a.hideApexVariantFromMake = !apexInfo.IsForPlatform()
Colin Cross56a83212020-09-15 18:30:11 -0700976
yangbill2af0b6e2024-03-15 09:29:29 +0000977 a.stem = proptools.StringDefault(a.overridableProperties.Stem, ctx.ModuleName())
Jihoon Kang1bfb6f22023-07-01 00:13:47 +0000978
Colin Cross4eae06d2023-06-20 22:40:02 -0700979 ctx.CheckbuildFile(a.aapt.proguardOptionsFile)
980 ctx.CheckbuildFile(a.aapt.exportPackage)
Colin Cross8f1b0332024-01-25 13:39:06 -0800981 if a.useResourceProcessorBusyBox(ctx) {
Colin Cross4eae06d2023-06-20 22:40:02 -0700982 ctx.CheckbuildFile(a.aapt.rJar)
Colin Crossf3b7bad2023-08-02 15:49:00 -0700983 } else {
984 ctx.CheckbuildFile(a.aapt.aaptSrcJar)
Colin Cross4eae06d2023-06-20 22:40:02 -0700985 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700986
987 // apps manifests are handled by aapt, don't let Module see them
988 a.properties.Manifest = nil
989
Colin Cross014489c2020-06-02 20:09:13 -0700990 a.linter.mergedManifest = a.aapt.mergedManifestFile
991 a.linter.manifest = a.aapt.manifestPath
992 a.linter.resources = a.aapt.resourceFiles
993
Sam Delmericoc8e040c2023-10-31 17:27:02 +0000994 proguardSpecInfo := a.collectProguardSpecInfo(ctx)
Colin Cross40213022023-12-13 15:19:49 -0800995 android.SetProvider(ctx, ProguardSpecInfoProvider, proguardSpecInfo)
Colin Cross312634e2023-11-21 15:13:56 -0800996 exportedProguardFlagsFiles := proguardSpecInfo.ProguardFlagsFiles.ToList()
997 a.extraProguardFlagsFiles = append(a.extraProguardFlagsFiles, exportedProguardFlagsFiles...)
998 a.extraProguardFlagsFiles = append(a.extraProguardFlagsFiles, a.proguardOptionsFile)
999
1000 combinedExportedProguardFlagFile := android.PathForModuleOut(ctx, "export_proguard_flags")
1001 writeCombinedProguardFlagsFile(ctx, combinedExportedProguardFlagFile, exportedProguardFlagsFiles)
1002 a.combinedExportedProguardFlagsFile = combinedExportedProguardFlagFile
Colin Crossa97c5d32018-03-28 14:58:31 -07001003
Colin Cross4eae06d2023-06-20 22:40:02 -07001004 var extraSrcJars android.Paths
1005 var extraCombinedJars android.Paths
1006 var extraClasspathJars android.Paths
Colin Cross8f1b0332024-01-25 13:39:06 -08001007 if a.useResourceProcessorBusyBox(ctx) {
Colin Cross4eae06d2023-06-20 22:40:02 -07001008 // When building a library with ResourceProcessorBusyBox enabled ResourceProcessorBusyBox for this
1009 // library and each of the transitive static android_library dependencies has already created an
1010 // R.class file for the appropriate package. Add all of those R.class files to the classpath.
1011 extraClasspathJars = a.transitiveAaptRJars
1012 } else {
1013 // When building a library without ResourceProcessorBusyBox the aapt2 rule creates R.srcjar containing
1014 // R.java files for the library's package and the packages from all transitive static android_library
1015 // dependencies. Compile the srcjar alongside the rest of the sources.
1016 extraSrcJars = android.Paths{a.aapt.aaptSrcJar}
1017 }
1018
Colin Crossfdaa6722024-08-23 11:58:08 -07001019 a.Module.compile(ctx, extraSrcJars, extraClasspathJars, extraCombinedJars, nil)
Colin Crossa97c5d32018-03-28 14:58:31 -07001020
Colin Crossf57c5782019-01-25 13:20:38 -08001021 a.aarFile = android.PathForModuleOut(ctx, ctx.ModuleName()+".aar")
Colin Crossa97c5d32018-03-28 14:58:31 -07001022 var res android.Paths
1023 if a.androidLibraryProperties.BuildAAR {
1024 BuildAAR(ctx, a.aarFile, a.outputFile, a.manifestPath, a.rTxt, res)
Colin Crossa97c5d32018-03-28 14:58:31 -07001025 }
Colin Cross89c31582018-04-30 15:55:11 -07001026
Sam Delmerico82602492022-06-10 17:05:42 +00001027 prebuiltJniPackages := android.Paths{}
1028 ctx.VisitDirectDeps(func(module android.Module) {
Colin Cross313aa542023-12-13 13:47:44 -08001029 if info, ok := android.OtherModuleProvider(ctx, module, JniPackageProvider); ok {
Sam Delmerico82602492022-06-10 17:05:42 +00001030 prebuiltJniPackages = append(prebuiltJniPackages, info.JniPackages...)
1031 }
1032 })
1033 if len(prebuiltJniPackages) > 0 {
Colin Cross40213022023-12-13 15:19:49 -08001034 android.SetProvider(ctx, JniPackageProvider, JniPackageInfo{
Sam Delmerico82602492022-06-10 17:05:42 +00001035 JniPackages: prebuiltJniPackages,
1036 })
1037 }
Jihoon Kang9aef7772024-06-14 23:45:06 +00001038
1039 android.SetProvider(ctx, FlagsPackagesProvider, FlagsPackages{
1040 AconfigTextFiles: aconfigTextFilePaths,
1041 })
mrziwang9f7b9f42024-07-10 12:18:06 -07001042
1043 a.setOutputFiles(ctx)
1044}
1045
1046func (a *AndroidLibrary) setOutputFiles(ctx android.ModuleContext) {
1047 ctx.SetOutputFiles([]android.Path{a.aarFile}, ".aar")
1048 setOutputFiles(ctx, a.Library.Module)
Colin Crossa97c5d32018-03-28 14:58:31 -07001049}
1050
Cole Faustb36d31d2024-08-27 16:04:28 -07001051func (a *AndroidLibrary) IDEInfo(ctx android.BaseModuleContext, dpInfo *android.IdeInfo) {
1052 a.Library.IDEInfo(ctx, dpInfo)
1053 a.aapt.IDEInfo(ctx, dpInfo)
Colin Cross95b53b82023-10-17 13:21:02 -07001054}
1055
Cole Faustb36d31d2024-08-27 16:04:28 -07001056func (a *aapt) IDEInfo(ctx android.BaseModuleContext, dpInfo *android.IdeInfo) {
Colin Cross8f1b0332024-01-25 13:39:06 -08001057 if a.rJar != nil {
Colin Cross95b53b82023-10-17 13:21:02 -07001058 dpInfo.Jars = append(dpInfo.Jars, a.rJar.String())
1059 }
1060}
1061
Colin Cross1b16b0e2019-02-12 14:41:32 -08001062// android_library builds and links sources into a `.jar` file for the device along with Android resources.
1063//
1064// An android_library has a single variant that produces a `.jar` file containing `.class` files that were
Sam Delmerico82602492022-06-10 17:05:42 +00001065// compiled against the device bootclasspath, along with a `package-res.apk` file containing Android resources compiled
Colin Cross1b16b0e2019-02-12 14:41:32 -08001066// with aapt2. This module is not suitable for installing on a device, but can be used as a `static_libs` dependency of
1067// an android_app module.
Colin Crossa97c5d32018-03-28 14:58:31 -07001068func AndroidLibraryFactory() android.Module {
1069 module := &AndroidLibrary{}
1070
Colin Crossce6734e2020-06-15 16:09:53 -07001071 module.Module.addHostAndDeviceProperties()
Colin Crossa97c5d32018-03-28 14:58:31 -07001072 module.AddProperties(
Colin Crossa97c5d32018-03-28 14:58:31 -07001073 &module.aaptProperties,
Ronald Braunsteincdc66f42024-04-12 11:23:19 -07001074 &module.androidLibraryProperties,
1075 &module.sourceProperties)
Colin Crossa97c5d32018-03-28 14:58:31 -07001076
1077 module.androidLibraryProperties.BuildAAR = true
Colin Cross014489c2020-06-02 20:09:13 -07001078 module.Module.linter.library = true
Colin Crossa97c5d32018-03-28 14:58:31 -07001079
Jooyung Hanacc7bbe2020-05-20 09:06:00 +09001080 android.InitApexModule(module)
Colin Cross48de9a42018-10-02 13:53:33 -07001081 InitJavaModule(module, android.DeviceSupported)
Colin Crossa97c5d32018-03-28 14:58:31 -07001082 return module
1083}
1084
Colin Crossfabb6082018-02-20 17:22:23 -08001085//
1086// AAR (android library) prebuilts
1087//
Colin Crossfabb6082018-02-20 17:22:23 -08001088
Vinh Trance0781f2022-04-13 01:30:44 +00001089// Properties for android_library_import
Colin Crossfabb6082018-02-20 17:22:23 -08001090type AARImportProperties struct {
Vinh Trance0781f2022-04-13 01:30:44 +00001091 // ARR (android library prebuilt) filepath. Exactly one ARR is required.
Colin Cross27b922f2019-03-04 22:35:41 -08001092 Aars []string `android:"path"`
Vinh Trance0781f2022-04-13 01:30:44 +00001093 // If not blank, set to the version of the sdk to compile against.
1094 // Defaults to private.
1095 // Values are of one of the following forms:
1096 // 1) numerical API level, "current", "none", or "core_platform"
1097 // 2) An SDK kind with an API level: "<sdk kind>_<API level>"
1098 // See build/soong/android/sdk_version.go for the complete and up to date list of SDK kinds.
1099 // If the SDK kind is empty, it will be set to public
1100 Sdk_version *string
1101 // If not blank, set the minimum version of the sdk that the compiled artifacts will run against.
1102 // Defaults to sdk_version if not set. See sdk_version for possible values.
Colin Cross479884c2018-07-10 13:39:30 -07001103 Min_sdk_version *string
Vinh Trance0781f2022-04-13 01:30:44 +00001104 // List of java static libraries that the included ARR (android library prebuilts) has dependencies to.
Cole Faustb7493472024-08-28 11:55:52 -07001105 Static_libs proptools.Configurable[[]string]
Vinh Trance0781f2022-04-13 01:30:44 +00001106 // List of java libraries that the included ARR (android library prebuilts) has dependencies to.
1107 Libs []string
1108 // If set to true, run Jetifier against .aar file. Defaults to false.
Colin Cross1001a792019-03-21 22:21:39 -07001109 Jetifier *bool
Sam Delmerico82602492022-06-10 17:05:42 +00001110 // If true, extract JNI libs from AAR archive. These libs will be accessible to android_app modules and
1111 // will be passed transitively through android_libraries to an android_app.
1112 //TODO(b/241138093) evaluate whether we can have this flag default to true for Bazel conversion
1113 Extract_jni *bool
Colin Cross21ed4692024-04-24 20:23:38 +00001114
1115 // If set, overrides the manifest extracted from the AAR with the provided path.
1116 Manifest *string `android:"path"`
Colin Crossfabb6082018-02-20 17:22:23 -08001117}
1118
1119type AARImport struct {
1120 android.ModuleBase
Colin Cross48de9a42018-10-02 13:53:33 -07001121 android.DefaultableModuleBase
Jooyung Hanacc7bbe2020-05-20 09:06:00 +09001122 android.ApexModuleBase
Colin Crossfabb6082018-02-20 17:22:23 -08001123 prebuilt android.Prebuilt
1124
Jooyung Hanacc7bbe2020-05-20 09:06:00 +09001125 // Functionality common to Module and Import.
1126 embeddableInModuleAndImport
1127
Colin Cross9ffaf282024-08-12 13:50:09 -07001128 providesTransitiveHeaderJarsForR8
Sam Delmerico9f9c0a22022-11-29 11:19:37 -05001129
Colin Crossfabb6082018-02-20 17:22:23 -08001130 properties AARImportProperties
1131
Colin Cross7707b242024-07-26 12:02:36 -07001132 headerJarFile android.Path
1133 implementationJarFile android.Path
1134 implementationAndResourcesJarFile android.Path
1135 proguardFlags android.Path
1136 exportPackage android.Path
Colin Cross312634e2023-11-21 15:13:56 -08001137 transitiveAaptResourcePackagesFile android.Path
Colin Cross7707b242024-07-26 12:02:36 -07001138 extraAaptPackagesFile android.Path
Colin Cross21ed4692024-04-24 20:23:38 +00001139 manifest android.Path
Colin Cross7707b242024-07-26 12:02:36 -07001140 assetsPackage android.Path
1141 rTxt android.Path
1142 rJar android.Path
Colin Cross66f78822018-05-02 12:58:28 -07001143
Colin Crossa14fb6a2024-10-23 16:57:06 -07001144 resourcesNodesDepSet depset.DepSet[*resourcesNode]
1145 manifestsDepSet depset.DepSet[android.Path]
Colin Cross56a83212020-09-15 18:30:11 -07001146
1147 hideApexVariantFromMake bool
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001148
Sam Delmerico82602492022-06-10 17:05:42 +00001149 aarPath android.Path
1150 jniPackages android.Paths
Jiyong Park92315372021-04-02 08:45:46 +09001151
1152 sdkVersion android.SdkSpec
Spandan Das8c9ae7e2023-03-03 21:20:36 +00001153 minSdkVersion android.ApiLevel
LaMont Jonesafe7baf2024-01-09 22:47:39 +00001154
Colin Cross9055e212024-03-23 04:43:41 +00001155 usesLibrary
1156 classLoaderContexts dexpreopt.ClassLoaderContextMap
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001157}
1158
Jiyong Park92315372021-04-02 08:45:46 +09001159func (a *AARImport) SdkVersion(ctx android.EarlyModuleContext) android.SdkSpec {
1160 return android.SdkSpecFrom(ctx, String(a.properties.Sdk_version))
Colin Cross83bb3162018-06-25 15:48:06 -07001161}
1162
Jiyong Parkf1691d22021-03-29 20:11:58 +09001163func (a *AARImport) SystemModules() string {
Paul Duffine25c6442019-10-11 13:50:28 +01001164 return ""
1165}
1166
Spandan Das8c9ae7e2023-03-03 21:20:36 +00001167func (a *AARImport) MinSdkVersion(ctx android.EarlyModuleContext) android.ApiLevel {
Colin Cross479884c2018-07-10 13:39:30 -07001168 if a.properties.Min_sdk_version != nil {
Spandan Das8c9ae7e2023-03-03 21:20:36 +00001169 return android.ApiLevelFrom(ctx, *a.properties.Min_sdk_version)
Colin Cross479884c2018-07-10 13:39:30 -07001170 }
Spandan Das8c9ae7e2023-03-03 21:20:36 +00001171 return a.SdkVersion(ctx).ApiLevel
Colin Cross83bb3162018-06-25 15:48:06 -07001172}
1173
Spandan Dasa26eda72023-03-02 00:56:06 +00001174func (a *AARImport) ReplaceMaxSdkVersionPlaceholder(ctx android.EarlyModuleContext) android.ApiLevel {
1175 return android.SdkSpecFrom(ctx, "").ApiLevel
William Loh5a082f92022-05-17 20:21:50 +00001176}
1177
Spandan Dasca70fc42023-03-01 23:38:49 +00001178func (a *AARImport) TargetSdkVersion(ctx android.EarlyModuleContext) android.ApiLevel {
1179 return a.SdkVersion(ctx).ApiLevel
Dan Willemsen419290a2018-10-31 15:28:47 -07001180}
1181
Colin Cross1e743852019-10-28 11:37:20 -07001182func (a *AARImport) javaVersion() string {
1183 return ""
1184}
1185
Colin Crossa97c5d32018-03-28 14:58:31 -07001186var _ AndroidLibraryDependency = (*AARImport)(nil)
1187
1188func (a *AARImport) ExportPackage() android.Path {
1189 return a.exportPackage
1190}
Colin Crossa14fb6a2024-10-23 16:57:06 -07001191func (a *AARImport) ResourcesNodeDepSet() depset.DepSet[*resourcesNode] {
Colin Crossab8d1382023-07-14 17:23:41 +00001192 return a.resourcesNodesDepSet
Colin Crossc1c37552019-01-31 11:42:41 -08001193}
1194
Colin Crossa14fb6a2024-10-23 16:57:06 -07001195func (a *AARImport) RRODirsDepSet() depset.DepSet[rroDir] {
1196 return depset.New[rroDir](depset.TOPOLOGICAL, nil, nil)
Colin Cross66f78822018-05-02 12:58:28 -07001197}
1198
Colin Crossa14fb6a2024-10-23 16:57:06 -07001199func (a *AARImport) ManifestsDepSet() depset.DepSet[android.Path] {
Colin Crossab8d1382023-07-14 17:23:41 +00001200 return a.manifestsDepSet
Jaewoong Jung6431ca72020-01-15 14:15:10 -08001201}
1202
Jaewoong Jungc779cd42020-10-06 18:56:10 -07001203// RRO enforcement is not available on aar_import since its RRO dirs are not
1204// exported.
1205func (a *AARImport) SetRROEnforcedForDependent(enforce bool) {
1206}
1207
1208// RRO enforcement is not available on aar_import since its RRO dirs are not
1209// exported.
1210func (a *AARImport) IsRROEnforced(ctx android.BaseModuleContext) bool {
1211 return false
1212}
1213
Colin Crossfabb6082018-02-20 17:22:23 -08001214func (a *AARImport) Prebuilt() *android.Prebuilt {
1215 return &a.prebuilt
1216}
1217
1218func (a *AARImport) Name() string {
1219 return a.prebuilt.Name(a.ModuleBase.Name())
1220}
1221
Jiyong Park618922e2020-01-08 13:35:43 +09001222func (a *AARImport) JacocoReportClassesFile() android.Path {
1223 return nil
1224}
1225
Colin Crossfabb6082018-02-20 17:22:23 -08001226func (a *AARImport) DepsMutator(ctx android.BottomUpMutatorContext) {
Jeongik Cha816a23a2020-07-08 01:09:23 +09001227 if !ctx.Config().AlwaysUsePrebuiltSdks() {
Jiyong Parkf1691d22021-03-29 20:11:58 +09001228 sdkDep := decodeSdkDep(ctx, android.SdkContext(a))
Colin Crossa97c5d32018-03-28 14:58:31 -07001229 if sdkDep.useModule && sdkDep.frameworkResModule != "" {
Colin Cross42d48b72018-08-29 14:10:52 -07001230 ctx.AddVariationDependencies(nil, frameworkResTag, sdkDep.frameworkResModule)
Colin Crossfabb6082018-02-20 17:22:23 -08001231 }
1232 }
Colin Crossa97c5d32018-03-28 14:58:31 -07001233
Colin Cross42d48b72018-08-29 14:10:52 -07001234 ctx.AddVariationDependencies(nil, libTag, a.properties.Libs...)
Cole Faustb7493472024-08-28 11:55:52 -07001235 ctx.AddVariationDependencies(nil, staticLibTag, a.properties.Static_libs.GetOrDefault(ctx, nil)...)
Colin Cross9055e212024-03-23 04:43:41 +00001236
1237 a.usesLibrary.deps(ctx, false)
Colin Crossfabb6082018-02-20 17:22:23 -08001238}
1239
Sam Delmerico82602492022-06-10 17:05:42 +00001240type JniPackageInfo struct {
1241 // List of zip files containing JNI libraries
1242 // Zip files should have directory structure jni/<arch>/*.so
1243 JniPackages android.Paths
1244}
1245
Colin Crossbc7d76c2023-12-12 16:39:03 -08001246var JniPackageProvider = blueprint.NewProvider[JniPackageInfo]()
Sam Delmerico82602492022-06-10 17:05:42 +00001247
1248// Unzip an AAR and extract the JNI libs for $archString.
1249var extractJNI = pctx.AndroidStaticRule("extractJNI",
1250 blueprint.RuleParams{
1251 Command: `rm -rf $out $outDir && touch $out && ` +
1252 `unzip -qoDD -d $outDir $in "jni/${archString}/*" && ` +
1253 `jni_files=$$(find $outDir/jni -type f) && ` +
1254 // print error message if there are no JNI libs for this arch
1255 `[ -n "$$jni_files" ] || (echo "ERROR: no JNI libs found for arch ${archString}" && exit 1) && ` +
Sam Delmerico80ee45c2023-06-22 15:36:02 -04001256 `${config.SoongZipCmd} -o $out -L 0 -P 'lib/${archString}' ` +
Sam Delmerico82602492022-06-10 17:05:42 +00001257 `-C $outDir/jni/${archString} $$(echo $$jni_files | xargs -n1 printf " -f %s")`,
1258 CommandDeps: []string{"${config.SoongZipCmd}"},
1259 },
1260 "outDir", "archString")
1261
Colin Crossfabb6082018-02-20 17:22:23 -08001262// Unzip an AAR into its constituent files and directories. Any files in Outputs that don't exist in the AAR will be
Dan Willemsen304cfec2019-05-28 14:49:06 -07001263// touched to create an empty file. The res directory is not extracted, as it will be extracted in its own rule.
Colin Crossfabb6082018-02-20 17:22:23 -08001264var unzipAAR = pctx.AndroidStaticRule("unzipAAR",
1265 blueprint.RuleParams{
Dan Willemsen304cfec2019-05-28 14:49:06 -07001266 Command: `rm -rf $outDir && mkdir -p $outDir && ` +
Colin Cross205e9112020-08-06 13:20:17 -07001267 `unzip -qoDD -d $outDir $in && rm -rf $outDir/res && touch $out && ` +
Michael Rosenfeld5ad15572021-12-03 13:25:10 -08001268 `${config.Zip2ZipCmd} -i $in -o $assetsPackage 'assets/**/*' && ` +
Colin Cross205e9112020-08-06 13:20:17 -07001269 `${config.MergeZipsCmd} $combinedClassesJar $$(ls $outDir/classes.jar 2> /dev/null) $$(ls $outDir/libs/*.jar 2> /dev/null)`,
Michael Rosenfeld5ad15572021-12-03 13:25:10 -08001270 CommandDeps: []string{"${config.MergeZipsCmd}", "${config.Zip2ZipCmd}"},
Colin Crossfabb6082018-02-20 17:22:23 -08001271 },
Michael Rosenfeld5ad15572021-12-03 13:25:10 -08001272 "outDir", "combinedClassesJar", "assetsPackage")
Colin Crossfabb6082018-02-20 17:22:23 -08001273
1274func (a *AARImport) GenerateAndroidBuildActions(ctx android.ModuleContext) {
1275 if len(a.properties.Aars) != 1 {
1276 ctx.PropertyErrorf("aars", "exactly one aar is required")
1277 return
1278 }
1279
Jiyong Park92315372021-04-02 08:45:46 +09001280 a.sdkVersion = a.SdkVersion(ctx)
1281 a.minSdkVersion = a.MinSdkVersion(ctx)
1282
Colin Crossff694a82023-12-13 15:54:49 -08001283 apexInfo, _ := android.ModuleProvider(ctx, android.ApexInfoProvider)
1284 a.hideApexVariantFromMake = !apexInfo.IsForPlatform()
Colin Cross56a83212020-09-15 18:30:11 -07001285
Nan Zhang4c819fb2018-08-27 18:31:46 -07001286 aarName := ctx.ModuleName() + ".aar"
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001287 a.aarPath = android.PathForModuleSrc(ctx, a.properties.Aars[0])
1288
Colin Cross1001a792019-03-21 22:21:39 -07001289 if Bool(a.properties.Jetifier) {
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001290 inputFile := a.aarPath
Colin Cross77965d92024-08-15 17:11:08 -07001291 jetifierPath := android.PathForModuleOut(ctx, "jetifier", aarName)
1292 TransformJetifier(ctx, jetifierPath, inputFile)
1293 a.aarPath = jetifierPath
Nan Zhang4c819fb2018-08-27 18:31:46 -07001294 }
Colin Crossfabb6082018-02-20 17:22:23 -08001295
Colin Cross607bbd62024-04-12 13:44:45 -07001296 jarName := ctx.ModuleName() + ".jar"
Colin Crossfabb6082018-02-20 17:22:23 -08001297 extractedAARDir := android.PathForModuleOut(ctx, "aar")
Colin Cross607bbd62024-04-12 13:44:45 -07001298 classpathFile := extractedAARDir.Join(ctx, jarName)
Colin Cross21ed4692024-04-24 20:23:38 +00001299
1300 extractedManifest := extractedAARDir.Join(ctx, "AndroidManifest.xml")
1301 providedManifest := android.OptionalPathForModuleSrc(ctx, a.properties.Manifest)
1302 if providedManifest.Valid() {
1303 a.manifest = providedManifest.Path()
1304 } else {
1305 a.manifest = extractedManifest
1306 }
1307
Colin Cross7707b242024-07-26 12:02:36 -07001308 rTxt := extractedAARDir.Join(ctx, "R.txt")
1309 assetsPackage := android.PathForModuleOut(ctx, "assets.zip")
1310 proguardFlags := extractedAARDir.Join(ctx, "proguard.txt")
Colin Crosscde55342024-03-27 14:11:51 -07001311 transitiveProguardFlags, transitiveUnconditionalExportedFlags := collectDepProguardSpecInfo(ctx)
Colin Cross40213022023-12-13 15:19:49 -08001312 android.SetProvider(ctx, ProguardSpecInfoProvider, ProguardSpecInfo{
Colin Crossa14fb6a2024-10-23 16:57:06 -07001313 ProguardFlagsFiles: depset.New[android.Path](
1314 depset.POSTORDER,
Colin Cross7707b242024-07-26 12:02:36 -07001315 android.Paths{proguardFlags},
Colin Crosscde55342024-03-27 14:11:51 -07001316 transitiveProguardFlags,
1317 ),
Colin Crossa14fb6a2024-10-23 16:57:06 -07001318 UnconditionallyExportedProguardFlags: depset.New[android.Path](
1319 depset.POSTORDER,
Sam Delmerico95d70942023-08-02 18:00:35 -04001320 nil,
Colin Crosscde55342024-03-27 14:11:51 -07001321 transitiveUnconditionalExportedFlags,
Sam Delmerico95d70942023-08-02 18:00:35 -04001322 ),
1323 })
Colin Crossfabb6082018-02-20 17:22:23 -08001324
1325 ctx.Build(pctx, android.BuildParams{
1326 Rule: unzipAAR,
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001327 Input: a.aarPath,
Colin Cross7707b242024-07-26 12:02:36 -07001328 Outputs: android.WritablePaths{classpathFile, proguardFlags, extractedManifest, assetsPackage, rTxt},
Colin Crossfabb6082018-02-20 17:22:23 -08001329 Description: "unzip AAR",
1330 Args: map[string]string{
Colin Cross205e9112020-08-06 13:20:17 -07001331 "outDir": extractedAARDir.String(),
Colin Cross9055e212024-03-23 04:43:41 +00001332 "combinedClassesJar": classpathFile.String(),
Colin Cross7707b242024-07-26 12:02:36 -07001333 "assetsPackage": assetsPackage.String(),
Colin Crossfabb6082018-02-20 17:22:23 -08001334 },
1335 })
1336
Colin Cross7707b242024-07-26 12:02:36 -07001337 a.proguardFlags = proguardFlags
1338 a.assetsPackage = assetsPackage
1339 a.rTxt = rTxt
1340
Colin Crossa0ba2f52019-06-22 12:59:27 -07001341 // Always set --pseudo-localize, it will be stripped out later for release
1342 // builds that don't want it.
1343 compileFlags := []string{"--pseudo-localize"}
Colin Crossfabb6082018-02-20 17:22:23 -08001344 compiledResDir := android.PathForModuleOut(ctx, "flat-res")
Colin Crossfabb6082018-02-20 17:22:23 -08001345 flata := compiledResDir.Join(ctx, "gen_res.flata")
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001346 aapt2CompileZip(ctx, flata, a.aarPath, "res", compileFlags)
Colin Crossfabb6082018-02-20 17:22:23 -08001347
Colin Cross7707b242024-07-26 12:02:36 -07001348 exportPackage := android.PathForModuleOut(ctx, "package-res.apk")
Colin Crossfabb6082018-02-20 17:22:23 -08001349 proguardOptionsFile := android.PathForModuleGen(ctx, "proguard.options")
Colin Crossbb77d8e2024-02-15 14:43:47 -08001350 aaptRTxt := android.PathForModuleOut(ctx, "R.txt")
Colin Cross7707b242024-07-26 12:02:36 -07001351 extraAaptPackagesFile := android.PathForModuleOut(ctx, "extra_packages")
Colin Crossfabb6082018-02-20 17:22:23 -08001352
1353 var linkDeps android.Paths
1354
1355 linkFlags := []string{
1356 "--static-lib",
Colin Cross4eae06d2023-06-20 22:40:02 -07001357 "--merge-only",
Colin Crossfabb6082018-02-20 17:22:23 -08001358 "--auto-add-overlay",
Colin Cross7c4dc5d2024-02-13 14:29:45 -08001359 "--no-static-lib-packages",
Colin Crossfabb6082018-02-20 17:22:23 -08001360 }
1361
Colin Cross10f7c4a2018-05-23 10:59:28 -07001362 linkFlags = append(linkFlags, "--manifest "+a.manifest.String())
1363 linkDeps = append(linkDeps, a.manifest)
Colin Crossfabb6082018-02-20 17:22:23 -08001364
Colin Cross8676c8c2023-10-12 15:58:57 -07001365 staticResourcesNodesDepSet, sharedResourcesNodesDepSet, staticRRODirsDepSet, staticManifestsDepSet, sharedLibs, libFlags :=
Jiakai Zhang36937082024-04-15 11:15:50 +00001366 aaptLibs(ctx, android.SdkContext(a), nil, nil)
Colin Cross31656952018-05-24 16:11:20 -07001367
Colin Cross8676c8c2023-10-12 15:58:57 -07001368 _ = sharedResourcesNodesDepSet
Colin Crossab8d1382023-07-14 17:23:41 +00001369 _ = staticRRODirsDepSet
Colin Cross8676c8c2023-10-12 15:58:57 -07001370
Colin Crossab8d1382023-07-14 17:23:41 +00001371 staticDeps := transitiveAarDeps(staticResourcesNodesDepSet.ToList())
Colin Crossfabb6082018-02-20 17:22:23 -08001372
Colin Crossab8d1382023-07-14 17:23:41 +00001373 linkDeps = append(linkDeps, sharedLibs...)
Colin Cross4eae06d2023-06-20 22:40:02 -07001374 linkDeps = append(linkDeps, staticDeps.resPackages()...)
Colin Crossa97c5d32018-03-28 14:58:31 -07001375 linkFlags = append(linkFlags, libFlags...)
Colin Crossfabb6082018-02-20 17:22:23 -08001376
Colin Cross4eae06d2023-06-20 22:40:02 -07001377 overlayRes := android.Paths{flata}
1378
1379 // Treat static library dependencies of static libraries as imports.
1380 transitiveStaticLibs := staticDeps.resPackages()
1381 linkDeps = append(linkDeps, transitiveStaticLibs...)
1382 for _, staticLib := range transitiveStaticLibs {
1383 linkFlags = append(linkFlags, "-I "+staticLib.String())
1384 }
Colin Crossfabb6082018-02-20 17:22:23 -08001385
Colin Crossab8d1382023-07-14 17:23:41 +00001386 transitiveAssets := android.ReverseSliceInPlace(staticDeps.assets())
Colin Cross7707b242024-07-26 12:02:36 -07001387 aapt2Link(ctx, exportPackage, nil, proguardOptionsFile, aaptRTxt,
Jihoon Kang84b25892023-12-01 22:01:06 +00001388 linkFlags, linkDeps, nil, overlayRes, transitiveAssets, nil, nil)
Colin Crossa6182ab2024-08-21 10:47:44 -07001389 ctx.CheckbuildFile(exportPackage)
Colin Cross7707b242024-07-26 12:02:36 -07001390 a.exportPackage = exportPackage
Colin Crossfabb6082018-02-20 17:22:23 -08001391
Colin Cross7707b242024-07-26 12:02:36 -07001392 rJar := android.PathForModuleOut(ctx, "busybox/R.jar")
1393 resourceProcessorBusyBoxGenerateBinaryR(ctx, a.rTxt, a.manifest, rJar, nil, true, nil, false)
Colin Crossa6182ab2024-08-21 10:47:44 -07001394 ctx.CheckbuildFile(rJar)
Colin Cross7707b242024-07-26 12:02:36 -07001395 a.rJar = rJar
Colin Cross4eae06d2023-06-20 22:40:02 -07001396
Colin Cross7707b242024-07-26 12:02:36 -07001397 aapt2ExtractExtraPackages(ctx, extraAaptPackagesFile, a.rJar)
1398 a.extraAaptPackagesFile = extraAaptPackagesFile
Colin Crossf3b7bad2023-08-02 15:49:00 -07001399
Colin Crossa14fb6a2024-10-23 16:57:06 -07001400 resourcesNodesDepSetBuilder := depset.NewBuilder[*resourcesNode](depset.TOPOLOGICAL)
Colin Crossab8d1382023-07-14 17:23:41 +00001401 resourcesNodesDepSetBuilder.Direct(&resourcesNode{
1402 resPackage: a.exportPackage,
1403 manifest: a.manifest,
Colin Cross4eae06d2023-06-20 22:40:02 -07001404 rTxt: a.rTxt,
1405 rJar: a.rJar,
Colin Crossab8d1382023-07-14 17:23:41 +00001406 assets: android.OptionalPathForPath(a.assetsPackage),
Colin Cross4eae06d2023-06-20 22:40:02 -07001407
1408 usedResourceProcessor: true,
Colin Crossab8d1382023-07-14 17:23:41 +00001409 })
1410 resourcesNodesDepSetBuilder.Transitive(staticResourcesNodesDepSet)
1411 a.resourcesNodesDepSet = resourcesNodesDepSetBuilder.Build()
1412
Colin Crossa14fb6a2024-10-23 16:57:06 -07001413 manifestDepSetBuilder := depset.NewBuilder[android.Path](depset.TOPOLOGICAL).Direct(a.manifest)
Colin Cross9055e212024-03-23 04:43:41 +00001414 manifestDepSetBuilder.Transitive(staticManifestsDepSet)
Colin Crossab8d1382023-07-14 17:23:41 +00001415 a.manifestsDepSet = manifestDepSetBuilder.Build()
Michael Rosenfeld5ad15572021-12-03 13:25:10 -08001416
Colin Cross312634e2023-11-21 15:13:56 -08001417 transitiveAaptResourcePackages := staticDeps.resPackages().Strings()
1418 transitiveAaptResourcePackages = slices.DeleteFunc(transitiveAaptResourcePackages, func(p string) bool {
1419 return p == a.exportPackage.String()
1420 })
1421 transitiveAaptResourcePackagesFile := android.PathForModuleOut(ctx, "transitive-res-packages")
1422 android.WriteFileRule(ctx, transitiveAaptResourcePackagesFile, strings.Join(transitiveAaptResourcePackages, "\n"))
1423 a.transitiveAaptResourcePackagesFile = transitiveAaptResourcePackagesFile
Colin Cross4eae06d2023-06-20 22:40:02 -07001424
Colin Cross9ffaf282024-08-12 13:50:09 -07001425 a.collectTransitiveHeaderJarsForR8(ctx)
Colin Cross9055e212024-03-23 04:43:41 +00001426
1427 a.classLoaderContexts = a.usesLibrary.classLoaderContextForUsesLibDeps(ctx)
1428
1429 var staticJars android.Paths
1430 var staticHeaderJars android.Paths
Colin Cross607bbd62024-04-12 13:44:45 -07001431 var staticResourceJars android.Paths
Colin Crossa14fb6a2024-10-23 16:57:06 -07001432 var transitiveStaticLibsHeaderJars []depset.DepSet[android.Path]
1433 var transitiveStaticLibsImplementationJars []depset.DepSet[android.Path]
1434 var transitiveStaticLibsResourceJars []depset.DepSet[android.Path]
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001435
Colin Cross9055e212024-03-23 04:43:41 +00001436 ctx.VisitDirectDeps(func(module android.Module) {
1437 if dep, ok := android.OtherModuleProvider(ctx, module, JavaInfoProvider); ok {
1438 tag := ctx.OtherModuleDependencyTag(module)
1439 switch tag {
1440 case staticLibTag:
1441 staticJars = append(staticJars, dep.ImplementationJars...)
1442 staticHeaderJars = append(staticHeaderJars, dep.HeaderJars...)
Colin Cross607bbd62024-04-12 13:44:45 -07001443 staticResourceJars = append(staticResourceJars, dep.ResourceJars...)
Colin Crossa14fb6a2024-10-23 16:57:06 -07001444 transitiveStaticLibsHeaderJars = append(transitiveStaticLibsHeaderJars, dep.TransitiveStaticLibsHeaderJars)
1445 transitiveStaticLibsImplementationJars = append(transitiveStaticLibsImplementationJars, dep.TransitiveStaticLibsImplementationJars)
1446 transitiveStaticLibsResourceJars = append(transitiveStaticLibsResourceJars, dep.TransitiveStaticLibsResourceJars)
Colin Cross9055e212024-03-23 04:43:41 +00001447 }
1448 }
1449 addCLCFromDep(ctx, module, a.classLoaderContexts)
Jiakai Zhang36937082024-04-15 11:15:50 +00001450 addMissingOptionalUsesLibsFromDep(ctx, module, &a.usesLibrary)
Colin Cross9055e212024-03-23 04:43:41 +00001451 })
1452
Colin Crossa14fb6a2024-10-23 16:57:06 -07001453 completeStaticLibsHeaderJars := depset.New(depset.PREORDER, android.Paths{classpathFile}, transitiveStaticLibsHeaderJars)
1454 completeStaticLibsImplementationJars := depset.New(depset.PREORDER, android.Paths{classpathFile}, transitiveStaticLibsImplementationJars)
1455 completeStaticLibsResourceJars := depset.New(depset.PREORDER, nil, transitiveStaticLibsResourceJars)
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001456
Colin Cross77965d92024-08-15 17:11:08 -07001457 var implementationJarFile android.Path
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001458 var combineJars android.Paths
1459 if ctx.Config().UseTransitiveJarsInClasspath() {
1460 combineJars = completeStaticLibsImplementationJars.ToList()
1461 } else {
1462 combineJars = append(android.Paths{classpathFile}, staticJars...)
1463 }
1464
1465 if len(combineJars) > 1 {
1466 implementationJarOutputPath := android.PathForModuleOut(ctx, "combined", jarName)
1467 TransformJarsToJar(ctx, implementationJarOutputPath, "combine", combineJars, android.OptionalPath{}, false, nil, nil)
1468 implementationJarFile = implementationJarOutputPath
Colin Cross9055e212024-03-23 04:43:41 +00001469 } else {
Colin Cross28ac2ff2024-04-02 12:21:34 -07001470 implementationJarFile = classpathFile
Colin Cross9055e212024-03-23 04:43:41 +00001471 }
1472
Colin Cross607bbd62024-04-12 13:44:45 -07001473 var resourceJarFile android.Path
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001474 var resourceJars android.Paths
1475 if ctx.Config().UseTransitiveJarsInClasspath() {
1476 resourceJars = completeStaticLibsResourceJars.ToList()
1477 } else {
1478 resourceJars = staticResourceJars
1479 }
1480 if len(resourceJars) > 1 {
Colin Cross607bbd62024-04-12 13:44:45 -07001481 combinedJar := android.PathForModuleOut(ctx, "res-combined", jarName)
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001482 TransformJarsToJar(ctx, combinedJar, "for resources", resourceJars, android.OptionalPath{},
Colin Cross607bbd62024-04-12 13:44:45 -07001483 false, nil, nil)
1484 resourceJarFile = combinedJar
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001485 } else if len(resourceJars) == 1 {
1486 resourceJarFile = resourceJars[0]
Colin Cross607bbd62024-04-12 13:44:45 -07001487 }
1488
1489 // merge implementation jar with resources if necessary
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001490 var implementationAndResourcesJars android.Paths
1491 if ctx.Config().UseTransitiveJarsInClasspath() {
1492 implementationAndResourcesJars = append(slices.Clone(resourceJars), combineJars...)
1493 } else {
1494 implementationAndResourcesJars = android.PathsIfNonNil(resourceJarFile, implementationJarFile)
1495 }
1496 var implementationAndResourcesJar android.Path
1497 if len(implementationAndResourcesJars) > 1 {
Colin Cross77965d92024-08-15 17:11:08 -07001498 combinedJar := android.PathForModuleOut(ctx, "withres", jarName)
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001499 TransformJarsToJar(ctx, combinedJar, "for resources", implementationAndResourcesJars, android.OptionalPath{},
Colin Cross607bbd62024-04-12 13:44:45 -07001500 false, nil, nil)
1501 implementationAndResourcesJar = combinedJar
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001502 } else {
1503 implementationAndResourcesJar = implementationAndResourcesJars[0]
Colin Cross607bbd62024-04-12 13:44:45 -07001504 }
1505
1506 a.implementationJarFile = implementationJarFile
Colin Cross28ac2ff2024-04-02 12:21:34 -07001507 // Save the output file with no relative path so that it doesn't end up in a subdirectory when used as a resource
Colin Cross607bbd62024-04-12 13:44:45 -07001508 a.implementationAndResourcesJarFile = implementationAndResourcesJar.WithoutRel()
Colin Cross28ac2ff2024-04-02 12:21:34 -07001509
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001510 var headerJars android.Paths
1511 if ctx.Config().UseTransitiveJarsInClasspath() {
1512 headerJars = completeStaticLibsHeaderJars.ToList()
1513 } else {
1514 headerJars = append(android.Paths{classpathFile}, staticHeaderJars...)
1515 }
1516 if len(headerJars) > 1 {
Colin Cross7707b242024-07-26 12:02:36 -07001517 headerJarFile := android.PathForModuleOut(ctx, "turbine-combined", jarName)
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001518 TransformJarsToJar(ctx, headerJarFile, "combine header jars", headerJars, android.OptionalPath{}, false, nil, nil)
Colin Cross7707b242024-07-26 12:02:36 -07001519 a.headerJarFile = headerJarFile
Colin Cross9055e212024-03-23 04:43:41 +00001520 } else {
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001521 a.headerJarFile = headerJars[0]
Colin Cross9055e212024-03-23 04:43:41 +00001522 }
1523
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001524 if ctx.Config().UseTransitiveJarsInClasspath() {
1525 ctx.CheckbuildFile(classpathFile)
1526 } else {
1527 ctx.CheckbuildFile(a.headerJarFile)
1528 ctx.CheckbuildFile(a.implementationJarFile)
1529 }
Colin Crossa6182ab2024-08-21 10:47:44 -07001530
Colin Cross7727c7f2024-07-18 15:36:32 -07001531 android.SetProvider(ctx, JavaInfoProvider, &JavaInfo{
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001532 HeaderJars: android.PathsIfNonNil(a.headerJarFile),
1533 LocalHeaderJars: android.PathsIfNonNil(classpathFile),
1534 TransitiveStaticLibsHeaderJars: completeStaticLibsHeaderJars,
1535 TransitiveStaticLibsImplementationJars: completeStaticLibsImplementationJars,
1536 TransitiveStaticLibsResourceJars: completeStaticLibsResourceJars,
1537 ResourceJars: android.PathsIfNonNil(resourceJarFile),
1538 TransitiveLibsHeaderJarsForR8: a.transitiveLibsHeaderJarsForR8,
1539 TransitiveStaticLibsHeaderJarsForR8: a.transitiveStaticLibsHeaderJarsForR8,
1540 ImplementationAndResourcesJars: android.PathsIfNonNil(a.implementationAndResourcesJarFile),
1541 ImplementationJars: android.PathsIfNonNil(a.implementationJarFile),
1542 StubsLinkType: Implementation,
Joe Onorato6fe59eb2023-07-16 13:20:33 -07001543 // TransitiveAconfigFiles: // TODO(b/289117800): LOCAL_ACONFIG_FILES for prebuilts
Colin Crossdcf71b22021-02-01 13:59:03 -08001544 })
Sam Delmerico82602492022-06-10 17:05:42 +00001545
1546 if proptools.Bool(a.properties.Extract_jni) {
1547 for _, t := range ctx.MultiTargets() {
1548 arch := t.Arch.Abi[0]
1549 path := android.PathForModuleOut(ctx, arch+"_jni.zip")
1550 a.jniPackages = append(a.jniPackages, path)
1551
1552 outDir := android.PathForModuleOut(ctx, "aarForJni")
1553 aarPath := android.PathForModuleSrc(ctx, a.properties.Aars[0])
1554 ctx.Build(pctx, android.BuildParams{
1555 Rule: extractJNI,
1556 Input: aarPath,
1557 Outputs: android.WritablePaths{path},
1558 Description: "extract JNI from AAR",
1559 Args: map[string]string{
1560 "outDir": outDir.String(),
1561 "archString": arch,
1562 },
1563 })
1564 }
Sam Delmerico82602492022-06-10 17:05:42 +00001565 }
Colin Crosse8eeec92023-12-14 14:50:05 -08001566
Colin Cross40213022023-12-13 15:19:49 -08001567 android.SetProvider(ctx, JniPackageProvider, JniPackageInfo{
Colin Crosse8eeec92023-12-14 14:50:05 -08001568 JniPackages: a.jniPackages,
1569 })
mrziwang68786d82024-07-09 10:41:55 -07001570
1571 ctx.SetOutputFiles([]android.Path{a.implementationAndResourcesJarFile}, "")
1572 ctx.SetOutputFiles([]android.Path{a.aarPath}, ".aar")
Colin Crossdcf71b22021-02-01 13:59:03 -08001573}
Colin Crossfabb6082018-02-20 17:22:23 -08001574
1575func (a *AARImport) HeaderJars() android.Paths {
Colin Cross9055e212024-03-23 04:43:41 +00001576 return android.Paths{a.headerJarFile}
Colin Crossfabb6082018-02-20 17:22:23 -08001577}
1578
Colin Cross331a1212018-08-15 20:40:52 -07001579func (a *AARImport) ImplementationAndResourcesJars() android.Paths {
Colin Cross607bbd62024-04-12 13:44:45 -07001580 return android.Paths{a.implementationAndResourcesJarFile}
Colin Cross331a1212018-08-15 20:40:52 -07001581}
1582
Colin Cross9055e212024-03-23 04:43:41 +00001583func (a *AARImport) DexJarBuildPath(ctx android.ModuleErrorfContext) OptionalDexJarPath {
1584 return OptionalDexJarPath{}
Colin Crossf24a22a2019-01-31 14:12:44 -08001585}
1586
Ulya Trafimovich9f3052c2020-06-09 14:31:19 +01001587func (a *AARImport) DexJarInstallPath() android.Path {
1588 return nil
1589}
1590
Ulya Trafimovichb23d28c2020-10-08 12:53:58 +01001591func (a *AARImport) ClassLoaderContexts() dexpreopt.ClassLoaderContextMap {
Colin Cross9055e212024-03-23 04:43:41 +00001592 return a.classLoaderContexts
Jiyong Park1be96912018-05-28 18:02:19 +09001593}
1594
Colin Cross9055e212024-03-23 04:43:41 +00001595var _ UsesLibraryDependency = (*AARImport)(nil)
1596
Jiyong Park45bf82e2020-12-15 22:29:02 +09001597var _ android.ApexModule = (*AARImport)(nil)
1598
1599// Implements android.ApexModule
Jooyung Hanacc7bbe2020-05-20 09:06:00 +09001600func (a *AARImport) DepIsInSameApex(ctx android.BaseModuleContext, dep android.Module) bool {
1601 return a.depIsInSameApex(ctx, dep)
1602}
1603
Jiyong Park45bf82e2020-12-15 22:29:02 +09001604// Implements android.ApexModule
Colin Cross9055e212024-03-23 04:43:41 +00001605func (a *AARImport) ShouldSupportSdkVersion(ctx android.BaseModuleContext,
Dan Albertc8060532020-07-22 22:32:17 -07001606 sdkVersion android.ApiLevel) error {
Jooyung Han749dc692020-04-15 11:03:39 +09001607 return nil
1608}
1609
Sam Delmericoaf8bb702022-07-25 15:39:32 -04001610var _ android.PrebuiltInterface = (*AARImport)(nil)
Colin Crossfabb6082018-02-20 17:22:23 -08001611
Jiakai Zhangf98da192024-04-15 11:15:41 +00001612func (a *AARImport) UsesLibrary() *usesLibrary {
1613 return &a.usesLibrary
1614}
1615
1616var _ ModuleWithUsesLibrary = (*AARImport)(nil)
1617
Colin Cross1b16b0e2019-02-12 14:41:32 -08001618// android_library_import imports an `.aar` file into the build graph as if it was built with android_library.
1619//
1620// This module is not suitable for installing on a device, but can be used as a `static_libs` dependency of
1621// an android_app module.
Colin Crossfabb6082018-02-20 17:22:23 -08001622func AARImportFactory() android.Module {
1623 module := &AARImport{}
1624
Colin Cross9055e212024-03-23 04:43:41 +00001625 module.AddProperties(
1626 &module.properties,
1627 &module.usesLibrary.usesLibraryProperties,
1628 )
Colin Crossfabb6082018-02-20 17:22:23 -08001629
1630 android.InitPrebuiltModule(module, &module.properties.Aars)
Jooyung Hanacc7bbe2020-05-20 09:06:00 +09001631 android.InitApexModule(module)
Sam Delmerico82602492022-06-10 17:05:42 +00001632 InitJavaModuleMultiTargets(module, android.DeviceSupported)
Colin Crossfabb6082018-02-20 17:22:23 -08001633 return module
1634}
Spandan Dasb08bf772024-08-07 21:55:42 +00001635
Cole Faustb36d31d2024-08-27 16:04:28 -07001636func (a *AARImport) IDEInfo(ctx android.BaseModuleContext, dpInfo *android.IdeInfo) {
Spandan Dasb08bf772024-08-07 21:55:42 +00001637 dpInfo.Jars = append(dpInfo.Jars, a.headerJarFile.String(), a.rJar.String())
1638}