blob: 752d6839fd235291fa3c9898d7031999f860b141 [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 (
Colin Crossa592e3e2019-02-19 16:59:53 -080018 "fmt"
Jaewoong Jung5b425e22019-06-17 17:40:56 -070019 "path/filepath"
Colin Cross312634e2023-11-21 15:13:56 -080020 "slices"
Colin Crossc20dc852020-11-10 12:27:45 -080021 "strconv"
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 Crossa97c5d32018-03-28 14:58:31 -070028 "github.com/google/blueprint/proptools"
Colin Crossfabb6082018-02-20 17:22:23 -080029)
30
Colin Crossa97c5d32018-03-28 14:58:31 -070031type AndroidLibraryDependency interface {
Colin Crossa97c5d32018-03-28 14:58:31 -070032 ExportPackage() android.Path
Colin Crossab8d1382023-07-14 17:23:41 +000033 ResourcesNodeDepSet() *android.DepSet[*resourcesNode]
34 RRODirsDepSet() *android.DepSet[rroDir]
35 ManifestsDepSet() *android.DepSet[android.Path]
Jaewoong Jungc779cd42020-10-06 18:56:10 -070036 SetRROEnforcedForDependent(enforce bool)
37 IsRROEnforced(ctx android.BaseModuleContext) bool
Colin Crossa97c5d32018-03-28 14:58:31 -070038}
39
40func init() {
Paul Duffinf9b1da02019-12-18 19:51:55 +000041 RegisterAARBuildComponents(android.InitRegistrationContext)
42}
43
44func RegisterAARBuildComponents(ctx android.RegistrationContext) {
45 ctx.RegisterModuleType("android_library_import", AARImportFactory)
46 ctx.RegisterModuleType("android_library", AndroidLibraryFactory)
Paul Duffin04ba70d2021-03-22 13:56:43 +000047 ctx.PostDepsMutators(func(ctx android.RegisterMutatorsContext) {
Colin Cross7e6a9012024-01-17 14:58:38 -080048 ctx.TopDown("propagate_rro_enforcement", propagateRROEnforcementMutator)
Paul Duffin04ba70d2021-03-22 13:56:43 +000049 })
Colin Crossa97c5d32018-03-28 14:58:31 -070050}
51
52//
53// AAR (android library)
54//
55
56type androidLibraryProperties struct {
57 BuildAAR bool `blueprint:"mutated"`
58}
59
60type aaptProperties struct {
61 // flags passed to aapt when creating the apk
62 Aaptflags []string
63
Dan Willemsen72be5902018-10-24 20:24:57 -070064 // include all resource configurations, not just the product-configured
65 // ones.
66 Aapt_include_all_resources *bool
67
Jiakai Zhangba82e282023-10-13 18:08:59 +010068 // list of files to use as assets.
69 Assets []string `android:"path"`
70
Colin Crossa97c5d32018-03-28 14:58:31 -070071 // list of directories relative to the Blueprints file containing assets.
Colin Cross0ddae7f2019-02-07 15:30:01 -080072 // Defaults to ["assets"] if a directory called assets exists. Set to []
73 // to disable the default.
Colin Crossa97c5d32018-03-28 14:58:31 -070074 Asset_dirs []string
75
76 // list of directories relative to the Blueprints file containing
Colin Cross0ddae7f2019-02-07 15:30:01 -080077 // Android resources. Defaults to ["res"] if a directory called res exists.
78 // Set to [] to disable the default.
Colin Crossa97c5d32018-03-28 14:58:31 -070079 Resource_dirs []string
80
Colin Crossa592e3e2019-02-19 16:59:53 -080081 // list of zip files containing Android resources.
Colin Cross27b922f2019-03-04 22:35:41 -080082 Resource_zips []string `android:"path"`
Colin Crossa592e3e2019-02-19 16:59:53 -080083
Colin Crossa97c5d32018-03-28 14:58:31 -070084 // path to AndroidManifest.xml. If unset, defaults to "AndroidManifest.xml".
Colin Cross27b922f2019-03-04 22:35:41 -080085 Manifest *string `android:"path"`
changho.shinb5432b72019-08-08 18:37:17 +090086
87 // paths to additional manifest files to merge with main manifest.
88 Additional_manifests []string `android:"path"`
Sasha Smundak541056c2019-10-28 15:50:06 -070089
90 // do not include AndroidManifest from dependent libraries
91 Dont_merge_manifests *bool
Jaewoong Jungc779cd42020-10-06 18:56:10 -070092
Colin Cross4eae06d2023-06-20 22:40:02 -070093 // If use_resource_processor is set, use Bazel's resource processor instead of aapt2 to generate R.class files.
94 // The resource processor produces more optimal R.class files that only list resources in the package of the
95 // library that provided them, as opposed to aapt2 which produces R.java files for every package containing
96 // every resource. Using the resource processor can provide significant build time speedups, but requires
97 // fixing the module to use the correct package to reference each resource, and to avoid having any other
98 // libraries in the tree that use the same package name. Defaults to false, but will default to true in the
99 // future.
100 Use_resource_processor *bool
101
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700102 // true if RRO is enforced for any of the dependent modules
103 RROEnforcedForDependent bool `blueprint:"mutated"`
Inseob Kim34dc4cd2023-11-07 13:37:14 +0900104
105 // Filter only specified product and ignore other products
106 Filter_product *string `blueprint:"mutated"`
Jihoon Kang9049c272024-03-19 21:57:36 +0000107
108 // Names of aconfig_declarations modules that specify aconfig flags that the module depends on.
109 Flags_packages []string
Colin Crossa97c5d32018-03-28 14:58:31 -0700110}
111
112type aapt struct {
Colin Cross312634e2023-11-21 15:13:56 -0800113 aaptSrcJar android.Path
114 transitiveAaptRJars android.Paths
115 transitiveAaptResourcePackagesFile android.Path
116 exportPackage android.Path
117 manifestPath android.Path
118 proguardOptionsFile android.Path
119 rTxt android.Path
120 rJar android.Path
121 extraAaptPackagesFile android.Path
122 mergedManifestFile android.Path
123 noticeFile android.OptionalPath
124 assetPackage android.OptionalPath
125 isLibrary bool
126 defaultManifestVersion string
127 useEmbeddedNativeLibs bool
128 useEmbeddedDex bool
129 usesNonSdkApis bool
130 hasNoCode bool
131 LoggingParent string
132 resourceFiles android.Paths
Colin Crossa97c5d32018-03-28 14:58:31 -0700133
Colin Crosse560c4a2019-03-19 16:03:11 -0700134 splitNames []string
135 splits []split
136
Colin Crossa97c5d32018-03-28 14:58:31 -0700137 aaptProperties aaptProperties
Colin Crossab8d1382023-07-14 17:23:41 +0000138
139 resourcesNodesDepSet *android.DepSet[*resourcesNode]
140 rroDirsDepSet *android.DepSet[rroDir]
141 manifestsDepSet *android.DepSet[android.Path]
Alix96ea88452023-08-31 15:48:23 +0000142
143 manifestValues struct {
144 applicationId string
145 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700146}
147
Colin Crosse560c4a2019-03-19 16:03:11 -0700148type split struct {
149 name string
150 suffix string
151 path android.Path
152}
153
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700154// Propagate RRO enforcement flag to static lib dependencies transitively.
155func propagateRROEnforcementMutator(ctx android.TopDownMutatorContext) {
156 m := ctx.Module()
157 if d, ok := m.(AndroidLibraryDependency); ok && d.IsRROEnforced(ctx) {
158 ctx.VisitDirectDepsWithTag(staticLibTag, func(d android.Module) {
159 if a, ok := d.(AndroidLibraryDependency); ok {
160 a.SetRROEnforcedForDependent(true)
161 }
162 })
163 }
164}
165
Colin Cross8f1b0332024-01-25 13:39:06 -0800166func (a *aapt) useResourceProcessorBusyBox(ctx android.BaseModuleContext) bool {
Colin Crosseee4ab12024-03-27 11:54:10 -0700167 return BoolDefault(a.aaptProperties.Use_resource_processor, ctx.Config().UseResourceProcessorByDefault()) &&
168 // TODO(b/331641946): remove this when ResourceProcessorBusyBox supports generating shared libraries.
169 !slices.Contains(a.aaptProperties.Aaptflags, "--shared-lib")
Colin Cross4eae06d2023-06-20 22:40:02 -0700170}
171
Inseob Kim34dc4cd2023-11-07 13:37:14 +0900172func (a *aapt) filterProduct() string {
173 return String(a.aaptProperties.Filter_product)
174}
175
Colin Crossa97c5d32018-03-28 14:58:31 -0700176func (a *aapt) ExportPackage() android.Path {
177 return a.exportPackage
178}
Colin Crossab8d1382023-07-14 17:23:41 +0000179func (a *aapt) ResourcesNodeDepSet() *android.DepSet[*resourcesNode] {
180 return a.resourcesNodesDepSet
Colin Crossc1c37552019-01-31 11:42:41 -0800181}
182
Colin Crossab8d1382023-07-14 17:23:41 +0000183func (a *aapt) RRODirsDepSet() *android.DepSet[rroDir] {
184 return a.rroDirsDepSet
Colin Crossc1c37552019-01-31 11:42:41 -0800185}
186
Colin Crossab8d1382023-07-14 17:23:41 +0000187func (a *aapt) ManifestsDepSet() *android.DepSet[android.Path] {
188 return a.manifestsDepSet
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800189}
190
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700191func (a *aapt) SetRROEnforcedForDependent(enforce bool) {
192 a.aaptProperties.RROEnforcedForDependent = enforce
193}
194
195func (a *aapt) IsRROEnforced(ctx android.BaseModuleContext) bool {
196 // True if RRO is enforced for this module or...
197 return ctx.Config().EnforceRROForModule(ctx.ModuleName()) ||
Jeongik Chacee5ba92021-02-19 12:11:51 +0900198 // if RRO is enforced for any of its dependents.
199 a.aaptProperties.RROEnforcedForDependent
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700200}
201
Jiyong Parkf1691d22021-03-29 20:11:58 +0900202func (a *aapt) aapt2Flags(ctx android.ModuleContext, sdkContext android.SdkContext,
Colin Crossa0ba2f52019-06-22 12:59:27 -0700203 manifestPath android.Path) (compileFlags, linkFlags []string, linkDeps android.Paths,
204 resDirs, overlayDirs []globbedResourceDir, rroDirs []rroDir, resZips android.Paths) {
Colin Crossa97c5d32018-03-28 14:58:31 -0700205
Jaewoong Jung3aff5782020-02-11 07:54:35 -0800206 hasVersionCode := android.PrefixInList(a.aaptProperties.Aaptflags, "--version-code")
207 hasVersionName := android.PrefixInList(a.aaptProperties.Aaptflags, "--version-name")
Colin Crossa97c5d32018-03-28 14:58:31 -0700208
Colin Crossa97c5d32018-03-28 14:58:31 -0700209 // Flags specified in Android.bp
210 linkFlags = append(linkFlags, a.aaptProperties.Aaptflags...)
211
Eric Miao40eab202023-03-30 16:57:17 +0000212 linkFlags = append(linkFlags, "--enable-compact-entries")
Colin Crossa97c5d32018-03-28 14:58:31 -0700213
214 // Find implicit or explicit asset and resource dirs
Jiakai Zhangba82e282023-10-13 18:08:59 +0100215 assets := android.PathsRelativeToModuleSourceDir(android.SourceInput{
216 Context: ctx,
217 Paths: a.aaptProperties.Assets,
218 IncludeDirs: false,
219 })
Colin Crossa97c5d32018-03-28 14:58:31 -0700220 assetDirs := android.PathsWithOptionalDefaultForModuleSrc(ctx, a.aaptProperties.Asset_dirs, "assets")
221 resourceDirs := android.PathsWithOptionalDefaultForModuleSrc(ctx, a.aaptProperties.Resource_dirs, "res")
Colin Cross8a497952019-03-05 22:25:09 -0800222 resourceZips := android.PathsForModuleSrc(ctx, a.aaptProperties.Resource_zips)
Colin Crossa97c5d32018-03-28 14:58:31 -0700223
Colin Crossa97c5d32018-03-28 14:58:31 -0700224 // Glob directories into lists of paths
225 for _, dir := range resourceDirs {
226 resDirs = append(resDirs, globbedResourceDir{
227 dir: dir,
228 files: androidResourceGlob(ctx, dir),
229 })
Jaewoong Jungc779cd42020-10-06 18:56:10 -0700230 resOverlayDirs, resRRODirs := overlayResourceGlob(ctx, a, dir)
Colin Crossa97c5d32018-03-28 14:58:31 -0700231 overlayDirs = append(overlayDirs, resOverlayDirs...)
232 rroDirs = append(rroDirs, resRRODirs...)
233 }
234
Colin Crossc20dc852020-11-10 12:27:45 -0800235 var assetDeps android.Paths
236 for i, dir := range assetDirs {
237 // Add a dependency on every file in the asset directory. This ensures the aapt2
238 // rule will be rerun if one of the files in the asset directory is modified.
239 assetDeps = append(assetDeps, androidResourceGlob(ctx, dir)...)
240
241 // Add a dependency on a file that contains a list of all the files in the asset directory.
242 // This ensures the aapt2 rule will be run if a file is removed from the asset directory,
243 // or a file is added whose timestamp is older than the output of aapt2.
244 assetFileListFile := android.PathForModuleOut(ctx, "asset_dir_globs", strconv.Itoa(i)+".glob")
245 androidResourceGlobList(ctx, dir, assetFileListFile)
246 assetDeps = append(assetDeps, assetFileListFile)
Colin Crossa97c5d32018-03-28 14:58:31 -0700247 }
248
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700249 assetDirStrings := assetDirs.Strings()
250 if a.noticeFile.Valid() {
251 assetDirStrings = append(assetDirStrings, filepath.Dir(a.noticeFile.Path().String()))
Colin Crossc20dc852020-11-10 12:27:45 -0800252 assetDeps = append(assetDeps, a.noticeFile.Path())
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700253 }
Jiakai Zhangba82e282023-10-13 18:08:59 +0100254 if len(assets) > 0 {
255 // aapt2 doesn't support adding individual asset files. Create a temp directory to hold asset
256 // files and pass it to aapt2.
257 tmpAssetDir := android.PathForModuleOut(ctx, "tmp_asset_dir")
258
259 rule := android.NewRuleBuilder(pctx, ctx)
260 rule.Command().
261 Text("rm -rf").Text(tmpAssetDir.String()).
262 Text("&&").
263 Text("mkdir -p").Text(tmpAssetDir.String())
264
265 for _, asset := range assets {
266 output := tmpAssetDir.Join(ctx, asset.Rel())
267 assetDeps = append(assetDeps, output)
268 rule.Command().Text("mkdir -p").Text(filepath.Dir(output.String()))
269 rule.Command().Text("cp").Input(asset).Output(output)
270 }
271
272 rule.Build("tmp_asset_dir", "tmp_asset_dir")
273
274 assetDirStrings = append(assetDirStrings, tmpAssetDir.String())
275 }
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700276
Colin Crossa97c5d32018-03-28 14:58:31 -0700277 linkFlags = append(linkFlags, "--manifest "+manifestPath.String())
278 linkDeps = append(linkDeps, manifestPath)
279
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700280 linkFlags = append(linkFlags, android.JoinWithPrefix(assetDirStrings, "-A "))
Colin Crossc20dc852020-11-10 12:27:45 -0800281 linkDeps = append(linkDeps, assetDeps...)
Colin Crossa97c5d32018-03-28 14:58:31 -0700282
Spandan Das50885c02023-02-23 21:31:33 +0000283 // Returns the effective version for {min|target}_sdk_version
Spandan Das8c9ae7e2023-03-03 21:20:36 +0000284 effectiveVersionString := func(sdkVersion android.SdkSpec, minSdkVersion android.ApiLevel) string {
Spandan Das50885c02023-02-23 21:31:33 +0000285 // If {min|target}_sdk_version is current, use sdk_version to determine the effective level
286 // This is necessary for vendor modules.
287 // The effective version does not _only_ depend on {min|target}_sdk_version(level),
288 // but also on the sdk_version (kind+level)
Spandan Das8c9ae7e2023-03-03 21:20:36 +0000289 if minSdkVersion.IsCurrent() {
Spandan Das50885c02023-02-23 21:31:33 +0000290 ret, err := sdkVersion.EffectiveVersionString(ctx)
291 if err != nil {
292 ctx.ModuleErrorf("invalid sdk_version: %s", err)
293 }
294 return ret
295 }
296 ret, err := minSdkVersion.EffectiveVersionString(ctx)
297 if err != nil {
298 ctx.ModuleErrorf("invalid min_sdk_version: %s", err)
299 }
300 return ret
Jiyong Park6a927c42020-01-21 02:03:43 +0900301 }
Spandan Das50885c02023-02-23 21:31:33 +0000302 // SDK version flags
303 sdkVersion := sdkContext.SdkVersion(ctx)
304 minSdkVersion := effectiveVersionString(sdkVersion, sdkContext.MinSdkVersion(ctx))
Colin Crossa97c5d32018-03-28 14:58:31 -0700305
Colin Cross83bb3162018-06-25 15:48:06 -0700306 linkFlags = append(linkFlags, "--min-sdk-version "+minSdkVersion)
Spandan Das6450b552023-02-23 19:27:07 +0000307 // Use minSdkVersion for target-sdk-version, even if `target_sdk_version` is set
308 // This behavior has been copied from Make.
Colin Cross83bb3162018-06-25 15:48:06 -0700309 linkFlags = append(linkFlags, "--target-sdk-version "+minSdkVersion)
Colin Crossa97c5d32018-03-28 14:58:31 -0700310
Colin Crossa97c5d32018-03-28 14:58:31 -0700311 // Version code
312 if !hasVersionCode {
Dan Albert4f378d72020-07-23 17:32:15 -0700313 linkFlags = append(linkFlags, "--version-code", ctx.Config().PlatformSdkVersion().String())
Colin Crossa97c5d32018-03-28 14:58:31 -0700314 }
315
316 if !hasVersionName {
Colin Cross402d5e02018-04-25 14:54:06 -0700317 var versionName string
318 if ctx.ModuleName() == "framework-res" {
319 // Some builds set AppsDefaultVersionName() to include the build number ("O-123456"). aapt2 copies the
320 // version name of framework-res into app manifests as compileSdkVersionCodename, which confuses things
Colin Crossbfd347d2018-05-09 11:11:35 -0700321 // if it contains the build number. Use the PlatformVersionName instead.
322 versionName = ctx.Config().PlatformVersionName()
Colin Cross402d5e02018-04-25 14:54:06 -0700323 } else {
324 versionName = ctx.Config().AppsDefaultVersionName()
325 }
Colin Cross0b9f31f2019-02-28 11:00:01 -0800326 versionName = proptools.NinjaEscape(versionName)
Colin Crossa97c5d32018-03-28 14:58:31 -0700327 linkFlags = append(linkFlags, "--version-name ", versionName)
328 }
329
Colin Crossa0ba2f52019-06-22 12:59:27 -0700330 linkFlags, compileFlags = android.FilterList(linkFlags, []string{"--legacy"})
331
332 // Always set --pseudo-localize, it will be stripped out later for release
333 // builds that don't want it.
334 compileFlags = append(compileFlags, "--pseudo-localize")
335
336 return compileFlags, linkFlags, linkDeps, resDirs, overlayDirs, rroDirs, resourceZips
Colin Crossa97c5d32018-03-28 14:58:31 -0700337}
338
Paul Duffin250e6192019-06-07 10:44:37 +0100339func (a *aapt) deps(ctx android.BottomUpMutatorContext, sdkDep sdkDep) {
Colin Cross42308aa2018-11-14 21:44:17 -0800340 if sdkDep.frameworkResModule != "" {
341 ctx.AddVariationDependencies(nil, frameworkResTag, sdkDep.frameworkResModule)
Colin Crossa97c5d32018-03-28 14:58:31 -0700342 }
343}
344
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800345var extractAssetsRule = pctx.AndroidStaticRule("extractAssets",
346 blueprint.RuleParams{
347 Command: `${config.Zip2ZipCmd} -i ${in} -o ${out} "assets/**/*"`,
348 CommandDeps: []string{"${config.Zip2ZipCmd}"},
349 })
350
Alixf7a10272023-09-27 16:47:56 +0000351type aaptBuildActionOptions struct {
352 sdkContext android.SdkContext
353 classLoaderContexts dexpreopt.ClassLoaderContextMap
354 excludedLibs []string
355 enforceDefaultTargetSdkVersion bool
Rico Winda2fa2632024-03-13 13:09:17 +0100356 forceNonFinalResourceIDs bool
Alixf7a10272023-09-27 16:47:56 +0000357 extraLinkFlags []string
Jihoon Kang84b25892023-12-01 22:01:06 +0000358 aconfigTextFiles android.Paths
Jiakai Zhang36937082024-04-15 11:15:50 +0000359 usesLibrary *usesLibrary
Alixf7a10272023-09-27 16:47:56 +0000360}
361
362func (a *aapt) buildActions(ctx android.ModuleContext, opts aaptBuildActionOptions) {
Colin Cross5446e882019-05-22 10:46:27 -0700363
Colin Cross8676c8c2023-10-12 15:58:57 -0700364 staticResourcesNodesDepSet, sharedResourcesNodesDepSet, staticRRODirsDepSet, staticManifestsDepSet, sharedExportPackages, libFlags :=
Jiakai Zhang36937082024-04-15 11:15:50 +0000365 aaptLibs(ctx, opts.sdkContext, opts.classLoaderContexts, opts.usesLibrary)
Ulya Trafimovich31e444e2020-08-14 17:32:16 +0100366
Paul Duffin06530572022-02-03 17:54:15 +0000367 // Exclude any libraries from the supplied list.
Alixf7a10272023-09-27 16:47:56 +0000368 opts.classLoaderContexts = opts.classLoaderContexts.ExcludeLibs(opts.excludedLibs)
Paul Duffin06530572022-02-03 17:54:15 +0000369
Colin Cross31656952018-05-24 16:11:20 -0700370 // App manifest file
371 manifestFile := proptools.StringDefault(a.aaptProperties.Manifest, "AndroidManifest.xml")
372 manifestSrcPath := android.PathForModuleSrc(ctx, manifestFile)
373
Gurpreet Singh7deabfa2022-02-10 13:28:35 +0000374 manifestPath := ManifestFixer(ctx, manifestSrcPath, ManifestFixerParams{
Alixf7a10272023-09-27 16:47:56 +0000375 SdkContext: opts.sdkContext,
376 ClassLoaderContexts: opts.classLoaderContexts,
Harshit Mahajan5b8b7302022-06-10 11:24:05 +0000377 IsLibrary: a.isLibrary,
378 DefaultManifestVersion: a.defaultManifestVersion,
379 UseEmbeddedNativeLibs: a.useEmbeddedNativeLibs,
380 UsesNonSdkApis: a.usesNonSdkApis,
381 UseEmbeddedDex: a.useEmbeddedDex,
382 HasNoCode: a.hasNoCode,
383 LoggingParent: a.LoggingParent,
Alixf7a10272023-09-27 16:47:56 +0000384 EnforceDefaultTargetSdkVersion: opts.enforceDefaultTargetSdkVersion,
Gurpreet Singh75d65f32022-01-24 17:44:05 +0000385 })
Colin Cross90c25c62019-04-19 16:22:57 -0700386
Colin Crossab8d1382023-07-14 17:23:41 +0000387 staticDeps := transitiveAarDeps(staticResourcesNodesDepSet.ToList())
Colin Cross8676c8c2023-10-12 15:58:57 -0700388 sharedDeps := transitiveAarDeps(sharedResourcesNodesDepSet.ToList())
Colin Crossab8d1382023-07-14 17:23:41 +0000389
Luca Stefanifd898822019-09-10 22:13:31 +0200390 // Add additional manifest files to transitive manifests.
391 additionalManifests := android.PathsForModuleSrc(ctx, a.aaptProperties.Additional_manifests)
Colin Crossab8d1382023-07-14 17:23:41 +0000392 transitiveManifestPaths := append(android.Paths{manifestPath}, additionalManifests...)
Colin Crossab8d1382023-07-14 17:23:41 +0000393 transitiveManifestPaths = append(transitiveManifestPaths, staticManifestsDepSet.ToList()...)
Colin Cross90c25c62019-04-19 16:22:57 -0700394
Colin Crossab8d1382023-07-14 17:23:41 +0000395 if len(transitiveManifestPaths) > 1 && !Bool(a.aaptProperties.Dont_merge_manifests) {
Alixf7a10272023-09-27 16:47:56 +0000396 manifestMergerParams := ManifestMergerParams{
397 staticLibManifests: transitiveManifestPaths[1:],
Alix96ea88452023-08-31 15:48:23 +0000398 isLibrary: a.isLibrary,
399 packageName: a.manifestValues.applicationId,
400 }
Alixf7a10272023-09-27 16:47:56 +0000401 a.mergedManifestFile = manifestMerger(ctx, transitiveManifestPaths[0], manifestMergerParams)
Colin Cross90c25c62019-04-19 16:22:57 -0700402 if !a.isLibrary {
403 // Only use the merged manifest for applications. For libraries, the transitive closure of manifests
404 // will be propagated to the final application and merged there. The merged manifest for libraries is
405 // only passed to Make, which can't handle transitive dependencies.
406 manifestPath = a.mergedManifestFile
407 }
408 } else {
409 a.mergedManifestFile = manifestPath
410 }
Colin Cross31656952018-05-24 16:11:20 -0700411
Alixf7a10272023-09-27 16:47:56 +0000412 compileFlags, linkFlags, linkDeps, resDirs, overlayDirs, rroDirs, resZips := a.aapt2Flags(ctx, opts.sdkContext, manifestPath)
Colin Cross31656952018-05-24 16:11:20 -0700413
414 linkFlags = append(linkFlags, libFlags...)
Colin Cross8676c8c2023-10-12 15:58:57 -0700415 linkDeps = append(linkDeps, sharedExportPackages...)
Colin Crossab8d1382023-07-14 17:23:41 +0000416 linkDeps = append(linkDeps, staticDeps.resPackages()...)
Alixf7a10272023-09-27 16:47:56 +0000417 linkFlags = append(linkFlags, opts.extraLinkFlags...)
Colin Cross1b6a3cf2018-07-24 14:51:30 -0700418 if a.isLibrary {
419 linkFlags = append(linkFlags, "--static-lib")
420 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700421
Colin Cross7c4dc5d2024-02-13 14:29:45 -0800422 linkFlags = append(linkFlags, "--no-static-lib-packages")
Colin Cross8f1b0332024-01-25 13:39:06 -0800423 if a.isLibrary && a.useResourceProcessorBusyBox(ctx) {
Colin Cross7c4dc5d2024-02-13 14:29:45 -0800424 // When building an android_library using ResourceProcessorBusyBox pass --merge-only to skip resource
425 // references validation until the final app link step when all static libraries are present.
Colin Cross4eae06d2023-06-20 22:40:02 -0700426 linkFlags = append(linkFlags, "--merge-only")
Colin Cross4eae06d2023-06-20 22:40:02 -0700427 }
428
Colin Crossa97c5d32018-03-28 14:58:31 -0700429 packageRes := android.PathForModuleOut(ctx, "package-res.apk")
Colin Crossa97c5d32018-03-28 14:58:31 -0700430 proguardOptionsFile := android.PathForModuleGen(ctx, "proguard.options")
431 rTxt := android.PathForModuleOut(ctx, "R.txt")
Colin Cross66f78822018-05-02 12:58:28 -0700432 // This file isn't used by Soong, but is generated for exporting
433 extraPackages := android.PathForModuleOut(ctx, "extra_packages")
Colin Cross4eae06d2023-06-20 22:40:02 -0700434 var transitiveRJars android.Paths
Colin Crossf3b7bad2023-08-02 15:49:00 -0700435 var srcJar android.WritablePath
Colin Crossa97c5d32018-03-28 14:58:31 -0700436
Colin Cross4aaa84a2018-08-21 15:14:37 -0700437 var compiledResDirs []android.Paths
Colin Crossa97c5d32018-03-28 14:58:31 -0700438 for _, dir := range resDirs {
Colin Cross014489c2020-06-02 20:09:13 -0700439 a.resourceFiles = append(a.resourceFiles, dir.files...)
Inseob Kim34dc4cd2023-11-07 13:37:14 +0900440 compiledResDirs = append(compiledResDirs, aapt2Compile(ctx, dir.dir, dir.files, compileFlags, a.filterProduct()).Paths())
Colin Crossa97c5d32018-03-28 14:58:31 -0700441 }
Colin Cross4aaa84a2018-08-21 15:14:37 -0700442
Colin Crossa592e3e2019-02-19 16:59:53 -0800443 for i, zip := range resZips {
444 flata := android.PathForModuleOut(ctx, fmt.Sprintf("reszip.%d.flata", i))
Colin Crossa0ba2f52019-06-22 12:59:27 -0700445 aapt2CompileZip(ctx, flata, zip, "", compileFlags)
Colin Crossa592e3e2019-02-19 16:59:53 -0800446 compiledResDirs = append(compiledResDirs, android.Paths{flata})
447 }
448
Colin Cross4aaa84a2018-08-21 15:14:37 -0700449 var compiledRes, compiledOverlay android.Paths
450
Colin Crossab8d1382023-07-14 17:23:41 +0000451 // AAPT2 overlays are in lowest to highest priority order, reverse the topological order
452 // of transitiveStaticLibs.
453 transitiveStaticLibs := android.ReversePaths(staticDeps.resPackages())
454
Colin Cross8f1b0332024-01-25 13:39:06 -0800455 if a.isLibrary && a.useResourceProcessorBusyBox(ctx) {
Colin Cross4eae06d2023-06-20 22:40:02 -0700456 // When building an android_library with ResourceProcessorBusyBox enabled treat static library dependencies
457 // as imports. The resources from dependencies will not be merged into this module's package-res.apk, and
458 // instead modules depending on this module will reference package-res.apk from all transitive static
459 // dependencies.
Colin Cross1d3f5902024-03-05 11:51:54 -0800460 for _, sharedDep := range sharedDeps {
461 if sharedDep.usedResourceProcessor {
462 transitiveRJars = append(transitiveRJars, sharedDep.rJar)
463 }
464 }
Colin Cross4eae06d2023-06-20 22:40:02 -0700465 for _, staticDep := range staticDeps {
466 linkDeps = append(linkDeps, staticDep.resPackage)
467 linkFlags = append(linkFlags, "-I "+staticDep.resPackage.String())
468 if staticDep.usedResourceProcessor {
469 transitiveRJars = append(transitiveRJars, staticDep.rJar)
470 }
471 }
472 } else {
473 // When building an app or building a library without ResourceProcessorBusyBox enabled all static
474 // dependencies are compiled into this module's package-res.apk as overlays.
475 compiledOverlay = append(compiledOverlay, transitiveStaticLibs...)
476 }
Colin Cross4aaa84a2018-08-21 15:14:37 -0700477
Colin Crossbec85302019-02-13 13:15:46 -0800478 if len(transitiveStaticLibs) > 0 {
Colin Cross4aaa84a2018-08-21 15:14:37 -0700479 // If we are using static android libraries, every source file becomes an overlay.
480 // This is to emulate old AAPT behavior which simulated library support.
481 for _, compiledResDir := range compiledResDirs {
482 compiledOverlay = append(compiledOverlay, compiledResDir...)
483 }
Colin Crossbec85302019-02-13 13:15:46 -0800484 } else if a.isLibrary {
485 // Otherwise, for a static library we treat all the resources equally with no overlay.
486 for _, compiledResDir := range compiledResDirs {
487 compiledRes = append(compiledRes, compiledResDir...)
488 }
Colin Cross4aaa84a2018-08-21 15:14:37 -0700489 } else if len(compiledResDirs) > 0 {
490 // Without static libraries, the first directory is our directory, which can then be
491 // overlaid by the rest.
492 compiledRes = append(compiledRes, compiledResDirs[0]...)
493 for _, compiledResDir := range compiledResDirs[1:] {
494 compiledOverlay = append(compiledOverlay, compiledResDir...)
495 }
496 }
497
Colin Crossa97c5d32018-03-28 14:58:31 -0700498 for _, dir := range overlayDirs {
Inseob Kim34dc4cd2023-11-07 13:37:14 +0900499 compiledOverlay = append(compiledOverlay, aapt2Compile(ctx, dir.dir, dir.files, compileFlags, a.filterProduct()).Paths()...)
Colin Crossa97c5d32018-03-28 14:58:31 -0700500 }
501
Colin Crosse560c4a2019-03-19 16:03:11 -0700502 var splitPackages android.WritablePaths
503 var splits []split
504
505 for _, s := range a.splitNames {
506 suffix := strings.Replace(s, ",", "_", -1)
507 path := android.PathForModuleOut(ctx, "package_"+suffix+".apk")
508 linkFlags = append(linkFlags, "--split", path.String()+":"+s)
509 splitPackages = append(splitPackages, path)
510 splits = append(splits, split{
511 name: s,
512 suffix: suffix,
513 path: path,
514 })
515 }
516
Colin Cross8f1b0332024-01-25 13:39:06 -0800517 if !a.useResourceProcessorBusyBox(ctx) {
Colin Crossf3b7bad2023-08-02 15:49:00 -0700518 // the subdir "android" is required to be filtered by package names
519 srcJar = android.PathForModuleGen(ctx, "android", "R.srcjar")
520 }
521
Colin Crossab8d1382023-07-14 17:23:41 +0000522 // No need to specify assets from dependencies to aapt2Link for libraries, all transitive assets will be
523 // provided to the final app aapt2Link step.
524 var transitiveAssets android.Paths
525 if !a.isLibrary {
526 transitiveAssets = android.ReverseSliceInPlace(staticDeps.assets())
527 }
Colin Crossf3b7bad2023-08-02 15:49:00 -0700528 aapt2Link(ctx, packageRes, srcJar, proguardOptionsFile, rTxt,
Jihoon Kang84b25892023-12-01 22:01:06 +0000529 linkFlags, linkDeps, compiledRes, compiledOverlay, transitiveAssets, splitPackages,
530 opts.aconfigTextFiles)
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800531 // Extract assets from the resource package output so that they can be used later in aapt2link
532 // for modules that depend on this one.
Colin Crossab8d1382023-07-14 17:23:41 +0000533 if android.PrefixInList(linkFlags, "-A ") {
Jaewoong Jung6431ca72020-01-15 14:15:10 -0800534 assets := android.PathForModuleOut(ctx, "assets.zip")
535 ctx.Build(pctx, android.BuildParams{
536 Rule: extractAssetsRule,
537 Input: packageRes,
538 Output: assets,
539 Description: "extract assets from built resource file",
540 })
541 a.assetPackage = android.OptionalPathForPath(assets)
542 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700543
Colin Cross8f1b0332024-01-25 13:39:06 -0800544 if a.useResourceProcessorBusyBox(ctx) {
Colin Cross4eae06d2023-06-20 22:40:02 -0700545 rJar := android.PathForModuleOut(ctx, "busybox/R.jar")
Rico Winda2fa2632024-03-13 13:09:17 +0100546 resourceProcessorBusyBoxGenerateBinaryR(ctx, rTxt, a.mergedManifestFile, rJar, staticDeps, a.isLibrary, a.aaptProperties.Aaptflags,
547 opts.forceNonFinalResourceIDs)
Colin Crossf3b7bad2023-08-02 15:49:00 -0700548 aapt2ExtractExtraPackages(ctx, extraPackages, rJar)
Colin Cross4eae06d2023-06-20 22:40:02 -0700549 transitiveRJars = append(transitiveRJars, rJar)
550 a.rJar = rJar
Colin Crossf3b7bad2023-08-02 15:49:00 -0700551 } else {
552 aapt2ExtractExtraPackages(ctx, extraPackages, srcJar)
Colin Cross4eae06d2023-06-20 22:40:02 -0700553 }
554
Colin Cross312634e2023-11-21 15:13:56 -0800555 transitiveAaptResourcePackages := staticDeps.resPackages().Strings()
556 transitiveAaptResourcePackages = slices.DeleteFunc(transitiveAaptResourcePackages, func(p string) bool {
557 return p == packageRes.String()
558 })
559 transitiveAaptResourcePackagesFile := android.PathForModuleOut(ctx, "transitive-res-packages")
560 android.WriteFileRule(ctx, transitiveAaptResourcePackagesFile, strings.Join(transitiveAaptResourcePackages, "\n"))
561
Colin Cross1d3f5902024-03-05 11:51:54 -0800562 // Reverse the list of R.jar files so that the current module comes first, and direct dependencies come before
563 // transitive dependencies.
564 transitiveRJars = android.ReversePaths(transitiveRJars)
565
Colin Crossa97c5d32018-03-28 14:58:31 -0700566 a.aaptSrcJar = srcJar
Colin Cross4eae06d2023-06-20 22:40:02 -0700567 a.transitiveAaptRJars = transitiveRJars
Colin Cross312634e2023-11-21 15:13:56 -0800568 a.transitiveAaptResourcePackagesFile = transitiveAaptResourcePackagesFile
Colin Crossa97c5d32018-03-28 14:58:31 -0700569 a.exportPackage = packageRes
570 a.manifestPath = manifestPath
571 a.proguardOptionsFile = proguardOptionsFile
Colin Cross66f78822018-05-02 12:58:28 -0700572 a.extraAaptPackagesFile = extraPackages
Colin Crossa97c5d32018-03-28 14:58:31 -0700573 a.rTxt = rTxt
Colin Crosse560c4a2019-03-19 16:03:11 -0700574 a.splits = splits
Colin Crossab8d1382023-07-14 17:23:41 +0000575 a.resourcesNodesDepSet = android.NewDepSetBuilder[*resourcesNode](android.TOPOLOGICAL).
576 Direct(&resourcesNode{
577 resPackage: a.exportPackage,
578 manifest: a.manifestPath,
579 additionalManifests: additionalManifests,
Colin Cross4eae06d2023-06-20 22:40:02 -0700580 rTxt: a.rTxt,
581 rJar: a.rJar,
Colin Crossab8d1382023-07-14 17:23:41 +0000582 assets: a.assetPackage,
Colin Cross4eae06d2023-06-20 22:40:02 -0700583
Colin Cross8f1b0332024-01-25 13:39:06 -0800584 usedResourceProcessor: a.useResourceProcessorBusyBox(ctx),
Colin Crossab8d1382023-07-14 17:23:41 +0000585 }).
586 Transitive(staticResourcesNodesDepSet).Build()
587 a.rroDirsDepSet = android.NewDepSetBuilder[rroDir](android.TOPOLOGICAL).
588 Direct(rroDirs...).
589 Transitive(staticRRODirsDepSet).Build()
590 a.manifestsDepSet = android.NewDepSetBuilder[android.Path](android.TOPOLOGICAL).
591 Direct(a.manifestPath).
592 DirectSlice(additionalManifests).
593 Transitive(staticManifestsDepSet).Build()
594}
595
Colin Cross4eae06d2023-06-20 22:40:02 -0700596var resourceProcessorBusyBox = pctx.AndroidStaticRule("resourceProcessorBusyBox",
597 blueprint.RuleParams{
598 Command: "${config.JavaCmd} -cp ${config.ResourceProcessorBusyBox} " +
599 "com.google.devtools.build.android.ResourceProcessorBusyBox --tool=GENERATE_BINARY_R -- @${out}.args && " +
600 "if cmp -s ${out}.tmp ${out} ; then rm ${out}.tmp ; else mv ${out}.tmp ${out}; fi",
601 CommandDeps: []string{"${config.ResourceProcessorBusyBox}"},
602 Rspfile: "${out}.args",
603 RspfileContent: "--primaryRTxt ${rTxt} --primaryManifest ${manifest} --classJarOutput ${out}.tmp ${args}",
604 Restat: true,
605 }, "rTxt", "manifest", "args")
606
607// resourceProcessorBusyBoxGenerateBinaryR converts the R.txt file produced by aapt2 into R.class files
608// using Bazel's ResourceProcessorBusyBox tool, which is faster than compiling the R.java files and
609// supports producing classes for static dependencies that only include resources from that dependency.
610func resourceProcessorBusyBoxGenerateBinaryR(ctx android.ModuleContext, rTxt, manifest android.Path,
Rico Winda2fa2632024-03-13 13:09:17 +0100611 rJar android.WritablePath, transitiveDeps transitiveAarDeps, isLibrary bool, aaptFlags []string,
612 forceNonFinalIds bool) {
Colin Cross4eae06d2023-06-20 22:40:02 -0700613
614 var args []string
615 var deps android.Paths
616
617 if !isLibrary {
618 // When compiling an app, pass all R.txt and AndroidManifest.xml from transitive static library dependencies
619 // to ResourceProcessorBusyBox so that it can regenerate R.class files with the final resource IDs for each
620 // package.
621 args, deps = transitiveDeps.resourceProcessorDeps()
Rico Winda2fa2632024-03-13 13:09:17 +0100622 if forceNonFinalIds {
623 args = append(args, "--finalFields=false")
624 }
Colin Cross4eae06d2023-06-20 22:40:02 -0700625 } else {
626 // When compiling a library don't pass any dependencies as it only needs to generate an R.class file for this
627 // library. Pass --finalFields=false so that the R.class file contains non-final fields so they don't get
628 // inlined into the library before the final IDs are assigned during app compilation.
629 args = append(args, "--finalFields=false")
630 }
631
Colin Crossd3f7d1a2024-01-03 19:42:25 -0800632 for i, arg := range aaptFlags {
633 const AAPT_CUSTOM_PACKAGE = "--custom-package"
634 if strings.HasPrefix(arg, AAPT_CUSTOM_PACKAGE) {
635 pkg := strings.TrimSpace(strings.TrimPrefix(arg, AAPT_CUSTOM_PACKAGE))
636 if pkg == "" && i+1 < len(aaptFlags) {
637 pkg = aaptFlags[i+1]
638 }
639 args = append(args, "--packageForR "+pkg)
640 }
641 }
642
Colin Cross4eae06d2023-06-20 22:40:02 -0700643 deps = append(deps, rTxt, manifest)
644
645 ctx.Build(pctx, android.BuildParams{
646 Rule: resourceProcessorBusyBox,
647 Output: rJar,
648 Implicits: deps,
649 Description: "ResourceProcessorBusyBox",
650 Args: map[string]string{
651 "rTxt": rTxt.String(),
652 "manifest": manifest.String(),
653 "args": strings.Join(args, " "),
654 },
655 })
656}
657
Colin Crossab8d1382023-07-14 17:23:41 +0000658type resourcesNode struct {
659 resPackage android.Path
660 manifest android.Path
661 additionalManifests android.Paths
Colin Cross4eae06d2023-06-20 22:40:02 -0700662 rTxt android.Path
663 rJar android.Path
Colin Crossab8d1382023-07-14 17:23:41 +0000664 assets android.OptionalPath
Colin Cross4eae06d2023-06-20 22:40:02 -0700665
666 usedResourceProcessor bool
Colin Crossab8d1382023-07-14 17:23:41 +0000667}
668
669type transitiveAarDeps []*resourcesNode
670
671func (t transitiveAarDeps) resPackages() android.Paths {
Colin Cross4eae06d2023-06-20 22:40:02 -0700672 paths := make(android.Paths, 0, len(t))
Colin Crossab8d1382023-07-14 17:23:41 +0000673 for _, dep := range t {
674 paths = append(paths, dep.resPackage)
675 }
Colin Cross4eae06d2023-06-20 22:40:02 -0700676 return paths
Colin Crossab8d1382023-07-14 17:23:41 +0000677}
678
679func (t transitiveAarDeps) manifests() android.Paths {
Colin Cross4eae06d2023-06-20 22:40:02 -0700680 paths := make(android.Paths, 0, len(t))
Colin Crossab8d1382023-07-14 17:23:41 +0000681 for _, dep := range t {
682 paths = append(paths, dep.manifest)
683 paths = append(paths, dep.additionalManifests...)
684 }
Colin Cross4eae06d2023-06-20 22:40:02 -0700685 return paths
686}
687
688func (t transitiveAarDeps) resourceProcessorDeps() (args []string, deps android.Paths) {
689 for _, dep := range t {
690 args = append(args, "--library="+dep.rTxt.String()+","+dep.manifest.String())
691 deps = append(deps, dep.rTxt, dep.manifest)
692 }
693 return args, deps
Colin Crossab8d1382023-07-14 17:23:41 +0000694}
695
696func (t transitiveAarDeps) assets() android.Paths {
Colin Cross4eae06d2023-06-20 22:40:02 -0700697 paths := make(android.Paths, 0, len(t))
Colin Crossab8d1382023-07-14 17:23:41 +0000698 for _, dep := range t {
699 if dep.assets.Valid() {
700 paths = append(paths, dep.assets.Path())
701 }
702 }
703 return paths
Colin Crossa97c5d32018-03-28 14:58:31 -0700704}
705
706// aaptLibs collects libraries from dependencies and sdk_version and converts them into paths
Jiakai Zhang36937082024-04-15 11:15:50 +0000707func aaptLibs(ctx android.ModuleContext, sdkContext android.SdkContext,
708 classLoaderContexts dexpreopt.ClassLoaderContextMap, usesLibrary *usesLibrary) (
Colin Cross8676c8c2023-10-12 15:58:57 -0700709 staticResourcesNodes, sharedResourcesNodes *android.DepSet[*resourcesNode], staticRRODirs *android.DepSet[rroDir],
Colin Crossab8d1382023-07-14 17:23:41 +0000710 staticManifests *android.DepSet[android.Path], sharedLibs android.Paths, flags []string) {
Colin Crossa97c5d32018-03-28 14:58:31 -0700711
Ulya Trafimovichb23d28c2020-10-08 12:53:58 +0100712 if classLoaderContexts == nil {
Ulya Trafimovich18554242020-11-03 15:55:11 +0000713 // Not all callers need to compute class loader context, those who don't just pass nil.
714 // Create a temporary class loader context here (it will be computed, but not used).
Ulya Trafimovichb23d28c2020-10-08 12:53:58 +0100715 classLoaderContexts = make(dexpreopt.ClassLoaderContextMap)
Ulya Trafimovich18554242020-11-03 15:55:11 +0000716 }
717
Colin Cross83bb3162018-06-25 15:48:06 -0700718 sdkDep := decodeSdkDep(ctx, sdkContext)
Colin Crossa97c5d32018-03-28 14:58:31 -0700719 if sdkDep.useFiles {
Colin Cross86a60ae2018-05-29 14:44:55 -0700720 sharedLibs = append(sharedLibs, sdkDep.jars...)
Colin Crossa97c5d32018-03-28 14:58:31 -0700721 }
722
Colin Cross8676c8c2023-10-12 15:58:57 -0700723 var staticResourcesNodeDepSets []*android.DepSet[*resourcesNode]
724 var sharedResourcesNodeDepSets []*android.DepSet[*resourcesNode]
Colin Crossab8d1382023-07-14 17:23:41 +0000725 rroDirsDepSetBuilder := android.NewDepSetBuilder[rroDir](android.TOPOLOGICAL)
726 manifestsDepSetBuilder := android.NewDepSetBuilder[android.Path](android.TOPOLOGICAL)
727
Colin Crossa97c5d32018-03-28 14:58:31 -0700728 ctx.VisitDirectDeps(func(module android.Module) {
Ulya Trafimovich65b03192020-12-03 16:50:22 +0000729 depTag := ctx.OtherModuleDependencyTag(module)
Ulya Trafimovich18554242020-11-03 15:55:11 +0000730
Colin Crossa97c5d32018-03-28 14:58:31 -0700731 var exportPackage android.Path
Colin Cross66f78822018-05-02 12:58:28 -0700732 aarDep, _ := module.(AndroidLibraryDependency)
733 if aarDep != nil {
Colin Crossa97c5d32018-03-28 14:58:31 -0700734 exportPackage = aarDep.ExportPackage()
735 }
736
Ulya Trafimovich65b03192020-12-03 16:50:22 +0000737 switch depTag {
Colin Cross4b964c02018-10-15 16:18:06 -0700738 case instrumentationForTag:
739 // Nothing, instrumentationForTag is treated as libTag for javac but not for aapt2.
Liz Kammeref28a4c2022-09-23 16:50:56 -0400740 case sdkLibTag, libTag:
Colin Cross5446e882019-05-22 10:46:27 -0700741 if exportPackage != nil {
Colin Cross8676c8c2023-10-12 15:58:57 -0700742 sharedResourcesNodeDepSets = append(sharedResourcesNodeDepSets, aarDep.ResourcesNodeDepSet())
Colin Cross5446e882019-05-22 10:46:27 -0700743 sharedLibs = append(sharedLibs, exportPackage)
744 }
Colin Cross5446e882019-05-22 10:46:27 -0700745 case frameworkResTag:
Colin Crossa97c5d32018-03-28 14:58:31 -0700746 if exportPackage != nil {
747 sharedLibs = append(sharedLibs, exportPackage)
748 }
749 case staticLibTag:
750 if exportPackage != nil {
Colin Cross8676c8c2023-10-12 15:58:57 -0700751 staticResourcesNodeDepSets = append(staticResourcesNodeDepSets, aarDep.ResourcesNodeDepSet())
Colin Crossab8d1382023-07-14 17:23:41 +0000752 rroDirsDepSetBuilder.Transitive(aarDep.RRODirsDepSet())
753 manifestsDepSetBuilder.Transitive(aarDep.ManifestsDepSet())
Colin Crossa97c5d32018-03-28 14:58:31 -0700754 }
755 }
Ulya Trafimovich18554242020-11-03 15:55:11 +0000756
Ulya Trafimovich88bb6f62020-12-16 16:16:11 +0000757 addCLCFromDep(ctx, module, classLoaderContexts)
Jiakai Zhang36937082024-04-15 11:15:50 +0000758 if usesLibrary != nil {
759 addMissingOptionalUsesLibsFromDep(ctx, module, usesLibrary)
760 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700761 })
762
Colin Crossab8d1382023-07-14 17:23:41 +0000763 // AAPT2 overlays are in lowest to highest priority order, the topological order will be reversed later.
764 // Reverse the dependency order now going into the depset so that it comes out in order after the second
765 // reverse later.
766 // NOTE: this is legacy and probably incorrect behavior, for most other cases (e.g. conflicting classes in
767 // dependencies) the highest priority dependency is listed first, but for resources the highest priority
Colin Cross9055e212024-03-23 04:43:41 +0000768 // dependency has to be listed last. This is also inconsistent with the way manifests from the same
769 // transitive dependencies are merged.
Colin Crossab8d1382023-07-14 17:23:41 +0000770 staticResourcesNodes = android.NewDepSet(android.TOPOLOGICAL, nil,
Colin Cross8676c8c2023-10-12 15:58:57 -0700771 android.ReverseSliceInPlace(staticResourcesNodeDepSets))
772 sharedResourcesNodes = android.NewDepSet(android.TOPOLOGICAL, nil,
773 android.ReverseSliceInPlace(sharedResourcesNodeDepSets))
Colin Crossa97c5d32018-03-28 14:58:31 -0700774
Colin Crossab8d1382023-07-14 17:23:41 +0000775 staticRRODirs = rroDirsDepSetBuilder.Build()
776 staticManifests = manifestsDepSetBuilder.Build()
777
778 if len(staticResourcesNodes.ToList()) > 0 {
Colin Crossa97c5d32018-03-28 14:58:31 -0700779 flags = append(flags, "--auto-add-overlay")
780 }
781
782 for _, sharedLib := range sharedLibs {
783 flags = append(flags, "-I "+sharedLib.String())
784 }
785
Colin Cross8676c8c2023-10-12 15:58:57 -0700786 return staticResourcesNodes, sharedResourcesNodes, staticRRODirs, staticManifests, sharedLibs, flags
Colin Crossa97c5d32018-03-28 14:58:31 -0700787}
788
789type AndroidLibrary struct {
790 Library
791 aapt
792
793 androidLibraryProperties androidLibraryProperties
794
795 aarFile android.WritablePath
Colin Cross89c31582018-04-30 15:55:11 -0700796}
797
Saeid Farivar Asanjan1fca3012021-09-14 18:40:19 +0000798var _ android.OutputFileProducer = (*AndroidLibrary)(nil)
799
800// For OutputFileProducer interface
801func (a *AndroidLibrary) OutputFiles(tag string) (android.Paths, error) {
802 switch tag {
803 case ".aar":
804 return []android.Path{a.aarFile}, nil
805 default:
806 return a.Library.OutputFiles(tag)
807 }
808}
809
Colin Crossa97c5d32018-03-28 14:58:31 -0700810var _ AndroidLibraryDependency = (*AndroidLibrary)(nil)
811
812func (a *AndroidLibrary) DepsMutator(ctx android.BottomUpMutatorContext) {
Jiakai Zhangf98da192024-04-15 11:15:41 +0000813 a.usesLibrary.deps(ctx, false)
Colin Crossa97c5d32018-03-28 14:58:31 -0700814 a.Module.deps(ctx)
Jiyong Parkf1691d22021-03-29 20:11:58 +0900815 sdkDep := decodeSdkDep(ctx, android.SdkContext(a))
Paul Duffin250e6192019-06-07 10:44:37 +0100816 if sdkDep.hasFrameworkLibs() {
817 a.aapt.deps(ctx, sdkDep)
Colin Crossa97c5d32018-03-28 14:58:31 -0700818 }
Jihoon Kang9049c272024-03-19 21:57:36 +0000819
820 for _, aconfig_declaration := range a.aaptProperties.Flags_packages {
821 ctx.AddDependency(ctx.Module(), aconfigDeclarationTag, aconfig_declaration)
822 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700823}
824
825func (a *AndroidLibrary) GenerateAndroidBuildActions(ctx android.ModuleContext) {
Colin Crosse4246ab2019-02-05 21:55:21 -0800826 a.aapt.isLibrary = true
Ulya Trafimovich42c7f0d2021-08-17 16:20:29 +0100827 a.classLoaderContexts = a.usesLibrary.classLoaderContextForUsesLibDeps(ctx)
Spandan Das0727ba72024-02-13 16:37:43 +0000828 if a.usesLibrary.shouldDisableDexpreopt {
829 a.dexpreopter.disableDexpreopt()
830 }
Alixf7a10272023-09-27 16:47:56 +0000831 a.aapt.buildActions(ctx,
832 aaptBuildActionOptions{
833 sdkContext: android.SdkContext(a),
834 classLoaderContexts: a.classLoaderContexts,
835 enforceDefaultTargetSdkVersion: false,
Jihoon Kang9049c272024-03-19 21:57:36 +0000836 aconfigTextFiles: getAconfigFilePaths(ctx),
Jiakai Zhang36937082024-04-15 11:15:50 +0000837 usesLibrary: &a.usesLibrary,
Alixf7a10272023-09-27 16:47:56 +0000838 },
839 )
Colin Crossa97c5d32018-03-28 14:58:31 -0700840
Colin Crossff694a82023-12-13 15:54:49 -0800841 apexInfo, _ := android.ModuleProvider(ctx, android.ApexInfoProvider)
842 a.hideApexVariantFromMake = !apexInfo.IsForPlatform()
Colin Cross56a83212020-09-15 18:30:11 -0700843
yangbill2af0b6e2024-03-15 09:29:29 +0000844 a.stem = proptools.StringDefault(a.overridableProperties.Stem, ctx.ModuleName())
Jihoon Kang1bfb6f22023-07-01 00:13:47 +0000845
Colin Cross4eae06d2023-06-20 22:40:02 -0700846 ctx.CheckbuildFile(a.aapt.proguardOptionsFile)
847 ctx.CheckbuildFile(a.aapt.exportPackage)
Colin Cross8f1b0332024-01-25 13:39:06 -0800848 if a.useResourceProcessorBusyBox(ctx) {
Colin Cross4eae06d2023-06-20 22:40:02 -0700849 ctx.CheckbuildFile(a.aapt.rJar)
Colin Crossf3b7bad2023-08-02 15:49:00 -0700850 } else {
851 ctx.CheckbuildFile(a.aapt.aaptSrcJar)
Colin Cross4eae06d2023-06-20 22:40:02 -0700852 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700853
854 // apps manifests are handled by aapt, don't let Module see them
855 a.properties.Manifest = nil
856
Colin Cross014489c2020-06-02 20:09:13 -0700857 a.linter.mergedManifest = a.aapt.mergedManifestFile
858 a.linter.manifest = a.aapt.manifestPath
859 a.linter.resources = a.aapt.resourceFiles
860
Sam Delmericoc8e040c2023-10-31 17:27:02 +0000861 proguardSpecInfo := a.collectProguardSpecInfo(ctx)
Colin Cross40213022023-12-13 15:19:49 -0800862 android.SetProvider(ctx, ProguardSpecInfoProvider, proguardSpecInfo)
Colin Cross312634e2023-11-21 15:13:56 -0800863 exportedProguardFlagsFiles := proguardSpecInfo.ProguardFlagsFiles.ToList()
864 a.extraProguardFlagsFiles = append(a.extraProguardFlagsFiles, exportedProguardFlagsFiles...)
865 a.extraProguardFlagsFiles = append(a.extraProguardFlagsFiles, a.proguardOptionsFile)
866
867 combinedExportedProguardFlagFile := android.PathForModuleOut(ctx, "export_proguard_flags")
868 writeCombinedProguardFlagsFile(ctx, combinedExportedProguardFlagFile, exportedProguardFlagsFiles)
869 a.combinedExportedProguardFlagsFile = combinedExportedProguardFlagFile
Colin Crossa97c5d32018-03-28 14:58:31 -0700870
Colin Cross4eae06d2023-06-20 22:40:02 -0700871 var extraSrcJars android.Paths
872 var extraCombinedJars android.Paths
873 var extraClasspathJars android.Paths
Colin Cross8f1b0332024-01-25 13:39:06 -0800874 if a.useResourceProcessorBusyBox(ctx) {
Colin Cross4eae06d2023-06-20 22:40:02 -0700875 // When building a library with ResourceProcessorBusyBox enabled ResourceProcessorBusyBox for this
876 // library and each of the transitive static android_library dependencies has already created an
877 // R.class file for the appropriate package. Add all of those R.class files to the classpath.
878 extraClasspathJars = a.transitiveAaptRJars
879 } else {
880 // When building a library without ResourceProcessorBusyBox the aapt2 rule creates R.srcjar containing
881 // R.java files for the library's package and the packages from all transitive static android_library
882 // dependencies. Compile the srcjar alongside the rest of the sources.
883 extraSrcJars = android.Paths{a.aapt.aaptSrcJar}
884 }
885
886 a.Module.compile(ctx, extraSrcJars, extraClasspathJars, extraCombinedJars)
Colin Crossa97c5d32018-03-28 14:58:31 -0700887
Colin Crossf57c5782019-01-25 13:20:38 -0800888 a.aarFile = android.PathForModuleOut(ctx, ctx.ModuleName()+".aar")
Colin Crossa97c5d32018-03-28 14:58:31 -0700889 var res android.Paths
890 if a.androidLibraryProperties.BuildAAR {
891 BuildAAR(ctx, a.aarFile, a.outputFile, a.manifestPath, a.rTxt, res)
892 ctx.CheckbuildFile(a.aarFile)
893 }
Colin Cross89c31582018-04-30 15:55:11 -0700894
Sam Delmerico82602492022-06-10 17:05:42 +0000895 prebuiltJniPackages := android.Paths{}
896 ctx.VisitDirectDeps(func(module android.Module) {
Colin Cross313aa542023-12-13 13:47:44 -0800897 if info, ok := android.OtherModuleProvider(ctx, module, JniPackageProvider); ok {
Sam Delmerico82602492022-06-10 17:05:42 +0000898 prebuiltJniPackages = append(prebuiltJniPackages, info.JniPackages...)
899 }
900 })
901 if len(prebuiltJniPackages) > 0 {
Colin Cross40213022023-12-13 15:19:49 -0800902 android.SetProvider(ctx, JniPackageProvider, JniPackageInfo{
Sam Delmerico82602492022-06-10 17:05:42 +0000903 JniPackages: prebuiltJniPackages,
904 })
905 }
Colin Crossa97c5d32018-03-28 14:58:31 -0700906}
907
Colin Cross95b53b82023-10-17 13:21:02 -0700908func (a *AndroidLibrary) IDEInfo(dpInfo *android.IdeInfo) {
909 a.Library.IDEInfo(dpInfo)
910 a.aapt.IDEInfo(dpInfo)
911}
912
913func (a *aapt) IDEInfo(dpInfo *android.IdeInfo) {
Colin Cross8f1b0332024-01-25 13:39:06 -0800914 if a.rJar != nil {
Colin Cross95b53b82023-10-17 13:21:02 -0700915 dpInfo.Jars = append(dpInfo.Jars, a.rJar.String())
916 }
917}
918
Colin Cross1b16b0e2019-02-12 14:41:32 -0800919// android_library builds and links sources into a `.jar` file for the device along with Android resources.
920//
921// An android_library has a single variant that produces a `.jar` file containing `.class` files that were
Sam Delmerico82602492022-06-10 17:05:42 +0000922// compiled against the device bootclasspath, along with a `package-res.apk` file containing Android resources compiled
Colin Cross1b16b0e2019-02-12 14:41:32 -0800923// with aapt2. This module is not suitable for installing on a device, but can be used as a `static_libs` dependency of
924// an android_app module.
Colin Crossa97c5d32018-03-28 14:58:31 -0700925func AndroidLibraryFactory() android.Module {
926 module := &AndroidLibrary{}
927
Colin Crossce6734e2020-06-15 16:09:53 -0700928 module.Module.addHostAndDeviceProperties()
Colin Crossa97c5d32018-03-28 14:58:31 -0700929 module.AddProperties(
Colin Crossa97c5d32018-03-28 14:58:31 -0700930 &module.aaptProperties,
931 &module.androidLibraryProperties)
932
933 module.androidLibraryProperties.BuildAAR = true
Colin Cross014489c2020-06-02 20:09:13 -0700934 module.Module.linter.library = true
Colin Crossa97c5d32018-03-28 14:58:31 -0700935
Jooyung Hanacc7bbe2020-05-20 09:06:00 +0900936 android.InitApexModule(module)
Colin Cross48de9a42018-10-02 13:53:33 -0700937 InitJavaModule(module, android.DeviceSupported)
Colin Crossa97c5d32018-03-28 14:58:31 -0700938 return module
939}
940
Colin Crossfabb6082018-02-20 17:22:23 -0800941//
942// AAR (android library) prebuilts
943//
Colin Crossfabb6082018-02-20 17:22:23 -0800944
Vinh Trance0781f2022-04-13 01:30:44 +0000945// Properties for android_library_import
Colin Crossfabb6082018-02-20 17:22:23 -0800946type AARImportProperties struct {
Vinh Trance0781f2022-04-13 01:30:44 +0000947 // ARR (android library prebuilt) filepath. Exactly one ARR is required.
Colin Cross27b922f2019-03-04 22:35:41 -0800948 Aars []string `android:"path"`
Vinh Trance0781f2022-04-13 01:30:44 +0000949 // If not blank, set to the version of the sdk to compile against.
950 // Defaults to private.
951 // Values are of one of the following forms:
952 // 1) numerical API level, "current", "none", or "core_platform"
953 // 2) An SDK kind with an API level: "<sdk kind>_<API level>"
954 // See build/soong/android/sdk_version.go for the complete and up to date list of SDK kinds.
955 // If the SDK kind is empty, it will be set to public
956 Sdk_version *string
957 // If not blank, set the minimum version of the sdk that the compiled artifacts will run against.
958 // Defaults to sdk_version if not set. See sdk_version for possible values.
Colin Cross479884c2018-07-10 13:39:30 -0700959 Min_sdk_version *string
Vinh Trance0781f2022-04-13 01:30:44 +0000960 // List of java static libraries that the included ARR (android library prebuilts) has dependencies to.
Colin Crossa97c5d32018-03-28 14:58:31 -0700961 Static_libs []string
Vinh Trance0781f2022-04-13 01:30:44 +0000962 // List of java libraries that the included ARR (android library prebuilts) has dependencies to.
963 Libs []string
964 // If set to true, run Jetifier against .aar file. Defaults to false.
Colin Cross1001a792019-03-21 22:21:39 -0700965 Jetifier *bool
Sam Delmerico82602492022-06-10 17:05:42 +0000966 // If true, extract JNI libs from AAR archive. These libs will be accessible to android_app modules and
967 // will be passed transitively through android_libraries to an android_app.
968 //TODO(b/241138093) evaluate whether we can have this flag default to true for Bazel conversion
969 Extract_jni *bool
Colin Crossfabb6082018-02-20 17:22:23 -0800970}
971
972type AARImport struct {
973 android.ModuleBase
Colin Cross48de9a42018-10-02 13:53:33 -0700974 android.DefaultableModuleBase
Jooyung Hanacc7bbe2020-05-20 09:06:00 +0900975 android.ApexModuleBase
Colin Crossfabb6082018-02-20 17:22:23 -0800976 prebuilt android.Prebuilt
977
Jooyung Hanacc7bbe2020-05-20 09:06:00 +0900978 // Functionality common to Module and Import.
979 embeddableInModuleAndImport
980
Sam Delmerico9f9c0a22022-11-29 11:19:37 -0500981 providesTransitiveHeaderJars
982
Colin Crossfabb6082018-02-20 17:22:23 -0800983 properties AARImportProperties
984
Colin Cross9055e212024-03-23 04:43:41 +0000985 headerJarFile android.WritablePath
986 implementationJarFile android.WritablePath
Colin Cross312634e2023-11-21 15:13:56 -0800987 proguardFlags android.WritablePath
988 exportPackage android.WritablePath
989 transitiveAaptResourcePackagesFile android.Path
990 extraAaptPackagesFile android.WritablePath
991 manifest android.WritablePath
992 assetsPackage android.WritablePath
993 rTxt android.WritablePath
994 rJar android.WritablePath
Colin Cross66f78822018-05-02 12:58:28 -0700995
Colin Crossab8d1382023-07-14 17:23:41 +0000996 resourcesNodesDepSet *android.DepSet[*resourcesNode]
997 manifestsDepSet *android.DepSet[android.Path]
Colin Cross56a83212020-09-15 18:30:11 -0700998
999 hideApexVariantFromMake bool
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001000
Sam Delmerico82602492022-06-10 17:05:42 +00001001 aarPath android.Path
1002 jniPackages android.Paths
Jiyong Park92315372021-04-02 08:45:46 +09001003
1004 sdkVersion android.SdkSpec
Spandan Das8c9ae7e2023-03-03 21:20:36 +00001005 minSdkVersion android.ApiLevel
LaMont Jonesafe7baf2024-01-09 22:47:39 +00001006
Colin Cross9055e212024-03-23 04:43:41 +00001007 usesLibrary
1008 classLoaderContexts dexpreopt.ClassLoaderContextMap
1009
LaMont Jonesafe7baf2024-01-09 22:47:39 +00001010 // Single aconfig "cache file" merged from this module and all dependencies.
1011 mergedAconfigFiles map[string]android.Paths
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001012}
1013
1014var _ android.OutputFileProducer = (*AARImport)(nil)
1015
1016// For OutputFileProducer interface
1017func (a *AARImport) OutputFiles(tag string) (android.Paths, error) {
1018 switch tag {
1019 case ".aar":
1020 return []android.Path{a.aarPath}, nil
1021 case "":
Colin Cross9055e212024-03-23 04:43:41 +00001022 return []android.Path{a.implementationJarFile}, nil
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001023 default:
1024 return nil, fmt.Errorf("unsupported module reference tag %q", tag)
1025 }
Colin Crossfabb6082018-02-20 17:22:23 -08001026}
1027
Jiyong Park92315372021-04-02 08:45:46 +09001028func (a *AARImport) SdkVersion(ctx android.EarlyModuleContext) android.SdkSpec {
1029 return android.SdkSpecFrom(ctx, String(a.properties.Sdk_version))
Colin Cross83bb3162018-06-25 15:48:06 -07001030}
1031
Jiyong Parkf1691d22021-03-29 20:11:58 +09001032func (a *AARImport) SystemModules() string {
Paul Duffine25c6442019-10-11 13:50:28 +01001033 return ""
1034}
1035
Spandan Das8c9ae7e2023-03-03 21:20:36 +00001036func (a *AARImport) MinSdkVersion(ctx android.EarlyModuleContext) android.ApiLevel {
Colin Cross479884c2018-07-10 13:39:30 -07001037 if a.properties.Min_sdk_version != nil {
Spandan Das8c9ae7e2023-03-03 21:20:36 +00001038 return android.ApiLevelFrom(ctx, *a.properties.Min_sdk_version)
Colin Cross479884c2018-07-10 13:39:30 -07001039 }
Spandan Das8c9ae7e2023-03-03 21:20:36 +00001040 return a.SdkVersion(ctx).ApiLevel
Colin Cross83bb3162018-06-25 15:48:06 -07001041}
1042
Spandan Dasa26eda72023-03-02 00:56:06 +00001043func (a *AARImport) ReplaceMaxSdkVersionPlaceholder(ctx android.EarlyModuleContext) android.ApiLevel {
1044 return android.SdkSpecFrom(ctx, "").ApiLevel
William Loh5a082f92022-05-17 20:21:50 +00001045}
1046
Spandan Dasca70fc42023-03-01 23:38:49 +00001047func (a *AARImport) TargetSdkVersion(ctx android.EarlyModuleContext) android.ApiLevel {
1048 return a.SdkVersion(ctx).ApiLevel
Dan Willemsen419290a2018-10-31 15:28:47 -07001049}
1050
Colin Cross1e743852019-10-28 11:37:20 -07001051func (a *AARImport) javaVersion() string {
1052 return ""
1053}
1054
Colin Crossa97c5d32018-03-28 14:58:31 -07001055var _ AndroidLibraryDependency = (*AARImport)(nil)
1056
1057func (a *AARImport) ExportPackage() android.Path {
1058 return a.exportPackage
1059}
Colin Crossab8d1382023-07-14 17:23:41 +00001060func (a *AARImport) ResourcesNodeDepSet() *android.DepSet[*resourcesNode] {
1061 return a.resourcesNodesDepSet
Colin Crossc1c37552019-01-31 11:42:41 -08001062}
1063
Colin Crossab8d1382023-07-14 17:23:41 +00001064func (a *AARImport) RRODirsDepSet() *android.DepSet[rroDir] {
1065 return android.NewDepSet[rroDir](android.TOPOLOGICAL, nil, nil)
Colin Cross66f78822018-05-02 12:58:28 -07001066}
1067
Colin Crossab8d1382023-07-14 17:23:41 +00001068func (a *AARImport) ManifestsDepSet() *android.DepSet[android.Path] {
1069 return a.manifestsDepSet
Jaewoong Jung6431ca72020-01-15 14:15:10 -08001070}
1071
Jaewoong Jungc779cd42020-10-06 18:56:10 -07001072// RRO enforcement is not available on aar_import since its RRO dirs are not
1073// exported.
1074func (a *AARImport) SetRROEnforcedForDependent(enforce bool) {
1075}
1076
1077// RRO enforcement is not available on aar_import since its RRO dirs are not
1078// exported.
1079func (a *AARImport) IsRROEnforced(ctx android.BaseModuleContext) bool {
1080 return false
1081}
1082
Colin Crossfabb6082018-02-20 17:22:23 -08001083func (a *AARImport) Prebuilt() *android.Prebuilt {
1084 return &a.prebuilt
1085}
1086
1087func (a *AARImport) Name() string {
1088 return a.prebuilt.Name(a.ModuleBase.Name())
1089}
1090
Jiyong Park618922e2020-01-08 13:35:43 +09001091func (a *AARImport) JacocoReportClassesFile() android.Path {
1092 return nil
1093}
1094
Colin Crossfabb6082018-02-20 17:22:23 -08001095func (a *AARImport) DepsMutator(ctx android.BottomUpMutatorContext) {
Jeongik Cha816a23a2020-07-08 01:09:23 +09001096 if !ctx.Config().AlwaysUsePrebuiltSdks() {
Jiyong Parkf1691d22021-03-29 20:11:58 +09001097 sdkDep := decodeSdkDep(ctx, android.SdkContext(a))
Colin Crossa97c5d32018-03-28 14:58:31 -07001098 if sdkDep.useModule && sdkDep.frameworkResModule != "" {
Colin Cross42d48b72018-08-29 14:10:52 -07001099 ctx.AddVariationDependencies(nil, frameworkResTag, sdkDep.frameworkResModule)
Colin Crossfabb6082018-02-20 17:22:23 -08001100 }
1101 }
Colin Crossa97c5d32018-03-28 14:58:31 -07001102
Colin Cross42d48b72018-08-29 14:10:52 -07001103 ctx.AddVariationDependencies(nil, libTag, a.properties.Libs...)
1104 ctx.AddVariationDependencies(nil, staticLibTag, a.properties.Static_libs...)
Colin Cross9055e212024-03-23 04:43:41 +00001105
1106 a.usesLibrary.deps(ctx, false)
Colin Crossfabb6082018-02-20 17:22:23 -08001107}
1108
Sam Delmerico82602492022-06-10 17:05:42 +00001109type JniPackageInfo struct {
1110 // List of zip files containing JNI libraries
1111 // Zip files should have directory structure jni/<arch>/*.so
1112 JniPackages android.Paths
1113}
1114
Colin Crossbc7d76c2023-12-12 16:39:03 -08001115var JniPackageProvider = blueprint.NewProvider[JniPackageInfo]()
Sam Delmerico82602492022-06-10 17:05:42 +00001116
1117// Unzip an AAR and extract the JNI libs for $archString.
1118var extractJNI = pctx.AndroidStaticRule("extractJNI",
1119 blueprint.RuleParams{
1120 Command: `rm -rf $out $outDir && touch $out && ` +
1121 `unzip -qoDD -d $outDir $in "jni/${archString}/*" && ` +
1122 `jni_files=$$(find $outDir/jni -type f) && ` +
1123 // print error message if there are no JNI libs for this arch
1124 `[ -n "$$jni_files" ] || (echo "ERROR: no JNI libs found for arch ${archString}" && exit 1) && ` +
Sam Delmerico80ee45c2023-06-22 15:36:02 -04001125 `${config.SoongZipCmd} -o $out -L 0 -P 'lib/${archString}' ` +
Sam Delmerico82602492022-06-10 17:05:42 +00001126 `-C $outDir/jni/${archString} $$(echo $$jni_files | xargs -n1 printf " -f %s")`,
1127 CommandDeps: []string{"${config.SoongZipCmd}"},
1128 },
1129 "outDir", "archString")
1130
Colin Crossfabb6082018-02-20 17:22:23 -08001131// 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 -07001132// 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 -08001133var unzipAAR = pctx.AndroidStaticRule("unzipAAR",
1134 blueprint.RuleParams{
Dan Willemsen304cfec2019-05-28 14:49:06 -07001135 Command: `rm -rf $outDir && mkdir -p $outDir && ` +
Colin Cross205e9112020-08-06 13:20:17 -07001136 `unzip -qoDD -d $outDir $in && rm -rf $outDir/res && touch $out && ` +
Michael Rosenfeld5ad15572021-12-03 13:25:10 -08001137 `${config.Zip2ZipCmd} -i $in -o $assetsPackage 'assets/**/*' && ` +
Colin Cross205e9112020-08-06 13:20:17 -07001138 `${config.MergeZipsCmd} $combinedClassesJar $$(ls $outDir/classes.jar 2> /dev/null) $$(ls $outDir/libs/*.jar 2> /dev/null)`,
Michael Rosenfeld5ad15572021-12-03 13:25:10 -08001139 CommandDeps: []string{"${config.MergeZipsCmd}", "${config.Zip2ZipCmd}"},
Colin Crossfabb6082018-02-20 17:22:23 -08001140 },
Michael Rosenfeld5ad15572021-12-03 13:25:10 -08001141 "outDir", "combinedClassesJar", "assetsPackage")
Colin Crossfabb6082018-02-20 17:22:23 -08001142
1143func (a *AARImport) GenerateAndroidBuildActions(ctx android.ModuleContext) {
1144 if len(a.properties.Aars) != 1 {
1145 ctx.PropertyErrorf("aars", "exactly one aar is required")
1146 return
1147 }
1148
Jiyong Park92315372021-04-02 08:45:46 +09001149 a.sdkVersion = a.SdkVersion(ctx)
1150 a.minSdkVersion = a.MinSdkVersion(ctx)
1151
Colin Crossff694a82023-12-13 15:54:49 -08001152 apexInfo, _ := android.ModuleProvider(ctx, android.ApexInfoProvider)
1153 a.hideApexVariantFromMake = !apexInfo.IsForPlatform()
Colin Cross56a83212020-09-15 18:30:11 -07001154
Nan Zhang4c819fb2018-08-27 18:31:46 -07001155 aarName := ctx.ModuleName() + ".aar"
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001156 a.aarPath = android.PathForModuleSrc(ctx, a.properties.Aars[0])
1157
Colin Cross1001a792019-03-21 22:21:39 -07001158 if Bool(a.properties.Jetifier) {
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001159 inputFile := a.aarPath
1160 a.aarPath = android.PathForModuleOut(ctx, "jetifier", aarName)
1161 TransformJetifier(ctx, a.aarPath.(android.WritablePath), inputFile)
Nan Zhang4c819fb2018-08-27 18:31:46 -07001162 }
Colin Crossfabb6082018-02-20 17:22:23 -08001163
1164 extractedAARDir := android.PathForModuleOut(ctx, "aar")
Colin Cross9055e212024-03-23 04:43:41 +00001165 classpathFile := extractedAARDir.Join(ctx, "classes-combined.jar")
Colin Cross10f7c4a2018-05-23 10:59:28 -07001166 a.manifest = extractedAARDir.Join(ctx, "AndroidManifest.xml")
Colin Crossbb77d8e2024-02-15 14:43:47 -08001167 a.rTxt = extractedAARDir.Join(ctx, "R.txt")
Michael Rosenfeld5ad15572021-12-03 13:25:10 -08001168 a.assetsPackage = android.PathForModuleOut(ctx, "assets.zip")
Sam Delmerico95d70942023-08-02 18:00:35 -04001169 a.proguardFlags = extractedAARDir.Join(ctx, "proguard.txt")
Colin Crosscde55342024-03-27 14:11:51 -07001170 transitiveProguardFlags, transitiveUnconditionalExportedFlags := collectDepProguardSpecInfo(ctx)
Colin Cross40213022023-12-13 15:19:49 -08001171 android.SetProvider(ctx, ProguardSpecInfoProvider, ProguardSpecInfo{
Sam Delmerico95d70942023-08-02 18:00:35 -04001172 ProguardFlagsFiles: android.NewDepSet[android.Path](
1173 android.POSTORDER,
1174 android.Paths{a.proguardFlags},
Colin Crosscde55342024-03-27 14:11:51 -07001175 transitiveProguardFlags,
1176 ),
1177 UnconditionallyExportedProguardFlags: android.NewDepSet[android.Path](
1178 android.POSTORDER,
Sam Delmerico95d70942023-08-02 18:00:35 -04001179 nil,
Colin Crosscde55342024-03-27 14:11:51 -07001180 transitiveUnconditionalExportedFlags,
Sam Delmerico95d70942023-08-02 18:00:35 -04001181 ),
1182 })
Colin Crossfabb6082018-02-20 17:22:23 -08001183
1184 ctx.Build(pctx, android.BuildParams{
1185 Rule: unzipAAR,
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001186 Input: a.aarPath,
Colin Cross9055e212024-03-23 04:43:41 +00001187 Outputs: android.WritablePaths{classpathFile, a.proguardFlags, a.manifest, a.assetsPackage, a.rTxt},
Colin Crossfabb6082018-02-20 17:22:23 -08001188 Description: "unzip AAR",
1189 Args: map[string]string{
Colin Cross205e9112020-08-06 13:20:17 -07001190 "outDir": extractedAARDir.String(),
Colin Cross9055e212024-03-23 04:43:41 +00001191 "combinedClassesJar": classpathFile.String(),
Michael Rosenfeld5ad15572021-12-03 13:25:10 -08001192 "assetsPackage": a.assetsPackage.String(),
Colin Crossfabb6082018-02-20 17:22:23 -08001193 },
1194 })
1195
Colin Crossa0ba2f52019-06-22 12:59:27 -07001196 // Always set --pseudo-localize, it will be stripped out later for release
1197 // builds that don't want it.
1198 compileFlags := []string{"--pseudo-localize"}
Colin Crossfabb6082018-02-20 17:22:23 -08001199 compiledResDir := android.PathForModuleOut(ctx, "flat-res")
Colin Crossfabb6082018-02-20 17:22:23 -08001200 flata := compiledResDir.Join(ctx, "gen_res.flata")
Saeid Farivar Asanjanf0436962020-10-05 19:09:09 +00001201 aapt2CompileZip(ctx, flata, a.aarPath, "res", compileFlags)
Colin Crossfabb6082018-02-20 17:22:23 -08001202
1203 a.exportPackage = android.PathForModuleOut(ctx, "package-res.apk")
Colin Crossfabb6082018-02-20 17:22:23 -08001204 proguardOptionsFile := android.PathForModuleGen(ctx, "proguard.options")
Colin Crossbb77d8e2024-02-15 14:43:47 -08001205 aaptRTxt := android.PathForModuleOut(ctx, "R.txt")
Colin Cross66f78822018-05-02 12:58:28 -07001206 a.extraAaptPackagesFile = android.PathForModuleOut(ctx, "extra_packages")
Colin Crossfabb6082018-02-20 17:22:23 -08001207
1208 var linkDeps android.Paths
1209
1210 linkFlags := []string{
1211 "--static-lib",
Colin Cross4eae06d2023-06-20 22:40:02 -07001212 "--merge-only",
Colin Crossfabb6082018-02-20 17:22:23 -08001213 "--auto-add-overlay",
Colin Cross7c4dc5d2024-02-13 14:29:45 -08001214 "--no-static-lib-packages",
Colin Crossfabb6082018-02-20 17:22:23 -08001215 }
1216
Colin Cross10f7c4a2018-05-23 10:59:28 -07001217 linkFlags = append(linkFlags, "--manifest "+a.manifest.String())
1218 linkDeps = append(linkDeps, a.manifest)
Colin Crossfabb6082018-02-20 17:22:23 -08001219
Colin Cross8676c8c2023-10-12 15:58:57 -07001220 staticResourcesNodesDepSet, sharedResourcesNodesDepSet, staticRRODirsDepSet, staticManifestsDepSet, sharedLibs, libFlags :=
Jiakai Zhang36937082024-04-15 11:15:50 +00001221 aaptLibs(ctx, android.SdkContext(a), nil, nil)
Colin Cross31656952018-05-24 16:11:20 -07001222
Colin Cross8676c8c2023-10-12 15:58:57 -07001223 _ = sharedResourcesNodesDepSet
Colin Crossab8d1382023-07-14 17:23:41 +00001224 _ = staticRRODirsDepSet
Colin Cross8676c8c2023-10-12 15:58:57 -07001225
Colin Crossab8d1382023-07-14 17:23:41 +00001226 staticDeps := transitiveAarDeps(staticResourcesNodesDepSet.ToList())
Colin Crossfabb6082018-02-20 17:22:23 -08001227
Colin Crossab8d1382023-07-14 17:23:41 +00001228 linkDeps = append(linkDeps, sharedLibs...)
Colin Cross4eae06d2023-06-20 22:40:02 -07001229 linkDeps = append(linkDeps, staticDeps.resPackages()...)
Colin Crossa97c5d32018-03-28 14:58:31 -07001230 linkFlags = append(linkFlags, libFlags...)
Colin Crossfabb6082018-02-20 17:22:23 -08001231
Colin Cross4eae06d2023-06-20 22:40:02 -07001232 overlayRes := android.Paths{flata}
1233
1234 // Treat static library dependencies of static libraries as imports.
1235 transitiveStaticLibs := staticDeps.resPackages()
1236 linkDeps = append(linkDeps, transitiveStaticLibs...)
1237 for _, staticLib := range transitiveStaticLibs {
1238 linkFlags = append(linkFlags, "-I "+staticLib.String())
1239 }
Colin Crossfabb6082018-02-20 17:22:23 -08001240
Colin Crossab8d1382023-07-14 17:23:41 +00001241 transitiveAssets := android.ReverseSliceInPlace(staticDeps.assets())
Colin Crossbb77d8e2024-02-15 14:43:47 -08001242 aapt2Link(ctx, a.exportPackage, nil, proguardOptionsFile, aaptRTxt,
Jihoon Kang84b25892023-12-01 22:01:06 +00001243 linkFlags, linkDeps, nil, overlayRes, transitiveAssets, nil, nil)
Colin Crossfabb6082018-02-20 17:22:23 -08001244
Colin Cross4eae06d2023-06-20 22:40:02 -07001245 a.rJar = android.PathForModuleOut(ctx, "busybox/R.jar")
Rico Winda2fa2632024-03-13 13:09:17 +01001246 resourceProcessorBusyBoxGenerateBinaryR(ctx, a.rTxt, a.manifest, a.rJar, nil, true, nil, false)
Colin Cross4eae06d2023-06-20 22:40:02 -07001247
Colin Crossf3b7bad2023-08-02 15:49:00 -07001248 aapt2ExtractExtraPackages(ctx, a.extraAaptPackagesFile, a.rJar)
1249
Colin Crossab8d1382023-07-14 17:23:41 +00001250 resourcesNodesDepSetBuilder := android.NewDepSetBuilder[*resourcesNode](android.TOPOLOGICAL)
1251 resourcesNodesDepSetBuilder.Direct(&resourcesNode{
1252 resPackage: a.exportPackage,
1253 manifest: a.manifest,
Colin Cross4eae06d2023-06-20 22:40:02 -07001254 rTxt: a.rTxt,
1255 rJar: a.rJar,
Colin Crossab8d1382023-07-14 17:23:41 +00001256 assets: android.OptionalPathForPath(a.assetsPackage),
Colin Cross4eae06d2023-06-20 22:40:02 -07001257
1258 usedResourceProcessor: true,
Colin Crossab8d1382023-07-14 17:23:41 +00001259 })
1260 resourcesNodesDepSetBuilder.Transitive(staticResourcesNodesDepSet)
1261 a.resourcesNodesDepSet = resourcesNodesDepSetBuilder.Build()
1262
1263 manifestDepSetBuilder := android.NewDepSetBuilder[android.Path](android.TOPOLOGICAL).Direct(a.manifest)
Colin Cross9055e212024-03-23 04:43:41 +00001264 manifestDepSetBuilder.Transitive(staticManifestsDepSet)
Colin Crossab8d1382023-07-14 17:23:41 +00001265 a.manifestsDepSet = manifestDepSetBuilder.Build()
Michael Rosenfeld5ad15572021-12-03 13:25:10 -08001266
Colin Cross312634e2023-11-21 15:13:56 -08001267 transitiveAaptResourcePackages := staticDeps.resPackages().Strings()
1268 transitiveAaptResourcePackages = slices.DeleteFunc(transitiveAaptResourcePackages, func(p string) bool {
1269 return p == a.exportPackage.String()
1270 })
1271 transitiveAaptResourcePackagesFile := android.PathForModuleOut(ctx, "transitive-res-packages")
1272 android.WriteFileRule(ctx, transitiveAaptResourcePackagesFile, strings.Join(transitiveAaptResourcePackages, "\n"))
1273 a.transitiveAaptResourcePackagesFile = transitiveAaptResourcePackagesFile
Colin Cross4eae06d2023-06-20 22:40:02 -07001274
Sam Delmerico9f9c0a22022-11-29 11:19:37 -05001275 a.collectTransitiveHeaderJars(ctx)
Colin Cross9055e212024-03-23 04:43:41 +00001276
1277 a.classLoaderContexts = a.usesLibrary.classLoaderContextForUsesLibDeps(ctx)
1278
1279 var staticJars android.Paths
1280 var staticHeaderJars android.Paths
1281 ctx.VisitDirectDeps(func(module android.Module) {
1282 if dep, ok := android.OtherModuleProvider(ctx, module, JavaInfoProvider); ok {
1283 tag := ctx.OtherModuleDependencyTag(module)
1284 switch tag {
1285 case staticLibTag:
1286 staticJars = append(staticJars, dep.ImplementationJars...)
1287 staticHeaderJars = append(staticHeaderJars, dep.HeaderJars...)
1288 }
1289 }
1290 addCLCFromDep(ctx, module, a.classLoaderContexts)
Jiakai Zhang36937082024-04-15 11:15:50 +00001291 addMissingOptionalUsesLibsFromDep(ctx, module, &a.usesLibrary)
Colin Cross9055e212024-03-23 04:43:41 +00001292 })
1293
1294 if len(staticJars) > 0 {
1295 combineJars := append(android.Paths{classpathFile}, staticJars...)
1296 a.implementationJarFile = android.PathForModuleOut(ctx, "combined", ctx.ModuleName()+".jar")
1297 TransformJarsToJar(ctx, a.implementationJarFile, "combine", combineJars, android.OptionalPath{}, false, nil, nil)
1298 } else {
1299 a.implementationJarFile = classpathFile
1300 }
1301
1302 if len(staticHeaderJars) > 0 {
1303 combineJars := append(android.Paths{classpathFile}, staticHeaderJars...)
1304 a.headerJarFile = android.PathForModuleOut(ctx, "turbine-combined", ctx.ModuleName()+".jar")
1305 TransformJarsToJar(ctx, a.headerJarFile, "combine header jars", combineJars, android.OptionalPath{}, false, nil, nil)
1306 } else {
1307 a.headerJarFile = classpathFile
1308 }
1309
Colin Cross40213022023-12-13 15:19:49 -08001310 android.SetProvider(ctx, JavaInfoProvider, JavaInfo{
Colin Cross9055e212024-03-23 04:43:41 +00001311 HeaderJars: android.PathsIfNonNil(a.headerJarFile),
Sam Delmerico9f9c0a22022-11-29 11:19:37 -05001312 TransitiveLibsHeaderJars: a.transitiveLibsHeaderJars,
1313 TransitiveStaticLibsHeaderJars: a.transitiveStaticLibsHeaderJars,
Colin Cross9055e212024-03-23 04:43:41 +00001314 ImplementationAndResourcesJars: android.PathsIfNonNil(a.implementationJarFile),
1315 ImplementationJars: android.PathsIfNonNil(a.implementationJarFile),
Jihoon Kangfe914ed2024-02-12 22:49:21 +00001316 StubsLinkType: Implementation,
Joe Onorato6fe59eb2023-07-16 13:20:33 -07001317 // TransitiveAconfigFiles: // TODO(b/289117800): LOCAL_ACONFIG_FILES for prebuilts
Colin Crossdcf71b22021-02-01 13:59:03 -08001318 })
Sam Delmerico82602492022-06-10 17:05:42 +00001319
1320 if proptools.Bool(a.properties.Extract_jni) {
1321 for _, t := range ctx.MultiTargets() {
1322 arch := t.Arch.Abi[0]
1323 path := android.PathForModuleOut(ctx, arch+"_jni.zip")
1324 a.jniPackages = append(a.jniPackages, path)
1325
1326 outDir := android.PathForModuleOut(ctx, "aarForJni")
1327 aarPath := android.PathForModuleSrc(ctx, a.properties.Aars[0])
1328 ctx.Build(pctx, android.BuildParams{
1329 Rule: extractJNI,
1330 Input: aarPath,
1331 Outputs: android.WritablePaths{path},
1332 Description: "extract JNI from AAR",
1333 Args: map[string]string{
1334 "outDir": outDir.String(),
1335 "archString": arch,
1336 },
1337 })
1338 }
Sam Delmerico82602492022-06-10 17:05:42 +00001339 }
Colin Crosse8eeec92023-12-14 14:50:05 -08001340
Colin Cross40213022023-12-13 15:19:49 -08001341 android.SetProvider(ctx, JniPackageProvider, JniPackageInfo{
Colin Crosse8eeec92023-12-14 14:50:05 -08001342 JniPackages: a.jniPackages,
1343 })
LaMont Jonesafe7baf2024-01-09 22:47:39 +00001344 android.CollectDependencyAconfigFiles(ctx, &a.mergedAconfigFiles)
Colin Crossdcf71b22021-02-01 13:59:03 -08001345}
Colin Crossfabb6082018-02-20 17:22:23 -08001346
1347func (a *AARImport) HeaderJars() android.Paths {
Colin Cross9055e212024-03-23 04:43:41 +00001348 return android.Paths{a.headerJarFile}
Colin Crossfabb6082018-02-20 17:22:23 -08001349}
1350
Colin Cross331a1212018-08-15 20:40:52 -07001351func (a *AARImport) ImplementationAndResourcesJars() android.Paths {
Colin Cross9055e212024-03-23 04:43:41 +00001352 return android.Paths{a.implementationJarFile}
Colin Cross331a1212018-08-15 20:40:52 -07001353}
1354
Colin Cross9055e212024-03-23 04:43:41 +00001355func (a *AARImport) DexJarBuildPath(ctx android.ModuleErrorfContext) OptionalDexJarPath {
1356 return OptionalDexJarPath{}
Colin Crossf24a22a2019-01-31 14:12:44 -08001357}
1358
Ulya Trafimovich9f3052c2020-06-09 14:31:19 +01001359func (a *AARImport) DexJarInstallPath() android.Path {
1360 return nil
1361}
1362
Ulya Trafimovichb23d28c2020-10-08 12:53:58 +01001363func (a *AARImport) ClassLoaderContexts() dexpreopt.ClassLoaderContextMap {
Colin Cross9055e212024-03-23 04:43:41 +00001364 return a.classLoaderContexts
Jiyong Park1be96912018-05-28 18:02:19 +09001365}
1366
Colin Cross9055e212024-03-23 04:43:41 +00001367var _ UsesLibraryDependency = (*AARImport)(nil)
1368
Jiyong Park45bf82e2020-12-15 22:29:02 +09001369var _ android.ApexModule = (*AARImport)(nil)
1370
1371// Implements android.ApexModule
Jooyung Hanacc7bbe2020-05-20 09:06:00 +09001372func (a *AARImport) DepIsInSameApex(ctx android.BaseModuleContext, dep android.Module) bool {
1373 return a.depIsInSameApex(ctx, dep)
1374}
1375
Jiyong Park45bf82e2020-12-15 22:29:02 +09001376// Implements android.ApexModule
Colin Cross9055e212024-03-23 04:43:41 +00001377func (a *AARImport) ShouldSupportSdkVersion(ctx android.BaseModuleContext,
Dan Albertc8060532020-07-22 22:32:17 -07001378 sdkVersion android.ApiLevel) error {
Jooyung Han749dc692020-04-15 11:03:39 +09001379 return nil
1380}
1381
Sam Delmericoaf8bb702022-07-25 15:39:32 -04001382var _ android.PrebuiltInterface = (*AARImport)(nil)
Colin Crossfabb6082018-02-20 17:22:23 -08001383
Jiakai Zhangf98da192024-04-15 11:15:41 +00001384func (a *AARImport) UsesLibrary() *usesLibrary {
1385 return &a.usesLibrary
1386}
1387
1388var _ ModuleWithUsesLibrary = (*AARImport)(nil)
1389
Colin Cross1b16b0e2019-02-12 14:41:32 -08001390// android_library_import imports an `.aar` file into the build graph as if it was built with android_library.
1391//
1392// This module is not suitable for installing on a device, but can be used as a `static_libs` dependency of
1393// an android_app module.
Colin Crossfabb6082018-02-20 17:22:23 -08001394func AARImportFactory() android.Module {
1395 module := &AARImport{}
1396
Colin Cross9055e212024-03-23 04:43:41 +00001397 module.AddProperties(
1398 &module.properties,
1399 &module.usesLibrary.usesLibraryProperties,
1400 )
Colin Crossfabb6082018-02-20 17:22:23 -08001401
1402 android.InitPrebuiltModule(module, &module.properties.Aars)
Jooyung Hanacc7bbe2020-05-20 09:06:00 +09001403 android.InitApexModule(module)
Sam Delmerico82602492022-06-10 17:05:42 +00001404 InitJavaModuleMultiTargets(module, android.DeviceSupported)
Colin Crossfabb6082018-02-20 17:22:23 -08001405 return module
1406}