blob: ebada655fc086312c75c77ee772a720596189255 [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 Cross13e17822025-02-19 15:07:52 -0800220 return BoolDefault(a.aaptProperties.Use_resource_processor, true) &&
Colin Crosseee4ab12024-03-27 11:54:10 -0700221 // TODO(b/331641946): remove this when ResourceProcessorBusyBox supports generating shared libraries.
Spandan Das15d7af52025-03-03 18:19:20 +0000222 !slices.Contains(a.aaptProperties.Aaptflags, "--shared-lib")
Colin Cross4eae06d2023-06-20 22:40:02 -0700223}
224
Inseob Kim34dc4cd2023-11-07 13:37:14 +0900225func (a *aapt) filterProduct() string {
226 return String(a.aaptProperties.Filter_product)
227}
228
Colin Crossa97c5d32018-03-28 14:58:31 -0700229func (a *aapt) ExportPackage() android.Path {
230 return a.exportPackage
231}
Colin Crossa14fb6a2024-10-23 16:57:06 -0700232func (a *aapt) ResourcesNodeDepSet() depset.DepSet[*resourcesNode] {
Colin Crossab8d1382023-07-14 17:23:41 +0000233 return a.resourcesNodesDepSet
Colin Crossc1c37552019-01-31 11:42:41 -0800234}
235
Colin Crossa14fb6a2024-10-23 16:57:06 -0700236func (a *aapt) RRODirsDepSet() depset.DepSet[rroDir] {
Colin Crossab8d1382023-07-14 17:23:41 +0000237 return a.rroDirsDepSet
Colin Crossc1c37552019-01-31 11:42:41 -0800238}
239
Colin Crossa14fb6a2024-10-23 16:57:06 -0700240func (a *aapt) ManifestsDepSet() depset.DepSet[android.Path] {
Colin Crossab8d1382023-07-14 17:23:41 +0000241 return a.manifestsDepSet
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800242}
243
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700244func (a *aapt) SetRROEnforcedForDependent(enforce bool) {
245 a.aaptProperties.RROEnforcedForDependent = enforce
246}
247
248func (a *aapt) IsRROEnforced(ctx android.BaseModuleContext) bool {
249 // True if RRO is enforced for this module or...
250 return ctx.Config().EnforceRROForModule(ctx.ModuleName()) ||
Jeongik Chacee5ba92021-02-19 12:11:51 +0900251 // if RRO is enforced for any of its dependents.
252 a.aaptProperties.RROEnforcedForDependent
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700253}
254
Jiyong Parkf1691d22021-03-29 20:11:58 +0900255func (a *aapt) aapt2Flags(ctx android.ModuleContext, sdkContext android.SdkContext,
Spandan Das82229552024-12-06 01:38:45 +0000256 manifestPath android.Path, doNotIncludeAssetDirImplicitly bool) (compileFlags, linkFlags []string, linkDeps android.Paths,
Colin Crossa0ba2f52019-06-22 12:59:27 -0700257 resDirs, overlayDirs []globbedResourceDir, rroDirs []rroDir, resZips android.Paths) {
Colin Crossa97c5d32018-03-28 14:58:31 -0700258
Jaewoong Jung3aff5782020-02-11 07:54:35 -0800259 hasVersionCode := android.PrefixInList(a.aaptProperties.Aaptflags, "--version-code")
260 hasVersionName := android.PrefixInList(a.aaptProperties.Aaptflags, "--version-name")
Colin Crossa97c5d32018-03-28 14:58:31 -0700261
Colin Crossa97c5d32018-03-28 14:58:31 -0700262 // Flags specified in Android.bp
263 linkFlags = append(linkFlags, a.aaptProperties.Aaptflags...)
264
Eric Miao40eab202023-03-30 16:57:17 +0000265 linkFlags = append(linkFlags, "--enable-compact-entries")
Colin Crossa97c5d32018-03-28 14:58:31 -0700266
267 // Find implicit or explicit asset and resource dirs
Jiakai Zhangba82e282023-10-13 18:08:59 +0100268 assets := android.PathsRelativeToModuleSourceDir(android.SourceInput{
269 Context: ctx,
270 Paths: a.aaptProperties.Assets,
271 IncludeDirs: false,
272 })
Spandan Das82229552024-12-06 01:38:45 +0000273 var assetDirs android.Paths
274 if doNotIncludeAssetDirImplicitly {
275 assetDirs = android.PathsForModuleSrc(ctx, a.aaptProperties.Asset_dirs)
276 } else {
277 assetDirs = android.PathsWithOptionalDefaultForModuleSrc(ctx, a.aaptProperties.Asset_dirs, "assets")
278 }
Cole Faust5b989772024-11-06 11:03:16 -0800279 resourceDirs := android.PathsWithOptionalDefaultForModuleSrc(ctx, a.aaptProperties.Resource_dirs.GetOrDefault(ctx, nil), "res")
Colin Cross8a497952019-03-05 22:25:09 -0800280 resourceZips := android.PathsForModuleSrc(ctx, a.aaptProperties.Resource_zips)
Colin Crossa97c5d32018-03-28 14:58:31 -0700281
Colin Crossa97c5d32018-03-28 14:58:31 -0700282 // Glob directories into lists of paths
283 for _, dir := range resourceDirs {
284 resDirs = append(resDirs, globbedResourceDir{
285 dir: dir,
286 files: androidResourceGlob(ctx, dir),
287 })
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700288 resOverlayDirs, resRRODirs := overlayResourceGlob(ctx, a, dir)
Colin Crossa97c5d32018-03-28 14:58:31 -0700289 overlayDirs = append(overlayDirs, resOverlayDirs...)
290 rroDirs = append(rroDirs, resRRODirs...)
291 }
292
Cole Faust6265120c2024-09-05 16:27:56 -0700293 assetDirsHasher := sha256.New()
Colin Crossc20dc852020-11-10 12:27:45 -0800294 var assetDeps android.Paths
Cole Faust6265120c2024-09-05 16:27:56 -0700295 for _, dir := range assetDirs {
Colin Crossc20dc852020-11-10 12:27:45 -0800296 // Add a dependency on every file in the asset directory. This ensures the aapt2
297 // rule will be rerun if one of the files in the asset directory is modified.
Cole Faust6265120c2024-09-05 16:27:56 -0700298 dirContents := androidResourceGlob(ctx, dir)
299 assetDeps = append(assetDeps, dirContents...)
Colin Crossc20dc852020-11-10 12:27:45 -0800300
Cole Faust6265120c2024-09-05 16:27:56 -0700301 // Add a hash of all the files in the asset directory to the command line.
Colin Crossc20dc852020-11-10 12:27:45 -0800302 // This ensures the aapt2 rule will be run if a file is removed from the asset directory,
303 // or a file is added whose timestamp is older than the output of aapt2.
Cole Faust6265120c2024-09-05 16:27:56 -0700304 for _, path := range dirContents.Strings() {
305 assetDirsHasher.Write([]byte(path))
306 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700307 }
308
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700309 assetDirStrings := assetDirs.Strings()
310 if a.noticeFile.Valid() {
311 assetDirStrings = append(assetDirStrings, filepath.Dir(a.noticeFile.Path().String()))
Colin Crossc20dc852020-11-10 12:27:45 -0800312 assetDeps = append(assetDeps, a.noticeFile.Path())
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700313 }
Jiakai Zhangba82e282023-10-13 18:08:59 +0100314 if len(assets) > 0 {
315 // aapt2 doesn't support adding individual asset files. Create a temp directory to hold asset
316 // files and pass it to aapt2.
317 tmpAssetDir := android.PathForModuleOut(ctx, "tmp_asset_dir")
318
319 rule := android.NewRuleBuilder(pctx, ctx)
320 rule.Command().
321 Text("rm -rf").Text(tmpAssetDir.String()).
322 Text("&&").
323 Text("mkdir -p").Text(tmpAssetDir.String())
324
325 for _, asset := range assets {
326 output := tmpAssetDir.Join(ctx, asset.Rel())
327 assetDeps = append(assetDeps, output)
328 rule.Command().Text("mkdir -p").Text(filepath.Dir(output.String()))
329 rule.Command().Text("cp").Input(asset).Output(output)
330 }
331
332 rule.Build("tmp_asset_dir", "tmp_asset_dir")
333
334 assetDirStrings = append(assetDirStrings, tmpAssetDir.String())
335 }
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700336
Colin Crossa97c5d32018-03-28 14:58:31 -0700337 linkFlags = append(linkFlags, "--manifest "+manifestPath.String())
338 linkDeps = append(linkDeps, manifestPath)
339
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700340 linkFlags = append(linkFlags, android.JoinWithPrefix(assetDirStrings, "-A "))
Cole Faust6265120c2024-09-05 16:27:56 -0700341 linkFlags = append(linkFlags, fmt.Sprintf("$$(: %x)", assetDirsHasher.Sum(nil)))
Colin Crossc20dc852020-11-10 12:27:45 -0800342 linkDeps = append(linkDeps, assetDeps...)
Colin Crossa97c5d32018-03-28 14:58:31 -0700343
Spandan Das50885c02023-02-23 21:31:33 +0000344 // Returns the effective version for {min|target}_sdk_version
Spandan Das8c9ae7e2023-03-03 21:20:36 +0000345 effectiveVersionString := func(sdkVersion android.SdkSpec, minSdkVersion android.ApiLevel) string {
Spandan Das50885c02023-02-23 21:31:33 +0000346 // If {min|target}_sdk_version is current, use sdk_version to determine the effective level
347 // This is necessary for vendor modules.
348 // The effective version does not _only_ depend on {min|target}_sdk_version(level),
349 // but also on the sdk_version (kind+level)
Spandan Das8c9ae7e2023-03-03 21:20:36 +0000350 if minSdkVersion.IsCurrent() {
Spandan Das50885c02023-02-23 21:31:33 +0000351 ret, err := sdkVersion.EffectiveVersionString(ctx)
352 if err != nil {
353 ctx.ModuleErrorf("invalid sdk_version: %s", err)
354 }
355 return ret
356 }
357 ret, err := minSdkVersion.EffectiveVersionString(ctx)
358 if err != nil {
359 ctx.ModuleErrorf("invalid min_sdk_version: %s", err)
360 }
361 return ret
Jiyong Park6a927c42020-01-21 02:03:43 +0900362 }
Spandan Das50885c02023-02-23 21:31:33 +0000363 // SDK version flags
364 sdkVersion := sdkContext.SdkVersion(ctx)
365 minSdkVersion := effectiveVersionString(sdkVersion, sdkContext.MinSdkVersion(ctx))
Colin Crossa97c5d32018-03-28 14:58:31 -0700366
Colin Cross83bb3162018-06-25 15:48:06 -0700367 linkFlags = append(linkFlags, "--min-sdk-version "+minSdkVersion)
Spandan Das6450b552023-02-23 19:27:07 +0000368 // Use minSdkVersion for target-sdk-version, even if `target_sdk_version` is set
369 // This behavior has been copied from Make.
Colin Cross83bb3162018-06-25 15:48:06 -0700370 linkFlags = append(linkFlags, "--target-sdk-version "+minSdkVersion)
Colin Crossa97c5d32018-03-28 14:58:31 -0700371
Colin Crossa97c5d32018-03-28 14:58:31 -0700372 // Version code
373 if !hasVersionCode {
Dan Albert4f378d72020-07-23 17:32:15 -0700374 linkFlags = append(linkFlags, "--version-code", ctx.Config().PlatformSdkVersion().String())
Colin Crossa97c5d32018-03-28 14:58:31 -0700375 }
376
377 if !hasVersionName {
Colin Cross402d5e02018-04-25 14:54:06 -0700378 var versionName string
379 if ctx.ModuleName() == "framework-res" {
380 // Some builds set AppsDefaultVersionName() to include the build number ("O-123456"). aapt2 copies the
381 // version name of framework-res into app manifests as compileSdkVersionCodename, which confuses things
Colin Crossbfd347d2018-05-09 11:11:35 -0700382 // if it contains the build number. Use the PlatformVersionName instead.
383 versionName = ctx.Config().PlatformVersionName()
Colin Cross402d5e02018-04-25 14:54:06 -0700384 } else {
385 versionName = ctx.Config().AppsDefaultVersionName()
386 }
Colin Cross0b9f31f2019-02-28 11:00:01 -0800387 versionName = proptools.NinjaEscape(versionName)
Colin Crossa97c5d32018-03-28 14:58:31 -0700388 linkFlags = append(linkFlags, "--version-name ", versionName)
389 }
Yurii Zubrytskyi3afd7952024-10-29 18:09:44 -0700390 // Split the flags by prefix, as --png-compression-level has the "=value" suffix.
391 linkFlags, compileFlags = android.FilterListByPrefix(linkFlags,
392 []string{"--legacy", "--png-compression-level"})
Colin Crossa0ba2f52019-06-22 12:59:27 -0700393
394 // Always set --pseudo-localize, it will be stripped out later for release
395 // builds that don't want it.
396 compileFlags = append(compileFlags, "--pseudo-localize")
397
398 return compileFlags, linkFlags, linkDeps, resDirs, overlayDirs, rroDirs, resourceZips
Colin Crossa97c5d32018-03-28 14:58:31 -0700399}
400
Paul Duffin250e6192019-06-07 10:44:37 +0100401func (a *aapt) deps(ctx android.BottomUpMutatorContext, sdkDep sdkDep) {
Colin Cross42308aa2018-11-14 21:44:17 -0800402 if sdkDep.frameworkResModule != "" {
403 ctx.AddVariationDependencies(nil, frameworkResTag, sdkDep.frameworkResModule)
Colin Crossa97c5d32018-03-28 14:58:31 -0700404 }
405}
406
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800407var extractAssetsRule = pctx.AndroidStaticRule("extractAssets",
408 blueprint.RuleParams{
409 Command: `${config.Zip2ZipCmd} -i ${in} -o ${out} "assets/**/*"`,
410 CommandDeps: []string{"${config.Zip2ZipCmd}"},
411 })
412
Alixf7a10272023-09-27 16:47:56 +0000413type aaptBuildActionOptions struct {
414 sdkContext android.SdkContext
415 classLoaderContexts dexpreopt.ClassLoaderContextMap
416 excludedLibs []string
417 enforceDefaultTargetSdkVersion bool
Rico Winda2fa2632024-03-13 13:09:17 +0100418 forceNonFinalResourceIDs bool
Alixf7a10272023-09-27 16:47:56 +0000419 extraLinkFlags []string
Jihoon Kang84b25892023-12-01 22:01:06 +0000420 aconfigTextFiles android.Paths
Jiakai Zhang36937082024-04-15 11:15:50 +0000421 usesLibrary *usesLibrary
Spandan Dase2f98da2024-11-18 19:22:39 +0000422 // If rroDirs is provided, it will be used to generate package-res.apk
423 rroDirs *android.Paths
424 // If manifestForAapt is not nil, it will be used for aapt instead of the default source manifest.
425 manifestForAapt android.Path
426}
427
428func filterRRO(rroDirsDepSet depset.DepSet[rroDir], filter overlayType) android.Paths {
429 var paths android.Paths
430 seen := make(map[android.Path]bool)
431 for _, d := range rroDirsDepSet.ToList() {
432 if d.overlayType == filter {
433 if seen[d.path] {
434 continue
435 }
436 seen[d.path] = true
437 paths = append(paths, d.path)
438 }
439 }
440 return paths
Alixf7a10272023-09-27 16:47:56 +0000441}
442
443func (a *aapt) buildActions(ctx android.ModuleContext, opts aaptBuildActionOptions) {
Colin Cross5446e882019-05-22 10:46:27 -0700444
Colin Cross8676c8c2023-10-12 15:58:57 -0700445 staticResourcesNodesDepSet, sharedResourcesNodesDepSet, staticRRODirsDepSet, staticManifestsDepSet, sharedExportPackages, libFlags :=
Jiakai Zhang36937082024-04-15 11:15:50 +0000446 aaptLibs(ctx, opts.sdkContext, opts.classLoaderContexts, opts.usesLibrary)
Ulya Trafimovich31e444e2020-08-14 17:32:16 +0100447
Paul Duffin06530572022-02-03 17:54:15 +0000448 // Exclude any libraries from the supplied list.
Alixf7a10272023-09-27 16:47:56 +0000449 opts.classLoaderContexts = opts.classLoaderContexts.ExcludeLibs(opts.excludedLibs)
Paul Duffin06530572022-02-03 17:54:15 +0000450
Colin Cross31656952018-05-24 16:11:20 -0700451 // App manifest file
Spandan Dase2f98da2024-11-18 19:22:39 +0000452 var manifestFilePath android.Path
453 if opts.manifestForAapt != nil {
454 manifestFilePath = opts.manifestForAapt
455 } else {
456 manifestFile := proptools.StringDefault(a.aaptProperties.Manifest, "AndroidManifest.xml")
457 manifestFilePath = android.PathForModuleSrc(ctx, manifestFile)
458 }
Colin Cross31656952018-05-24 16:11:20 -0700459
Spandan Dase2f98da2024-11-18 19:22:39 +0000460 manifestPath := ManifestFixer(ctx, manifestFilePath, ManifestFixerParams{
Alixf7a10272023-09-27 16:47:56 +0000461 SdkContext: opts.sdkContext,
462 ClassLoaderContexts: opts.classLoaderContexts,
Harshit Mahajan5b8b7302022-06-10 11:24:05 +0000463 IsLibrary: a.isLibrary,
464 DefaultManifestVersion: a.defaultManifestVersion,
465 UseEmbeddedNativeLibs: a.useEmbeddedNativeLibs,
466 UsesNonSdkApis: a.usesNonSdkApis,
467 UseEmbeddedDex: a.useEmbeddedDex,
468 HasNoCode: a.hasNoCode,
469 LoggingParent: a.LoggingParent,
Alixf7a10272023-09-27 16:47:56 +0000470 EnforceDefaultTargetSdkVersion: opts.enforceDefaultTargetSdkVersion,
Gurpreet Singh75d65f32022-01-24 17:44:05 +0000471 })
Colin Cross90c25c62019-04-19 16:22:57 -0700472
Colin Crossab8d1382023-07-14 17:23:41 +0000473 staticDeps := transitiveAarDeps(staticResourcesNodesDepSet.ToList())
Colin Cross8676c8c2023-10-12 15:58:57 -0700474 sharedDeps := transitiveAarDeps(sharedResourcesNodesDepSet.ToList())
Colin Crossab8d1382023-07-14 17:23:41 +0000475
Luca Stefanifd898822019-09-10 22:13:31 +0200476 // Add additional manifest files to transitive manifests.
477 additionalManifests := android.PathsForModuleSrc(ctx, a.aaptProperties.Additional_manifests)
Colin Crossab8d1382023-07-14 17:23:41 +0000478 transitiveManifestPaths := append(android.Paths{manifestPath}, additionalManifests...)
Colin Crossab8d1382023-07-14 17:23:41 +0000479 transitiveManifestPaths = append(transitiveManifestPaths, staticManifestsDepSet.ToList()...)
Colin Cross90c25c62019-04-19 16:22:57 -0700480
Colin Crossab8d1382023-07-14 17:23:41 +0000481 if len(transitiveManifestPaths) > 1 && !Bool(a.aaptProperties.Dont_merge_manifests) {
Alixf7a10272023-09-27 16:47:56 +0000482 manifestMergerParams := ManifestMergerParams{
483 staticLibManifests: transitiveManifestPaths[1:],
Alix96ea88452023-08-31 15:48:23 +0000484 isLibrary: a.isLibrary,
485 packageName: a.manifestValues.applicationId,
486 }
Alixf7a10272023-09-27 16:47:56 +0000487 a.mergedManifestFile = manifestMerger(ctx, transitiveManifestPaths[0], manifestMergerParams)
Colin Crossa6182ab2024-08-21 10:47:44 -0700488 ctx.CheckbuildFile(a.mergedManifestFile)
Colin Cross90c25c62019-04-19 16:22:57 -0700489 if !a.isLibrary {
490 // Only use the merged manifest for applications. For libraries, the transitive closure of manifests
491 // will be propagated to the final application and merged there. The merged manifest for libraries is
492 // only passed to Make, which can't handle transitive dependencies.
493 manifestPath = a.mergedManifestFile
494 }
495 } else {
496 a.mergedManifestFile = manifestPath
497 }
Colin Cross31656952018-05-24 16:11:20 -0700498
Spandan Das82229552024-12-06 01:38:45 +0000499 // do not include assets in autogenerated RRO.
500 compileFlags, linkFlags, linkDeps, resDirs, overlayDirs, rroDirs, resZips := a.aapt2Flags(ctx, opts.sdkContext, manifestPath, opts.rroDirs != nil)
Colin Cross31656952018-05-24 16:11:20 -0700501
Spandan Dase2f98da2024-11-18 19:22:39 +0000502 a.rroDirsDepSet = depset.NewBuilder[rroDir](depset.TOPOLOGICAL).
503 Direct(rroDirs...).
504 Transitive(staticRRODirsDepSet).Build()
505
Colin Cross31656952018-05-24 16:11:20 -0700506 linkFlags = append(linkFlags, libFlags...)
Colin Cross8676c8c2023-10-12 15:58:57 -0700507 linkDeps = append(linkDeps, sharedExportPackages...)
Colin Crossab8d1382023-07-14 17:23:41 +0000508 linkDeps = append(linkDeps, staticDeps.resPackages()...)
Alixf7a10272023-09-27 16:47:56 +0000509 linkFlags = append(linkFlags, opts.extraLinkFlags...)
Colin Cross1b6a3cf2018-07-24 14:51:30 -0700510 if a.isLibrary {
511 linkFlags = append(linkFlags, "--static-lib")
512 }
Rico Wind7152e822024-04-19 08:14:15 +0200513 if opts.forceNonFinalResourceIDs {
514 linkFlags = append(linkFlags, "--non-final-ids")
515 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700516
Colin Cross7c4dc5d2024-02-13 14:29:45 -0800517 linkFlags = append(linkFlags, "--no-static-lib-packages")
Spandan Das457e5062024-10-05 01:00:34 +0000518 if a.isLibrary {
519 // Pass --merge-only to skip resource references validation until the final
520 // app link step when when all static libraries are present.
Colin Cross4eae06d2023-06-20 22:40:02 -0700521 linkFlags = append(linkFlags, "--merge-only")
Colin Cross4eae06d2023-06-20 22:40:02 -0700522 }
523
Colin Crossa97c5d32018-03-28 14:58:31 -0700524 packageRes := android.PathForModuleOut(ctx, "package-res.apk")
Colin Crossa97c5d32018-03-28 14:58:31 -0700525 proguardOptionsFile := android.PathForModuleGen(ctx, "proguard.options")
526 rTxt := android.PathForModuleOut(ctx, "R.txt")
Colin Cross66f78822018-05-02 12:58:28 -0700527 // This file isn't used by Soong, but is generated for exporting
528 extraPackages := android.PathForModuleOut(ctx, "extra_packages")
Colin Cross4eae06d2023-06-20 22:40:02 -0700529 var transitiveRJars android.Paths
Colin Crossf3b7bad2023-08-02 15:49:00 -0700530 var srcJar android.WritablePath
Colin Crossa97c5d32018-03-28 14:58:31 -0700531
Colin Cross4aaa84a2018-08-21 15:14:37 -0700532 var compiledResDirs []android.Paths
Colin Crossa97c5d32018-03-28 14:58:31 -0700533 for _, dir := range resDirs {
Colin Cross014489c2020-06-02 20:09:13 -0700534 a.resourceFiles = append(a.resourceFiles, dir.files...)
Jihoon Kang98ea8362024-07-16 18:20:03 +0000535 compiledResDirs = append(compiledResDirs, aapt2Compile(ctx, dir.dir, dir.files,
536 compileFlags, a.filterProduct(), opts.aconfigTextFiles).Paths())
Colin Crossa97c5d32018-03-28 14:58:31 -0700537 }
Colin Cross4aaa84a2018-08-21 15:14:37 -0700538
Colin Crossa592e3e2019-02-19 16:59:53 -0800539 for i, zip := range resZips {
540 flata := android.PathForModuleOut(ctx, fmt.Sprintf("reszip.%d.flata", i))
Colin Crossa0ba2f52019-06-22 12:59:27 -0700541 aapt2CompileZip(ctx, flata, zip, "", compileFlags)
Colin Crossa592e3e2019-02-19 16:59:53 -0800542 compiledResDirs = append(compiledResDirs, android.Paths{flata})
543 }
544
Colin Cross4aaa84a2018-08-21 15:14:37 -0700545 var compiledRes, compiledOverlay android.Paths
546
Colin Crossab8d1382023-07-14 17:23:41 +0000547 // AAPT2 overlays are in lowest to highest priority order, reverse the topological order
548 // of transitiveStaticLibs.
549 transitiveStaticLibs := android.ReversePaths(staticDeps.resPackages())
550
Colin Cross8f1b0332024-01-25 13:39:06 -0800551 if a.isLibrary && a.useResourceProcessorBusyBox(ctx) {
Colin Cross4eae06d2023-06-20 22:40:02 -0700552 // When building an android_library with ResourceProcessorBusyBox enabled treat static library dependencies
553 // as imports. The resources from dependencies will not be merged into this module's package-res.apk, and
554 // instead modules depending on this module will reference package-res.apk from all transitive static
555 // dependencies.
Colin Cross1d3f5902024-03-05 11:51:54 -0800556 for _, sharedDep := range sharedDeps {
557 if sharedDep.usedResourceProcessor {
558 transitiveRJars = append(transitiveRJars, sharedDep.rJar)
559 }
560 }
Colin Cross4eae06d2023-06-20 22:40:02 -0700561 for _, staticDep := range staticDeps {
562 linkDeps = append(linkDeps, staticDep.resPackage)
563 linkFlags = append(linkFlags, "-I "+staticDep.resPackage.String())
564 if staticDep.usedResourceProcessor {
565 transitiveRJars = append(transitiveRJars, staticDep.rJar)
566 }
567 }
568 } else {
569 // When building an app or building a library without ResourceProcessorBusyBox enabled all static
570 // dependencies are compiled into this module's package-res.apk as overlays.
571 compiledOverlay = append(compiledOverlay, transitiveStaticLibs...)
572 }
Colin Cross4aaa84a2018-08-21 15:14:37 -0700573
Colin Crossbec85302019-02-13 13:15:46 -0800574 if len(transitiveStaticLibs) > 0 {
Colin Cross4aaa84a2018-08-21 15:14:37 -0700575 // If we are using static android libraries, every source file becomes an overlay.
576 // This is to emulate old AAPT behavior which simulated library support.
577 for _, compiledResDir := range compiledResDirs {
578 compiledOverlay = append(compiledOverlay, compiledResDir...)
579 }
Colin Crossbec85302019-02-13 13:15:46 -0800580 } else if a.isLibrary {
581 // Otherwise, for a static library we treat all the resources equally with no overlay.
582 for _, compiledResDir := range compiledResDirs {
583 compiledRes = append(compiledRes, compiledResDir...)
584 }
Colin Cross4aaa84a2018-08-21 15:14:37 -0700585 } else if len(compiledResDirs) > 0 {
586 // Without static libraries, the first directory is our directory, which can then be
587 // overlaid by the rest.
588 compiledRes = append(compiledRes, compiledResDirs[0]...)
589 for _, compiledResDir := range compiledResDirs[1:] {
590 compiledOverlay = append(compiledOverlay, compiledResDir...)
591 }
592 }
593
Spandan Dase2f98da2024-11-18 19:22:39 +0000594 var compiledRro, compiledRroOverlay android.Paths
595 if opts.rroDirs != nil {
596 compiledRro, compiledRroOverlay = a.compileResInDir(ctx, *opts.rroDirs, compileFlags, opts.aconfigTextFiles)
Spandan Dasde588a32024-12-03 22:52:24 +0000597 } else {
598 // RRO enforcement is done based on module name. Compile the overlayDirs only if rroDirs is nil.
599 // This ensures that the autogenerated RROs do not compile the overlay dirs twice.
600 for _, dir := range overlayDirs {
601 compiledOverlay = append(compiledOverlay, aapt2Compile(ctx, dir.dir, dir.files,
602 compileFlags, a.filterProduct(), opts.aconfigTextFiles).Paths()...)
603 }
Spandan Dase2f98da2024-11-18 19:22:39 +0000604 }
605
Colin Crosse560c4a2019-03-19 16:03:11 -0700606 var splitPackages android.WritablePaths
607 var splits []split
608
609 for _, s := range a.splitNames {
610 suffix := strings.Replace(s, ",", "_", -1)
611 path := android.PathForModuleOut(ctx, "package_"+suffix+".apk")
612 linkFlags = append(linkFlags, "--split", path.String()+":"+s)
613 splitPackages = append(splitPackages, path)
614 splits = append(splits, split{
615 name: s,
616 suffix: suffix,
617 path: path,
618 })
619 }
620
Colin Cross8f1b0332024-01-25 13:39:06 -0800621 if !a.useResourceProcessorBusyBox(ctx) {
Colin Crossf3b7bad2023-08-02 15:49:00 -0700622 // the subdir "android" is required to be filtered by package names
623 srcJar = android.PathForModuleGen(ctx, "android", "R.srcjar")
624 }
625
Colin Crossab8d1382023-07-14 17:23:41 +0000626 // No need to specify assets from dependencies to aapt2Link for libraries, all transitive assets will be
627 // provided to the final app aapt2Link step.
628 var transitiveAssets android.Paths
629 if !a.isLibrary {
630 transitiveAssets = android.ReverseSliceInPlace(staticDeps.assets())
631 }
Spandan Dase2f98da2024-11-18 19:22:39 +0000632 if opts.rroDirs == nil { // link resources and overlay
633 aapt2Link(ctx, packageRes, srcJar, proguardOptionsFile, rTxt,
634 linkFlags, linkDeps, compiledRes, compiledOverlay, transitiveAssets, splitPackages,
635 opts.aconfigTextFiles)
636 ctx.CheckbuildFile(packageRes)
637 } else { // link autogenerated rro
638 if len(compiledRro) == 0 {
639 return
640 }
641 aapt2Link(ctx, packageRes, srcJar, proguardOptionsFile, rTxt,
642 linkFlags, linkDeps, compiledRro, compiledRroOverlay, nil, nil,
643 opts.aconfigTextFiles)
644 ctx.CheckbuildFile(packageRes)
645 }
Colin Crossa6182ab2024-08-21 10:47:44 -0700646
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800647 // Extract assets from the resource package output so that they can be used later in aapt2link
648 // for modules that depend on this one.
Colin Crossab8d1382023-07-14 17:23:41 +0000649 if android.PrefixInList(linkFlags, "-A ") {
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800650 assets := android.PathForModuleOut(ctx, "assets.zip")
651 ctx.Build(pctx, android.BuildParams{
652 Rule: extractAssetsRule,
653 Input: packageRes,
654 Output: assets,
655 Description: "extract assets from built resource file",
656 })
657 a.assetPackage = android.OptionalPathForPath(assets)
658 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700659
Colin Cross8f1b0332024-01-25 13:39:06 -0800660 if a.useResourceProcessorBusyBox(ctx) {
Colin Cross4eae06d2023-06-20 22:40:02 -0700661 rJar := android.PathForModuleOut(ctx, "busybox/R.jar")
Rico Winda2fa2632024-03-13 13:09:17 +0100662 resourceProcessorBusyBoxGenerateBinaryR(ctx, rTxt, a.mergedManifestFile, rJar, staticDeps, a.isLibrary, a.aaptProperties.Aaptflags,
663 opts.forceNonFinalResourceIDs)
Colin Crossf3b7bad2023-08-02 15:49:00 -0700664 aapt2ExtractExtraPackages(ctx, extraPackages, rJar)
Colin Cross4eae06d2023-06-20 22:40:02 -0700665 transitiveRJars = append(transitiveRJars, rJar)
666 a.rJar = rJar
Colin Crossf3b7bad2023-08-02 15:49:00 -0700667 } else {
668 aapt2ExtractExtraPackages(ctx, extraPackages, srcJar)
Colin Cross4eae06d2023-06-20 22:40:02 -0700669 }
670
Colin Cross312634e2023-11-21 15:13:56 -0800671 transitiveAaptResourcePackages := staticDeps.resPackages().Strings()
672 transitiveAaptResourcePackages = slices.DeleteFunc(transitiveAaptResourcePackages, func(p string) bool {
673 return p == packageRes.String()
674 })
675 transitiveAaptResourcePackagesFile := android.PathForModuleOut(ctx, "transitive-res-packages")
676 android.WriteFileRule(ctx, transitiveAaptResourcePackagesFile, strings.Join(transitiveAaptResourcePackages, "\n"))
677
Colin Cross1d3f5902024-03-05 11:51:54 -0800678 // Reverse the list of R.jar files so that the current module comes first, and direct dependencies come before
679 // transitive dependencies.
680 transitiveRJars = android.ReversePaths(transitiveRJars)
681
Colin Crossa97c5d32018-03-28 14:58:31 -0700682 a.aaptSrcJar = srcJar
Colin Cross4eae06d2023-06-20 22:40:02 -0700683 a.transitiveAaptRJars = transitiveRJars
Colin Cross312634e2023-11-21 15:13:56 -0800684 a.transitiveAaptResourcePackagesFile = transitiveAaptResourcePackagesFile
Colin Crossa97c5d32018-03-28 14:58:31 -0700685 a.exportPackage = packageRes
686 a.manifestPath = manifestPath
687 a.proguardOptionsFile = proguardOptionsFile
Colin Cross66f78822018-05-02 12:58:28 -0700688 a.extraAaptPackagesFile = extraPackages
Colin Crossa97c5d32018-03-28 14:58:31 -0700689 a.rTxt = rTxt
Colin Crosse560c4a2019-03-19 16:03:11 -0700690 a.splits = splits
Colin Crossa14fb6a2024-10-23 16:57:06 -0700691 a.resourcesNodesDepSet = depset.NewBuilder[*resourcesNode](depset.TOPOLOGICAL).
Colin Crossab8d1382023-07-14 17:23:41 +0000692 Direct(&resourcesNode{
693 resPackage: a.exportPackage,
694 manifest: a.manifestPath,
695 additionalManifests: additionalManifests,
Colin Cross4eae06d2023-06-20 22:40:02 -0700696 rTxt: a.rTxt,
697 rJar: a.rJar,
Colin Crossab8d1382023-07-14 17:23:41 +0000698 assets: a.assetPackage,
Colin Cross4eae06d2023-06-20 22:40:02 -0700699
Colin Cross8f1b0332024-01-25 13:39:06 -0800700 usedResourceProcessor: a.useResourceProcessorBusyBox(ctx),
Colin Crossab8d1382023-07-14 17:23:41 +0000701 }).
702 Transitive(staticResourcesNodesDepSet).Build()
Colin Crossa14fb6a2024-10-23 16:57:06 -0700703 a.manifestsDepSet = depset.NewBuilder[android.Path](depset.TOPOLOGICAL).
Colin Crossab8d1382023-07-14 17:23:41 +0000704 Direct(a.manifestPath).
705 DirectSlice(additionalManifests).
706 Transitive(staticManifestsDepSet).Build()
707}
708
Spandan Dase2f98da2024-11-18 19:22:39 +0000709// comileResInDir finds the resource files in dirs by globbing and then compiles them using aapt2
710// returns the file paths of compiled resources
711// dirs[0] is used as compileRes
712// dirs[1:] is used as compileOverlay
713func (a *aapt) compileResInDir(ctx android.ModuleContext, dirs android.Paths, compileFlags []string, aconfig android.Paths) (android.Paths, android.Paths) {
714 filesInDir := func(dir android.Path) android.Paths {
715 files, err := ctx.GlobWithDeps(filepath.Join(dir.String(), "**/*"), androidResourceIgnoreFilenames)
716 if err != nil {
717 ctx.ModuleErrorf("failed to glob overlay resource dir %q: %s", dir, err.Error())
718 return nil
719 }
720 var filePaths android.Paths
721 for _, file := range files {
722 if strings.HasSuffix(file, "/") {
723 continue // ignore directories
724 }
725 filePaths = append(filePaths, android.PathForSource(ctx, file))
726 }
727 return filePaths
728 }
729
730 var compiledRes, compiledOverlay android.Paths
731 if len(dirs) == 0 {
732 return nil, nil
733 }
734 compiledRes = append(compiledRes, aapt2Compile(ctx, dirs[0], filesInDir(dirs[0]), compileFlags, a.filterProduct(), aconfig).Paths()...)
735 if len(dirs) > 0 {
736 for _, dir := range dirs[1:] {
737 compiledOverlay = append(compiledOverlay, aapt2Compile(ctx, dir, filesInDir(dir), compileFlags, a.filterProduct(), aconfig).Paths()...)
738 }
739 }
740 return compiledRes, compiledOverlay
741}
742
Colin Cross4eae06d2023-06-20 22:40:02 -0700743var resourceProcessorBusyBox = pctx.AndroidStaticRule("resourceProcessorBusyBox",
744 blueprint.RuleParams{
745 Command: "${config.JavaCmd} -cp ${config.ResourceProcessorBusyBox} " +
746 "com.google.devtools.build.android.ResourceProcessorBusyBox --tool=GENERATE_BINARY_R -- @${out}.args && " +
747 "if cmp -s ${out}.tmp ${out} ; then rm ${out}.tmp ; else mv ${out}.tmp ${out}; fi",
748 CommandDeps: []string{"${config.ResourceProcessorBusyBox}"},
749 Rspfile: "${out}.args",
750 RspfileContent: "--primaryRTxt ${rTxt} --primaryManifest ${manifest} --classJarOutput ${out}.tmp ${args}",
751 Restat: true,
752 }, "rTxt", "manifest", "args")
753
754// resourceProcessorBusyBoxGenerateBinaryR converts the R.txt file produced by aapt2 into R.class files
755// using Bazel's ResourceProcessorBusyBox tool, which is faster than compiling the R.java files and
756// supports producing classes for static dependencies that only include resources from that dependency.
757func resourceProcessorBusyBoxGenerateBinaryR(ctx android.ModuleContext, rTxt, manifest android.Path,
Rico Winda2fa2632024-03-13 13:09:17 +0100758 rJar android.WritablePath, transitiveDeps transitiveAarDeps, isLibrary bool, aaptFlags []string,
759 forceNonFinalIds bool) {
Colin Cross4eae06d2023-06-20 22:40:02 -0700760
761 var args []string
762 var deps android.Paths
763
764 if !isLibrary {
765 // When compiling an app, pass all R.txt and AndroidManifest.xml from transitive static library dependencies
766 // to ResourceProcessorBusyBox so that it can regenerate R.class files with the final resource IDs for each
767 // package.
768 args, deps = transitiveDeps.resourceProcessorDeps()
Rico Winda2fa2632024-03-13 13:09:17 +0100769 if forceNonFinalIds {
770 args = append(args, "--finalFields=false")
771 }
Colin Cross4eae06d2023-06-20 22:40:02 -0700772 } else {
773 // When compiling a library don't pass any dependencies as it only needs to generate an R.class file for this
774 // library. Pass --finalFields=false so that the R.class file contains non-final fields so they don't get
775 // inlined into the library before the final IDs are assigned during app compilation.
776 args = append(args, "--finalFields=false")
777 }
778
Colin Crossd3f7d1a2024-01-03 19:42:25 -0800779 for i, arg := range aaptFlags {
780 const AAPT_CUSTOM_PACKAGE = "--custom-package"
781 if strings.HasPrefix(arg, AAPT_CUSTOM_PACKAGE) {
782 pkg := strings.TrimSpace(strings.TrimPrefix(arg, AAPT_CUSTOM_PACKAGE))
783 if pkg == "" && i+1 < len(aaptFlags) {
784 pkg = aaptFlags[i+1]
785 }
786 args = append(args, "--packageForR "+pkg)
787 }
788 }
789
Colin Cross4eae06d2023-06-20 22:40:02 -0700790 deps = append(deps, rTxt, manifest)
791
792 ctx.Build(pctx, android.BuildParams{
793 Rule: resourceProcessorBusyBox,
794 Output: rJar,
795 Implicits: deps,
796 Description: "ResourceProcessorBusyBox",
797 Args: map[string]string{
798 "rTxt": rTxt.String(),
799 "manifest": manifest.String(),
800 "args": strings.Join(args, " "),
801 },
802 })
803}
804
Colin Crossab8d1382023-07-14 17:23:41 +0000805type resourcesNode struct {
806 resPackage android.Path
807 manifest android.Path
808 additionalManifests android.Paths
Colin Cross4eae06d2023-06-20 22:40:02 -0700809 rTxt android.Path
810 rJar android.Path
Colin Crossab8d1382023-07-14 17:23:41 +0000811 assets android.OptionalPath
Colin Cross4eae06d2023-06-20 22:40:02 -0700812
813 usedResourceProcessor bool
Colin Crossab8d1382023-07-14 17:23:41 +0000814}
815
816type transitiveAarDeps []*resourcesNode
817
818func (t transitiveAarDeps) resPackages() android.Paths {
Colin Cross4eae06d2023-06-20 22:40:02 -0700819 paths := make(android.Paths, 0, len(t))
Colin Crossab8d1382023-07-14 17:23:41 +0000820 for _, dep := range t {
821 paths = append(paths, dep.resPackage)
822 }
Colin Cross4eae06d2023-06-20 22:40:02 -0700823 return paths
Colin Crossab8d1382023-07-14 17:23:41 +0000824}
825
826func (t transitiveAarDeps) manifests() android.Paths {
Colin Cross4eae06d2023-06-20 22:40:02 -0700827 paths := make(android.Paths, 0, len(t))
Colin Crossab8d1382023-07-14 17:23:41 +0000828 for _, dep := range t {
829 paths = append(paths, dep.manifest)
830 paths = append(paths, dep.additionalManifests...)
831 }
Colin Cross4eae06d2023-06-20 22:40:02 -0700832 return paths
833}
834
835func (t transitiveAarDeps) resourceProcessorDeps() (args []string, deps android.Paths) {
836 for _, dep := range t {
837 args = append(args, "--library="+dep.rTxt.String()+","+dep.manifest.String())
838 deps = append(deps, dep.rTxt, dep.manifest)
839 }
840 return args, deps
Colin Crossab8d1382023-07-14 17:23:41 +0000841}
842
843func (t transitiveAarDeps) assets() android.Paths {
Colin Cross4eae06d2023-06-20 22:40:02 -0700844 paths := make(android.Paths, 0, len(t))
Colin Crossab8d1382023-07-14 17:23:41 +0000845 for _, dep := range t {
846 if dep.assets.Valid() {
847 paths = append(paths, dep.assets.Path())
848 }
849 }
850 return paths
Colin Crossa97c5d32018-03-28 14:58:31 -0700851}
852
853// aaptLibs collects libraries from dependencies and sdk_version and converts them into paths
Jiakai Zhang36937082024-04-15 11:15:50 +0000854func aaptLibs(ctx android.ModuleContext, sdkContext android.SdkContext,
855 classLoaderContexts dexpreopt.ClassLoaderContextMap, usesLibrary *usesLibrary) (
Colin Crossa14fb6a2024-10-23 16:57:06 -0700856 staticResourcesNodes, sharedResourcesNodes depset.DepSet[*resourcesNode], staticRRODirs depset.DepSet[rroDir],
857 staticManifests depset.DepSet[android.Path], sharedLibs android.Paths, flags []string) {
Colin Crossa97c5d32018-03-28 14:58:31 -0700858
Ulya Trafimovichb23d28c2020-10-08 12:53:58 +0100859 if classLoaderContexts == nil {
Ulya Trafimovich18554242020-11-03 15:55:11 +0000860 // Not all callers need to compute class loader context, those who don't just pass nil.
861 // Create a temporary class loader context here (it will be computed, but not used).
Ulya Trafimovichb23d28c2020-10-08 12:53:58 +0100862 classLoaderContexts = make(dexpreopt.ClassLoaderContextMap)
Ulya Trafimovich18554242020-11-03 15:55:11 +0000863 }
864
Colin Cross83bb3162018-06-25 15:48:06 -0700865 sdkDep := decodeSdkDep(ctx, sdkContext)
Colin Crossa97c5d32018-03-28 14:58:31 -0700866 if sdkDep.useFiles {
Colin Cross86a60ae2018-05-29 14:44:55 -0700867 sharedLibs = append(sharedLibs, sdkDep.jars...)
Colin Crossa97c5d32018-03-28 14:58:31 -0700868 }
869
Colin Crossa14fb6a2024-10-23 16:57:06 -0700870 var staticResourcesNodeDepSets []depset.DepSet[*resourcesNode]
871 var sharedResourcesNodeDepSets []depset.DepSet[*resourcesNode]
872 rroDirsDepSetBuilder := depset.NewBuilder[rroDir](depset.TOPOLOGICAL)
873 manifestsDepSetBuilder := depset.NewBuilder[android.Path](depset.TOPOLOGICAL)
Colin Crossab8d1382023-07-14 17:23:41 +0000874
Yu Liu460cf372025-01-10 00:34:06 +0000875 ctx.VisitDirectDepsProxy(func(module android.ModuleProxy) {
Ulya Trafimovich65b03192020-12-03 16:50:22 +0000876 depTag := ctx.OtherModuleDependencyTag(module)
Ulya Trafimovich18554242020-11-03 15:55:11 +0000877
Colin Crossa97c5d32018-03-28 14:58:31 -0700878 var exportPackage android.Path
Yu Liu460cf372025-01-10 00:34:06 +0000879 var aarDep *AndroidLibraryDependencyInfo
880 javaInfo, ok := android.OtherModuleProvider(ctx, module, JavaInfoProvider)
881 if ok && javaInfo.AndroidLibraryDependencyInfo != nil {
882 aarDep = javaInfo.AndroidLibraryDependencyInfo
883 exportPackage = aarDep.ExportPackage
Colin Crossa97c5d32018-03-28 14:58:31 -0700884 }
885
Ulya Trafimovich65b03192020-12-03 16:50:22 +0000886 switch depTag {
Colin Cross4b964c02018-10-15 16:18:06 -0700887 case instrumentationForTag:
888 // Nothing, instrumentationForTag is treated as libTag for javac but not for aapt2.
Spandan Dase2f98da2024-11-18 19:22:39 +0000889 case sdkLibTag, libTag, rroDepTag:
Colin Cross5446e882019-05-22 10:46:27 -0700890 if exportPackage != nil {
Yu Liu460cf372025-01-10 00:34:06 +0000891 sharedResourcesNodeDepSets = append(sharedResourcesNodeDepSets, aarDep.ResourcesNodeDepSet)
Colin Cross5446e882019-05-22 10:46:27 -0700892 sharedLibs = append(sharedLibs, exportPackage)
893 }
Colin Cross5446e882019-05-22 10:46:27 -0700894 case frameworkResTag:
Colin Crossa97c5d32018-03-28 14:58:31 -0700895 if exportPackage != nil {
896 sharedLibs = append(sharedLibs, exportPackage)
897 }
898 case staticLibTag:
899 if exportPackage != nil {
Yu Liu460cf372025-01-10 00:34:06 +0000900 staticResourcesNodeDepSets = append(staticResourcesNodeDepSets, aarDep.ResourcesNodeDepSet)
901 rroDirsDepSetBuilder.Transitive(aarDep.RRODirsDepSet)
902 manifestsDepSetBuilder.Transitive(aarDep.ManifestsDepSet)
Colin Crossa97c5d32018-03-28 14:58:31 -0700903 }
904 }
Ulya Trafimovich18554242020-11-03 15:55:11 +0000905
Ulya Trafimovich88bb6f62020-12-16 16:16:11 +0000906 addCLCFromDep(ctx, module, classLoaderContexts)
Jiakai Zhang36937082024-04-15 11:15:50 +0000907 if usesLibrary != nil {
908 addMissingOptionalUsesLibsFromDep(ctx, module, usesLibrary)
909 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700910 })
911
Colin Crossab8d1382023-07-14 17:23:41 +0000912 // AAPT2 overlays are in lowest to highest priority order, the topological order will be reversed later.
913 // Reverse the dependency order now going into the depset so that it comes out in order after the second
914 // reverse later.
915 // NOTE: this is legacy and probably incorrect behavior, for most other cases (e.g. conflicting classes in
916 // dependencies) the highest priority dependency is listed first, but for resources the highest priority
Colin Cross9055e212024-03-23 04:43:41 +0000917 // dependency has to be listed last. This is also inconsistent with the way manifests from the same
918 // transitive dependencies are merged.
Colin Crossa14fb6a2024-10-23 16:57:06 -0700919 staticResourcesNodes = depset.New(depset.TOPOLOGICAL, nil,
Colin Cross8676c8c2023-10-12 15:58:57 -0700920 android.ReverseSliceInPlace(staticResourcesNodeDepSets))
Colin Crossa14fb6a2024-10-23 16:57:06 -0700921 sharedResourcesNodes = depset.New(depset.TOPOLOGICAL, nil,
Colin Cross8676c8c2023-10-12 15:58:57 -0700922 android.ReverseSliceInPlace(sharedResourcesNodeDepSets))
Colin Crossa97c5d32018-03-28 14:58:31 -0700923
Colin Crossab8d1382023-07-14 17:23:41 +0000924 staticRRODirs = rroDirsDepSetBuilder.Build()
925 staticManifests = manifestsDepSetBuilder.Build()
926
927 if len(staticResourcesNodes.ToList()) > 0 {
Colin Crossa97c5d32018-03-28 14:58:31 -0700928 flags = append(flags, "--auto-add-overlay")
929 }
930
931 for _, sharedLib := range sharedLibs {
932 flags = append(flags, "-I "+sharedLib.String())
933 }
934
Colin Cross8676c8c2023-10-12 15:58:57 -0700935 return staticResourcesNodes, sharedResourcesNodes, staticRRODirs, staticManifests, sharedLibs, flags
Colin Crossa97c5d32018-03-28 14:58:31 -0700936}
937
Yu Liuc0a36302025-01-10 22:26:01 +0000938type AndroidLibraryInfo struct {
939 // Empty for now
940}
941
942var AndroidLibraryInfoProvider = blueprint.NewProvider[AndroidLibraryInfo]()
943
Yu Liu2a815b62025-02-21 20:46:25 +0000944type AARImportInfo struct {
945 // Empty for now
946}
947
948var AARImportInfoProvider = blueprint.NewProvider[AARImportInfo]()
949
Colin Crossa97c5d32018-03-28 14:58:31 -0700950type AndroidLibrary struct {
951 Library
952 aapt
953
954 androidLibraryProperties androidLibraryProperties
955
956 aarFile android.WritablePath
Colin Cross89c31582018-04-30 15:55:11 -0700957}
958
Colin Crossa97c5d32018-03-28 14:58:31 -0700959var _ AndroidLibraryDependency = (*AndroidLibrary)(nil)
960
961func (a *AndroidLibrary) DepsMutator(ctx android.BottomUpMutatorContext) {
Jiakai Zhangf98da192024-04-15 11:15:41 +0000962 a.usesLibrary.deps(ctx, false)
Colin Crossa97c5d32018-03-28 14:58:31 -0700963 a.Module.deps(ctx)
Jiyong Parkf1691d22021-03-29 20:11:58 +0900964 sdkDep := decodeSdkDep(ctx, android.SdkContext(a))
Paul Duffin250e6192019-06-07 10:44:37 +0100965 if sdkDep.hasFrameworkLibs() {
966 a.aapt.deps(ctx, sdkDep)
Colin Crossa97c5d32018-03-28 14:58:31 -0700967 }
Jihoon Kang9049c272024-03-19 21:57:36 +0000968
969 for _, aconfig_declaration := range a.aaptProperties.Flags_packages {
970 ctx.AddDependency(ctx.Module(), aconfigDeclarationTag, aconfig_declaration)
971 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700972}
973
974func (a *AndroidLibrary) GenerateAndroidBuildActions(ctx android.ModuleContext) {
Colin Crosse4246ab2019-02-05 21:55:21 -0800975 a.aapt.isLibrary = true
Ulya Trafimovich42c7f0d2021-08-17 16:20:29 +0100976 a.classLoaderContexts = a.usesLibrary.classLoaderContextForUsesLibDeps(ctx)
Spandan Das0727ba72024-02-13 16:37:43 +0000977 if a.usesLibrary.shouldDisableDexpreopt {
978 a.dexpreopter.disableDexpreopt()
979 }
Jihoon Kang9aef7772024-06-14 23:45:06 +0000980 aconfigTextFilePaths := getAconfigFilePaths(ctx)
Alixf7a10272023-09-27 16:47:56 +0000981 a.aapt.buildActions(ctx,
982 aaptBuildActionOptions{
983 sdkContext: android.SdkContext(a),
984 classLoaderContexts: a.classLoaderContexts,
985 enforceDefaultTargetSdkVersion: false,
Jihoon Kang9aef7772024-06-14 23:45:06 +0000986 aconfigTextFiles: aconfigTextFilePaths,
Jiakai Zhang36937082024-04-15 11:15:50 +0000987 usesLibrary: &a.usesLibrary,
Alixf7a10272023-09-27 16:47:56 +0000988 },
989 )
Colin Crossa97c5d32018-03-28 14:58:31 -0700990
Colin Crossff694a82023-12-13 15:54:49 -0800991 apexInfo, _ := android.ModuleProvider(ctx, android.ApexInfoProvider)
992 a.hideApexVariantFromMake = !apexInfo.IsForPlatform()
Colin Cross56a83212020-09-15 18:30:11 -0700993
yangbill2af0b6e2024-03-15 09:29:29 +0000994 a.stem = proptools.StringDefault(a.overridableProperties.Stem, ctx.ModuleName())
Jihoon Kang1bfb6f22023-07-01 00:13:47 +0000995
Colin Cross4eae06d2023-06-20 22:40:02 -0700996 ctx.CheckbuildFile(a.aapt.proguardOptionsFile)
997 ctx.CheckbuildFile(a.aapt.exportPackage)
Colin Cross8f1b0332024-01-25 13:39:06 -0800998 if a.useResourceProcessorBusyBox(ctx) {
Colin Cross4eae06d2023-06-20 22:40:02 -0700999 ctx.CheckbuildFile(a.aapt.rJar)
Colin Crossf3b7bad2023-08-02 15:49:00 -07001000 } else {
1001 ctx.CheckbuildFile(a.aapt.aaptSrcJar)
Colin Cross4eae06d2023-06-20 22:40:02 -07001002 }
Colin Crossa97c5d32018-03-28 14:58:31 -07001003
1004 // apps manifests are handled by aapt, don't let Module see them
1005 a.properties.Manifest = nil
1006
Colin Cross014489c2020-06-02 20:09:13 -07001007 a.linter.mergedManifest = a.aapt.mergedManifestFile
1008 a.linter.manifest = a.aapt.manifestPath
1009 a.linter.resources = a.aapt.resourceFiles
1010
Sam Delmericoc8e040c2023-10-31 17:27:02 +00001011 proguardSpecInfo := a.collectProguardSpecInfo(ctx)
Colin Cross40213022023-12-13 15:19:49 -08001012 android.SetProvider(ctx, ProguardSpecInfoProvider, proguardSpecInfo)
Colin Cross312634e2023-11-21 15:13:56 -08001013 exportedProguardFlagsFiles := proguardSpecInfo.ProguardFlagsFiles.ToList()
1014 a.extraProguardFlagsFiles = append(a.extraProguardFlagsFiles, exportedProguardFlagsFiles...)
1015 a.extraProguardFlagsFiles = append(a.extraProguardFlagsFiles, a.proguardOptionsFile)
1016
1017 combinedExportedProguardFlagFile := android.PathForModuleOut(ctx, "export_proguard_flags")
1018 writeCombinedProguardFlagsFile(ctx, combinedExportedProguardFlagFile, exportedProguardFlagsFiles)
1019 a.combinedExportedProguardFlagsFile = combinedExportedProguardFlagFile
Colin Crossa97c5d32018-03-28 14:58:31 -07001020
Colin Cross4eae06d2023-06-20 22:40:02 -07001021 var extraSrcJars android.Paths
1022 var extraCombinedJars android.Paths
1023 var extraClasspathJars android.Paths
Colin Cross8f1b0332024-01-25 13:39:06 -08001024 if a.useResourceProcessorBusyBox(ctx) {
Colin Cross4eae06d2023-06-20 22:40:02 -07001025 // When building a library with ResourceProcessorBusyBox enabled ResourceProcessorBusyBox for this
1026 // library and each of the transitive static android_library dependencies has already created an
1027 // R.class file for the appropriate package. Add all of those R.class files to the classpath.
1028 extraClasspathJars = a.transitiveAaptRJars
1029 } else {
1030 // When building a library without ResourceProcessorBusyBox the aapt2 rule creates R.srcjar containing
1031 // R.java files for the library's package and the packages from all transitive static android_library
1032 // dependencies. Compile the srcjar alongside the rest of the sources.
1033 extraSrcJars = android.Paths{a.aapt.aaptSrcJar}
1034 }
1035
Yu Liu460cf372025-01-10 00:34:06 +00001036 javaInfo := a.Module.compile(ctx, extraSrcJars, extraClasspathJars, extraCombinedJars, nil)
Colin Crossa97c5d32018-03-28 14:58:31 -07001037
Colin Crossf57c5782019-01-25 13:20:38 -08001038 a.aarFile = android.PathForModuleOut(ctx, ctx.ModuleName()+".aar")
Colin Crossa97c5d32018-03-28 14:58:31 -07001039 var res android.Paths
1040 if a.androidLibraryProperties.BuildAAR {
1041 BuildAAR(ctx, a.aarFile, a.outputFile, a.manifestPath, a.rTxt, res)
Colin Crossa97c5d32018-03-28 14:58:31 -07001042 }
Colin Cross89c31582018-04-30 15:55:11 -07001043
Sam Delmerico82602492022-06-10 17:05:42 +00001044 prebuiltJniPackages := android.Paths{}
Yu Liucbb50c22025-01-15 20:57:49 +00001045 ctx.VisitDirectDepsProxy(func(module android.ModuleProxy) {
Colin Cross313aa542023-12-13 13:47:44 -08001046 if info, ok := android.OtherModuleProvider(ctx, module, JniPackageProvider); ok {
Sam Delmerico82602492022-06-10 17:05:42 +00001047 prebuiltJniPackages = append(prebuiltJniPackages, info.JniPackages...)
1048 }
1049 })
1050 if len(prebuiltJniPackages) > 0 {
Colin Cross40213022023-12-13 15:19:49 -08001051 android.SetProvider(ctx, JniPackageProvider, JniPackageInfo{
Sam Delmerico82602492022-06-10 17:05:42 +00001052 JniPackages: prebuiltJniPackages,
1053 })
1054 }
Jihoon Kang9aef7772024-06-14 23:45:06 +00001055
1056 android.SetProvider(ctx, FlagsPackagesProvider, FlagsPackages{
1057 AconfigTextFiles: aconfigTextFilePaths,
1058 })
mrziwang9f7b9f42024-07-10 12:18:06 -07001059
Yu Liu42f69972024-12-13 22:46:12 +00001060 android.SetProvider(ctx, AndroidLibraryInfoProvider, AndroidLibraryInfo{})
1061
Yu Liu460cf372025-01-10 00:34:06 +00001062 if javaInfo != nil {
1063 setExtraJavaInfo(ctx, a, javaInfo)
1064 android.SetProvider(ctx, JavaInfoProvider, javaInfo)
1065 }
1066
mrziwang9f7b9f42024-07-10 12:18:06 -07001067 a.setOutputFiles(ctx)
Wei Li986fe742025-01-30 15:14:42 -08001068
1069 buildComplianceMetadata(ctx)
mrziwang9f7b9f42024-07-10 12:18:06 -07001070}
1071
1072func (a *AndroidLibrary) setOutputFiles(ctx android.ModuleContext) {
1073 ctx.SetOutputFiles([]android.Path{a.aarFile}, ".aar")
1074 setOutputFiles(ctx, a.Library.Module)
Colin Crossa97c5d32018-03-28 14:58:31 -07001075}
1076
Cole Faustb36d31d2024-08-27 16:04:28 -07001077func (a *AndroidLibrary) IDEInfo(ctx android.BaseModuleContext, dpInfo *android.IdeInfo) {
1078 a.Library.IDEInfo(ctx, dpInfo)
1079 a.aapt.IDEInfo(ctx, dpInfo)
Colin Cross95b53b82023-10-17 13:21:02 -07001080}
1081
Cole Faustb36d31d2024-08-27 16:04:28 -07001082func (a *aapt) IDEInfo(ctx android.BaseModuleContext, dpInfo *android.IdeInfo) {
Colin Cross8f1b0332024-01-25 13:39:06 -08001083 if a.rJar != nil {
Colin Cross95b53b82023-10-17 13:21:02 -07001084 dpInfo.Jars = append(dpInfo.Jars, a.rJar.String())
1085 }
1086}
1087
Colin Cross1b16b0e2019-02-12 14:41:32 -08001088// android_library builds and links sources into a `.jar` file for the device along with Android resources.
1089//
1090// An android_library has a single variant that produces a `.jar` file containing `.class` files that were
Sam Delmerico82602492022-06-10 17:05:42 +00001091// compiled against the device bootclasspath, along with a `package-res.apk` file containing Android resources compiled
Colin Cross1b16b0e2019-02-12 14:41:32 -08001092// with aapt2. This module is not suitable for installing on a device, but can be used as a `static_libs` dependency of
1093// an android_app module.
Colin Crossa97c5d32018-03-28 14:58:31 -07001094func AndroidLibraryFactory() android.Module {
1095 module := &AndroidLibrary{}
1096
Colin Crossce6734e2020-06-15 16:09:53 -07001097 module.Module.addHostAndDeviceProperties()
Colin Crossa97c5d32018-03-28 14:58:31 -07001098 module.AddProperties(
Colin Crossa97c5d32018-03-28 14:58:31 -07001099 &module.aaptProperties,
Ronald Braunsteincdc66f42024-04-12 11:23:19 -07001100 &module.androidLibraryProperties,
1101 &module.sourceProperties)
Colin Crossa97c5d32018-03-28 14:58:31 -07001102
1103 module.androidLibraryProperties.BuildAAR = true
Colin Cross014489c2020-06-02 20:09:13 -07001104 module.Module.linter.library = true
Colin Crossa97c5d32018-03-28 14:58:31 -07001105
Jooyung Hanacc7bbe2020-05-20 09:06:00 +09001106 android.InitApexModule(module)
Colin Cross48de9a42018-10-02 13:53:33 -07001107 InitJavaModule(module, android.DeviceSupported)
Colin Crossa97c5d32018-03-28 14:58:31 -07001108 return module
1109}
1110
Colin Crossfabb6082018-02-20 17:22:23 -08001111//
1112// AAR (android library) prebuilts
1113//
Colin Crossfabb6082018-02-20 17:22:23 -08001114
Vinh Trance0781f2022-04-13 01:30:44 +00001115// Properties for android_library_import
Colin Crossfabb6082018-02-20 17:22:23 -08001116type AARImportProperties struct {
Vinh Trance0781f2022-04-13 01:30:44 +00001117 // ARR (android library prebuilt) filepath. Exactly one ARR is required.
Colin Cross27b922f2019-03-04 22:35:41 -08001118 Aars []string `android:"path"`
Vinh Trance0781f2022-04-13 01:30:44 +00001119 // If not blank, set to the version of the sdk to compile against.
1120 // Defaults to private.
1121 // Values are of one of the following forms:
1122 // 1) numerical API level, "current", "none", or "core_platform"
1123 // 2) An SDK kind with an API level: "<sdk kind>_<API level>"
1124 // See build/soong/android/sdk_version.go for the complete and up to date list of SDK kinds.
1125 // If the SDK kind is empty, it will be set to public
1126 Sdk_version *string
1127 // If not blank, set the minimum version of the sdk that the compiled artifacts will run against.
1128 // Defaults to sdk_version if not set. See sdk_version for possible values.
Colin Cross479884c2018-07-10 13:39:30 -07001129 Min_sdk_version *string
Vinh Trance0781f2022-04-13 01:30:44 +00001130 // List of java static libraries that the included ARR (android library prebuilts) has dependencies to.
Cole Faustb7493472024-08-28 11:55:52 -07001131 Static_libs proptools.Configurable[[]string]
Vinh Trance0781f2022-04-13 01:30:44 +00001132 // List of java libraries that the included ARR (android library prebuilts) has dependencies to.
1133 Libs []string
1134 // If set to true, run Jetifier against .aar file. Defaults to false.
Colin Cross1001a792019-03-21 22:21:39 -07001135 Jetifier *bool
Sam Delmerico82602492022-06-10 17:05:42 +00001136 // If true, extract JNI libs from AAR archive. These libs will be accessible to android_app modules and
1137 // will be passed transitively through android_libraries to an android_app.
1138 //TODO(b/241138093) evaluate whether we can have this flag default to true for Bazel conversion
1139 Extract_jni *bool
Colin Cross21ed4692024-04-24 20:23:38 +00001140
1141 // If set, overrides the manifest extracted from the AAR with the provided path.
1142 Manifest *string `android:"path"`
Colin Crossfabb6082018-02-20 17:22:23 -08001143}
1144
1145type AARImport struct {
1146 android.ModuleBase
Colin Cross48de9a42018-10-02 13:53:33 -07001147 android.DefaultableModuleBase
Jooyung Hanacc7bbe2020-05-20 09:06:00 +09001148 android.ApexModuleBase
Colin Crossfabb6082018-02-20 17:22:23 -08001149 prebuilt android.Prebuilt
1150
Jooyung Hanacc7bbe2020-05-20 09:06:00 +09001151 // Functionality common to Module and Import.
1152 embeddableInModuleAndImport
1153
Colin Cross9ffaf282024-08-12 13:50:09 -07001154 providesTransitiveHeaderJarsForR8
Sam Delmerico9f9c0a22022-11-29 11:19:37 -05001155
Colin Crossfabb6082018-02-20 17:22:23 -08001156 properties AARImportProperties
1157
Colin Cross7707b242024-07-26 12:02:36 -07001158 headerJarFile android.Path
1159 implementationJarFile android.Path
1160 implementationAndResourcesJarFile android.Path
1161 proguardFlags android.Path
1162 exportPackage android.Path
Colin Cross312634e2023-11-21 15:13:56 -08001163 transitiveAaptResourcePackagesFile android.Path
Colin Cross7707b242024-07-26 12:02:36 -07001164 extraAaptPackagesFile android.Path
Colin Cross21ed4692024-04-24 20:23:38 +00001165 manifest android.Path
Colin Cross7707b242024-07-26 12:02:36 -07001166 assetsPackage android.Path
1167 rTxt android.Path
1168 rJar android.Path
Colin Cross66f78822018-05-02 12:58:28 -07001169
Colin Crossa14fb6a2024-10-23 16:57:06 -07001170 resourcesNodesDepSet depset.DepSet[*resourcesNode]
1171 manifestsDepSet depset.DepSet[android.Path]
Colin Cross56a83212020-09-15 18:30:11 -07001172
1173 hideApexVariantFromMake bool
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001174
Sam Delmerico82602492022-06-10 17:05:42 +00001175 aarPath android.Path
1176 jniPackages android.Paths
Jiyong Park92315372021-04-02 08:45:46 +09001177
1178 sdkVersion android.SdkSpec
Spandan Das8c9ae7e2023-03-03 21:20:36 +00001179 minSdkVersion android.ApiLevel
LaMont Jonesafe7baf2024-01-09 22:47:39 +00001180
Colin Cross9055e212024-03-23 04:43:41 +00001181 usesLibrary
1182 classLoaderContexts dexpreopt.ClassLoaderContextMap
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001183}
1184
Jiyong Park92315372021-04-02 08:45:46 +09001185func (a *AARImport) SdkVersion(ctx android.EarlyModuleContext) android.SdkSpec {
1186 return android.SdkSpecFrom(ctx, String(a.properties.Sdk_version))
Colin Cross83bb3162018-06-25 15:48:06 -07001187}
1188
Jiyong Parkf1691d22021-03-29 20:11:58 +09001189func (a *AARImport) SystemModules() string {
Paul Duffine25c6442019-10-11 13:50:28 +01001190 return ""
1191}
1192
Spandan Das8c9ae7e2023-03-03 21:20:36 +00001193func (a *AARImport) MinSdkVersion(ctx android.EarlyModuleContext) android.ApiLevel {
Colin Cross479884c2018-07-10 13:39:30 -07001194 if a.properties.Min_sdk_version != nil {
Spandan Das8c9ae7e2023-03-03 21:20:36 +00001195 return android.ApiLevelFrom(ctx, *a.properties.Min_sdk_version)
Colin Cross479884c2018-07-10 13:39:30 -07001196 }
Spandan Das8c9ae7e2023-03-03 21:20:36 +00001197 return a.SdkVersion(ctx).ApiLevel
Colin Cross83bb3162018-06-25 15:48:06 -07001198}
1199
Spandan Dasa26eda72023-03-02 00:56:06 +00001200func (a *AARImport) ReplaceMaxSdkVersionPlaceholder(ctx android.EarlyModuleContext) android.ApiLevel {
1201 return android.SdkSpecFrom(ctx, "").ApiLevel
William Loh5a082f92022-05-17 20:21:50 +00001202}
1203
Spandan Dasca70fc42023-03-01 23:38:49 +00001204func (a *AARImport) TargetSdkVersion(ctx android.EarlyModuleContext) android.ApiLevel {
1205 return a.SdkVersion(ctx).ApiLevel
Dan Willemsen419290a2018-10-31 15:28:47 -07001206}
1207
Colin Cross1e743852019-10-28 11:37:20 -07001208func (a *AARImport) javaVersion() string {
1209 return ""
1210}
1211
Colin Crossa97c5d32018-03-28 14:58:31 -07001212var _ AndroidLibraryDependency = (*AARImport)(nil)
1213
1214func (a *AARImport) ExportPackage() android.Path {
1215 return a.exportPackage
1216}
Colin Crossa14fb6a2024-10-23 16:57:06 -07001217func (a *AARImport) ResourcesNodeDepSet() depset.DepSet[*resourcesNode] {
Colin Crossab8d1382023-07-14 17:23:41 +00001218 return a.resourcesNodesDepSet
Colin Crossc1c37552019-01-31 11:42:41 -08001219}
1220
Colin Crossa14fb6a2024-10-23 16:57:06 -07001221func (a *AARImport) RRODirsDepSet() depset.DepSet[rroDir] {
1222 return depset.New[rroDir](depset.TOPOLOGICAL, nil, nil)
Colin Cross66f78822018-05-02 12:58:28 -07001223}
1224
Colin Crossa14fb6a2024-10-23 16:57:06 -07001225func (a *AARImport) ManifestsDepSet() depset.DepSet[android.Path] {
Colin Crossab8d1382023-07-14 17:23:41 +00001226 return a.manifestsDepSet
Jaewoong Jung6431ca72020-01-15 14:15:10 -08001227}
1228
Jaewoong Jungc779cd42020-10-06 18:56:10 -07001229// RRO enforcement is not available on aar_import since its RRO dirs are not
1230// exported.
1231func (a *AARImport) SetRROEnforcedForDependent(enforce bool) {
1232}
1233
1234// RRO enforcement is not available on aar_import since its RRO dirs are not
1235// exported.
1236func (a *AARImport) IsRROEnforced(ctx android.BaseModuleContext) bool {
1237 return false
1238}
1239
Colin Crossfabb6082018-02-20 17:22:23 -08001240func (a *AARImport) Prebuilt() *android.Prebuilt {
1241 return &a.prebuilt
1242}
1243
1244func (a *AARImport) Name() string {
1245 return a.prebuilt.Name(a.ModuleBase.Name())
1246}
1247
Jiyong Park618922e2020-01-08 13:35:43 +09001248func (a *AARImport) JacocoReportClassesFile() android.Path {
1249 return nil
1250}
1251
Colin Crossfabb6082018-02-20 17:22:23 -08001252func (a *AARImport) DepsMutator(ctx android.BottomUpMutatorContext) {
Jeongik Cha816a23a2020-07-08 01:09:23 +09001253 if !ctx.Config().AlwaysUsePrebuiltSdks() {
Jiyong Parkf1691d22021-03-29 20:11:58 +09001254 sdkDep := decodeSdkDep(ctx, android.SdkContext(a))
Colin Crossa97c5d32018-03-28 14:58:31 -07001255 if sdkDep.useModule && sdkDep.frameworkResModule != "" {
Colin Cross42d48b72018-08-29 14:10:52 -07001256 ctx.AddVariationDependencies(nil, frameworkResTag, sdkDep.frameworkResModule)
Colin Crossfabb6082018-02-20 17:22:23 -08001257 }
1258 }
Colin Crossa97c5d32018-03-28 14:58:31 -07001259
Colin Cross42d48b72018-08-29 14:10:52 -07001260 ctx.AddVariationDependencies(nil, libTag, a.properties.Libs...)
Cole Faustb7493472024-08-28 11:55:52 -07001261 ctx.AddVariationDependencies(nil, staticLibTag, a.properties.Static_libs.GetOrDefault(ctx, nil)...)
Colin Cross9055e212024-03-23 04:43:41 +00001262
1263 a.usesLibrary.deps(ctx, false)
Colin Crossfabb6082018-02-20 17:22:23 -08001264}
1265
Sam Delmerico82602492022-06-10 17:05:42 +00001266type JniPackageInfo struct {
1267 // List of zip files containing JNI libraries
1268 // Zip files should have directory structure jni/<arch>/*.so
1269 JniPackages android.Paths
1270}
1271
Colin Crossbc7d76c2023-12-12 16:39:03 -08001272var JniPackageProvider = blueprint.NewProvider[JniPackageInfo]()
Sam Delmerico82602492022-06-10 17:05:42 +00001273
1274// Unzip an AAR and extract the JNI libs for $archString.
1275var extractJNI = pctx.AndroidStaticRule("extractJNI",
1276 blueprint.RuleParams{
1277 Command: `rm -rf $out $outDir && touch $out && ` +
1278 `unzip -qoDD -d $outDir $in "jni/${archString}/*" && ` +
1279 `jni_files=$$(find $outDir/jni -type f) && ` +
1280 // print error message if there are no JNI libs for this arch
1281 `[ -n "$$jni_files" ] || (echo "ERROR: no JNI libs found for arch ${archString}" && exit 1) && ` +
Sam Delmerico80ee45c2023-06-22 15:36:02 -04001282 `${config.SoongZipCmd} -o $out -L 0 -P 'lib/${archString}' ` +
Sam Delmerico82602492022-06-10 17:05:42 +00001283 `-C $outDir/jni/${archString} $$(echo $$jni_files | xargs -n1 printf " -f %s")`,
1284 CommandDeps: []string{"${config.SoongZipCmd}"},
1285 },
1286 "outDir", "archString")
1287
Colin Crossfabb6082018-02-20 17:22:23 -08001288// 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 -07001289// 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 -08001290var unzipAAR = pctx.AndroidStaticRule("unzipAAR",
1291 blueprint.RuleParams{
Dan Willemsen304cfec2019-05-28 14:49:06 -07001292 Command: `rm -rf $outDir && mkdir -p $outDir && ` +
Colin Cross205e9112020-08-06 13:20:17 -07001293 `unzip -qoDD -d $outDir $in && rm -rf $outDir/res && touch $out && ` +
Michael Rosenfeld5ad15572021-12-03 13:25:10 -08001294 `${config.Zip2ZipCmd} -i $in -o $assetsPackage 'assets/**/*' && ` +
Colin Cross205e9112020-08-06 13:20:17 -07001295 `${config.MergeZipsCmd} $combinedClassesJar $$(ls $outDir/classes.jar 2> /dev/null) $$(ls $outDir/libs/*.jar 2> /dev/null)`,
Michael Rosenfeld5ad15572021-12-03 13:25:10 -08001296 CommandDeps: []string{"${config.MergeZipsCmd}", "${config.Zip2ZipCmd}"},
Colin Crossfabb6082018-02-20 17:22:23 -08001297 },
Michael Rosenfeld5ad15572021-12-03 13:25:10 -08001298 "outDir", "combinedClassesJar", "assetsPackage")
Colin Crossfabb6082018-02-20 17:22:23 -08001299
1300func (a *AARImport) GenerateAndroidBuildActions(ctx android.ModuleContext) {
1301 if len(a.properties.Aars) != 1 {
1302 ctx.PropertyErrorf("aars", "exactly one aar is required")
1303 return
1304 }
1305
Jiyong Park92315372021-04-02 08:45:46 +09001306 a.sdkVersion = a.SdkVersion(ctx)
1307 a.minSdkVersion = a.MinSdkVersion(ctx)
1308
Colin Crossff694a82023-12-13 15:54:49 -08001309 apexInfo, _ := android.ModuleProvider(ctx, android.ApexInfoProvider)
1310 a.hideApexVariantFromMake = !apexInfo.IsForPlatform()
Colin Cross56a83212020-09-15 18:30:11 -07001311
Nan Zhang4c819fb2018-08-27 18:31:46 -07001312 aarName := ctx.ModuleName() + ".aar"
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001313 a.aarPath = android.PathForModuleSrc(ctx, a.properties.Aars[0])
1314
Colin Cross1001a792019-03-21 22:21:39 -07001315 if Bool(a.properties.Jetifier) {
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001316 inputFile := a.aarPath
Colin Cross77965d92024-08-15 17:11:08 -07001317 jetifierPath := android.PathForModuleOut(ctx, "jetifier", aarName)
1318 TransformJetifier(ctx, jetifierPath, inputFile)
1319 a.aarPath = jetifierPath
Nan Zhang4c819fb2018-08-27 18:31:46 -07001320 }
Colin Crossfabb6082018-02-20 17:22:23 -08001321
Colin Cross607bbd62024-04-12 13:44:45 -07001322 jarName := ctx.ModuleName() + ".jar"
Colin Crossfabb6082018-02-20 17:22:23 -08001323 extractedAARDir := android.PathForModuleOut(ctx, "aar")
Colin Cross607bbd62024-04-12 13:44:45 -07001324 classpathFile := extractedAARDir.Join(ctx, jarName)
Colin Cross21ed4692024-04-24 20:23:38 +00001325
1326 extractedManifest := extractedAARDir.Join(ctx, "AndroidManifest.xml")
1327 providedManifest := android.OptionalPathForModuleSrc(ctx, a.properties.Manifest)
1328 if providedManifest.Valid() {
1329 a.manifest = providedManifest.Path()
1330 } else {
1331 a.manifest = extractedManifest
1332 }
1333
Colin Cross7707b242024-07-26 12:02:36 -07001334 rTxt := extractedAARDir.Join(ctx, "R.txt")
1335 assetsPackage := android.PathForModuleOut(ctx, "assets.zip")
1336 proguardFlags := extractedAARDir.Join(ctx, "proguard.txt")
Colin Crosscde55342024-03-27 14:11:51 -07001337 transitiveProguardFlags, transitiveUnconditionalExportedFlags := collectDepProguardSpecInfo(ctx)
Colin Cross40213022023-12-13 15:19:49 -08001338 android.SetProvider(ctx, ProguardSpecInfoProvider, ProguardSpecInfo{
Colin Crossa14fb6a2024-10-23 16:57:06 -07001339 ProguardFlagsFiles: depset.New[android.Path](
1340 depset.POSTORDER,
Colin Cross7707b242024-07-26 12:02:36 -07001341 android.Paths{proguardFlags},
Colin Crosscde55342024-03-27 14:11:51 -07001342 transitiveProguardFlags,
1343 ),
Colin Crossa14fb6a2024-10-23 16:57:06 -07001344 UnconditionallyExportedProguardFlags: depset.New[android.Path](
1345 depset.POSTORDER,
Sam Delmerico95d70942023-08-02 18:00:35 -04001346 nil,
Colin Crosscde55342024-03-27 14:11:51 -07001347 transitiveUnconditionalExportedFlags,
Sam Delmerico95d70942023-08-02 18:00:35 -04001348 ),
1349 })
Colin Crossfabb6082018-02-20 17:22:23 -08001350
1351 ctx.Build(pctx, android.BuildParams{
1352 Rule: unzipAAR,
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001353 Input: a.aarPath,
Colin Cross7707b242024-07-26 12:02:36 -07001354 Outputs: android.WritablePaths{classpathFile, proguardFlags, extractedManifest, assetsPackage, rTxt},
Colin Crossfabb6082018-02-20 17:22:23 -08001355 Description: "unzip AAR",
1356 Args: map[string]string{
Colin Cross205e9112020-08-06 13:20:17 -07001357 "outDir": extractedAARDir.String(),
Colin Cross9055e212024-03-23 04:43:41 +00001358 "combinedClassesJar": classpathFile.String(),
Colin Cross7707b242024-07-26 12:02:36 -07001359 "assetsPackage": assetsPackage.String(),
Colin Crossfabb6082018-02-20 17:22:23 -08001360 },
1361 })
1362
Colin Cross7707b242024-07-26 12:02:36 -07001363 a.proguardFlags = proguardFlags
1364 a.assetsPackage = assetsPackage
1365 a.rTxt = rTxt
1366
Colin Crossa0ba2f52019-06-22 12:59:27 -07001367 // Always set --pseudo-localize, it will be stripped out later for release
1368 // builds that don't want it.
1369 compileFlags := []string{"--pseudo-localize"}
Colin Crossfabb6082018-02-20 17:22:23 -08001370 compiledResDir := android.PathForModuleOut(ctx, "flat-res")
Colin Crossfabb6082018-02-20 17:22:23 -08001371 flata := compiledResDir.Join(ctx, "gen_res.flata")
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001372 aapt2CompileZip(ctx, flata, a.aarPath, "res", compileFlags)
Colin Crossfabb6082018-02-20 17:22:23 -08001373
Colin Cross7707b242024-07-26 12:02:36 -07001374 exportPackage := android.PathForModuleOut(ctx, "package-res.apk")
Colin Crossfabb6082018-02-20 17:22:23 -08001375 proguardOptionsFile := android.PathForModuleGen(ctx, "proguard.options")
Colin Crossbb77d8e2024-02-15 14:43:47 -08001376 aaptRTxt := android.PathForModuleOut(ctx, "R.txt")
Colin Cross7707b242024-07-26 12:02:36 -07001377 extraAaptPackagesFile := android.PathForModuleOut(ctx, "extra_packages")
Colin Crossfabb6082018-02-20 17:22:23 -08001378
1379 var linkDeps android.Paths
1380
1381 linkFlags := []string{
1382 "--static-lib",
Colin Cross4eae06d2023-06-20 22:40:02 -07001383 "--merge-only",
Colin Crossfabb6082018-02-20 17:22:23 -08001384 "--auto-add-overlay",
Colin Cross7c4dc5d2024-02-13 14:29:45 -08001385 "--no-static-lib-packages",
Colin Crossfabb6082018-02-20 17:22:23 -08001386 }
1387
Colin Cross10f7c4a2018-05-23 10:59:28 -07001388 linkFlags = append(linkFlags, "--manifest "+a.manifest.String())
1389 linkDeps = append(linkDeps, a.manifest)
Colin Crossfabb6082018-02-20 17:22:23 -08001390
Colin Cross8676c8c2023-10-12 15:58:57 -07001391 staticResourcesNodesDepSet, sharedResourcesNodesDepSet, staticRRODirsDepSet, staticManifestsDepSet, sharedLibs, libFlags :=
Jiakai Zhang36937082024-04-15 11:15:50 +00001392 aaptLibs(ctx, android.SdkContext(a), nil, nil)
Colin Cross31656952018-05-24 16:11:20 -07001393
Colin Cross8676c8c2023-10-12 15:58:57 -07001394 _ = sharedResourcesNodesDepSet
Colin Crossab8d1382023-07-14 17:23:41 +00001395 _ = staticRRODirsDepSet
Colin Cross8676c8c2023-10-12 15:58:57 -07001396
Colin Crossab8d1382023-07-14 17:23:41 +00001397 staticDeps := transitiveAarDeps(staticResourcesNodesDepSet.ToList())
Colin Crossfabb6082018-02-20 17:22:23 -08001398
Colin Crossab8d1382023-07-14 17:23:41 +00001399 linkDeps = append(linkDeps, sharedLibs...)
Colin Cross4eae06d2023-06-20 22:40:02 -07001400 linkDeps = append(linkDeps, staticDeps.resPackages()...)
Colin Crossa97c5d32018-03-28 14:58:31 -07001401 linkFlags = append(linkFlags, libFlags...)
Colin Crossfabb6082018-02-20 17:22:23 -08001402
Colin Cross4eae06d2023-06-20 22:40:02 -07001403 overlayRes := android.Paths{flata}
1404
1405 // Treat static library dependencies of static libraries as imports.
1406 transitiveStaticLibs := staticDeps.resPackages()
1407 linkDeps = append(linkDeps, transitiveStaticLibs...)
1408 for _, staticLib := range transitiveStaticLibs {
1409 linkFlags = append(linkFlags, "-I "+staticLib.String())
1410 }
Colin Crossfabb6082018-02-20 17:22:23 -08001411
Colin Crossab8d1382023-07-14 17:23:41 +00001412 transitiveAssets := android.ReverseSliceInPlace(staticDeps.assets())
Colin Cross7707b242024-07-26 12:02:36 -07001413 aapt2Link(ctx, exportPackage, nil, proguardOptionsFile, aaptRTxt,
Jihoon Kang84b25892023-12-01 22:01:06 +00001414 linkFlags, linkDeps, nil, overlayRes, transitiveAssets, nil, nil)
Colin Crossa6182ab2024-08-21 10:47:44 -07001415 ctx.CheckbuildFile(exportPackage)
Colin Cross7707b242024-07-26 12:02:36 -07001416 a.exportPackage = exportPackage
Colin Crossfabb6082018-02-20 17:22:23 -08001417
Colin Cross7707b242024-07-26 12:02:36 -07001418 rJar := android.PathForModuleOut(ctx, "busybox/R.jar")
1419 resourceProcessorBusyBoxGenerateBinaryR(ctx, a.rTxt, a.manifest, rJar, nil, true, nil, false)
Colin Crossa6182ab2024-08-21 10:47:44 -07001420 ctx.CheckbuildFile(rJar)
Colin Cross7707b242024-07-26 12:02:36 -07001421 a.rJar = rJar
Colin Cross4eae06d2023-06-20 22:40:02 -07001422
Colin Cross7707b242024-07-26 12:02:36 -07001423 aapt2ExtractExtraPackages(ctx, extraAaptPackagesFile, a.rJar)
1424 a.extraAaptPackagesFile = extraAaptPackagesFile
Colin Crossf3b7bad2023-08-02 15:49:00 -07001425
Colin Crossa14fb6a2024-10-23 16:57:06 -07001426 resourcesNodesDepSetBuilder := depset.NewBuilder[*resourcesNode](depset.TOPOLOGICAL)
Colin Crossab8d1382023-07-14 17:23:41 +00001427 resourcesNodesDepSetBuilder.Direct(&resourcesNode{
1428 resPackage: a.exportPackage,
1429 manifest: a.manifest,
Colin Cross4eae06d2023-06-20 22:40:02 -07001430 rTxt: a.rTxt,
1431 rJar: a.rJar,
Colin Crossab8d1382023-07-14 17:23:41 +00001432 assets: android.OptionalPathForPath(a.assetsPackage),
Colin Cross4eae06d2023-06-20 22:40:02 -07001433
1434 usedResourceProcessor: true,
Colin Crossab8d1382023-07-14 17:23:41 +00001435 })
1436 resourcesNodesDepSetBuilder.Transitive(staticResourcesNodesDepSet)
1437 a.resourcesNodesDepSet = resourcesNodesDepSetBuilder.Build()
1438
Colin Crossa14fb6a2024-10-23 16:57:06 -07001439 manifestDepSetBuilder := depset.NewBuilder[android.Path](depset.TOPOLOGICAL).Direct(a.manifest)
Colin Cross9055e212024-03-23 04:43:41 +00001440 manifestDepSetBuilder.Transitive(staticManifestsDepSet)
Colin Crossab8d1382023-07-14 17:23:41 +00001441 a.manifestsDepSet = manifestDepSetBuilder.Build()
Michael Rosenfeld5ad15572021-12-03 13:25:10 -08001442
Colin Cross312634e2023-11-21 15:13:56 -08001443 transitiveAaptResourcePackages := staticDeps.resPackages().Strings()
1444 transitiveAaptResourcePackages = slices.DeleteFunc(transitiveAaptResourcePackages, func(p string) bool {
1445 return p == a.exportPackage.String()
1446 })
1447 transitiveAaptResourcePackagesFile := android.PathForModuleOut(ctx, "transitive-res-packages")
1448 android.WriteFileRule(ctx, transitiveAaptResourcePackagesFile, strings.Join(transitiveAaptResourcePackages, "\n"))
1449 a.transitiveAaptResourcePackagesFile = transitiveAaptResourcePackagesFile
Colin Cross4eae06d2023-06-20 22:40:02 -07001450
Colin Cross9ffaf282024-08-12 13:50:09 -07001451 a.collectTransitiveHeaderJarsForR8(ctx)
Colin Cross9055e212024-03-23 04:43:41 +00001452
1453 a.classLoaderContexts = a.usesLibrary.classLoaderContextForUsesLibDeps(ctx)
1454
1455 var staticJars android.Paths
1456 var staticHeaderJars android.Paths
Colin Cross607bbd62024-04-12 13:44:45 -07001457 var staticResourceJars android.Paths
Colin Crossa14fb6a2024-10-23 16:57:06 -07001458 var transitiveStaticLibsHeaderJars []depset.DepSet[android.Path]
1459 var transitiveStaticLibsImplementationJars []depset.DepSet[android.Path]
1460 var transitiveStaticLibsResourceJars []depset.DepSet[android.Path]
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001461
Yu Liu39f5fb32025-01-13 23:52:19 +00001462 ctx.VisitDirectDepsProxy(func(module android.ModuleProxy) {
Colin Cross9055e212024-03-23 04:43:41 +00001463 if dep, ok := android.OtherModuleProvider(ctx, module, JavaInfoProvider); ok {
1464 tag := ctx.OtherModuleDependencyTag(module)
1465 switch tag {
1466 case staticLibTag:
1467 staticJars = append(staticJars, dep.ImplementationJars...)
1468 staticHeaderJars = append(staticHeaderJars, dep.HeaderJars...)
Colin Cross607bbd62024-04-12 13:44:45 -07001469 staticResourceJars = append(staticResourceJars, dep.ResourceJars...)
Colin Crossa14fb6a2024-10-23 16:57:06 -07001470 transitiveStaticLibsHeaderJars = append(transitiveStaticLibsHeaderJars, dep.TransitiveStaticLibsHeaderJars)
1471 transitiveStaticLibsImplementationJars = append(transitiveStaticLibsImplementationJars, dep.TransitiveStaticLibsImplementationJars)
1472 transitiveStaticLibsResourceJars = append(transitiveStaticLibsResourceJars, dep.TransitiveStaticLibsResourceJars)
Colin Cross9055e212024-03-23 04:43:41 +00001473 }
1474 }
1475 addCLCFromDep(ctx, module, a.classLoaderContexts)
Jiakai Zhang36937082024-04-15 11:15:50 +00001476 addMissingOptionalUsesLibsFromDep(ctx, module, &a.usesLibrary)
Colin Cross9055e212024-03-23 04:43:41 +00001477 })
1478
Colin Crossa14fb6a2024-10-23 16:57:06 -07001479 completeStaticLibsHeaderJars := depset.New(depset.PREORDER, android.Paths{classpathFile}, transitiveStaticLibsHeaderJars)
1480 completeStaticLibsImplementationJars := depset.New(depset.PREORDER, android.Paths{classpathFile}, transitiveStaticLibsImplementationJars)
1481 completeStaticLibsResourceJars := depset.New(depset.PREORDER, nil, transitiveStaticLibsResourceJars)
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001482
Colin Cross77965d92024-08-15 17:11:08 -07001483 var implementationJarFile android.Path
Colin Cross8ff4af32025-02-19 15:17:02 -08001484 combineJars := completeStaticLibsImplementationJars.ToList()
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001485
1486 if len(combineJars) > 1 {
1487 implementationJarOutputPath := android.PathForModuleOut(ctx, "combined", jarName)
1488 TransformJarsToJar(ctx, implementationJarOutputPath, "combine", combineJars, android.OptionalPath{}, false, nil, nil)
1489 implementationJarFile = implementationJarOutputPath
Colin Cross9055e212024-03-23 04:43:41 +00001490 } else {
Colin Cross28ac2ff2024-04-02 12:21:34 -07001491 implementationJarFile = classpathFile
Colin Cross9055e212024-03-23 04:43:41 +00001492 }
1493
Colin Cross607bbd62024-04-12 13:44:45 -07001494 var resourceJarFile android.Path
Colin Cross8ff4af32025-02-19 15:17:02 -08001495 resourceJars := completeStaticLibsResourceJars.ToList()
1496
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001497 if len(resourceJars) > 1 {
Colin Cross607bbd62024-04-12 13:44:45 -07001498 combinedJar := android.PathForModuleOut(ctx, "res-combined", jarName)
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001499 TransformJarsToJar(ctx, combinedJar, "for resources", resourceJars, android.OptionalPath{},
Colin Cross607bbd62024-04-12 13:44:45 -07001500 false, nil, nil)
1501 resourceJarFile = combinedJar
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001502 } else if len(resourceJars) == 1 {
1503 resourceJarFile = resourceJars[0]
Colin Cross607bbd62024-04-12 13:44:45 -07001504 }
1505
1506 // merge implementation jar with resources if necessary
Colin Cross8ff4af32025-02-19 15:17:02 -08001507 implementationAndResourcesJars := append(slices.Clone(resourceJars), combineJars...)
1508
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001509 var implementationAndResourcesJar android.Path
1510 if len(implementationAndResourcesJars) > 1 {
Colin Cross77965d92024-08-15 17:11:08 -07001511 combinedJar := android.PathForModuleOut(ctx, "withres", jarName)
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001512 TransformJarsToJar(ctx, combinedJar, "for resources", implementationAndResourcesJars, android.OptionalPath{},
Colin Cross607bbd62024-04-12 13:44:45 -07001513 false, nil, nil)
1514 implementationAndResourcesJar = combinedJar
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001515 } else {
1516 implementationAndResourcesJar = implementationAndResourcesJars[0]
Colin Cross607bbd62024-04-12 13:44:45 -07001517 }
1518
1519 a.implementationJarFile = implementationJarFile
Colin Cross28ac2ff2024-04-02 12:21:34 -07001520 // 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 -07001521 a.implementationAndResourcesJarFile = implementationAndResourcesJar.WithoutRel()
Colin Cross28ac2ff2024-04-02 12:21:34 -07001522
Colin Cross8ff4af32025-02-19 15:17:02 -08001523 headerJars := completeStaticLibsHeaderJars.ToList()
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001524 if len(headerJars) > 1 {
Colin Cross7707b242024-07-26 12:02:36 -07001525 headerJarFile := android.PathForModuleOut(ctx, "turbine-combined", jarName)
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001526 TransformJarsToJar(ctx, headerJarFile, "combine header jars", headerJars, android.OptionalPath{}, false, nil, nil)
Colin Cross7707b242024-07-26 12:02:36 -07001527 a.headerJarFile = headerJarFile
Colin Cross9055e212024-03-23 04:43:41 +00001528 } else {
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001529 a.headerJarFile = headerJars[0]
Colin Cross9055e212024-03-23 04:43:41 +00001530 }
1531
Colin Cross8ff4af32025-02-19 15:17:02 -08001532 ctx.CheckbuildFile(classpathFile)
Colin Crossa6182ab2024-08-21 10:47:44 -07001533
Yu Liu460cf372025-01-10 00:34:06 +00001534 javaInfo := &JavaInfo{
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001535 HeaderJars: android.PathsIfNonNil(a.headerJarFile),
1536 LocalHeaderJars: android.PathsIfNonNil(classpathFile),
1537 TransitiveStaticLibsHeaderJars: completeStaticLibsHeaderJars,
1538 TransitiveStaticLibsImplementationJars: completeStaticLibsImplementationJars,
1539 TransitiveStaticLibsResourceJars: completeStaticLibsResourceJars,
1540 ResourceJars: android.PathsIfNonNil(resourceJarFile),
1541 TransitiveLibsHeaderJarsForR8: a.transitiveLibsHeaderJarsForR8,
1542 TransitiveStaticLibsHeaderJarsForR8: a.transitiveStaticLibsHeaderJarsForR8,
1543 ImplementationAndResourcesJars: android.PathsIfNonNil(a.implementationAndResourcesJarFile),
1544 ImplementationJars: android.PathsIfNonNil(a.implementationJarFile),
1545 StubsLinkType: Implementation,
Joe Onorato6fe59eb2023-07-16 13:20:33 -07001546 // TransitiveAconfigFiles: // TODO(b/289117800): LOCAL_ACONFIG_FILES for prebuilts
Yu Liu460cf372025-01-10 00:34:06 +00001547 }
1548 setExtraJavaInfo(ctx, a, javaInfo)
1549 android.SetProvider(ctx, JavaInfoProvider, javaInfo)
Sam Delmerico82602492022-06-10 17:05:42 +00001550
1551 if proptools.Bool(a.properties.Extract_jni) {
1552 for _, t := range ctx.MultiTargets() {
1553 arch := t.Arch.Abi[0]
1554 path := android.PathForModuleOut(ctx, arch+"_jni.zip")
1555 a.jniPackages = append(a.jniPackages, path)
1556
1557 outDir := android.PathForModuleOut(ctx, "aarForJni")
1558 aarPath := android.PathForModuleSrc(ctx, a.properties.Aars[0])
1559 ctx.Build(pctx, android.BuildParams{
1560 Rule: extractJNI,
1561 Input: aarPath,
1562 Outputs: android.WritablePaths{path},
1563 Description: "extract JNI from AAR",
1564 Args: map[string]string{
1565 "outDir": outDir.String(),
1566 "archString": arch,
1567 },
1568 })
1569 }
Sam Delmerico82602492022-06-10 17:05:42 +00001570 }
Colin Crosse8eeec92023-12-14 14:50:05 -08001571
Colin Cross40213022023-12-13 15:19:49 -08001572 android.SetProvider(ctx, JniPackageProvider, JniPackageInfo{
Colin Crosse8eeec92023-12-14 14:50:05 -08001573 JniPackages: a.jniPackages,
1574 })
mrziwang68786d82024-07-09 10:41:55 -07001575
Yu Liu2a815b62025-02-21 20:46:25 +00001576 android.SetProvider(ctx, AARImportInfoProvider, AARImportInfo{})
1577
mrziwang68786d82024-07-09 10:41:55 -07001578 ctx.SetOutputFiles([]android.Path{a.implementationAndResourcesJarFile}, "")
1579 ctx.SetOutputFiles([]android.Path{a.aarPath}, ".aar")
Wei Li986fe742025-01-30 15:14:42 -08001580
1581 buildComplianceMetadata(ctx)
Colin Crossdcf71b22021-02-01 13:59:03 -08001582}
Colin Crossfabb6082018-02-20 17:22:23 -08001583
1584func (a *AARImport) HeaderJars() android.Paths {
Colin Cross9055e212024-03-23 04:43:41 +00001585 return android.Paths{a.headerJarFile}
Colin Crossfabb6082018-02-20 17:22:23 -08001586}
1587
Colin Cross331a1212018-08-15 20:40:52 -07001588func (a *AARImport) ImplementationAndResourcesJars() android.Paths {
Colin Cross607bbd62024-04-12 13:44:45 -07001589 return android.Paths{a.implementationAndResourcesJarFile}
Colin Cross331a1212018-08-15 20:40:52 -07001590}
1591
Colin Cross9055e212024-03-23 04:43:41 +00001592func (a *AARImport) DexJarBuildPath(ctx android.ModuleErrorfContext) OptionalDexJarPath {
1593 return OptionalDexJarPath{}
Colin Crossf24a22a2019-01-31 14:12:44 -08001594}
1595
Ulya Trafimovich9f3052c2020-06-09 14:31:19 +01001596func (a *AARImport) DexJarInstallPath() android.Path {
1597 return nil
1598}
1599
Ulya Trafimovichb23d28c2020-10-08 12:53:58 +01001600func (a *AARImport) ClassLoaderContexts() dexpreopt.ClassLoaderContextMap {
Colin Cross9055e212024-03-23 04:43:41 +00001601 return a.classLoaderContexts
Jiyong Park1be96912018-05-28 18:02:19 +09001602}
1603
Colin Cross9055e212024-03-23 04:43:41 +00001604var _ UsesLibraryDependency = (*AARImport)(nil)
1605
Jiyong Park45bf82e2020-12-15 22:29:02 +09001606var _ android.ApexModule = (*AARImport)(nil)
1607
1608// Implements android.ApexModule
Yu Liuf1806032025-02-07 00:23:34 +00001609func (m *AARImport) GetDepInSameApexChecker() android.DepInSameApexChecker {
1610 return AARImportDepInSameApexChecker{}
1611}
1612
1613type AARImportDepInSameApexChecker struct {
1614 android.BaseDepInSameApexChecker
1615}
1616
1617func (m AARImportDepInSameApexChecker) OutgoingDepIsInSameApex(tag blueprint.DependencyTag) bool {
1618 return depIsInSameApex(tag)
Jooyung Hanacc7bbe2020-05-20 09:06:00 +09001619}
1620
Jiyong Park45bf82e2020-12-15 22:29:02 +09001621// Implements android.ApexModule
Yu Liudf0b8392025-02-12 18:27:03 +00001622func (a *AARImport) MinSdkVersionSupported(ctx android.BaseModuleContext) android.ApiLevel {
1623 return android.MinApiLevel
Jooyung Han749dc692020-04-15 11:03:39 +09001624}
1625
Sam Delmericoaf8bb702022-07-25 15:39:32 -04001626var _ android.PrebuiltInterface = (*AARImport)(nil)
Colin Crossfabb6082018-02-20 17:22:23 -08001627
Jiakai Zhangf98da192024-04-15 11:15:41 +00001628func (a *AARImport) UsesLibrary() *usesLibrary {
1629 return &a.usesLibrary
1630}
1631
1632var _ ModuleWithUsesLibrary = (*AARImport)(nil)
1633
Colin Cross1b16b0e2019-02-12 14:41:32 -08001634// android_library_import imports an `.aar` file into the build graph as if it was built with android_library.
1635//
1636// This module is not suitable for installing on a device, but can be used as a `static_libs` dependency of
1637// an android_app module.
Colin Crossfabb6082018-02-20 17:22:23 -08001638func AARImportFactory() android.Module {
1639 module := &AARImport{}
1640
Colin Cross9055e212024-03-23 04:43:41 +00001641 module.AddProperties(
1642 &module.properties,
1643 &module.usesLibrary.usesLibraryProperties,
1644 )
Colin Crossfabb6082018-02-20 17:22:23 -08001645
1646 android.InitPrebuiltModule(module, &module.properties.Aars)
Jooyung Hanacc7bbe2020-05-20 09:06:00 +09001647 android.InitApexModule(module)
Sam Delmerico82602492022-06-10 17:05:42 +00001648 InitJavaModuleMultiTargets(module, android.DeviceSupported)
Colin Crossfabb6082018-02-20 17:22:23 -08001649 return module
1650}
Spandan Dasb08bf772024-08-07 21:55:42 +00001651
Cole Faustb36d31d2024-08-27 16:04:28 -07001652func (a *AARImport) IDEInfo(ctx android.BaseModuleContext, dpInfo *android.IdeInfo) {
Spandan Das65bfc292025-01-02 22:55:56 +00001653 dpInfo.Jars = append(dpInfo.Jars, a.implementationJarFile.String(), a.rJar.String())
Spandan Dasb08bf772024-08-07 21:55:42 +00001654}