blob: b66876663ad18bc7b9ab39cd6b0d74315a0fa77c [file] [log] [blame]
Colin Crossfabb6082018-02-20 17:22:23 -08001// Copyright 2018 Google Inc. All rights reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package java
16
17import (
Cole Faust6265120c2024-09-05 16:27:56 -070018 "crypto/sha256"
Colin Crossa592e3e2019-02-19 16:59:53 -080019 "fmt"
Jaewoong Jung5b425e22019-06-17 17:40:56 -070020 "path/filepath"
Colin Cross312634e2023-11-21 15:13:56 -080021 "slices"
Colin Crossa97c5d32018-03-28 14:58:31 -070022 "strings"
Colin Crossfabb6082018-02-20 17:22:23 -080023
Jaewoong Jung9befb0c2020-01-18 10:33:43 -080024 "android/soong/android"
Ulya Trafimovich31e444e2020-08-14 17:32:16 +010025 "android/soong/dexpreopt"
Jihoon Kangfe914ed2024-02-12 22:49:21 +000026
Colin Crossfabb6082018-02-20 17:22:23 -080027 "github.com/google/blueprint"
Colin Crossa14fb6a2024-10-23 16:57:06 -070028 "github.com/google/blueprint/depset"
Colin Crossa97c5d32018-03-28 14:58:31 -070029 "github.com/google/blueprint/proptools"
Colin Crossfabb6082018-02-20 17:22:23 -080030)
31
Colin Crossa97c5d32018-03-28 14:58:31 -070032type AndroidLibraryDependency interface {
Colin Crossa97c5d32018-03-28 14:58:31 -070033 ExportPackage() android.Path
Colin Crossa14fb6a2024-10-23 16:57:06 -070034 ResourcesNodeDepSet() depset.DepSet[*resourcesNode]
35 RRODirsDepSet() depset.DepSet[rroDir]
36 ManifestsDepSet() depset.DepSet[android.Path]
Jaewoong Jungc779cd42020-10-06 18:56:10 -070037 SetRROEnforcedForDependent(enforce bool)
38 IsRROEnforced(ctx android.BaseModuleContext) bool
Colin Crossa97c5d32018-03-28 14:58:31 -070039}
40
41func init() {
Paul Duffinf9b1da02019-12-18 19:51:55 +000042 RegisterAARBuildComponents(android.InitRegistrationContext)
43}
44
45func RegisterAARBuildComponents(ctx android.RegistrationContext) {
46 ctx.RegisterModuleType("android_library_import", AARImportFactory)
47 ctx.RegisterModuleType("android_library", AndroidLibraryFactory)
Paul Duffin04ba70d2021-03-22 13:56:43 +000048 ctx.PostDepsMutators(func(ctx android.RegisterMutatorsContext) {
Colin Crossc4441622024-09-18 14:55:49 -070049 ctx.Transition("propagate_rro_enforcement", &propagateRROEnforcementTransitionMutator{})
Paul Duffin04ba70d2021-03-22 13:56:43 +000050 })
Colin Crossa97c5d32018-03-28 14:58:31 -070051}
52
53//
54// AAR (android library)
55//
56
57type androidLibraryProperties struct {
58 BuildAAR bool `blueprint:"mutated"`
59}
60
61type aaptProperties struct {
62 // flags passed to aapt when creating the apk
63 Aaptflags []string
64
Dan Willemsen72be5902018-10-24 20:24:57 -070065 // include all resource configurations, not just the product-configured
66 // ones.
67 Aapt_include_all_resources *bool
68
Jiakai Zhangba82e282023-10-13 18:08:59 +010069 // list of files to use as assets.
70 Assets []string `android:"path"`
71
Colin Crossa97c5d32018-03-28 14:58:31 -070072 // list of directories relative to the Blueprints file containing assets.
Colin Cross0ddae7f2019-02-07 15:30:01 -080073 // Defaults to ["assets"] if a directory called assets exists. Set to []
74 // to disable the default.
Colin Crossa97c5d32018-03-28 14:58:31 -070075 Asset_dirs []string
76
77 // list of directories relative to the Blueprints file containing
Colin Cross0ddae7f2019-02-07 15:30:01 -080078 // Android resources. Defaults to ["res"] if a directory called res exists.
79 // Set to [] to disable the default.
Cole Faust5b989772024-11-06 11:03:16 -080080 Resource_dirs proptools.Configurable[[]string] `android:"path"`
Colin Crossa97c5d32018-03-28 14:58:31 -070081
Colin Crossa592e3e2019-02-19 16:59:53 -080082 // list of zip files containing Android resources.
Colin Cross27b922f2019-03-04 22:35:41 -080083 Resource_zips []string `android:"path"`
Colin Crossa592e3e2019-02-19 16:59:53 -080084
Colin Crossa97c5d32018-03-28 14:58:31 -070085 // path to AndroidManifest.xml. If unset, defaults to "AndroidManifest.xml".
Colin Cross27b922f2019-03-04 22:35:41 -080086 Manifest *string `android:"path"`
changho.shinb5432b72019-08-08 18:37:17 +090087
88 // paths to additional manifest files to merge with main manifest.
89 Additional_manifests []string `android:"path"`
Sasha Smundak541056c2019-10-28 15:50:06 -070090
91 // do not include AndroidManifest from dependent libraries
92 Dont_merge_manifests *bool
Jaewoong Jungc779cd42020-10-06 18:56:10 -070093
Colin Cross4eae06d2023-06-20 22:40:02 -070094 // If use_resource_processor is set, use Bazel's resource processor instead of aapt2 to generate R.class files.
95 // The resource processor produces more optimal R.class files that only list resources in the package of the
96 // library that provided them, as opposed to aapt2 which produces R.java files for every package containing
97 // every resource. Using the resource processor can provide significant build time speedups, but requires
98 // fixing the module to use the correct package to reference each resource, and to avoid having any other
99 // libraries in the tree that use the same package name. Defaults to false, but will default to true in the
100 // future.
101 Use_resource_processor *bool
102
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700103 // true if RRO is enforced for any of the dependent modules
104 RROEnforcedForDependent bool `blueprint:"mutated"`
Inseob Kim34dc4cd2023-11-07 13:37:14 +0900105
106 // Filter only specified product and ignore other products
107 Filter_product *string `blueprint:"mutated"`
Jihoon Kang9049c272024-03-19 21:57:36 +0000108
109 // Names of aconfig_declarations modules that specify aconfig flags that the module depends on.
110 Flags_packages []string
Colin Crossa97c5d32018-03-28 14:58:31 -0700111}
112
113type aapt struct {
Colin Cross312634e2023-11-21 15:13:56 -0800114 aaptSrcJar android.Path
115 transitiveAaptRJars android.Paths
116 transitiveAaptResourcePackagesFile android.Path
117 exportPackage android.Path
118 manifestPath android.Path
119 proguardOptionsFile android.Path
120 rTxt android.Path
121 rJar android.Path
122 extraAaptPackagesFile android.Path
123 mergedManifestFile android.Path
124 noticeFile android.OptionalPath
125 assetPackage android.OptionalPath
126 isLibrary bool
127 defaultManifestVersion string
128 useEmbeddedNativeLibs bool
129 useEmbeddedDex bool
130 usesNonSdkApis bool
131 hasNoCode bool
132 LoggingParent string
133 resourceFiles android.Paths
Colin Crossa97c5d32018-03-28 14:58:31 -0700134
Colin Crosse560c4a2019-03-19 16:03:11 -0700135 splitNames []string
136 splits []split
137
Colin Crossa97c5d32018-03-28 14:58:31 -0700138 aaptProperties aaptProperties
Colin Crossab8d1382023-07-14 17:23:41 +0000139
Colin Crossa14fb6a2024-10-23 16:57:06 -0700140 resourcesNodesDepSet depset.DepSet[*resourcesNode]
141 rroDirsDepSet depset.DepSet[rroDir]
142 manifestsDepSet depset.DepSet[android.Path]
Alix96ea88452023-08-31 15:48:23 +0000143
144 manifestValues struct {
145 applicationId string
146 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700147}
148
Colin Crosse560c4a2019-03-19 16:03:11 -0700149type split struct {
150 name string
151 suffix string
152 path android.Path
153}
154
Colin Crossc4441622024-09-18 14:55:49 -0700155// Propagate RRO enforcement flag to static lib dependencies transitively. If EnforceRROGlobally is set then
156// all modules will use the "" variant. If specific modules have RRO enforced, then modules (usually apps) with
157// RRO enabled will use the "" variation for themselves, but use the "rro" variant of direct and transitive static
158// android_library dependencies.
159type propagateRROEnforcementTransitionMutator struct{}
160
161func (p propagateRROEnforcementTransitionMutator) Split(ctx android.BaseModuleContext) []string {
162 // Never split modules, apps with or without RRO enabled use the "" variant, static android_library dependencies
163 // will use create the "rro" variant from incoming tranisitons.
164 return []string{""}
165}
166
167func (p propagateRROEnforcementTransitionMutator) OutgoingTransition(ctx android.OutgoingTransitionContext, sourceVariation string) string {
168 // Non-static dependencies are not involved in RRO and always use the empty variant.
169 if ctx.DepTag() != staticLibTag {
170 return ""
171 }
172
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700173 m := ctx.Module()
Colin Crossc4441622024-09-18 14:55:49 -0700174 if _, ok := m.(AndroidLibraryDependency); ok {
175 // If RRO is enforced globally don't bother using "rro" variants, the empty variant will have RRO enabled.
176 if ctx.Config().EnforceRROGlobally() {
177 return ""
178 }
179
180 // If RRO is enabled for this module use the "rro" variants of static dependencies. IncomingTransition will
181 // rewrite this back to "" if the dependency is not an android_library.
182 if ctx.Config().EnforceRROForModule(ctx.Module().Name()) {
183 return "rro"
184 }
185 }
186
187 return sourceVariation
188}
189
190func (p propagateRROEnforcementTransitionMutator) IncomingTransition(ctx android.IncomingTransitionContext, incomingVariation string) string {
191 // Propagate the "rro" variant to android_library modules, but use the empty variant for everything else.
192 if incomingVariation == "rro" {
193 m := ctx.Module()
194 if _, ok := m.(AndroidLibraryDependency); ok {
195 return "rro"
196 }
197 return ""
198 }
199
200 return ""
201}
202
203func (p propagateRROEnforcementTransitionMutator) Mutate(ctx android.BottomUpMutatorContext, variation string) {
204 m := ctx.Module()
205 if d, ok := m.(AndroidLibraryDependency); ok {
206 if variation == "rro" {
207 // This is the "rro" variant of a module that has both variants, mark this one as RRO enabled and
208 // hide it from make to avoid collisions with the non-RRO empty variant.
209 d.SetRROEnforcedForDependent(true)
210 m.HideFromMake()
211 } else if ctx.Config().EnforceRROGlobally() {
212 // RRO is enabled globally, mark it enabled for this module, but there is only one variant so no
213 // need to hide it from make.
214 d.SetRROEnforcedForDependent(true)
215 }
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700216 }
217}
218
Colin Cross8f1b0332024-01-25 13:39:06 -0800219func (a *aapt) useResourceProcessorBusyBox(ctx android.BaseModuleContext) bool {
Colin Crosseee4ab12024-03-27 11:54:10 -0700220 return BoolDefault(a.aaptProperties.Use_resource_processor, ctx.Config().UseResourceProcessorByDefault()) &&
221 // TODO(b/331641946): remove this when ResourceProcessorBusyBox supports generating shared libraries.
Spandan Dasa62d9562024-08-06 00:13:31 +0000222 !slices.Contains(a.aaptProperties.Aaptflags, "--shared-lib") &&
223 // Use the legacy resource processor in kythe builds.
224 // The legacy resource processor creates an R.srcjar, which kythe can use for generating crossrefs.
225 // TODO(b/354854007): Re-enable BusyBox in kythe builds
226 !ctx.Config().EmitXrefRules()
Colin Cross4eae06d2023-06-20 22:40:02 -0700227}
228
Inseob Kim34dc4cd2023-11-07 13:37:14 +0900229func (a *aapt) filterProduct() string {
230 return String(a.aaptProperties.Filter_product)
231}
232
Colin Crossa97c5d32018-03-28 14:58:31 -0700233func (a *aapt) ExportPackage() android.Path {
234 return a.exportPackage
235}
Colin Crossa14fb6a2024-10-23 16:57:06 -0700236func (a *aapt) ResourcesNodeDepSet() depset.DepSet[*resourcesNode] {
Colin Crossab8d1382023-07-14 17:23:41 +0000237 return a.resourcesNodesDepSet
Colin Crossc1c37552019-01-31 11:42:41 -0800238}
239
Colin Crossa14fb6a2024-10-23 16:57:06 -0700240func (a *aapt) RRODirsDepSet() depset.DepSet[rroDir] {
Colin Crossab8d1382023-07-14 17:23:41 +0000241 return a.rroDirsDepSet
Colin Crossc1c37552019-01-31 11:42:41 -0800242}
243
Colin Crossa14fb6a2024-10-23 16:57:06 -0700244func (a *aapt) ManifestsDepSet() depset.DepSet[android.Path] {
Colin Crossab8d1382023-07-14 17:23:41 +0000245 return a.manifestsDepSet
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800246}
247
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700248func (a *aapt) SetRROEnforcedForDependent(enforce bool) {
249 a.aaptProperties.RROEnforcedForDependent = enforce
250}
251
252func (a *aapt) IsRROEnforced(ctx android.BaseModuleContext) bool {
253 // True if RRO is enforced for this module or...
254 return ctx.Config().EnforceRROForModule(ctx.ModuleName()) ||
Jeongik Chacee5ba92021-02-19 12:11:51 +0900255 // if RRO is enforced for any of its dependents.
256 a.aaptProperties.RROEnforcedForDependent
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700257}
258
Jiyong Parkf1691d22021-03-29 20:11:58 +0900259func (a *aapt) aapt2Flags(ctx android.ModuleContext, sdkContext android.SdkContext,
Spandan Das82229552024-12-06 01:38:45 +0000260 manifestPath android.Path, doNotIncludeAssetDirImplicitly bool) (compileFlags, linkFlags []string, linkDeps android.Paths,
Colin Crossa0ba2f52019-06-22 12:59:27 -0700261 resDirs, overlayDirs []globbedResourceDir, rroDirs []rroDir, resZips android.Paths) {
Colin Crossa97c5d32018-03-28 14:58:31 -0700262
Jaewoong Jung3aff5782020-02-11 07:54:35 -0800263 hasVersionCode := android.PrefixInList(a.aaptProperties.Aaptflags, "--version-code")
264 hasVersionName := android.PrefixInList(a.aaptProperties.Aaptflags, "--version-name")
Colin Crossa97c5d32018-03-28 14:58:31 -0700265
Colin Crossa97c5d32018-03-28 14:58:31 -0700266 // Flags specified in Android.bp
267 linkFlags = append(linkFlags, a.aaptProperties.Aaptflags...)
268
Eric Miao40eab202023-03-30 16:57:17 +0000269 linkFlags = append(linkFlags, "--enable-compact-entries")
Colin Crossa97c5d32018-03-28 14:58:31 -0700270
271 // Find implicit or explicit asset and resource dirs
Jiakai Zhangba82e282023-10-13 18:08:59 +0100272 assets := android.PathsRelativeToModuleSourceDir(android.SourceInput{
273 Context: ctx,
274 Paths: a.aaptProperties.Assets,
275 IncludeDirs: false,
276 })
Spandan Das82229552024-12-06 01:38:45 +0000277 var assetDirs android.Paths
278 if doNotIncludeAssetDirImplicitly {
279 assetDirs = android.PathsForModuleSrc(ctx, a.aaptProperties.Asset_dirs)
280 } else {
281 assetDirs = android.PathsWithOptionalDefaultForModuleSrc(ctx, a.aaptProperties.Asset_dirs, "assets")
282 }
Cole Faust5b989772024-11-06 11:03:16 -0800283 resourceDirs := android.PathsWithOptionalDefaultForModuleSrc(ctx, a.aaptProperties.Resource_dirs.GetOrDefault(ctx, nil), "res")
Colin Cross8a497952019-03-05 22:25:09 -0800284 resourceZips := android.PathsForModuleSrc(ctx, a.aaptProperties.Resource_zips)
Colin Crossa97c5d32018-03-28 14:58:31 -0700285
Colin Crossa97c5d32018-03-28 14:58:31 -0700286 // Glob directories into lists of paths
287 for _, dir := range resourceDirs {
288 resDirs = append(resDirs, globbedResourceDir{
289 dir: dir,
290 files: androidResourceGlob(ctx, dir),
291 })
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700292 resOverlayDirs, resRRODirs := overlayResourceGlob(ctx, a, dir)
Colin Crossa97c5d32018-03-28 14:58:31 -0700293 overlayDirs = append(overlayDirs, resOverlayDirs...)
294 rroDirs = append(rroDirs, resRRODirs...)
295 }
296
Cole Faust6265120c2024-09-05 16:27:56 -0700297 assetDirsHasher := sha256.New()
Colin Crossc20dc852020-11-10 12:27:45 -0800298 var assetDeps android.Paths
Cole Faust6265120c2024-09-05 16:27:56 -0700299 for _, dir := range assetDirs {
Colin Crossc20dc852020-11-10 12:27:45 -0800300 // Add a dependency on every file in the asset directory. This ensures the aapt2
301 // rule will be rerun if one of the files in the asset directory is modified.
Cole Faust6265120c2024-09-05 16:27:56 -0700302 dirContents := androidResourceGlob(ctx, dir)
303 assetDeps = append(assetDeps, dirContents...)
Colin Crossc20dc852020-11-10 12:27:45 -0800304
Cole Faust6265120c2024-09-05 16:27:56 -0700305 // Add a hash of all the files in the asset directory to the command line.
Colin Crossc20dc852020-11-10 12:27:45 -0800306 // This ensures the aapt2 rule will be run if a file is removed from the asset directory,
307 // or a file is added whose timestamp is older than the output of aapt2.
Cole Faust6265120c2024-09-05 16:27:56 -0700308 for _, path := range dirContents.Strings() {
309 assetDirsHasher.Write([]byte(path))
310 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700311 }
312
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700313 assetDirStrings := assetDirs.Strings()
314 if a.noticeFile.Valid() {
315 assetDirStrings = append(assetDirStrings, filepath.Dir(a.noticeFile.Path().String()))
Colin Crossc20dc852020-11-10 12:27:45 -0800316 assetDeps = append(assetDeps, a.noticeFile.Path())
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700317 }
Jiakai Zhangba82e282023-10-13 18:08:59 +0100318 if len(assets) > 0 {
319 // aapt2 doesn't support adding individual asset files. Create a temp directory to hold asset
320 // files and pass it to aapt2.
321 tmpAssetDir := android.PathForModuleOut(ctx, "tmp_asset_dir")
322
323 rule := android.NewRuleBuilder(pctx, ctx)
324 rule.Command().
325 Text("rm -rf").Text(tmpAssetDir.String()).
326 Text("&&").
327 Text("mkdir -p").Text(tmpAssetDir.String())
328
329 for _, asset := range assets {
330 output := tmpAssetDir.Join(ctx, asset.Rel())
331 assetDeps = append(assetDeps, output)
332 rule.Command().Text("mkdir -p").Text(filepath.Dir(output.String()))
333 rule.Command().Text("cp").Input(asset).Output(output)
334 }
335
336 rule.Build("tmp_asset_dir", "tmp_asset_dir")
337
338 assetDirStrings = append(assetDirStrings, tmpAssetDir.String())
339 }
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700340
Colin Crossa97c5d32018-03-28 14:58:31 -0700341 linkFlags = append(linkFlags, "--manifest "+manifestPath.String())
342 linkDeps = append(linkDeps, manifestPath)
343
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700344 linkFlags = append(linkFlags, android.JoinWithPrefix(assetDirStrings, "-A "))
Cole Faust6265120c2024-09-05 16:27:56 -0700345 linkFlags = append(linkFlags, fmt.Sprintf("$$(: %x)", assetDirsHasher.Sum(nil)))
Colin Crossc20dc852020-11-10 12:27:45 -0800346 linkDeps = append(linkDeps, assetDeps...)
Colin Crossa97c5d32018-03-28 14:58:31 -0700347
Spandan Das50885c02023-02-23 21:31:33 +0000348 // Returns the effective version for {min|target}_sdk_version
Spandan Das8c9ae7e2023-03-03 21:20:36 +0000349 effectiveVersionString := func(sdkVersion android.SdkSpec, minSdkVersion android.ApiLevel) string {
Spandan Das50885c02023-02-23 21:31:33 +0000350 // If {min|target}_sdk_version is current, use sdk_version to determine the effective level
351 // This is necessary for vendor modules.
352 // The effective version does not _only_ depend on {min|target}_sdk_version(level),
353 // but also on the sdk_version (kind+level)
Spandan Das8c9ae7e2023-03-03 21:20:36 +0000354 if minSdkVersion.IsCurrent() {
Spandan Das50885c02023-02-23 21:31:33 +0000355 ret, err := sdkVersion.EffectiveVersionString(ctx)
356 if err != nil {
357 ctx.ModuleErrorf("invalid sdk_version: %s", err)
358 }
359 return ret
360 }
361 ret, err := minSdkVersion.EffectiveVersionString(ctx)
362 if err != nil {
363 ctx.ModuleErrorf("invalid min_sdk_version: %s", err)
364 }
365 return ret
Jiyong Park6a927c42020-01-21 02:03:43 +0900366 }
Spandan Das50885c02023-02-23 21:31:33 +0000367 // SDK version flags
368 sdkVersion := sdkContext.SdkVersion(ctx)
369 minSdkVersion := effectiveVersionString(sdkVersion, sdkContext.MinSdkVersion(ctx))
Colin Crossa97c5d32018-03-28 14:58:31 -0700370
Colin Cross83bb3162018-06-25 15:48:06 -0700371 linkFlags = append(linkFlags, "--min-sdk-version "+minSdkVersion)
Spandan Das6450b552023-02-23 19:27:07 +0000372 // Use minSdkVersion for target-sdk-version, even if `target_sdk_version` is set
373 // This behavior has been copied from Make.
Colin Cross83bb3162018-06-25 15:48:06 -0700374 linkFlags = append(linkFlags, "--target-sdk-version "+minSdkVersion)
Colin Crossa97c5d32018-03-28 14:58:31 -0700375
Colin Crossa97c5d32018-03-28 14:58:31 -0700376 // Version code
377 if !hasVersionCode {
Dan Albert4f378d72020-07-23 17:32:15 -0700378 linkFlags = append(linkFlags, "--version-code", ctx.Config().PlatformSdkVersion().String())
Colin Crossa97c5d32018-03-28 14:58:31 -0700379 }
380
381 if !hasVersionName {
Colin Cross402d5e02018-04-25 14:54:06 -0700382 var versionName string
383 if ctx.ModuleName() == "framework-res" {
384 // Some builds set AppsDefaultVersionName() to include the build number ("O-123456"). aapt2 copies the
385 // version name of framework-res into app manifests as compileSdkVersionCodename, which confuses things
Colin Crossbfd347d2018-05-09 11:11:35 -0700386 // if it contains the build number. Use the PlatformVersionName instead.
387 versionName = ctx.Config().PlatformVersionName()
Colin Cross402d5e02018-04-25 14:54:06 -0700388 } else {
389 versionName = ctx.Config().AppsDefaultVersionName()
390 }
Colin Cross0b9f31f2019-02-28 11:00:01 -0800391 versionName = proptools.NinjaEscape(versionName)
Colin Crossa97c5d32018-03-28 14:58:31 -0700392 linkFlags = append(linkFlags, "--version-name ", versionName)
393 }
394
Colin Crossa0ba2f52019-06-22 12:59:27 -0700395 linkFlags, compileFlags = android.FilterList(linkFlags, []string{"--legacy"})
396
397 // Always set --pseudo-localize, it will be stripped out later for release
398 // builds that don't want it.
399 compileFlags = append(compileFlags, "--pseudo-localize")
400
401 return compileFlags, linkFlags, linkDeps, resDirs, overlayDirs, rroDirs, resourceZips
Colin Crossa97c5d32018-03-28 14:58:31 -0700402}
403
Paul Duffin250e6192019-06-07 10:44:37 +0100404func (a *aapt) deps(ctx android.BottomUpMutatorContext, sdkDep sdkDep) {
Colin Cross42308aa2018-11-14 21:44:17 -0800405 if sdkDep.frameworkResModule != "" {
406 ctx.AddVariationDependencies(nil, frameworkResTag, sdkDep.frameworkResModule)
Colin Crossa97c5d32018-03-28 14:58:31 -0700407 }
408}
409
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800410var extractAssetsRule = pctx.AndroidStaticRule("extractAssets",
411 blueprint.RuleParams{
412 Command: `${config.Zip2ZipCmd} -i ${in} -o ${out} "assets/**/*"`,
413 CommandDeps: []string{"${config.Zip2ZipCmd}"},
414 })
415
Alixf7a10272023-09-27 16:47:56 +0000416type aaptBuildActionOptions struct {
417 sdkContext android.SdkContext
418 classLoaderContexts dexpreopt.ClassLoaderContextMap
419 excludedLibs []string
420 enforceDefaultTargetSdkVersion bool
Rico Winda2fa2632024-03-13 13:09:17 +0100421 forceNonFinalResourceIDs bool
Alixf7a10272023-09-27 16:47:56 +0000422 extraLinkFlags []string
Jihoon Kang84b25892023-12-01 22:01:06 +0000423 aconfigTextFiles android.Paths
Jiakai Zhang36937082024-04-15 11:15:50 +0000424 usesLibrary *usesLibrary
Spandan Dase2f98da2024-11-18 19:22:39 +0000425 // If rroDirs is provided, it will be used to generate package-res.apk
426 rroDirs *android.Paths
427 // If manifestForAapt is not nil, it will be used for aapt instead of the default source manifest.
428 manifestForAapt android.Path
429}
430
431func filterRRO(rroDirsDepSet depset.DepSet[rroDir], filter overlayType) android.Paths {
432 var paths android.Paths
433 seen := make(map[android.Path]bool)
434 for _, d := range rroDirsDepSet.ToList() {
435 if d.overlayType == filter {
436 if seen[d.path] {
437 continue
438 }
439 seen[d.path] = true
440 paths = append(paths, d.path)
441 }
442 }
443 return paths
Alixf7a10272023-09-27 16:47:56 +0000444}
445
446func (a *aapt) buildActions(ctx android.ModuleContext, opts aaptBuildActionOptions) {
Colin Cross5446e882019-05-22 10:46:27 -0700447
Colin Cross8676c8c2023-10-12 15:58:57 -0700448 staticResourcesNodesDepSet, sharedResourcesNodesDepSet, staticRRODirsDepSet, staticManifestsDepSet, sharedExportPackages, libFlags :=
Jiakai Zhang36937082024-04-15 11:15:50 +0000449 aaptLibs(ctx, opts.sdkContext, opts.classLoaderContexts, opts.usesLibrary)
Ulya Trafimovich31e444e2020-08-14 17:32:16 +0100450
Paul Duffin06530572022-02-03 17:54:15 +0000451 // Exclude any libraries from the supplied list.
Alixf7a10272023-09-27 16:47:56 +0000452 opts.classLoaderContexts = opts.classLoaderContexts.ExcludeLibs(opts.excludedLibs)
Paul Duffin06530572022-02-03 17:54:15 +0000453
Colin Cross31656952018-05-24 16:11:20 -0700454 // App manifest file
Spandan Dase2f98da2024-11-18 19:22:39 +0000455 var manifestFilePath android.Path
456 if opts.manifestForAapt != nil {
457 manifestFilePath = opts.manifestForAapt
458 } else {
459 manifestFile := proptools.StringDefault(a.aaptProperties.Manifest, "AndroidManifest.xml")
460 manifestFilePath = android.PathForModuleSrc(ctx, manifestFile)
461 }
Colin Cross31656952018-05-24 16:11:20 -0700462
Spandan Dase2f98da2024-11-18 19:22:39 +0000463 manifestPath := ManifestFixer(ctx, manifestFilePath, ManifestFixerParams{
Alixf7a10272023-09-27 16:47:56 +0000464 SdkContext: opts.sdkContext,
465 ClassLoaderContexts: opts.classLoaderContexts,
Harshit Mahajan5b8b7302022-06-10 11:24:05 +0000466 IsLibrary: a.isLibrary,
467 DefaultManifestVersion: a.defaultManifestVersion,
468 UseEmbeddedNativeLibs: a.useEmbeddedNativeLibs,
469 UsesNonSdkApis: a.usesNonSdkApis,
470 UseEmbeddedDex: a.useEmbeddedDex,
471 HasNoCode: a.hasNoCode,
472 LoggingParent: a.LoggingParent,
Alixf7a10272023-09-27 16:47:56 +0000473 EnforceDefaultTargetSdkVersion: opts.enforceDefaultTargetSdkVersion,
Gurpreet Singh75d65f32022-01-24 17:44:05 +0000474 })
Colin Cross90c25c62019-04-19 16:22:57 -0700475
Colin Crossab8d1382023-07-14 17:23:41 +0000476 staticDeps := transitiveAarDeps(staticResourcesNodesDepSet.ToList())
Colin Cross8676c8c2023-10-12 15:58:57 -0700477 sharedDeps := transitiveAarDeps(sharedResourcesNodesDepSet.ToList())
Colin Crossab8d1382023-07-14 17:23:41 +0000478
Luca Stefanifd898822019-09-10 22:13:31 +0200479 // Add additional manifest files to transitive manifests.
480 additionalManifests := android.PathsForModuleSrc(ctx, a.aaptProperties.Additional_manifests)
Colin Crossab8d1382023-07-14 17:23:41 +0000481 transitiveManifestPaths := append(android.Paths{manifestPath}, additionalManifests...)
Colin Crossab8d1382023-07-14 17:23:41 +0000482 transitiveManifestPaths = append(transitiveManifestPaths, staticManifestsDepSet.ToList()...)
Colin Cross90c25c62019-04-19 16:22:57 -0700483
Colin Crossab8d1382023-07-14 17:23:41 +0000484 if len(transitiveManifestPaths) > 1 && !Bool(a.aaptProperties.Dont_merge_manifests) {
Alixf7a10272023-09-27 16:47:56 +0000485 manifestMergerParams := ManifestMergerParams{
486 staticLibManifests: transitiveManifestPaths[1:],
Alix96ea88452023-08-31 15:48:23 +0000487 isLibrary: a.isLibrary,
488 packageName: a.manifestValues.applicationId,
489 }
Alixf7a10272023-09-27 16:47:56 +0000490 a.mergedManifestFile = manifestMerger(ctx, transitiveManifestPaths[0], manifestMergerParams)
Colin Crossa6182ab2024-08-21 10:47:44 -0700491 ctx.CheckbuildFile(a.mergedManifestFile)
Colin Cross90c25c62019-04-19 16:22:57 -0700492 if !a.isLibrary {
493 // Only use the merged manifest for applications. For libraries, the transitive closure of manifests
494 // will be propagated to the final application and merged there. The merged manifest for libraries is
495 // only passed to Make, which can't handle transitive dependencies.
496 manifestPath = a.mergedManifestFile
497 }
498 } else {
499 a.mergedManifestFile = manifestPath
500 }
Colin Cross31656952018-05-24 16:11:20 -0700501
Spandan Das82229552024-12-06 01:38:45 +0000502 // do not include assets in autogenerated RRO.
503 compileFlags, linkFlags, linkDeps, resDirs, overlayDirs, rroDirs, resZips := a.aapt2Flags(ctx, opts.sdkContext, manifestPath, opts.rroDirs != nil)
Colin Cross31656952018-05-24 16:11:20 -0700504
Spandan Dase2f98da2024-11-18 19:22:39 +0000505 a.rroDirsDepSet = depset.NewBuilder[rroDir](depset.TOPOLOGICAL).
506 Direct(rroDirs...).
507 Transitive(staticRRODirsDepSet).Build()
508
Colin Cross31656952018-05-24 16:11:20 -0700509 linkFlags = append(linkFlags, libFlags...)
Colin Cross8676c8c2023-10-12 15:58:57 -0700510 linkDeps = append(linkDeps, sharedExportPackages...)
Colin Crossab8d1382023-07-14 17:23:41 +0000511 linkDeps = append(linkDeps, staticDeps.resPackages()...)
Alixf7a10272023-09-27 16:47:56 +0000512 linkFlags = append(linkFlags, opts.extraLinkFlags...)
Colin Cross1b6a3cf2018-07-24 14:51:30 -0700513 if a.isLibrary {
514 linkFlags = append(linkFlags, "--static-lib")
515 }
Rico Wind7152e822024-04-19 08:14:15 +0200516 if opts.forceNonFinalResourceIDs {
517 linkFlags = append(linkFlags, "--non-final-ids")
518 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700519
Colin Cross7c4dc5d2024-02-13 14:29:45 -0800520 linkFlags = append(linkFlags, "--no-static-lib-packages")
Spandan Das457e5062024-10-05 01:00:34 +0000521 if a.isLibrary {
522 // Pass --merge-only to skip resource references validation until the final
523 // app link step when when all static libraries are present.
Colin Cross4eae06d2023-06-20 22:40:02 -0700524 linkFlags = append(linkFlags, "--merge-only")
Colin Cross4eae06d2023-06-20 22:40:02 -0700525 }
526
Colin Crossa97c5d32018-03-28 14:58:31 -0700527 packageRes := android.PathForModuleOut(ctx, "package-res.apk")
Colin Crossa97c5d32018-03-28 14:58:31 -0700528 proguardOptionsFile := android.PathForModuleGen(ctx, "proguard.options")
529 rTxt := android.PathForModuleOut(ctx, "R.txt")
Colin Cross66f78822018-05-02 12:58:28 -0700530 // This file isn't used by Soong, but is generated for exporting
531 extraPackages := android.PathForModuleOut(ctx, "extra_packages")
Colin Cross4eae06d2023-06-20 22:40:02 -0700532 var transitiveRJars android.Paths
Colin Crossf3b7bad2023-08-02 15:49:00 -0700533 var srcJar android.WritablePath
Colin Crossa97c5d32018-03-28 14:58:31 -0700534
Colin Cross4aaa84a2018-08-21 15:14:37 -0700535 var compiledResDirs []android.Paths
Colin Crossa97c5d32018-03-28 14:58:31 -0700536 for _, dir := range resDirs {
Colin Cross014489c2020-06-02 20:09:13 -0700537 a.resourceFiles = append(a.resourceFiles, dir.files...)
Jihoon Kang98ea8362024-07-16 18:20:03 +0000538 compiledResDirs = append(compiledResDirs, aapt2Compile(ctx, dir.dir, dir.files,
539 compileFlags, a.filterProduct(), opts.aconfigTextFiles).Paths())
Colin Crossa97c5d32018-03-28 14:58:31 -0700540 }
Colin Cross4aaa84a2018-08-21 15:14:37 -0700541
Colin Crossa592e3e2019-02-19 16:59:53 -0800542 for i, zip := range resZips {
543 flata := android.PathForModuleOut(ctx, fmt.Sprintf("reszip.%d.flata", i))
Colin Crossa0ba2f52019-06-22 12:59:27 -0700544 aapt2CompileZip(ctx, flata, zip, "", compileFlags)
Colin Crossa592e3e2019-02-19 16:59:53 -0800545 compiledResDirs = append(compiledResDirs, android.Paths{flata})
546 }
547
Colin Cross4aaa84a2018-08-21 15:14:37 -0700548 var compiledRes, compiledOverlay android.Paths
549
Colin Crossab8d1382023-07-14 17:23:41 +0000550 // AAPT2 overlays are in lowest to highest priority order, reverse the topological order
551 // of transitiveStaticLibs.
552 transitiveStaticLibs := android.ReversePaths(staticDeps.resPackages())
553
Colin Cross8f1b0332024-01-25 13:39:06 -0800554 if a.isLibrary && a.useResourceProcessorBusyBox(ctx) {
Colin Cross4eae06d2023-06-20 22:40:02 -0700555 // When building an android_library with ResourceProcessorBusyBox enabled treat static library dependencies
556 // as imports. The resources from dependencies will not be merged into this module's package-res.apk, and
557 // instead modules depending on this module will reference package-res.apk from all transitive static
558 // dependencies.
Colin Cross1d3f5902024-03-05 11:51:54 -0800559 for _, sharedDep := range sharedDeps {
560 if sharedDep.usedResourceProcessor {
561 transitiveRJars = append(transitiveRJars, sharedDep.rJar)
562 }
563 }
Colin Cross4eae06d2023-06-20 22:40:02 -0700564 for _, staticDep := range staticDeps {
565 linkDeps = append(linkDeps, staticDep.resPackage)
566 linkFlags = append(linkFlags, "-I "+staticDep.resPackage.String())
567 if staticDep.usedResourceProcessor {
568 transitiveRJars = append(transitiveRJars, staticDep.rJar)
569 }
570 }
571 } else {
572 // When building an app or building a library without ResourceProcessorBusyBox enabled all static
573 // dependencies are compiled into this module's package-res.apk as overlays.
574 compiledOverlay = append(compiledOverlay, transitiveStaticLibs...)
575 }
Colin Cross4aaa84a2018-08-21 15:14:37 -0700576
Colin Crossbec85302019-02-13 13:15:46 -0800577 if len(transitiveStaticLibs) > 0 {
Colin Cross4aaa84a2018-08-21 15:14:37 -0700578 // If we are using static android libraries, every source file becomes an overlay.
579 // This is to emulate old AAPT behavior which simulated library support.
580 for _, compiledResDir := range compiledResDirs {
581 compiledOverlay = append(compiledOverlay, compiledResDir...)
582 }
Colin Crossbec85302019-02-13 13:15:46 -0800583 } else if a.isLibrary {
584 // Otherwise, for a static library we treat all the resources equally with no overlay.
585 for _, compiledResDir := range compiledResDirs {
586 compiledRes = append(compiledRes, compiledResDir...)
587 }
Colin Cross4aaa84a2018-08-21 15:14:37 -0700588 } else if len(compiledResDirs) > 0 {
589 // Without static libraries, the first directory is our directory, which can then be
590 // overlaid by the rest.
591 compiledRes = append(compiledRes, compiledResDirs[0]...)
592 for _, compiledResDir := range compiledResDirs[1:] {
593 compiledOverlay = append(compiledOverlay, compiledResDir...)
594 }
595 }
596
Spandan Dase2f98da2024-11-18 19:22:39 +0000597 var compiledRro, compiledRroOverlay android.Paths
598 if opts.rroDirs != nil {
599 compiledRro, compiledRroOverlay = a.compileResInDir(ctx, *opts.rroDirs, compileFlags, opts.aconfigTextFiles)
Spandan Dasde588a32024-12-03 22:52:24 +0000600 } else {
601 // RRO enforcement is done based on module name. Compile the overlayDirs only if rroDirs is nil.
602 // This ensures that the autogenerated RROs do not compile the overlay dirs twice.
603 for _, dir := range overlayDirs {
604 compiledOverlay = append(compiledOverlay, aapt2Compile(ctx, dir.dir, dir.files,
605 compileFlags, a.filterProduct(), opts.aconfigTextFiles).Paths()...)
606 }
Spandan Dase2f98da2024-11-18 19:22:39 +0000607 }
608
Colin Crosse560c4a2019-03-19 16:03:11 -0700609 var splitPackages android.WritablePaths
610 var splits []split
611
612 for _, s := range a.splitNames {
613 suffix := strings.Replace(s, ",", "_", -1)
614 path := android.PathForModuleOut(ctx, "package_"+suffix+".apk")
615 linkFlags = append(linkFlags, "--split", path.String()+":"+s)
616 splitPackages = append(splitPackages, path)
617 splits = append(splits, split{
618 name: s,
619 suffix: suffix,
620 path: path,
621 })
622 }
623
Colin Cross8f1b0332024-01-25 13:39:06 -0800624 if !a.useResourceProcessorBusyBox(ctx) {
Colin Crossf3b7bad2023-08-02 15:49:00 -0700625 // the subdir "android" is required to be filtered by package names
626 srcJar = android.PathForModuleGen(ctx, "android", "R.srcjar")
627 }
628
Colin Crossab8d1382023-07-14 17:23:41 +0000629 // No need to specify assets from dependencies to aapt2Link for libraries, all transitive assets will be
630 // provided to the final app aapt2Link step.
631 var transitiveAssets android.Paths
632 if !a.isLibrary {
633 transitiveAssets = android.ReverseSliceInPlace(staticDeps.assets())
634 }
Spandan Dase2f98da2024-11-18 19:22:39 +0000635 if opts.rroDirs == nil { // link resources and overlay
636 aapt2Link(ctx, packageRes, srcJar, proguardOptionsFile, rTxt,
637 linkFlags, linkDeps, compiledRes, compiledOverlay, transitiveAssets, splitPackages,
638 opts.aconfigTextFiles)
639 ctx.CheckbuildFile(packageRes)
640 } else { // link autogenerated rro
641 if len(compiledRro) == 0 {
642 return
643 }
644 aapt2Link(ctx, packageRes, srcJar, proguardOptionsFile, rTxt,
645 linkFlags, linkDeps, compiledRro, compiledRroOverlay, nil, nil,
646 opts.aconfigTextFiles)
647 ctx.CheckbuildFile(packageRes)
648 }
Colin Crossa6182ab2024-08-21 10:47:44 -0700649
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800650 // Extract assets from the resource package output so that they can be used later in aapt2link
651 // for modules that depend on this one.
Colin Crossab8d1382023-07-14 17:23:41 +0000652 if android.PrefixInList(linkFlags, "-A ") {
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800653 assets := android.PathForModuleOut(ctx, "assets.zip")
654 ctx.Build(pctx, android.BuildParams{
655 Rule: extractAssetsRule,
656 Input: packageRes,
657 Output: assets,
658 Description: "extract assets from built resource file",
659 })
660 a.assetPackage = android.OptionalPathForPath(assets)
661 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700662
Colin Cross8f1b0332024-01-25 13:39:06 -0800663 if a.useResourceProcessorBusyBox(ctx) {
Colin Cross4eae06d2023-06-20 22:40:02 -0700664 rJar := android.PathForModuleOut(ctx, "busybox/R.jar")
Rico Winda2fa2632024-03-13 13:09:17 +0100665 resourceProcessorBusyBoxGenerateBinaryR(ctx, rTxt, a.mergedManifestFile, rJar, staticDeps, a.isLibrary, a.aaptProperties.Aaptflags,
666 opts.forceNonFinalResourceIDs)
Colin Crossf3b7bad2023-08-02 15:49:00 -0700667 aapt2ExtractExtraPackages(ctx, extraPackages, rJar)
Colin Cross4eae06d2023-06-20 22:40:02 -0700668 transitiveRJars = append(transitiveRJars, rJar)
669 a.rJar = rJar
Colin Crossf3b7bad2023-08-02 15:49:00 -0700670 } else {
671 aapt2ExtractExtraPackages(ctx, extraPackages, srcJar)
Colin Cross4eae06d2023-06-20 22:40:02 -0700672 }
673
Colin Cross312634e2023-11-21 15:13:56 -0800674 transitiveAaptResourcePackages := staticDeps.resPackages().Strings()
675 transitiveAaptResourcePackages = slices.DeleteFunc(transitiveAaptResourcePackages, func(p string) bool {
676 return p == packageRes.String()
677 })
678 transitiveAaptResourcePackagesFile := android.PathForModuleOut(ctx, "transitive-res-packages")
679 android.WriteFileRule(ctx, transitiveAaptResourcePackagesFile, strings.Join(transitiveAaptResourcePackages, "\n"))
680
Colin Cross1d3f5902024-03-05 11:51:54 -0800681 // Reverse the list of R.jar files so that the current module comes first, and direct dependencies come before
682 // transitive dependencies.
683 transitiveRJars = android.ReversePaths(transitiveRJars)
684
Colin Crossa97c5d32018-03-28 14:58:31 -0700685 a.aaptSrcJar = srcJar
Colin Cross4eae06d2023-06-20 22:40:02 -0700686 a.transitiveAaptRJars = transitiveRJars
Colin Cross312634e2023-11-21 15:13:56 -0800687 a.transitiveAaptResourcePackagesFile = transitiveAaptResourcePackagesFile
Colin Crossa97c5d32018-03-28 14:58:31 -0700688 a.exportPackage = packageRes
689 a.manifestPath = manifestPath
690 a.proguardOptionsFile = proguardOptionsFile
Colin Cross66f78822018-05-02 12:58:28 -0700691 a.extraAaptPackagesFile = extraPackages
Colin Crossa97c5d32018-03-28 14:58:31 -0700692 a.rTxt = rTxt
Colin Crosse560c4a2019-03-19 16:03:11 -0700693 a.splits = splits
Colin Crossa14fb6a2024-10-23 16:57:06 -0700694 a.resourcesNodesDepSet = depset.NewBuilder[*resourcesNode](depset.TOPOLOGICAL).
Colin Crossab8d1382023-07-14 17:23:41 +0000695 Direct(&resourcesNode{
696 resPackage: a.exportPackage,
697 manifest: a.manifestPath,
698 additionalManifests: additionalManifests,
Colin Cross4eae06d2023-06-20 22:40:02 -0700699 rTxt: a.rTxt,
700 rJar: a.rJar,
Colin Crossab8d1382023-07-14 17:23:41 +0000701 assets: a.assetPackage,
Colin Cross4eae06d2023-06-20 22:40:02 -0700702
Colin Cross8f1b0332024-01-25 13:39:06 -0800703 usedResourceProcessor: a.useResourceProcessorBusyBox(ctx),
Colin Crossab8d1382023-07-14 17:23:41 +0000704 }).
705 Transitive(staticResourcesNodesDepSet).Build()
Colin Crossa14fb6a2024-10-23 16:57:06 -0700706 a.manifestsDepSet = depset.NewBuilder[android.Path](depset.TOPOLOGICAL).
Colin Crossab8d1382023-07-14 17:23:41 +0000707 Direct(a.manifestPath).
708 DirectSlice(additionalManifests).
709 Transitive(staticManifestsDepSet).Build()
710}
711
Spandan Dase2f98da2024-11-18 19:22:39 +0000712// comileResInDir finds the resource files in dirs by globbing and then compiles them using aapt2
713// returns the file paths of compiled resources
714// dirs[0] is used as compileRes
715// dirs[1:] is used as compileOverlay
716func (a *aapt) compileResInDir(ctx android.ModuleContext, dirs android.Paths, compileFlags []string, aconfig android.Paths) (android.Paths, android.Paths) {
717 filesInDir := func(dir android.Path) android.Paths {
718 files, err := ctx.GlobWithDeps(filepath.Join(dir.String(), "**/*"), androidResourceIgnoreFilenames)
719 if err != nil {
720 ctx.ModuleErrorf("failed to glob overlay resource dir %q: %s", dir, err.Error())
721 return nil
722 }
723 var filePaths android.Paths
724 for _, file := range files {
725 if strings.HasSuffix(file, "/") {
726 continue // ignore directories
727 }
728 filePaths = append(filePaths, android.PathForSource(ctx, file))
729 }
730 return filePaths
731 }
732
733 var compiledRes, compiledOverlay android.Paths
734 if len(dirs) == 0 {
735 return nil, nil
736 }
737 compiledRes = append(compiledRes, aapt2Compile(ctx, dirs[0], filesInDir(dirs[0]), compileFlags, a.filterProduct(), aconfig).Paths()...)
738 if len(dirs) > 0 {
739 for _, dir := range dirs[1:] {
740 compiledOverlay = append(compiledOverlay, aapt2Compile(ctx, dir, filesInDir(dir), compileFlags, a.filterProduct(), aconfig).Paths()...)
741 }
742 }
743 return compiledRes, compiledOverlay
744}
745
Colin Cross4eae06d2023-06-20 22:40:02 -0700746var resourceProcessorBusyBox = pctx.AndroidStaticRule("resourceProcessorBusyBox",
747 blueprint.RuleParams{
748 Command: "${config.JavaCmd} -cp ${config.ResourceProcessorBusyBox} " +
749 "com.google.devtools.build.android.ResourceProcessorBusyBox --tool=GENERATE_BINARY_R -- @${out}.args && " +
750 "if cmp -s ${out}.tmp ${out} ; then rm ${out}.tmp ; else mv ${out}.tmp ${out}; fi",
751 CommandDeps: []string{"${config.ResourceProcessorBusyBox}"},
752 Rspfile: "${out}.args",
753 RspfileContent: "--primaryRTxt ${rTxt} --primaryManifest ${manifest} --classJarOutput ${out}.tmp ${args}",
754 Restat: true,
755 }, "rTxt", "manifest", "args")
756
757// resourceProcessorBusyBoxGenerateBinaryR converts the R.txt file produced by aapt2 into R.class files
758// using Bazel's ResourceProcessorBusyBox tool, which is faster than compiling the R.java files and
759// supports producing classes for static dependencies that only include resources from that dependency.
760func resourceProcessorBusyBoxGenerateBinaryR(ctx android.ModuleContext, rTxt, manifest android.Path,
Rico Winda2fa2632024-03-13 13:09:17 +0100761 rJar android.WritablePath, transitiveDeps transitiveAarDeps, isLibrary bool, aaptFlags []string,
762 forceNonFinalIds bool) {
Colin Cross4eae06d2023-06-20 22:40:02 -0700763
764 var args []string
765 var deps android.Paths
766
767 if !isLibrary {
768 // When compiling an app, pass all R.txt and AndroidManifest.xml from transitive static library dependencies
769 // to ResourceProcessorBusyBox so that it can regenerate R.class files with the final resource IDs for each
770 // package.
771 args, deps = transitiveDeps.resourceProcessorDeps()
Rico Winda2fa2632024-03-13 13:09:17 +0100772 if forceNonFinalIds {
773 args = append(args, "--finalFields=false")
774 }
Colin Cross4eae06d2023-06-20 22:40:02 -0700775 } else {
776 // When compiling a library don't pass any dependencies as it only needs to generate an R.class file for this
777 // library. Pass --finalFields=false so that the R.class file contains non-final fields so they don't get
778 // inlined into the library before the final IDs are assigned during app compilation.
779 args = append(args, "--finalFields=false")
780 }
781
Colin Crossd3f7d1a2024-01-03 19:42:25 -0800782 for i, arg := range aaptFlags {
783 const AAPT_CUSTOM_PACKAGE = "--custom-package"
784 if strings.HasPrefix(arg, AAPT_CUSTOM_PACKAGE) {
785 pkg := strings.TrimSpace(strings.TrimPrefix(arg, AAPT_CUSTOM_PACKAGE))
786 if pkg == "" && i+1 < len(aaptFlags) {
787 pkg = aaptFlags[i+1]
788 }
789 args = append(args, "--packageForR "+pkg)
790 }
791 }
792
Colin Cross4eae06d2023-06-20 22:40:02 -0700793 deps = append(deps, rTxt, manifest)
794
795 ctx.Build(pctx, android.BuildParams{
796 Rule: resourceProcessorBusyBox,
797 Output: rJar,
798 Implicits: deps,
799 Description: "ResourceProcessorBusyBox",
800 Args: map[string]string{
801 "rTxt": rTxt.String(),
802 "manifest": manifest.String(),
803 "args": strings.Join(args, " "),
804 },
805 })
806}
807
Colin Crossab8d1382023-07-14 17:23:41 +0000808type resourcesNode struct {
809 resPackage android.Path
810 manifest android.Path
811 additionalManifests android.Paths
Colin Cross4eae06d2023-06-20 22:40:02 -0700812 rTxt android.Path
813 rJar android.Path
Colin Crossab8d1382023-07-14 17:23:41 +0000814 assets android.OptionalPath
Colin Cross4eae06d2023-06-20 22:40:02 -0700815
816 usedResourceProcessor bool
Colin Crossab8d1382023-07-14 17:23:41 +0000817}
818
819type transitiveAarDeps []*resourcesNode
820
821func (t transitiveAarDeps) resPackages() android.Paths {
Colin Cross4eae06d2023-06-20 22:40:02 -0700822 paths := make(android.Paths, 0, len(t))
Colin Crossab8d1382023-07-14 17:23:41 +0000823 for _, dep := range t {
824 paths = append(paths, dep.resPackage)
825 }
Colin Cross4eae06d2023-06-20 22:40:02 -0700826 return paths
Colin Crossab8d1382023-07-14 17:23:41 +0000827}
828
829func (t transitiveAarDeps) manifests() android.Paths {
Colin Cross4eae06d2023-06-20 22:40:02 -0700830 paths := make(android.Paths, 0, len(t))
Colin Crossab8d1382023-07-14 17:23:41 +0000831 for _, dep := range t {
832 paths = append(paths, dep.manifest)
833 paths = append(paths, dep.additionalManifests...)
834 }
Colin Cross4eae06d2023-06-20 22:40:02 -0700835 return paths
836}
837
838func (t transitiveAarDeps) resourceProcessorDeps() (args []string, deps android.Paths) {
839 for _, dep := range t {
840 args = append(args, "--library="+dep.rTxt.String()+","+dep.manifest.String())
841 deps = append(deps, dep.rTxt, dep.manifest)
842 }
843 return args, deps
Colin Crossab8d1382023-07-14 17:23:41 +0000844}
845
846func (t transitiveAarDeps) assets() android.Paths {
Colin Cross4eae06d2023-06-20 22:40:02 -0700847 paths := make(android.Paths, 0, len(t))
Colin Crossab8d1382023-07-14 17:23:41 +0000848 for _, dep := range t {
849 if dep.assets.Valid() {
850 paths = append(paths, dep.assets.Path())
851 }
852 }
853 return paths
Colin Crossa97c5d32018-03-28 14:58:31 -0700854}
855
856// aaptLibs collects libraries from dependencies and sdk_version and converts them into paths
Jiakai Zhang36937082024-04-15 11:15:50 +0000857func aaptLibs(ctx android.ModuleContext, sdkContext android.SdkContext,
858 classLoaderContexts dexpreopt.ClassLoaderContextMap, usesLibrary *usesLibrary) (
Colin Crossa14fb6a2024-10-23 16:57:06 -0700859 staticResourcesNodes, sharedResourcesNodes depset.DepSet[*resourcesNode], staticRRODirs depset.DepSet[rroDir],
860 staticManifests depset.DepSet[android.Path], sharedLibs android.Paths, flags []string) {
Colin Crossa97c5d32018-03-28 14:58:31 -0700861
Ulya Trafimovichb23d28c2020-10-08 12:53:58 +0100862 if classLoaderContexts == nil {
Ulya Trafimovich18554242020-11-03 15:55:11 +0000863 // Not all callers need to compute class loader context, those who don't just pass nil.
864 // Create a temporary class loader context here (it will be computed, but not used).
Ulya Trafimovichb23d28c2020-10-08 12:53:58 +0100865 classLoaderContexts = make(dexpreopt.ClassLoaderContextMap)
Ulya Trafimovich18554242020-11-03 15:55:11 +0000866 }
867
Colin Cross83bb3162018-06-25 15:48:06 -0700868 sdkDep := decodeSdkDep(ctx, sdkContext)
Colin Crossa97c5d32018-03-28 14:58:31 -0700869 if sdkDep.useFiles {
Colin Cross86a60ae2018-05-29 14:44:55 -0700870 sharedLibs = append(sharedLibs, sdkDep.jars...)
Colin Crossa97c5d32018-03-28 14:58:31 -0700871 }
872
Colin Crossa14fb6a2024-10-23 16:57:06 -0700873 var staticResourcesNodeDepSets []depset.DepSet[*resourcesNode]
874 var sharedResourcesNodeDepSets []depset.DepSet[*resourcesNode]
875 rroDirsDepSetBuilder := depset.NewBuilder[rroDir](depset.TOPOLOGICAL)
876 manifestsDepSetBuilder := depset.NewBuilder[android.Path](depset.TOPOLOGICAL)
Colin Crossab8d1382023-07-14 17:23:41 +0000877
Colin Crossa97c5d32018-03-28 14:58:31 -0700878 ctx.VisitDirectDeps(func(module android.Module) {
Ulya Trafimovich65b03192020-12-03 16:50:22 +0000879 depTag := ctx.OtherModuleDependencyTag(module)
Ulya Trafimovich18554242020-11-03 15:55:11 +0000880
Colin Crossa97c5d32018-03-28 14:58:31 -0700881 var exportPackage android.Path
Colin Cross66f78822018-05-02 12:58:28 -0700882 aarDep, _ := module.(AndroidLibraryDependency)
883 if aarDep != nil {
Colin Crossa97c5d32018-03-28 14:58:31 -0700884 exportPackage = aarDep.ExportPackage()
885 }
886
Ulya Trafimovich65b03192020-12-03 16:50:22 +0000887 switch depTag {
Colin Cross4b964c02018-10-15 16:18:06 -0700888 case instrumentationForTag:
889 // Nothing, instrumentationForTag is treated as libTag for javac but not for aapt2.
Spandan Dase2f98da2024-11-18 19:22:39 +0000890 case sdkLibTag, libTag, rroDepTag:
Colin Cross5446e882019-05-22 10:46:27 -0700891 if exportPackage != nil {
Colin Cross8676c8c2023-10-12 15:58:57 -0700892 sharedResourcesNodeDepSets = append(sharedResourcesNodeDepSets, aarDep.ResourcesNodeDepSet())
Colin Cross5446e882019-05-22 10:46:27 -0700893 sharedLibs = append(sharedLibs, exportPackage)
894 }
Colin Cross5446e882019-05-22 10:46:27 -0700895 case frameworkResTag:
Colin Crossa97c5d32018-03-28 14:58:31 -0700896 if exportPackage != nil {
897 sharedLibs = append(sharedLibs, exportPackage)
898 }
899 case staticLibTag:
900 if exportPackage != nil {
Colin Cross8676c8c2023-10-12 15:58:57 -0700901 staticResourcesNodeDepSets = append(staticResourcesNodeDepSets, aarDep.ResourcesNodeDepSet())
Colin Crossab8d1382023-07-14 17:23:41 +0000902 rroDirsDepSetBuilder.Transitive(aarDep.RRODirsDepSet())
903 manifestsDepSetBuilder.Transitive(aarDep.ManifestsDepSet())
Colin Crossa97c5d32018-03-28 14:58:31 -0700904 }
905 }
Ulya Trafimovich18554242020-11-03 15:55:11 +0000906
Ulya Trafimovich88bb6f62020-12-16 16:16:11 +0000907 addCLCFromDep(ctx, module, classLoaderContexts)
Jiakai Zhang36937082024-04-15 11:15:50 +0000908 if usesLibrary != nil {
909 addMissingOptionalUsesLibsFromDep(ctx, module, usesLibrary)
910 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700911 })
912
Colin Crossab8d1382023-07-14 17:23:41 +0000913 // AAPT2 overlays are in lowest to highest priority order, the topological order will be reversed later.
914 // Reverse the dependency order now going into the depset so that it comes out in order after the second
915 // reverse later.
916 // NOTE: this is legacy and probably incorrect behavior, for most other cases (e.g. conflicting classes in
917 // dependencies) the highest priority dependency is listed first, but for resources the highest priority
Colin Cross9055e212024-03-23 04:43:41 +0000918 // dependency has to be listed last. This is also inconsistent with the way manifests from the same
919 // transitive dependencies are merged.
Colin Crossa14fb6a2024-10-23 16:57:06 -0700920 staticResourcesNodes = depset.New(depset.TOPOLOGICAL, nil,
Colin Cross8676c8c2023-10-12 15:58:57 -0700921 android.ReverseSliceInPlace(staticResourcesNodeDepSets))
Colin Crossa14fb6a2024-10-23 16:57:06 -0700922 sharedResourcesNodes = depset.New(depset.TOPOLOGICAL, nil,
Colin Cross8676c8c2023-10-12 15:58:57 -0700923 android.ReverseSliceInPlace(sharedResourcesNodeDepSets))
Colin Crossa97c5d32018-03-28 14:58:31 -0700924
Colin Crossab8d1382023-07-14 17:23:41 +0000925 staticRRODirs = rroDirsDepSetBuilder.Build()
926 staticManifests = manifestsDepSetBuilder.Build()
927
928 if len(staticResourcesNodes.ToList()) > 0 {
Colin Crossa97c5d32018-03-28 14:58:31 -0700929 flags = append(flags, "--auto-add-overlay")
930 }
931
932 for _, sharedLib := range sharedLibs {
933 flags = append(flags, "-I "+sharedLib.String())
934 }
935
Colin Cross8676c8c2023-10-12 15:58:57 -0700936 return staticResourcesNodes, sharedResourcesNodes, staticRRODirs, staticManifests, sharedLibs, flags
Colin Crossa97c5d32018-03-28 14:58:31 -0700937}
938
939type AndroidLibrary struct {
940 Library
941 aapt
942
943 androidLibraryProperties androidLibraryProperties
944
945 aarFile android.WritablePath
Colin Cross89c31582018-04-30 15:55:11 -0700946}
947
Colin Crossa97c5d32018-03-28 14:58:31 -0700948var _ AndroidLibraryDependency = (*AndroidLibrary)(nil)
949
950func (a *AndroidLibrary) DepsMutator(ctx android.BottomUpMutatorContext) {
Jiakai Zhangf98da192024-04-15 11:15:41 +0000951 a.usesLibrary.deps(ctx, false)
Colin Crossa97c5d32018-03-28 14:58:31 -0700952 a.Module.deps(ctx)
Jiyong Parkf1691d22021-03-29 20:11:58 +0900953 sdkDep := decodeSdkDep(ctx, android.SdkContext(a))
Paul Duffin250e6192019-06-07 10:44:37 +0100954 if sdkDep.hasFrameworkLibs() {
955 a.aapt.deps(ctx, sdkDep)
Colin Crossa97c5d32018-03-28 14:58:31 -0700956 }
Jihoon Kang9049c272024-03-19 21:57:36 +0000957
958 for _, aconfig_declaration := range a.aaptProperties.Flags_packages {
959 ctx.AddDependency(ctx.Module(), aconfigDeclarationTag, aconfig_declaration)
960 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700961}
962
963func (a *AndroidLibrary) GenerateAndroidBuildActions(ctx android.ModuleContext) {
Colin Crosse4246ab2019-02-05 21:55:21 -0800964 a.aapt.isLibrary = true
Ulya Trafimovich42c7f0d2021-08-17 16:20:29 +0100965 a.classLoaderContexts = a.usesLibrary.classLoaderContextForUsesLibDeps(ctx)
Spandan Das0727ba72024-02-13 16:37:43 +0000966 if a.usesLibrary.shouldDisableDexpreopt {
967 a.dexpreopter.disableDexpreopt()
968 }
Jihoon Kang9aef7772024-06-14 23:45:06 +0000969 aconfigTextFilePaths := getAconfigFilePaths(ctx)
Alixf7a10272023-09-27 16:47:56 +0000970 a.aapt.buildActions(ctx,
971 aaptBuildActionOptions{
972 sdkContext: android.SdkContext(a),
973 classLoaderContexts: a.classLoaderContexts,
974 enforceDefaultTargetSdkVersion: false,
Jihoon Kang9aef7772024-06-14 23:45:06 +0000975 aconfigTextFiles: aconfigTextFilePaths,
Jiakai Zhang36937082024-04-15 11:15:50 +0000976 usesLibrary: &a.usesLibrary,
Alixf7a10272023-09-27 16:47:56 +0000977 },
978 )
Colin Crossa97c5d32018-03-28 14:58:31 -0700979
Colin Crossff694a82023-12-13 15:54:49 -0800980 apexInfo, _ := android.ModuleProvider(ctx, android.ApexInfoProvider)
981 a.hideApexVariantFromMake = !apexInfo.IsForPlatform()
Colin Cross56a83212020-09-15 18:30:11 -0700982
yangbill2af0b6e2024-03-15 09:29:29 +0000983 a.stem = proptools.StringDefault(a.overridableProperties.Stem, ctx.ModuleName())
Jihoon Kang1bfb6f22023-07-01 00:13:47 +0000984
Colin Cross4eae06d2023-06-20 22:40:02 -0700985 ctx.CheckbuildFile(a.aapt.proguardOptionsFile)
986 ctx.CheckbuildFile(a.aapt.exportPackage)
Colin Cross8f1b0332024-01-25 13:39:06 -0800987 if a.useResourceProcessorBusyBox(ctx) {
Colin Cross4eae06d2023-06-20 22:40:02 -0700988 ctx.CheckbuildFile(a.aapt.rJar)
Colin Crossf3b7bad2023-08-02 15:49:00 -0700989 } else {
990 ctx.CheckbuildFile(a.aapt.aaptSrcJar)
Colin Cross4eae06d2023-06-20 22:40:02 -0700991 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700992
993 // apps manifests are handled by aapt, don't let Module see them
994 a.properties.Manifest = nil
995
Colin Cross014489c2020-06-02 20:09:13 -0700996 a.linter.mergedManifest = a.aapt.mergedManifestFile
997 a.linter.manifest = a.aapt.manifestPath
998 a.linter.resources = a.aapt.resourceFiles
999
Sam Delmericoc8e040c2023-10-31 17:27:02 +00001000 proguardSpecInfo := a.collectProguardSpecInfo(ctx)
Colin Cross40213022023-12-13 15:19:49 -08001001 android.SetProvider(ctx, ProguardSpecInfoProvider, proguardSpecInfo)
Colin Cross312634e2023-11-21 15:13:56 -08001002 exportedProguardFlagsFiles := proguardSpecInfo.ProguardFlagsFiles.ToList()
1003 a.extraProguardFlagsFiles = append(a.extraProguardFlagsFiles, exportedProguardFlagsFiles...)
1004 a.extraProguardFlagsFiles = append(a.extraProguardFlagsFiles, a.proguardOptionsFile)
1005
1006 combinedExportedProguardFlagFile := android.PathForModuleOut(ctx, "export_proguard_flags")
1007 writeCombinedProguardFlagsFile(ctx, combinedExportedProguardFlagFile, exportedProguardFlagsFiles)
1008 a.combinedExportedProguardFlagsFile = combinedExportedProguardFlagFile
Colin Crossa97c5d32018-03-28 14:58:31 -07001009
Colin Cross4eae06d2023-06-20 22:40:02 -07001010 var extraSrcJars android.Paths
1011 var extraCombinedJars android.Paths
1012 var extraClasspathJars android.Paths
Colin Cross8f1b0332024-01-25 13:39:06 -08001013 if a.useResourceProcessorBusyBox(ctx) {
Colin Cross4eae06d2023-06-20 22:40:02 -07001014 // When building a library with ResourceProcessorBusyBox enabled ResourceProcessorBusyBox for this
1015 // library and each of the transitive static android_library dependencies has already created an
1016 // R.class file for the appropriate package. Add all of those R.class files to the classpath.
1017 extraClasspathJars = a.transitiveAaptRJars
1018 } else {
1019 // When building a library without ResourceProcessorBusyBox the aapt2 rule creates R.srcjar containing
1020 // R.java files for the library's package and the packages from all transitive static android_library
1021 // dependencies. Compile the srcjar alongside the rest of the sources.
1022 extraSrcJars = android.Paths{a.aapt.aaptSrcJar}
1023 }
1024
Colin Crossfdaa6722024-08-23 11:58:08 -07001025 a.Module.compile(ctx, extraSrcJars, extraClasspathJars, extraCombinedJars, nil)
Colin Crossa97c5d32018-03-28 14:58:31 -07001026
Colin Crossf57c5782019-01-25 13:20:38 -08001027 a.aarFile = android.PathForModuleOut(ctx, ctx.ModuleName()+".aar")
Colin Crossa97c5d32018-03-28 14:58:31 -07001028 var res android.Paths
1029 if a.androidLibraryProperties.BuildAAR {
1030 BuildAAR(ctx, a.aarFile, a.outputFile, a.manifestPath, a.rTxt, res)
Colin Crossa97c5d32018-03-28 14:58:31 -07001031 }
Colin Cross89c31582018-04-30 15:55:11 -07001032
Sam Delmerico82602492022-06-10 17:05:42 +00001033 prebuiltJniPackages := android.Paths{}
1034 ctx.VisitDirectDeps(func(module android.Module) {
Colin Cross313aa542023-12-13 13:47:44 -08001035 if info, ok := android.OtherModuleProvider(ctx, module, JniPackageProvider); ok {
Sam Delmerico82602492022-06-10 17:05:42 +00001036 prebuiltJniPackages = append(prebuiltJniPackages, info.JniPackages...)
1037 }
1038 })
1039 if len(prebuiltJniPackages) > 0 {
Colin Cross40213022023-12-13 15:19:49 -08001040 android.SetProvider(ctx, JniPackageProvider, JniPackageInfo{
Sam Delmerico82602492022-06-10 17:05:42 +00001041 JniPackages: prebuiltJniPackages,
1042 })
1043 }
Jihoon Kang9aef7772024-06-14 23:45:06 +00001044
1045 android.SetProvider(ctx, FlagsPackagesProvider, FlagsPackages{
1046 AconfigTextFiles: aconfigTextFilePaths,
1047 })
mrziwang9f7b9f42024-07-10 12:18:06 -07001048
1049 a.setOutputFiles(ctx)
1050}
1051
1052func (a *AndroidLibrary) setOutputFiles(ctx android.ModuleContext) {
1053 ctx.SetOutputFiles([]android.Path{a.aarFile}, ".aar")
1054 setOutputFiles(ctx, a.Library.Module)
Colin Crossa97c5d32018-03-28 14:58:31 -07001055}
1056
Cole Faustb36d31d2024-08-27 16:04:28 -07001057func (a *AndroidLibrary) IDEInfo(ctx android.BaseModuleContext, dpInfo *android.IdeInfo) {
1058 a.Library.IDEInfo(ctx, dpInfo)
1059 a.aapt.IDEInfo(ctx, dpInfo)
Colin Cross95b53b82023-10-17 13:21:02 -07001060}
1061
Cole Faustb36d31d2024-08-27 16:04:28 -07001062func (a *aapt) IDEInfo(ctx android.BaseModuleContext, dpInfo *android.IdeInfo) {
Colin Cross8f1b0332024-01-25 13:39:06 -08001063 if a.rJar != nil {
Colin Cross95b53b82023-10-17 13:21:02 -07001064 dpInfo.Jars = append(dpInfo.Jars, a.rJar.String())
1065 }
1066}
1067
Colin Cross1b16b0e2019-02-12 14:41:32 -08001068// android_library builds and links sources into a `.jar` file for the device along with Android resources.
1069//
1070// An android_library has a single variant that produces a `.jar` file containing `.class` files that were
Sam Delmerico82602492022-06-10 17:05:42 +00001071// compiled against the device bootclasspath, along with a `package-res.apk` file containing Android resources compiled
Colin Cross1b16b0e2019-02-12 14:41:32 -08001072// with aapt2. This module is not suitable for installing on a device, but can be used as a `static_libs` dependency of
1073// an android_app module.
Colin Crossa97c5d32018-03-28 14:58:31 -07001074func AndroidLibraryFactory() android.Module {
1075 module := &AndroidLibrary{}
1076
Colin Crossce6734e2020-06-15 16:09:53 -07001077 module.Module.addHostAndDeviceProperties()
Colin Crossa97c5d32018-03-28 14:58:31 -07001078 module.AddProperties(
Colin Crossa97c5d32018-03-28 14:58:31 -07001079 &module.aaptProperties,
Ronald Braunsteincdc66f42024-04-12 11:23:19 -07001080 &module.androidLibraryProperties,
1081 &module.sourceProperties)
Colin Crossa97c5d32018-03-28 14:58:31 -07001082
1083 module.androidLibraryProperties.BuildAAR = true
Colin Cross014489c2020-06-02 20:09:13 -07001084 module.Module.linter.library = true
Colin Crossa97c5d32018-03-28 14:58:31 -07001085
Jooyung Hanacc7bbe2020-05-20 09:06:00 +09001086 android.InitApexModule(module)
Colin Cross48de9a42018-10-02 13:53:33 -07001087 InitJavaModule(module, android.DeviceSupported)
Colin Crossa97c5d32018-03-28 14:58:31 -07001088 return module
1089}
1090
Colin Crossfabb6082018-02-20 17:22:23 -08001091//
1092// AAR (android library) prebuilts
1093//
Colin Crossfabb6082018-02-20 17:22:23 -08001094
Vinh Trance0781f2022-04-13 01:30:44 +00001095// Properties for android_library_import
Colin Crossfabb6082018-02-20 17:22:23 -08001096type AARImportProperties struct {
Vinh Trance0781f2022-04-13 01:30:44 +00001097 // ARR (android library prebuilt) filepath. Exactly one ARR is required.
Colin Cross27b922f2019-03-04 22:35:41 -08001098 Aars []string `android:"path"`
Vinh Trance0781f2022-04-13 01:30:44 +00001099 // If not blank, set to the version of the sdk to compile against.
1100 // Defaults to private.
1101 // Values are of one of the following forms:
1102 // 1) numerical API level, "current", "none", or "core_platform"
1103 // 2) An SDK kind with an API level: "<sdk kind>_<API level>"
1104 // See build/soong/android/sdk_version.go for the complete and up to date list of SDK kinds.
1105 // If the SDK kind is empty, it will be set to public
1106 Sdk_version *string
1107 // If not blank, set the minimum version of the sdk that the compiled artifacts will run against.
1108 // Defaults to sdk_version if not set. See sdk_version for possible values.
Colin Cross479884c2018-07-10 13:39:30 -07001109 Min_sdk_version *string
Vinh Trance0781f2022-04-13 01:30:44 +00001110 // List of java static libraries that the included ARR (android library prebuilts) has dependencies to.
Cole Faustb7493472024-08-28 11:55:52 -07001111 Static_libs proptools.Configurable[[]string]
Vinh Trance0781f2022-04-13 01:30:44 +00001112 // List of java libraries that the included ARR (android library prebuilts) has dependencies to.
1113 Libs []string
1114 // If set to true, run Jetifier against .aar file. Defaults to false.
Colin Cross1001a792019-03-21 22:21:39 -07001115 Jetifier *bool
Sam Delmerico82602492022-06-10 17:05:42 +00001116 // If true, extract JNI libs from AAR archive. These libs will be accessible to android_app modules and
1117 // will be passed transitively through android_libraries to an android_app.
1118 //TODO(b/241138093) evaluate whether we can have this flag default to true for Bazel conversion
1119 Extract_jni *bool
Colin Cross21ed4692024-04-24 20:23:38 +00001120
1121 // If set, overrides the manifest extracted from the AAR with the provided path.
1122 Manifest *string `android:"path"`
Colin Crossfabb6082018-02-20 17:22:23 -08001123}
1124
1125type AARImport struct {
1126 android.ModuleBase
Colin Cross48de9a42018-10-02 13:53:33 -07001127 android.DefaultableModuleBase
Jooyung Hanacc7bbe2020-05-20 09:06:00 +09001128 android.ApexModuleBase
Colin Crossfabb6082018-02-20 17:22:23 -08001129 prebuilt android.Prebuilt
1130
Jooyung Hanacc7bbe2020-05-20 09:06:00 +09001131 // Functionality common to Module and Import.
1132 embeddableInModuleAndImport
1133
Colin Cross9ffaf282024-08-12 13:50:09 -07001134 providesTransitiveHeaderJarsForR8
Sam Delmerico9f9c0a22022-11-29 11:19:37 -05001135
Colin Crossfabb6082018-02-20 17:22:23 -08001136 properties AARImportProperties
1137
Colin Cross7707b242024-07-26 12:02:36 -07001138 headerJarFile android.Path
1139 implementationJarFile android.Path
1140 implementationAndResourcesJarFile android.Path
1141 proguardFlags android.Path
1142 exportPackage android.Path
Colin Cross312634e2023-11-21 15:13:56 -08001143 transitiveAaptResourcePackagesFile android.Path
Colin Cross7707b242024-07-26 12:02:36 -07001144 extraAaptPackagesFile android.Path
Colin Cross21ed4692024-04-24 20:23:38 +00001145 manifest android.Path
Colin Cross7707b242024-07-26 12:02:36 -07001146 assetsPackage android.Path
1147 rTxt android.Path
1148 rJar android.Path
Colin Cross66f78822018-05-02 12:58:28 -07001149
Colin Crossa14fb6a2024-10-23 16:57:06 -07001150 resourcesNodesDepSet depset.DepSet[*resourcesNode]
1151 manifestsDepSet depset.DepSet[android.Path]
Colin Cross56a83212020-09-15 18:30:11 -07001152
1153 hideApexVariantFromMake bool
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001154
Sam Delmerico82602492022-06-10 17:05:42 +00001155 aarPath android.Path
1156 jniPackages android.Paths
Jiyong Park92315372021-04-02 08:45:46 +09001157
1158 sdkVersion android.SdkSpec
Spandan Das8c9ae7e2023-03-03 21:20:36 +00001159 minSdkVersion android.ApiLevel
LaMont Jonesafe7baf2024-01-09 22:47:39 +00001160
Colin Cross9055e212024-03-23 04:43:41 +00001161 usesLibrary
1162 classLoaderContexts dexpreopt.ClassLoaderContextMap
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001163}
1164
Jiyong Park92315372021-04-02 08:45:46 +09001165func (a *AARImport) SdkVersion(ctx android.EarlyModuleContext) android.SdkSpec {
1166 return android.SdkSpecFrom(ctx, String(a.properties.Sdk_version))
Colin Cross83bb3162018-06-25 15:48:06 -07001167}
1168
Jiyong Parkf1691d22021-03-29 20:11:58 +09001169func (a *AARImport) SystemModules() string {
Paul Duffine25c6442019-10-11 13:50:28 +01001170 return ""
1171}
1172
Spandan Das8c9ae7e2023-03-03 21:20:36 +00001173func (a *AARImport) MinSdkVersion(ctx android.EarlyModuleContext) android.ApiLevel {
Colin Cross479884c2018-07-10 13:39:30 -07001174 if a.properties.Min_sdk_version != nil {
Spandan Das8c9ae7e2023-03-03 21:20:36 +00001175 return android.ApiLevelFrom(ctx, *a.properties.Min_sdk_version)
Colin Cross479884c2018-07-10 13:39:30 -07001176 }
Spandan Das8c9ae7e2023-03-03 21:20:36 +00001177 return a.SdkVersion(ctx).ApiLevel
Colin Cross83bb3162018-06-25 15:48:06 -07001178}
1179
Spandan Dasa26eda72023-03-02 00:56:06 +00001180func (a *AARImport) ReplaceMaxSdkVersionPlaceholder(ctx android.EarlyModuleContext) android.ApiLevel {
1181 return android.SdkSpecFrom(ctx, "").ApiLevel
William Loh5a082f92022-05-17 20:21:50 +00001182}
1183
Spandan Dasca70fc42023-03-01 23:38:49 +00001184func (a *AARImport) TargetSdkVersion(ctx android.EarlyModuleContext) android.ApiLevel {
1185 return a.SdkVersion(ctx).ApiLevel
Dan Willemsen419290a2018-10-31 15:28:47 -07001186}
1187
Colin Cross1e743852019-10-28 11:37:20 -07001188func (a *AARImport) javaVersion() string {
1189 return ""
1190}
1191
Colin Crossa97c5d32018-03-28 14:58:31 -07001192var _ AndroidLibraryDependency = (*AARImport)(nil)
1193
1194func (a *AARImport) ExportPackage() android.Path {
1195 return a.exportPackage
1196}
Colin Crossa14fb6a2024-10-23 16:57:06 -07001197func (a *AARImport) ResourcesNodeDepSet() depset.DepSet[*resourcesNode] {
Colin Crossab8d1382023-07-14 17:23:41 +00001198 return a.resourcesNodesDepSet
Colin Crossc1c37552019-01-31 11:42:41 -08001199}
1200
Colin Crossa14fb6a2024-10-23 16:57:06 -07001201func (a *AARImport) RRODirsDepSet() depset.DepSet[rroDir] {
1202 return depset.New[rroDir](depset.TOPOLOGICAL, nil, nil)
Colin Cross66f78822018-05-02 12:58:28 -07001203}
1204
Colin Crossa14fb6a2024-10-23 16:57:06 -07001205func (a *AARImport) ManifestsDepSet() depset.DepSet[android.Path] {
Colin Crossab8d1382023-07-14 17:23:41 +00001206 return a.manifestsDepSet
Jaewoong Jung6431ca72020-01-15 14:15:10 -08001207}
1208
Jaewoong Jungc779cd42020-10-06 18:56:10 -07001209// RRO enforcement is not available on aar_import since its RRO dirs are not
1210// exported.
1211func (a *AARImport) SetRROEnforcedForDependent(enforce bool) {
1212}
1213
1214// RRO enforcement is not available on aar_import since its RRO dirs are not
1215// exported.
1216func (a *AARImport) IsRROEnforced(ctx android.BaseModuleContext) bool {
1217 return false
1218}
1219
Colin Crossfabb6082018-02-20 17:22:23 -08001220func (a *AARImport) Prebuilt() *android.Prebuilt {
1221 return &a.prebuilt
1222}
1223
1224func (a *AARImport) Name() string {
1225 return a.prebuilt.Name(a.ModuleBase.Name())
1226}
1227
Jiyong Park618922e2020-01-08 13:35:43 +09001228func (a *AARImport) JacocoReportClassesFile() android.Path {
1229 return nil
1230}
1231
Colin Crossfabb6082018-02-20 17:22:23 -08001232func (a *AARImport) DepsMutator(ctx android.BottomUpMutatorContext) {
Jeongik Cha816a23a2020-07-08 01:09:23 +09001233 if !ctx.Config().AlwaysUsePrebuiltSdks() {
Jiyong Parkf1691d22021-03-29 20:11:58 +09001234 sdkDep := decodeSdkDep(ctx, android.SdkContext(a))
Colin Crossa97c5d32018-03-28 14:58:31 -07001235 if sdkDep.useModule && sdkDep.frameworkResModule != "" {
Colin Cross42d48b72018-08-29 14:10:52 -07001236 ctx.AddVariationDependencies(nil, frameworkResTag, sdkDep.frameworkResModule)
Colin Crossfabb6082018-02-20 17:22:23 -08001237 }
1238 }
Colin Crossa97c5d32018-03-28 14:58:31 -07001239
Colin Cross42d48b72018-08-29 14:10:52 -07001240 ctx.AddVariationDependencies(nil, libTag, a.properties.Libs...)
Cole Faustb7493472024-08-28 11:55:52 -07001241 ctx.AddVariationDependencies(nil, staticLibTag, a.properties.Static_libs.GetOrDefault(ctx, nil)...)
Colin Cross9055e212024-03-23 04:43:41 +00001242
1243 a.usesLibrary.deps(ctx, false)
Colin Crossfabb6082018-02-20 17:22:23 -08001244}
1245
Sam Delmerico82602492022-06-10 17:05:42 +00001246type JniPackageInfo struct {
1247 // List of zip files containing JNI libraries
1248 // Zip files should have directory structure jni/<arch>/*.so
1249 JniPackages android.Paths
1250}
1251
Colin Crossbc7d76c2023-12-12 16:39:03 -08001252var JniPackageProvider = blueprint.NewProvider[JniPackageInfo]()
Sam Delmerico82602492022-06-10 17:05:42 +00001253
1254// Unzip an AAR and extract the JNI libs for $archString.
1255var extractJNI = pctx.AndroidStaticRule("extractJNI",
1256 blueprint.RuleParams{
1257 Command: `rm -rf $out $outDir && touch $out && ` +
1258 `unzip -qoDD -d $outDir $in "jni/${archString}/*" && ` +
1259 `jni_files=$$(find $outDir/jni -type f) && ` +
1260 // print error message if there are no JNI libs for this arch
1261 `[ -n "$$jni_files" ] || (echo "ERROR: no JNI libs found for arch ${archString}" && exit 1) && ` +
Sam Delmerico80ee45c2023-06-22 15:36:02 -04001262 `${config.SoongZipCmd} -o $out -L 0 -P 'lib/${archString}' ` +
Sam Delmerico82602492022-06-10 17:05:42 +00001263 `-C $outDir/jni/${archString} $$(echo $$jni_files | xargs -n1 printf " -f %s")`,
1264 CommandDeps: []string{"${config.SoongZipCmd}"},
1265 },
1266 "outDir", "archString")
1267
Colin Crossfabb6082018-02-20 17:22:23 -08001268// 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 -07001269// 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 -08001270var unzipAAR = pctx.AndroidStaticRule("unzipAAR",
1271 blueprint.RuleParams{
Dan Willemsen304cfec2019-05-28 14:49:06 -07001272 Command: `rm -rf $outDir && mkdir -p $outDir && ` +
Colin Cross205e9112020-08-06 13:20:17 -07001273 `unzip -qoDD -d $outDir $in && rm -rf $outDir/res && touch $out && ` +
Michael Rosenfeld5ad15572021-12-03 13:25:10 -08001274 `${config.Zip2ZipCmd} -i $in -o $assetsPackage 'assets/**/*' && ` +
Colin Cross205e9112020-08-06 13:20:17 -07001275 `${config.MergeZipsCmd} $combinedClassesJar $$(ls $outDir/classes.jar 2> /dev/null) $$(ls $outDir/libs/*.jar 2> /dev/null)`,
Michael Rosenfeld5ad15572021-12-03 13:25:10 -08001276 CommandDeps: []string{"${config.MergeZipsCmd}", "${config.Zip2ZipCmd}"},
Colin Crossfabb6082018-02-20 17:22:23 -08001277 },
Michael Rosenfeld5ad15572021-12-03 13:25:10 -08001278 "outDir", "combinedClassesJar", "assetsPackage")
Colin Crossfabb6082018-02-20 17:22:23 -08001279
1280func (a *AARImport) GenerateAndroidBuildActions(ctx android.ModuleContext) {
1281 if len(a.properties.Aars) != 1 {
1282 ctx.PropertyErrorf("aars", "exactly one aar is required")
1283 return
1284 }
1285
Jiyong Park92315372021-04-02 08:45:46 +09001286 a.sdkVersion = a.SdkVersion(ctx)
1287 a.minSdkVersion = a.MinSdkVersion(ctx)
1288
Colin Crossff694a82023-12-13 15:54:49 -08001289 apexInfo, _ := android.ModuleProvider(ctx, android.ApexInfoProvider)
1290 a.hideApexVariantFromMake = !apexInfo.IsForPlatform()
Colin Cross56a83212020-09-15 18:30:11 -07001291
Nan Zhang4c819fb2018-08-27 18:31:46 -07001292 aarName := ctx.ModuleName() + ".aar"
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001293 a.aarPath = android.PathForModuleSrc(ctx, a.properties.Aars[0])
1294
Colin Cross1001a792019-03-21 22:21:39 -07001295 if Bool(a.properties.Jetifier) {
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001296 inputFile := a.aarPath
Colin Cross77965d92024-08-15 17:11:08 -07001297 jetifierPath := android.PathForModuleOut(ctx, "jetifier", aarName)
1298 TransformJetifier(ctx, jetifierPath, inputFile)
1299 a.aarPath = jetifierPath
Nan Zhang4c819fb2018-08-27 18:31:46 -07001300 }
Colin Crossfabb6082018-02-20 17:22:23 -08001301
Colin Cross607bbd62024-04-12 13:44:45 -07001302 jarName := ctx.ModuleName() + ".jar"
Colin Crossfabb6082018-02-20 17:22:23 -08001303 extractedAARDir := android.PathForModuleOut(ctx, "aar")
Colin Cross607bbd62024-04-12 13:44:45 -07001304 classpathFile := extractedAARDir.Join(ctx, jarName)
Colin Cross21ed4692024-04-24 20:23:38 +00001305
1306 extractedManifest := extractedAARDir.Join(ctx, "AndroidManifest.xml")
1307 providedManifest := android.OptionalPathForModuleSrc(ctx, a.properties.Manifest)
1308 if providedManifest.Valid() {
1309 a.manifest = providedManifest.Path()
1310 } else {
1311 a.manifest = extractedManifest
1312 }
1313
Colin Cross7707b242024-07-26 12:02:36 -07001314 rTxt := extractedAARDir.Join(ctx, "R.txt")
1315 assetsPackage := android.PathForModuleOut(ctx, "assets.zip")
1316 proguardFlags := extractedAARDir.Join(ctx, "proguard.txt")
Colin Crosscde55342024-03-27 14:11:51 -07001317 transitiveProguardFlags, transitiveUnconditionalExportedFlags := collectDepProguardSpecInfo(ctx)
Colin Cross40213022023-12-13 15:19:49 -08001318 android.SetProvider(ctx, ProguardSpecInfoProvider, ProguardSpecInfo{
Colin Crossa14fb6a2024-10-23 16:57:06 -07001319 ProguardFlagsFiles: depset.New[android.Path](
1320 depset.POSTORDER,
Colin Cross7707b242024-07-26 12:02:36 -07001321 android.Paths{proguardFlags},
Colin Crosscde55342024-03-27 14:11:51 -07001322 transitiveProguardFlags,
1323 ),
Colin Crossa14fb6a2024-10-23 16:57:06 -07001324 UnconditionallyExportedProguardFlags: depset.New[android.Path](
1325 depset.POSTORDER,
Sam Delmerico95d70942023-08-02 18:00:35 -04001326 nil,
Colin Crosscde55342024-03-27 14:11:51 -07001327 transitiveUnconditionalExportedFlags,
Sam Delmerico95d70942023-08-02 18:00:35 -04001328 ),
1329 })
Colin Crossfabb6082018-02-20 17:22:23 -08001330
1331 ctx.Build(pctx, android.BuildParams{
1332 Rule: unzipAAR,
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001333 Input: a.aarPath,
Colin Cross7707b242024-07-26 12:02:36 -07001334 Outputs: android.WritablePaths{classpathFile, proguardFlags, extractedManifest, assetsPackage, rTxt},
Colin Crossfabb6082018-02-20 17:22:23 -08001335 Description: "unzip AAR",
1336 Args: map[string]string{
Colin Cross205e9112020-08-06 13:20:17 -07001337 "outDir": extractedAARDir.String(),
Colin Cross9055e212024-03-23 04:43:41 +00001338 "combinedClassesJar": classpathFile.String(),
Colin Cross7707b242024-07-26 12:02:36 -07001339 "assetsPackage": assetsPackage.String(),
Colin Crossfabb6082018-02-20 17:22:23 -08001340 },
1341 })
1342
Colin Cross7707b242024-07-26 12:02:36 -07001343 a.proguardFlags = proguardFlags
1344 a.assetsPackage = assetsPackage
1345 a.rTxt = rTxt
1346
Colin Crossa0ba2f52019-06-22 12:59:27 -07001347 // Always set --pseudo-localize, it will be stripped out later for release
1348 // builds that don't want it.
1349 compileFlags := []string{"--pseudo-localize"}
Colin Crossfabb6082018-02-20 17:22:23 -08001350 compiledResDir := android.PathForModuleOut(ctx, "flat-res")
Colin Crossfabb6082018-02-20 17:22:23 -08001351 flata := compiledResDir.Join(ctx, "gen_res.flata")
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001352 aapt2CompileZip(ctx, flata, a.aarPath, "res", compileFlags)
Colin Crossfabb6082018-02-20 17:22:23 -08001353
Colin Cross7707b242024-07-26 12:02:36 -07001354 exportPackage := android.PathForModuleOut(ctx, "package-res.apk")
Colin Crossfabb6082018-02-20 17:22:23 -08001355 proguardOptionsFile := android.PathForModuleGen(ctx, "proguard.options")
Colin Crossbb77d8e2024-02-15 14:43:47 -08001356 aaptRTxt := android.PathForModuleOut(ctx, "R.txt")
Colin Cross7707b242024-07-26 12:02:36 -07001357 extraAaptPackagesFile := android.PathForModuleOut(ctx, "extra_packages")
Colin Crossfabb6082018-02-20 17:22:23 -08001358
1359 var linkDeps android.Paths
1360
1361 linkFlags := []string{
1362 "--static-lib",
Colin Cross4eae06d2023-06-20 22:40:02 -07001363 "--merge-only",
Colin Crossfabb6082018-02-20 17:22:23 -08001364 "--auto-add-overlay",
Colin Cross7c4dc5d2024-02-13 14:29:45 -08001365 "--no-static-lib-packages",
Colin Crossfabb6082018-02-20 17:22:23 -08001366 }
1367
Colin Cross10f7c4a2018-05-23 10:59:28 -07001368 linkFlags = append(linkFlags, "--manifest "+a.manifest.String())
1369 linkDeps = append(linkDeps, a.manifest)
Colin Crossfabb6082018-02-20 17:22:23 -08001370
Colin Cross8676c8c2023-10-12 15:58:57 -07001371 staticResourcesNodesDepSet, sharedResourcesNodesDepSet, staticRRODirsDepSet, staticManifestsDepSet, sharedLibs, libFlags :=
Jiakai Zhang36937082024-04-15 11:15:50 +00001372 aaptLibs(ctx, android.SdkContext(a), nil, nil)
Colin Cross31656952018-05-24 16:11:20 -07001373
Colin Cross8676c8c2023-10-12 15:58:57 -07001374 _ = sharedResourcesNodesDepSet
Colin Crossab8d1382023-07-14 17:23:41 +00001375 _ = staticRRODirsDepSet
Colin Cross8676c8c2023-10-12 15:58:57 -07001376
Colin Crossab8d1382023-07-14 17:23:41 +00001377 staticDeps := transitiveAarDeps(staticResourcesNodesDepSet.ToList())
Colin Crossfabb6082018-02-20 17:22:23 -08001378
Colin Crossab8d1382023-07-14 17:23:41 +00001379 linkDeps = append(linkDeps, sharedLibs...)
Colin Cross4eae06d2023-06-20 22:40:02 -07001380 linkDeps = append(linkDeps, staticDeps.resPackages()...)
Colin Crossa97c5d32018-03-28 14:58:31 -07001381 linkFlags = append(linkFlags, libFlags...)
Colin Crossfabb6082018-02-20 17:22:23 -08001382
Colin Cross4eae06d2023-06-20 22:40:02 -07001383 overlayRes := android.Paths{flata}
1384
1385 // Treat static library dependencies of static libraries as imports.
1386 transitiveStaticLibs := staticDeps.resPackages()
1387 linkDeps = append(linkDeps, transitiveStaticLibs...)
1388 for _, staticLib := range transitiveStaticLibs {
1389 linkFlags = append(linkFlags, "-I "+staticLib.String())
1390 }
Colin Crossfabb6082018-02-20 17:22:23 -08001391
Colin Crossab8d1382023-07-14 17:23:41 +00001392 transitiveAssets := android.ReverseSliceInPlace(staticDeps.assets())
Colin Cross7707b242024-07-26 12:02:36 -07001393 aapt2Link(ctx, exportPackage, nil, proguardOptionsFile, aaptRTxt,
Jihoon Kang84b25892023-12-01 22:01:06 +00001394 linkFlags, linkDeps, nil, overlayRes, transitiveAssets, nil, nil)
Colin Crossa6182ab2024-08-21 10:47:44 -07001395 ctx.CheckbuildFile(exportPackage)
Colin Cross7707b242024-07-26 12:02:36 -07001396 a.exportPackage = exportPackage
Colin Crossfabb6082018-02-20 17:22:23 -08001397
Colin Cross7707b242024-07-26 12:02:36 -07001398 rJar := android.PathForModuleOut(ctx, "busybox/R.jar")
1399 resourceProcessorBusyBoxGenerateBinaryR(ctx, a.rTxt, a.manifest, rJar, nil, true, nil, false)
Colin Crossa6182ab2024-08-21 10:47:44 -07001400 ctx.CheckbuildFile(rJar)
Colin Cross7707b242024-07-26 12:02:36 -07001401 a.rJar = rJar
Colin Cross4eae06d2023-06-20 22:40:02 -07001402
Colin Cross7707b242024-07-26 12:02:36 -07001403 aapt2ExtractExtraPackages(ctx, extraAaptPackagesFile, a.rJar)
1404 a.extraAaptPackagesFile = extraAaptPackagesFile
Colin Crossf3b7bad2023-08-02 15:49:00 -07001405
Colin Crossa14fb6a2024-10-23 16:57:06 -07001406 resourcesNodesDepSetBuilder := depset.NewBuilder[*resourcesNode](depset.TOPOLOGICAL)
Colin Crossab8d1382023-07-14 17:23:41 +00001407 resourcesNodesDepSetBuilder.Direct(&resourcesNode{
1408 resPackage: a.exportPackage,
1409 manifest: a.manifest,
Colin Cross4eae06d2023-06-20 22:40:02 -07001410 rTxt: a.rTxt,
1411 rJar: a.rJar,
Colin Crossab8d1382023-07-14 17:23:41 +00001412 assets: android.OptionalPathForPath(a.assetsPackage),
Colin Cross4eae06d2023-06-20 22:40:02 -07001413
1414 usedResourceProcessor: true,
Colin Crossab8d1382023-07-14 17:23:41 +00001415 })
1416 resourcesNodesDepSetBuilder.Transitive(staticResourcesNodesDepSet)
1417 a.resourcesNodesDepSet = resourcesNodesDepSetBuilder.Build()
1418
Colin Crossa14fb6a2024-10-23 16:57:06 -07001419 manifestDepSetBuilder := depset.NewBuilder[android.Path](depset.TOPOLOGICAL).Direct(a.manifest)
Colin Cross9055e212024-03-23 04:43:41 +00001420 manifestDepSetBuilder.Transitive(staticManifestsDepSet)
Colin Crossab8d1382023-07-14 17:23:41 +00001421 a.manifestsDepSet = manifestDepSetBuilder.Build()
Michael Rosenfeld5ad15572021-12-03 13:25:10 -08001422
Colin Cross312634e2023-11-21 15:13:56 -08001423 transitiveAaptResourcePackages := staticDeps.resPackages().Strings()
1424 transitiveAaptResourcePackages = slices.DeleteFunc(transitiveAaptResourcePackages, func(p string) bool {
1425 return p == a.exportPackage.String()
1426 })
1427 transitiveAaptResourcePackagesFile := android.PathForModuleOut(ctx, "transitive-res-packages")
1428 android.WriteFileRule(ctx, transitiveAaptResourcePackagesFile, strings.Join(transitiveAaptResourcePackages, "\n"))
1429 a.transitiveAaptResourcePackagesFile = transitiveAaptResourcePackagesFile
Colin Cross4eae06d2023-06-20 22:40:02 -07001430
Colin Cross9ffaf282024-08-12 13:50:09 -07001431 a.collectTransitiveHeaderJarsForR8(ctx)
Colin Cross9055e212024-03-23 04:43:41 +00001432
1433 a.classLoaderContexts = a.usesLibrary.classLoaderContextForUsesLibDeps(ctx)
1434
1435 var staticJars android.Paths
1436 var staticHeaderJars android.Paths
Colin Cross607bbd62024-04-12 13:44:45 -07001437 var staticResourceJars android.Paths
Colin Crossa14fb6a2024-10-23 16:57:06 -07001438 var transitiveStaticLibsHeaderJars []depset.DepSet[android.Path]
1439 var transitiveStaticLibsImplementationJars []depset.DepSet[android.Path]
1440 var transitiveStaticLibsResourceJars []depset.DepSet[android.Path]
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001441
Colin Cross9055e212024-03-23 04:43:41 +00001442 ctx.VisitDirectDeps(func(module android.Module) {
1443 if dep, ok := android.OtherModuleProvider(ctx, module, JavaInfoProvider); ok {
1444 tag := ctx.OtherModuleDependencyTag(module)
1445 switch tag {
1446 case staticLibTag:
1447 staticJars = append(staticJars, dep.ImplementationJars...)
1448 staticHeaderJars = append(staticHeaderJars, dep.HeaderJars...)
Colin Cross607bbd62024-04-12 13:44:45 -07001449 staticResourceJars = append(staticResourceJars, dep.ResourceJars...)
Colin Crossa14fb6a2024-10-23 16:57:06 -07001450 transitiveStaticLibsHeaderJars = append(transitiveStaticLibsHeaderJars, dep.TransitiveStaticLibsHeaderJars)
1451 transitiveStaticLibsImplementationJars = append(transitiveStaticLibsImplementationJars, dep.TransitiveStaticLibsImplementationJars)
1452 transitiveStaticLibsResourceJars = append(transitiveStaticLibsResourceJars, dep.TransitiveStaticLibsResourceJars)
Colin Cross9055e212024-03-23 04:43:41 +00001453 }
1454 }
1455 addCLCFromDep(ctx, module, a.classLoaderContexts)
Jiakai Zhang36937082024-04-15 11:15:50 +00001456 addMissingOptionalUsesLibsFromDep(ctx, module, &a.usesLibrary)
Colin Cross9055e212024-03-23 04:43:41 +00001457 })
1458
Colin Crossa14fb6a2024-10-23 16:57:06 -07001459 completeStaticLibsHeaderJars := depset.New(depset.PREORDER, android.Paths{classpathFile}, transitiveStaticLibsHeaderJars)
1460 completeStaticLibsImplementationJars := depset.New(depset.PREORDER, android.Paths{classpathFile}, transitiveStaticLibsImplementationJars)
1461 completeStaticLibsResourceJars := depset.New(depset.PREORDER, nil, transitiveStaticLibsResourceJars)
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001462
Colin Cross77965d92024-08-15 17:11:08 -07001463 var implementationJarFile android.Path
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001464 var combineJars android.Paths
1465 if ctx.Config().UseTransitiveJarsInClasspath() {
1466 combineJars = completeStaticLibsImplementationJars.ToList()
1467 } else {
1468 combineJars = append(android.Paths{classpathFile}, staticJars...)
1469 }
1470
1471 if len(combineJars) > 1 {
1472 implementationJarOutputPath := android.PathForModuleOut(ctx, "combined", jarName)
1473 TransformJarsToJar(ctx, implementationJarOutputPath, "combine", combineJars, android.OptionalPath{}, false, nil, nil)
1474 implementationJarFile = implementationJarOutputPath
Colin Cross9055e212024-03-23 04:43:41 +00001475 } else {
Colin Cross28ac2ff2024-04-02 12:21:34 -07001476 implementationJarFile = classpathFile
Colin Cross9055e212024-03-23 04:43:41 +00001477 }
1478
Colin Cross607bbd62024-04-12 13:44:45 -07001479 var resourceJarFile android.Path
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001480 var resourceJars android.Paths
1481 if ctx.Config().UseTransitiveJarsInClasspath() {
1482 resourceJars = completeStaticLibsResourceJars.ToList()
1483 } else {
1484 resourceJars = staticResourceJars
1485 }
1486 if len(resourceJars) > 1 {
Colin Cross607bbd62024-04-12 13:44:45 -07001487 combinedJar := android.PathForModuleOut(ctx, "res-combined", jarName)
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001488 TransformJarsToJar(ctx, combinedJar, "for resources", resourceJars, android.OptionalPath{},
Colin Cross607bbd62024-04-12 13:44:45 -07001489 false, nil, nil)
1490 resourceJarFile = combinedJar
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001491 } else if len(resourceJars) == 1 {
1492 resourceJarFile = resourceJars[0]
Colin Cross607bbd62024-04-12 13:44:45 -07001493 }
1494
1495 // merge implementation jar with resources if necessary
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001496 var implementationAndResourcesJars android.Paths
1497 if ctx.Config().UseTransitiveJarsInClasspath() {
1498 implementationAndResourcesJars = append(slices.Clone(resourceJars), combineJars...)
1499 } else {
1500 implementationAndResourcesJars = android.PathsIfNonNil(resourceJarFile, implementationJarFile)
1501 }
1502 var implementationAndResourcesJar android.Path
1503 if len(implementationAndResourcesJars) > 1 {
Colin Cross77965d92024-08-15 17:11:08 -07001504 combinedJar := android.PathForModuleOut(ctx, "withres", jarName)
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001505 TransformJarsToJar(ctx, combinedJar, "for resources", implementationAndResourcesJars, android.OptionalPath{},
Colin Cross607bbd62024-04-12 13:44:45 -07001506 false, nil, nil)
1507 implementationAndResourcesJar = combinedJar
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001508 } else {
1509 implementationAndResourcesJar = implementationAndResourcesJars[0]
Colin Cross607bbd62024-04-12 13:44:45 -07001510 }
1511
1512 a.implementationJarFile = implementationJarFile
Colin Cross28ac2ff2024-04-02 12:21:34 -07001513 // 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 -07001514 a.implementationAndResourcesJarFile = implementationAndResourcesJar.WithoutRel()
Colin Cross28ac2ff2024-04-02 12:21:34 -07001515
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001516 var headerJars android.Paths
1517 if ctx.Config().UseTransitiveJarsInClasspath() {
1518 headerJars = completeStaticLibsHeaderJars.ToList()
1519 } else {
1520 headerJars = append(android.Paths{classpathFile}, staticHeaderJars...)
1521 }
1522 if len(headerJars) > 1 {
Colin Cross7707b242024-07-26 12:02:36 -07001523 headerJarFile := android.PathForModuleOut(ctx, "turbine-combined", jarName)
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001524 TransformJarsToJar(ctx, headerJarFile, "combine header jars", headerJars, android.OptionalPath{}, false, nil, nil)
Colin Cross7707b242024-07-26 12:02:36 -07001525 a.headerJarFile = headerJarFile
Colin Cross9055e212024-03-23 04:43:41 +00001526 } else {
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001527 a.headerJarFile = headerJars[0]
Colin Cross9055e212024-03-23 04:43:41 +00001528 }
1529
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001530 if ctx.Config().UseTransitiveJarsInClasspath() {
1531 ctx.CheckbuildFile(classpathFile)
1532 } else {
1533 ctx.CheckbuildFile(a.headerJarFile)
1534 ctx.CheckbuildFile(a.implementationJarFile)
1535 }
Colin Crossa6182ab2024-08-21 10:47:44 -07001536
Colin Cross7727c7f2024-07-18 15:36:32 -07001537 android.SetProvider(ctx, JavaInfoProvider, &JavaInfo{
Colin Crossc9b4f6b2024-07-26 15:25:46 -07001538 HeaderJars: android.PathsIfNonNil(a.headerJarFile),
1539 LocalHeaderJars: android.PathsIfNonNil(classpathFile),
1540 TransitiveStaticLibsHeaderJars: completeStaticLibsHeaderJars,
1541 TransitiveStaticLibsImplementationJars: completeStaticLibsImplementationJars,
1542 TransitiveStaticLibsResourceJars: completeStaticLibsResourceJars,
1543 ResourceJars: android.PathsIfNonNil(resourceJarFile),
1544 TransitiveLibsHeaderJarsForR8: a.transitiveLibsHeaderJarsForR8,
1545 TransitiveStaticLibsHeaderJarsForR8: a.transitiveStaticLibsHeaderJarsForR8,
1546 ImplementationAndResourcesJars: android.PathsIfNonNil(a.implementationAndResourcesJarFile),
1547 ImplementationJars: android.PathsIfNonNil(a.implementationJarFile),
1548 StubsLinkType: Implementation,
Joe Onorato6fe59eb2023-07-16 13:20:33 -07001549 // TransitiveAconfigFiles: // TODO(b/289117800): LOCAL_ACONFIG_FILES for prebuilts
Colin Crossdcf71b22021-02-01 13:59:03 -08001550 })
Sam Delmerico82602492022-06-10 17:05:42 +00001551
1552 if proptools.Bool(a.properties.Extract_jni) {
1553 for _, t := range ctx.MultiTargets() {
1554 arch := t.Arch.Abi[0]
1555 path := android.PathForModuleOut(ctx, arch+"_jni.zip")
1556 a.jniPackages = append(a.jniPackages, path)
1557
1558 outDir := android.PathForModuleOut(ctx, "aarForJni")
1559 aarPath := android.PathForModuleSrc(ctx, a.properties.Aars[0])
1560 ctx.Build(pctx, android.BuildParams{
1561 Rule: extractJNI,
1562 Input: aarPath,
1563 Outputs: android.WritablePaths{path},
1564 Description: "extract JNI from AAR",
1565 Args: map[string]string{
1566 "outDir": outDir.String(),
1567 "archString": arch,
1568 },
1569 })
1570 }
Sam Delmerico82602492022-06-10 17:05:42 +00001571 }
Colin Crosse8eeec92023-12-14 14:50:05 -08001572
Colin Cross40213022023-12-13 15:19:49 -08001573 android.SetProvider(ctx, JniPackageProvider, JniPackageInfo{
Colin Crosse8eeec92023-12-14 14:50:05 -08001574 JniPackages: a.jniPackages,
1575 })
mrziwang68786d82024-07-09 10:41:55 -07001576
1577 ctx.SetOutputFiles([]android.Path{a.implementationAndResourcesJarFile}, "")
1578 ctx.SetOutputFiles([]android.Path{a.aarPath}, ".aar")
Colin Crossdcf71b22021-02-01 13:59:03 -08001579}
Colin Crossfabb6082018-02-20 17:22:23 -08001580
1581func (a *AARImport) HeaderJars() android.Paths {
Colin Cross9055e212024-03-23 04:43:41 +00001582 return android.Paths{a.headerJarFile}
Colin Crossfabb6082018-02-20 17:22:23 -08001583}
1584
Colin Cross331a1212018-08-15 20:40:52 -07001585func (a *AARImport) ImplementationAndResourcesJars() android.Paths {
Colin Cross607bbd62024-04-12 13:44:45 -07001586 return android.Paths{a.implementationAndResourcesJarFile}
Colin Cross331a1212018-08-15 20:40:52 -07001587}
1588
Colin Cross9055e212024-03-23 04:43:41 +00001589func (a *AARImport) DexJarBuildPath(ctx android.ModuleErrorfContext) OptionalDexJarPath {
1590 return OptionalDexJarPath{}
Colin Crossf24a22a2019-01-31 14:12:44 -08001591}
1592
Ulya Trafimovich9f3052c2020-06-09 14:31:19 +01001593func (a *AARImport) DexJarInstallPath() android.Path {
1594 return nil
1595}
1596
Ulya Trafimovichb23d28c2020-10-08 12:53:58 +01001597func (a *AARImport) ClassLoaderContexts() dexpreopt.ClassLoaderContextMap {
Colin Cross9055e212024-03-23 04:43:41 +00001598 return a.classLoaderContexts
Jiyong Park1be96912018-05-28 18:02:19 +09001599}
1600
Colin Cross9055e212024-03-23 04:43:41 +00001601var _ UsesLibraryDependency = (*AARImport)(nil)
1602
Jiyong Park45bf82e2020-12-15 22:29:02 +09001603var _ android.ApexModule = (*AARImport)(nil)
1604
1605// Implements android.ApexModule
Jooyung Hanacc7bbe2020-05-20 09:06:00 +09001606func (a *AARImport) DepIsInSameApex(ctx android.BaseModuleContext, dep android.Module) bool {
1607 return a.depIsInSameApex(ctx, dep)
1608}
1609
Jiyong Park45bf82e2020-12-15 22:29:02 +09001610// Implements android.ApexModule
Colin Cross9055e212024-03-23 04:43:41 +00001611func (a *AARImport) ShouldSupportSdkVersion(ctx android.BaseModuleContext,
Dan Albertc8060532020-07-22 22:32:17 -07001612 sdkVersion android.ApiLevel) error {
Jooyung Han749dc692020-04-15 11:03:39 +09001613 return nil
1614}
1615
Sam Delmericoaf8bb702022-07-25 15:39:32 -04001616var _ android.PrebuiltInterface = (*AARImport)(nil)
Colin Crossfabb6082018-02-20 17:22:23 -08001617
Jiakai Zhangf98da192024-04-15 11:15:41 +00001618func (a *AARImport) UsesLibrary() *usesLibrary {
1619 return &a.usesLibrary
1620}
1621
1622var _ ModuleWithUsesLibrary = (*AARImport)(nil)
1623
Colin Cross1b16b0e2019-02-12 14:41:32 -08001624// android_library_import imports an `.aar` file into the build graph as if it was built with android_library.
1625//
1626// This module is not suitable for installing on a device, but can be used as a `static_libs` dependency of
1627// an android_app module.
Colin Crossfabb6082018-02-20 17:22:23 -08001628func AARImportFactory() android.Module {
1629 module := &AARImport{}
1630
Colin Cross9055e212024-03-23 04:43:41 +00001631 module.AddProperties(
1632 &module.properties,
1633 &module.usesLibrary.usesLibraryProperties,
1634 )
Colin Crossfabb6082018-02-20 17:22:23 -08001635
1636 android.InitPrebuiltModule(module, &module.properties.Aars)
Jooyung Hanacc7bbe2020-05-20 09:06:00 +09001637 android.InitApexModule(module)
Sam Delmerico82602492022-06-10 17:05:42 +00001638 InitJavaModuleMultiTargets(module, android.DeviceSupported)
Colin Crossfabb6082018-02-20 17:22:23 -08001639 return module
1640}
Spandan Dasb08bf772024-08-07 21:55:42 +00001641
Cole Faustb36d31d2024-08-27 16:04:28 -07001642func (a *AARImport) IDEInfo(ctx android.BaseModuleContext, dpInfo *android.IdeInfo) {
Spandan Dasb08bf772024-08-07 21:55:42 +00001643 dpInfo.Jars = append(dpInfo.Jars, a.headerJarFile.String(), a.rJar.String())
1644}