blob: 5af89b9c8b39c3a20bb28dc524365df15c100e69 [file] [log] [blame]
Colin Cross30e076a2015-04-13 13:58:27 -07001// Copyright 2015 Google Inc. All rights reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package java
16
17// This file contains the module types for compiling Android apps.
18
19import (
Jaewoong Junga5e5abc2019-04-26 14:31:50 -070020 "path/filepath"
21 "reflect"
Jaewoong Jung5b425e22019-06-17 17:40:56 -070022 "sort"
Sasha Smundak4de27a52020-04-23 09:49:59 -070023 "strconv"
Jaewoong Junga5e5abc2019-04-26 14:31:50 -070024 "strings"
Colin Cross30e076a2015-04-13 13:58:27 -070025
Colin Cross50ddcc42019-05-16 12:28:22 -070026 "github.com/google/blueprint"
27 "github.com/google/blueprint/proptools"
28
Colin Cross635c3b02016-05-18 15:37:25 -070029 "android/soong/android"
Colin Crossa4f08812018-10-02 22:03:40 -070030 "android/soong/cc"
Colin Cross303e21f2018-08-07 16:49:25 -070031 "android/soong/tradefed"
Colin Cross30e076a2015-04-13 13:58:27 -070032)
33
Jaewoong Jung3e18b192019-06-11 12:25:34 -070034var supportedDpis = []string{"ldpi", "mdpi", "hdpi", "xhdpi", "xxhdpi", "xxxhdpi"}
Jaewoong Junga5e5abc2019-04-26 14:31:50 -070035
Colin Cross3bc7ffa2017-11-22 16:19:37 -080036func init() {
Paul Duffinf9b1da02019-12-18 19:51:55 +000037 RegisterAppBuildComponents(android.InitRegistrationContext)
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -070038
39 initAndroidAppImportVariantGroupTypes()
Colin Cross3bc7ffa2017-11-22 16:19:37 -080040}
41
Paul Duffinf9b1da02019-12-18 19:51:55 +000042func RegisterAppBuildComponents(ctx android.RegistrationContext) {
43 ctx.RegisterModuleType("android_app", AndroidAppFactory)
44 ctx.RegisterModuleType("android_test", AndroidTestFactory)
45 ctx.RegisterModuleType("android_test_helper_app", AndroidTestHelperAppFactory)
46 ctx.RegisterModuleType("android_app_certificate", AndroidAppCertificateFactory)
47 ctx.RegisterModuleType("override_android_app", OverrideAndroidAppModuleFactory)
48 ctx.RegisterModuleType("override_android_test", OverrideAndroidTestModuleFactory)
Roshan Piusb8307962020-04-27 09:42:27 -070049 ctx.RegisterModuleType("override_runtime_resource_overlay", OverrideRuntimeResourceOverlayModuleFactory)
Paul Duffinf9b1da02019-12-18 19:51:55 +000050 ctx.RegisterModuleType("android_app_import", AndroidAppImportFactory)
51 ctx.RegisterModuleType("android_test_import", AndroidTestImportFactory)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -080052 ctx.RegisterModuleType("runtime_resource_overlay", RuntimeResourceOverlayFactory)
Sasha Smundak4de27a52020-04-23 09:49:59 -070053 ctx.RegisterModuleType("android_app_set", AndroidApkSetFactory)
54}
55
56type AndroidAppSetProperties struct {
57 // APK Set path
58 Set *string
59
60 // Specifies that this app should be installed to the priv-app directory,
61 // where the system will grant it additional privileges not available to
62 // normal apps.
63 Privileged *bool
64
65 // APKs in this set use prerelease SDK version
66 Prerelease *bool
67
68 // Names of modules to be overridden. Listed modules can only be other apps
69 // (in Make or Soong).
70 Overrides []string
71}
72
73type AndroidAppSet struct {
74 android.ModuleBase
75 android.DefaultableModuleBase
76 prebuilt android.Prebuilt
77
78 properties AndroidAppSetProperties
79 packedOutput android.WritablePath
80 masterFile string
Jaewoong Jung8bec0262020-06-29 19:18:44 -070081 apkcertsFile android.ModuleOutPath
Sasha Smundak4de27a52020-04-23 09:49:59 -070082}
83
84func (as *AndroidAppSet) Name() string {
85 return as.prebuilt.Name(as.ModuleBase.Name())
86}
87
88func (as *AndroidAppSet) IsInstallable() bool {
89 return true
90}
91
92func (as *AndroidAppSet) Prebuilt() *android.Prebuilt {
93 return &as.prebuilt
94}
95
96func (as *AndroidAppSet) Privileged() bool {
97 return Bool(as.properties.Privileged)
98}
99
Sasha Smundakc4f0ff12020-05-27 16:36:07 -0700100func (as *AndroidAppSet) OutputFile() android.Path {
101 return as.packedOutput
102}
103
104func (as *AndroidAppSet) MasterFile() string {
105 return as.masterFile
106}
107
Colin Cross7e2b36c2020-07-09 19:05:35 -0700108func (as *AndroidAppSet) APKCertsFile() android.Path {
109 return as.apkcertsFile
110}
111
Jaewoong Jung8cf307e2020-05-14 14:15:24 -0700112var TargetCpuAbi = map[string]string{
Sasha Smundak4de27a52020-04-23 09:49:59 -0700113 "arm": "ARMEABI_V7A",
114 "arm64": "ARM64_V8A",
115 "x86": "X86",
116 "x86_64": "X86_64",
117}
118
Jaewoong Jung8cf307e2020-05-14 14:15:24 -0700119func SupportedAbis(ctx android.ModuleContext) []string {
Jaewoong Jung829b7132020-06-10 12:23:32 -0700120 abiName := func(targetIdx int, deviceArch string) string {
Jaewoong Jung8cf307e2020-05-14 14:15:24 -0700121 if abi, found := TargetCpuAbi[deviceArch]; found {
Sasha Smundak4de27a52020-04-23 09:49:59 -0700122 return abi
123 }
Jaewoong Jung829b7132020-06-10 12:23:32 -0700124 ctx.ModuleErrorf("Target %d has invalid Arch: %s", targetIdx, deviceArch)
Sasha Smundak4de27a52020-04-23 09:49:59 -0700125 return "BAD_ABI"
126 }
127
Jaewoong Jung829b7132020-06-10 12:23:32 -0700128 var result []string
129 for i, target := range ctx.Config().Targets[android.Android] {
130 result = append(result, abiName(i, target.Arch.ArchType.String()))
Sasha Smundak4de27a52020-04-23 09:49:59 -0700131 }
132 return result
133}
134
135func (as *AndroidAppSet) GenerateAndroidBuildActions(ctx android.ModuleContext) {
Sasha Smundakc4f0ff12020-05-27 16:36:07 -0700136 as.packedOutput = android.PathForModuleOut(ctx, ctx.ModuleName()+".zip")
Jaewoong Jung8bec0262020-06-29 19:18:44 -0700137 as.apkcertsFile = android.PathForModuleOut(ctx, "apkcerts.txt")
Sasha Smundak4de27a52020-04-23 09:49:59 -0700138 // We are assuming here that the master file in the APK
139 // set has `.apk` suffix. If it doesn't the build will fail.
140 // APK sets containing APEX files are handled elsewhere.
Sasha Smundak854c14f2020-06-16 10:28:22 -0700141 as.masterFile = as.BaseModuleName() + ".apk"
Sasha Smundak4de27a52020-04-23 09:49:59 -0700142 screenDensities := "all"
143 if dpis := ctx.Config().ProductAAPTPrebuiltDPI(); len(dpis) > 0 {
144 screenDensities = strings.ToUpper(strings.Join(dpis, ","))
145 }
146 // TODO(asmundak): handle locales.
147 // TODO(asmundak): do we support device features
148 ctx.Build(pctx,
149 android.BuildParams{
Jaewoong Jung8bec0262020-06-29 19:18:44 -0700150 Rule: extractMatchingApks,
151 Description: "Extract APKs from APK set",
152 Output: as.packedOutput,
153 ImplicitOutput: as.apkcertsFile,
154 Inputs: android.Paths{as.prebuilt.SingleSourcePath(ctx)},
Sasha Smundak4de27a52020-04-23 09:49:59 -0700155 Args: map[string]string{
Jaewoong Jung8cf307e2020-05-14 14:15:24 -0700156 "abis": strings.Join(SupportedAbis(ctx), ","),
Sasha Smundak4de27a52020-04-23 09:49:59 -0700157 "allow-prereleased": strconv.FormatBool(proptools.Bool(as.properties.Prerelease)),
158 "screen-densities": screenDensities,
159 "sdk-version": ctx.Config().PlatformSdkVersion(),
Sasha Smundak3c904e82020-06-22 16:53:33 -0700160 "stem": as.BaseModuleName(),
Jaewoong Jung8bec0262020-06-29 19:18:44 -0700161 "apkcerts": as.apkcertsFile.String(),
162 "partition": as.PartitionTag(ctx.DeviceConfig()),
Sasha Smundak4de27a52020-04-23 09:49:59 -0700163 },
164 })
Sasha Smundak4de27a52020-04-23 09:49:59 -0700165}
166
167// android_app_set extracts a set of APKs based on the target device
168// configuration and installs this set as "split APKs".
Sasha Smundakc4f0ff12020-05-27 16:36:07 -0700169// The extracted set always contains 'master' APK whose name is
170// _module_name_.apk and every split APK matching target device.
171// The extraction of the density-specific splits depends on
172// PRODUCT_AAPT_PREBUILT_DPI variable. If present (its value should
173// be a list density names: LDPI, MDPI, HDPI, etc.), only listed
174// splits will be extracted. Otherwise all density-specific splits
175// will be extracted.
Sasha Smundak4de27a52020-04-23 09:49:59 -0700176func AndroidApkSetFactory() android.Module {
177 module := &AndroidAppSet{}
178 module.AddProperties(&module.properties)
179 InitJavaModule(module, android.DeviceSupported)
180 android.InitSingleSourcePrebuiltModule(module, &module.properties, "Set")
181 return module
Paul Duffinf9b1da02019-12-18 19:51:55 +0000182}
183
Colin Cross30e076a2015-04-13 13:58:27 -0700184// AndroidManifest.xml merging
185// package splits
186
Colin Crossfabb6082018-02-20 17:22:23 -0800187type appProperties struct {
Colin Crossbd01e2a2018-10-04 15:21:03 -0700188 // Names of extra android_app_certificate modules to sign the apk with in the form ":module".
Colin Cross7d5136f2015-05-11 13:39:40 -0700189 Additional_certificates []string
190
191 // If set, create package-export.apk, which other packages can
192 // use to get PRODUCT-agnostic resource data like IDs and type definitions.
Nan Zhangea568a42017-11-08 21:20:04 -0800193 Export_package_resources *bool
Colin Cross7d5136f2015-05-11 13:39:40 -0700194
Colin Cross16056062017-12-13 22:46:28 -0800195 // Specifies that this app should be installed to the priv-app directory,
196 // where the system will grant it additional privileges not available to
197 // normal apps.
198 Privileged *bool
Colin Crossa97c5d32018-03-28 14:58:31 -0700199
200 // list of resource labels to generate individual resource packages
201 Package_splits []string
Jason Monkd4122be2018-08-10 09:33:36 -0400202
203 // Names of modules to be overridden. Listed modules can only be other binaries
204 // (in Make or Soong).
205 // This does not completely prevent installation of the overridden binaries, but if both
206 // binaries would be installed by default (in PRODUCT_PACKAGES) the other binary will be removed
207 // from PRODUCT_PACKAGES.
208 Overrides []string
Colin Crossa4f08812018-10-02 22:03:40 -0700209
210 // list of native libraries that will be provided in or alongside the resulting jar
211 Jni_libs []string `android:"arch_variant"`
212
Colin Cross76583a42020-05-06 17:51:39 -0700213 // if true, use JNI libraries that link against platform APIs even if this module sets
Colin Crossee87c602020-02-19 16:57:15 -0800214 // sdk_version.
215 Jni_uses_platform_apis *bool
216
Colin Cross76583a42020-05-06 17:51:39 -0700217 // if true, use JNI libraries that link against SDK APIs even if this module does not set
218 // sdk_version.
219 Jni_uses_sdk_apis *bool
220
Jaewoong Jungbc625cd2019-05-06 15:48:44 -0700221 // STL library to use for JNI libraries.
222 Stl *string `android:"arch_variant"`
223
Colin Crosse4246ab2019-02-05 21:55:21 -0800224 // Store native libraries uncompressed in the APK and set the android:extractNativeLibs="false" manifest
225 // flag so that they are used from inside the APK at runtime. Defaults to true for android_test modules unless
Jiyong Park52cd06f2019-11-11 10:14:32 +0900226 // sdk_version or min_sdk_version is set to a version that doesn't support it (<23), defaults to true for
227 // android_app modules that are embedded to APEXes, defaults to false for other module types where the native
228 // libraries are generally preinstalled outside the APK.
Colin Crosse4246ab2019-02-05 21:55:21 -0800229 Use_embedded_native_libs *bool
Colin Cross46abdad2019-02-07 13:07:08 -0800230
231 // Store dex files uncompressed in the APK and set the android:useEmbeddedDex="true" manifest attribute so that
232 // they are used from inside the APK at runtime.
233 Use_embedded_dex *bool
Colin Cross47fa9d32019-03-26 10:51:39 -0700234
235 // Forces native libraries to always be packaged into the APK,
236 // Use_embedded_native_libs still selects whether they are stored uncompressed and aligned or compressed.
237 // True for android_test* modules.
238 AlwaysPackageNativeLibs bool `blueprint:"mutated"`
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700239
240 // If set, find and merge all NOTICE files that this module and its dependencies have and store
241 // it in the APK as an asset.
242 Embed_notices *bool
Jaewoong Jung37ca4a12020-03-26 14:01:48 -0700243
244 // cc.Coverage related properties
245 PreventInstall bool `blueprint:"mutated"`
246 HideFromMake bool `blueprint:"mutated"`
247 IsCoverageVariant bool `blueprint:"mutated"`
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100248
249 // Whether this app is considered mainline updatable or not. When set to true, this will enforce
Artur Satayev11962102020-04-16 13:43:02 +0100250 // additional rules to make sure an app can safely be updated. Default is false.
251 // Prefer using other specific properties if build behaviour must be changed; avoid using this
252 // flag for anything but neverallow rules (unless the behaviour change is invisible to owners).
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100253 Updatable *bool
Colin Cross7d5136f2015-05-11 13:39:40 -0700254}
255
Jaewoong Jung525443a2019-02-28 15:35:54 -0800256// android_app properties that can be overridden by override_android_app
257type overridableAppProperties struct {
258 // The name of a certificate in the default certificate directory, blank to use the default product certificate,
259 // or an android_app_certificate module name in the form ":module".
260 Certificate *string
Jaewoong Jung6f373f62019-03-13 10:13:24 -0700261
Liz Kammer70dd74d2020-05-07 13:24:05 -0700262 // Name of the signing certificate lineage file.
263 Lineage *string
264
Jaewoong Jung6f373f62019-03-13 10:13:24 -0700265 // the package name of this app. The package name in the manifest file is used if one was not given.
266 Package_name *string
Baligh Uddin5b16dfb2020-02-11 17:27:19 -0800267
268 // the logging parent of this app.
269 Logging_parent *string
Jaewoong Jung525443a2019-02-28 15:35:54 -0800270}
271
Roshan Piusb8307962020-04-27 09:42:27 -0700272// runtime_resource_overlay properties that can be overridden by override_runtime_resource_overlay
273type OverridableRuntimeResourceOverlayProperties struct {
274 // the package name of this app. The package name in the manifest file is used if one was not given.
275 Package_name *string
276
277 // the target package name of this overlay app. The target package name in the manifest file is used if one was not given.
278 Target_package_name *string
279}
280
Colin Cross30e076a2015-04-13 13:58:27 -0700281type AndroidApp struct {
Colin Crossa97c5d32018-03-28 14:58:31 -0700282 Library
283 aapt
Jaewoong Jung525443a2019-02-28 15:35:54 -0800284 android.OverridableModuleBase
Colin Crossa97c5d32018-03-28 14:58:31 -0700285
Colin Cross50ddcc42019-05-16 12:28:22 -0700286 usesLibrary usesLibrary
287
Jiyong Parkc00cbd92018-10-30 21:20:05 +0900288 certificate Certificate
Colin Cross30e076a2015-04-13 13:58:27 -0700289
Colin Crossfabb6082018-02-20 17:22:23 -0800290 appProperties appProperties
Colin Crossae5caf52018-05-22 11:11:52 -0700291
Jaewoong Jung525443a2019-02-28 15:35:54 -0800292 overridableAppProperties overridableAppProperties
293
Jaewoong Jung37ca4a12020-03-26 14:01:48 -0700294 installJniLibs []jniLib
295 jniCoverageOutputs android.Paths
Colin Crossf6237212018-10-29 23:14:58 -0700296
297 bundleFile android.Path
Jaewoong Jung9d22a912019-01-23 16:27:47 -0800298
299 // the install APK name is normally the same as the module name, but can be overridden with PRODUCT_PACKAGE_NAME_OVERRIDES.
300 installApkName string
Jaewoong Jung4102e5d2019-02-27 16:26:28 -0800301
Colin Cross70dda7e2019-10-01 22:05:35 -0700302 installDir android.InstallPath
Jaewoong Jung0949f312019-09-11 10:25:18 -0700303
Jaewoong Jung7dd4ae22019-09-27 17:13:15 -0700304 onDeviceDir string
305
Jaewoong Jung4102e5d2019-02-27 16:26:28 -0800306 additionalAaptFlags []string
Jaewoong Jung98772792019-07-01 17:15:13 -0700307
308 noticeOutputs android.NoticeOutputs
Jiyong Parkaf8998c2020-02-28 16:51:07 +0900309
310 overriddenManifestPackageName string
Artur Satayevd9b503a2020-04-27 19:05:28 +0100311
312 android.ApexBundleDepsInfo
Colin Crosse1731a52017-12-14 11:22:55 -0800313}
314
Martin Stjernholm6d415272020-01-31 17:10:36 +0000315func (a *AndroidApp) IsInstallable() bool {
316 return Bool(a.properties.Installable)
317}
318
Colin Cross89c31582018-04-30 15:55:11 -0700319func (a *AndroidApp) ExportedProguardFlagFiles() android.Paths {
320 return nil
321}
322
Colin Cross66f78822018-05-02 12:58:28 -0700323func (a *AndroidApp) ExportedStaticPackages() android.Paths {
324 return nil
325}
326
Sundong Ahne1f05aa2019-08-27 13:55:42 +0900327func (a *AndroidApp) OutputFile() android.Path {
328 return a.outputFile
329}
330
Colin Cross503c1d02020-01-28 14:00:53 -0800331func (a *AndroidApp) Certificate() Certificate {
332 return a.certificate
333}
334
Jaewoong Jung37ca4a12020-03-26 14:01:48 -0700335func (a *AndroidApp) JniCoverageOutputs() android.Paths {
336 return a.jniCoverageOutputs
337}
338
Colin Crossa97c5d32018-03-28 14:58:31 -0700339var _ AndroidLibraryDependency = (*AndroidApp)(nil)
340
Jiyong Parkc00cbd92018-10-30 21:20:05 +0900341type Certificate struct {
Colin Cross503c1d02020-01-28 14:00:53 -0800342 Pem, Key android.Path
343 presigned bool
344}
345
Sasha Smundakc4f0ff12020-05-27 16:36:07 -0700346var PresignedCertificate = Certificate{presigned: true}
Colin Cross503c1d02020-01-28 14:00:53 -0800347
348func (c Certificate) AndroidMkString() string {
349 if c.presigned {
350 return "PRESIGNED"
351 } else {
352 return c.Pem.String()
353 }
Colin Cross30e076a2015-04-13 13:58:27 -0700354}
355
Colin Cross46c9b8b2017-06-22 16:51:17 -0700356func (a *AndroidApp) DepsMutator(ctx android.BottomUpMutatorContext) {
357 a.Module.deps(ctx)
Colin Crossa4f08812018-10-02 22:03:40 -0700358
Jiyong Park6a927c42020-01-21 02:03:43 +0900359 if String(a.appProperties.Stl) == "c++_shared" && !a.sdkVersion().specified() {
Jaewoong Jungbc625cd2019-05-06 15:48:44 -0700360 ctx.PropertyErrorf("stl", "sdk_version must be set in order to use c++_shared")
361 }
362
Paul Duffin250e6192019-06-07 10:44:37 +0100363 sdkDep := decodeSdkDep(ctx, sdkContext(a))
364 if sdkDep.hasFrameworkLibs() {
365 a.aapt.deps(ctx, sdkDep)
Colin Cross30e076a2015-04-13 13:58:27 -0700366 }
Colin Crossa4f08812018-10-02 22:03:40 -0700367
Colin Cross1dd9c442020-05-08 11:20:24 -0700368 usesSDK := a.sdkVersion().specified() && a.sdkVersion().kind != sdkCorePlatform
369
370 if usesSDK && Bool(a.appProperties.Jni_uses_sdk_apis) {
371 ctx.PropertyErrorf("jni_uses_sdk_apis",
372 "can only be set for modules that do not set sdk_version")
373 } else if !usesSDK && Bool(a.appProperties.Jni_uses_platform_apis) {
374 ctx.PropertyErrorf("jni_uses_platform_apis",
375 "can only be set for modules that set sdk_version")
376 }
377
Peter Collingbournead84f972019-12-17 16:46:18 -0800378 tag := &jniDependencyTag{}
Colin Crossa4f08812018-10-02 22:03:40 -0700379 for _, jniTarget := range ctx.MultiTargets() {
Colin Cross0f7d2ef2019-10-16 11:03:10 -0700380 variation := append(jniTarget.Variations(),
381 blueprint.Variation{Mutator: "link", Variation: "shared"})
Colin Cross01fd7cc2020-02-19 16:54:04 -0800382
383 // If the app builds against an Android SDK use the SDK variant of JNI dependencies
384 // unless jni_uses_platform_apis is set.
Colin Crosseb032962020-05-13 11:05:02 -0700385 // Don't require the SDK variant for apps that are shipped on vendor, etc., as they already
386 // have stable APIs through the VNDK.
387 if (usesSDK && !a.RequiresStableAPIs(ctx) &&
388 !Bool(a.appProperties.Jni_uses_platform_apis)) ||
Colin Cross76583a42020-05-06 17:51:39 -0700389 Bool(a.appProperties.Jni_uses_sdk_apis) {
Colin Cross01fd7cc2020-02-19 16:54:04 -0800390 variation = append(variation, blueprint.Variation{Mutator: "sdk", Variation: "sdk"})
391 }
Colin Crossa4f08812018-10-02 22:03:40 -0700392 ctx.AddFarVariationDependencies(variation, tag, a.appProperties.Jni_libs...)
393 }
Colin Cross50ddcc42019-05-16 12:28:22 -0700394
Paul Duffin250e6192019-06-07 10:44:37 +0100395 a.usesLibrary.deps(ctx, sdkDep.hasFrameworkLibs())
Jaewoong Jungb639a6a2019-05-10 15:16:29 -0700396}
Colin Crossbd01e2a2018-10-04 15:21:03 -0700397
Jaewoong Jungb639a6a2019-05-10 15:16:29 -0700398func (a *AndroidApp) OverridablePropertiesDepsMutator(ctx android.BottomUpMutatorContext) {
Jaewoong Jung2ad817c2019-01-18 14:27:16 -0800399 cert := android.SrcIsModule(a.getCertString(ctx))
Colin Crossbd01e2a2018-10-04 15:21:03 -0700400 if cert != "" {
401 ctx.AddDependency(ctx.Module(), certificateTag, cert)
402 }
403
404 for _, cert := range a.appProperties.Additional_certificates {
405 cert = android.SrcIsModule(cert)
406 if cert != "" {
407 ctx.AddDependency(ctx.Module(), certificateTag, cert)
408 } else {
409 ctx.PropertyErrorf("additional_certificates",
410 `must be names of android_app_certificate modules in the form ":module"`)
411 }
412 }
Colin Cross30e076a2015-04-13 13:58:27 -0700413}
414
Jeongik Cha538c0d02019-07-11 15:54:27 +0900415func (a *AndroidTestHelperApp) GenerateAndroidBuildActions(ctx android.ModuleContext) {
416 a.generateAndroidBuildActions(ctx)
417}
418
Colin Cross46c9b8b2017-06-22 16:51:17 -0700419func (a *AndroidApp) GenerateAndroidBuildActions(ctx android.ModuleContext) {
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100420 a.checkAppSdkVersions(ctx)
Colin Crossae5caf52018-05-22 11:11:52 -0700421 a.generateAndroidBuildActions(ctx)
422}
423
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100424func (a *AndroidApp) checkAppSdkVersions(ctx android.ModuleContext) {
Artur Satayev2b4b7bb2020-04-28 14:57:42 +0100425 if a.Updatable() {
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100426 if !a.sdkVersion().stable() {
427 ctx.PropertyErrorf("sdk_version", "Updatable apps must use stable SDKs, found %v", a.sdkVersion())
428 }
Artur Satayev11962102020-04-16 13:43:02 +0100429 if String(a.deviceProperties.Min_sdk_version) == "" {
430 ctx.PropertyErrorf("updatable", "updatable apps must set min_sdk_version.")
431 }
Jooyung Hanaf7f91f2020-04-29 14:01:06 +0900432 if minSdkVersion, err := a.minSdkVersion().effectiveVersion(ctx); err == nil {
433 a.checkJniLibsSdkVersion(ctx, minSdkVersion)
434 } else {
435 ctx.PropertyErrorf("min_sdk_version", "%s", err.Error())
436 }
Artur Satayeve5ac15a2020-04-08 19:09:30 +0100437 }
438
439 a.checkPlatformAPI(ctx)
440 a.checkSdkVersions(ctx)
441}
442
Jooyung Hanaf7f91f2020-04-29 14:01:06 +0900443// If an updatable APK sets min_sdk_version, min_sdk_vesion of JNI libs should match with it.
444// This check is enforced for "updatable" APKs (including APK-in-APEX).
445// b/155209650: until min_sdk_version is properly supported, use sdk_version instead.
446// because, sdk_version is overridden by min_sdk_version (if set as smaller)
447// and linkType is checked with dependencies so we can be sure that the whole dependency tree
448// will meet the requirements.
449func (a *AndroidApp) checkJniLibsSdkVersion(ctx android.ModuleContext, minSdkVersion sdkVersion) {
450 // It's enough to check direct JNI deps' sdk_version because all transitive deps from JNI deps are checked in cc.checkLinkType()
451 ctx.VisitDirectDeps(func(m android.Module) {
452 if !IsJniDepTag(ctx.OtherModuleDependencyTag(m)) {
453 return
454 }
455 dep, _ := m.(*cc.Module)
Jooyung Han9d2c0f72020-05-20 17:12:13 +0900456 // The domain of cc.sdk_version is "current" and <number>
457 // We can rely on sdkSpec to convert it to <number> so that "current" is handled
458 // properly regardless of sdk finalization.
459 jniSdkVersion, err := sdkSpecFrom(dep.SdkVersion()).effectiveVersion(ctx)
460 if err != nil || minSdkVersion < jniSdkVersion {
Jooyung Hanaf7f91f2020-04-29 14:01:06 +0900461 ctx.OtherModuleErrorf(dep, "sdk_version(%v) is higher than min_sdk_version(%v) of the containing android_app(%v)",
462 dep.SdkVersion(), minSdkVersion, ctx.ModuleName())
463 return
464 }
465
466 })
467}
468
Sasha Smundak6ad77252019-05-01 13:16:22 -0700469// Returns true if the native libraries should be stored in the APK uncompressed and the
Colin Crosse4246ab2019-02-05 21:55:21 -0800470// extractNativeLibs application flag should be set to false in the manifest.
Sasha Smundak6ad77252019-05-01 13:16:22 -0700471func (a *AndroidApp) useEmbeddedNativeLibs(ctx android.ModuleContext) bool {
Jiyong Park6a927c42020-01-21 02:03:43 +0900472 minSdkVersion, err := a.minSdkVersion().effectiveVersion(ctx)
Colin Crosse4246ab2019-02-05 21:55:21 -0800473 if err != nil {
474 ctx.PropertyErrorf("min_sdk_version", "invalid value %q: %s", a.minSdkVersion(), err)
475 }
476
Jiyong Park52cd06f2019-11-11 10:14:32 +0900477 return (minSdkVersion >= 23 && Bool(a.appProperties.Use_embedded_native_libs)) ||
478 !a.IsForPlatform()
Colin Crosse4246ab2019-02-05 21:55:21 -0800479}
480
Colin Cross43f08db2018-11-12 10:13:39 -0800481// Returns whether this module should have the dex file stored uncompressed in the APK.
482func (a *AndroidApp) shouldUncompressDex(ctx android.ModuleContext) bool {
Colin Cross46abdad2019-02-07 13:07:08 -0800483 if Bool(a.appProperties.Use_embedded_dex) {
484 return true
485 }
486
Colin Cross53a87f52019-06-25 13:35:30 -0700487 // Uncompress dex in APKs of privileged apps (even for unbundled builds, they may
488 // be preinstalled as prebuilts).
Jiyong Parkf7487312019-10-17 12:54:30 +0900489 if ctx.Config().UncompressPrivAppDex() && a.Privileged() {
Nicolas Geoffrayfa6e9ec2019-02-12 13:12:16 +0000490 return true
491 }
492
Colin Cross53a87f52019-06-25 13:35:30 -0700493 if ctx.Config().UnbundledBuild() {
494 return false
495 }
496
Jaewoong Jungacf18d72019-05-02 14:55:29 -0700497 return shouldUncompressDex(ctx, &a.dexpreopter)
Colin Cross5a0dcd52018-10-05 14:20:06 -0700498}
499
Jaewoong Jungbc625cd2019-05-06 15:48:44 -0700500func (a *AndroidApp) shouldEmbedJnis(ctx android.BaseModuleContext) bool {
501 return ctx.Config().UnbundledBuild() || Bool(a.appProperties.Use_embedded_native_libs) ||
Jiyong Park52cd06f2019-11-11 10:14:32 +0900502 !a.IsForPlatform() || a.appProperties.AlwaysPackageNativeLibs
Jaewoong Jungbc625cd2019-05-06 15:48:44 -0700503}
504
Jiyong Parkaf8998c2020-02-28 16:51:07 +0900505func (a *AndroidApp) OverriddenManifestPackageName() string {
506 return a.overriddenManifestPackageName
507}
508
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800509func (a *AndroidApp) aaptBuildActions(ctx android.ModuleContext) {
David Brazdild25060a2019-02-18 18:24:16 +0000510 a.aapt.usesNonSdkApis = Bool(a.Module.deviceProperties.Platform_apis)
511
Jaewoong Jungc27ab662019-05-30 15:51:14 -0700512 // Ask manifest_fixer to add or update the application element indicating this app has no code.
513 a.aapt.hasNoCode = !a.hasCode(ctx)
514
Jaewoong Jungde4c02f2019-01-22 11:19:56 -0800515 aaptLinkFlags := []string{}
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800516
Jaewoong Jungde4c02f2019-01-22 11:19:56 -0800517 // Add TARGET_AAPT_CHARACTERISTICS values to AAPT link flags if they exist and --product flags were not provided.
Jaewoong Jung3aff5782020-02-11 07:54:35 -0800518 hasProduct := android.PrefixInList(a.aaptProperties.Aaptflags, "--product")
Colin Crosse78dcd32018-04-19 15:25:19 -0700519 if !hasProduct && len(ctx.Config().ProductAAPTCharacteristics()) > 0 {
Jaewoong Jungde4c02f2019-01-22 11:19:56 -0800520 aaptLinkFlags = append(aaptLinkFlags, "--product", ctx.Config().ProductAAPTCharacteristics())
Colin Crosse78dcd32018-04-19 15:25:19 -0700521 }
522
Dan Willemsen72be5902018-10-24 20:24:57 -0700523 if !Bool(a.aaptProperties.Aapt_include_all_resources) {
524 // Product AAPT config
525 for _, aaptConfig := range ctx.Config().ProductAAPTConfig() {
Jaewoong Jungde4c02f2019-01-22 11:19:56 -0800526 aaptLinkFlags = append(aaptLinkFlags, "-c", aaptConfig)
Dan Willemsen72be5902018-10-24 20:24:57 -0700527 }
Colin Crosse78dcd32018-04-19 15:25:19 -0700528
Dan Willemsen72be5902018-10-24 20:24:57 -0700529 // Product AAPT preferred config
530 if len(ctx.Config().ProductAAPTPreferredConfig()) > 0 {
Jaewoong Jungde4c02f2019-01-22 11:19:56 -0800531 aaptLinkFlags = append(aaptLinkFlags, "--preferred-density", ctx.Config().ProductAAPTPreferredConfig())
Dan Willemsen72be5902018-10-24 20:24:57 -0700532 }
Colin Crosse78dcd32018-04-19 15:25:19 -0700533 }
534
Jiyong Park7f67f482019-01-05 12:57:48 +0900535 manifestPackageName, overridden := ctx.DeviceConfig().OverrideManifestPackageNameFor(ctx.ModuleName())
Jaewoong Jung6f373f62019-03-13 10:13:24 -0700536 if overridden || a.overridableAppProperties.Package_name != nil {
537 // The product override variable has a priority over the package_name property.
538 if !overridden {
539 manifestPackageName = *a.overridableAppProperties.Package_name
540 }
Jaewoong Jungde4c02f2019-01-22 11:19:56 -0800541 aaptLinkFlags = append(aaptLinkFlags, "--rename-manifest-package "+manifestPackageName)
Jiyong Parkaf8998c2020-02-28 16:51:07 +0900542 a.overriddenManifestPackageName = manifestPackageName
Jiyong Park7f67f482019-01-05 12:57:48 +0900543 }
544
Jaewoong Jung4102e5d2019-02-27 16:26:28 -0800545 aaptLinkFlags = append(aaptLinkFlags, a.additionalAaptFlags...)
546
Colin Crosse560c4a2019-03-19 16:03:11 -0700547 a.aapt.splitNames = a.appProperties.Package_splits
Colin Cross50ddcc42019-05-16 12:28:22 -0700548 a.aapt.sdkLibraries = a.exportedSdkLibs
Baligh Uddin5b16dfb2020-02-11 17:27:19 -0800549 a.aapt.LoggingParent = String(a.overridableAppProperties.Logging_parent)
Jaewoong Jungde4c02f2019-01-22 11:19:56 -0800550 a.aapt.buildActions(ctx, sdkContext(a), aaptLinkFlags...)
Colin Cross30e076a2015-04-13 13:58:27 -0700551
Colin Cross46c9b8b2017-06-22 16:51:17 -0700552 // apps manifests are handled by aapt, don't let Module see them
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700553 a.properties.Manifest = nil
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800554}
Colin Cross30e076a2015-04-13 13:58:27 -0700555
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800556func (a *AndroidApp) proguardBuildActions(ctx android.ModuleContext) {
Colin Cross89c31582018-04-30 15:55:11 -0700557 var staticLibProguardFlagFiles android.Paths
558 ctx.VisitDirectDeps(func(m android.Module) {
559 if lib, ok := m.(AndroidLibraryDependency); ok && ctx.OtherModuleDependencyTag(m) == staticLibTag {
560 staticLibProguardFlagFiles = append(staticLibProguardFlagFiles, lib.ExportedProguardFlagFiles()...)
561 }
562 })
563
564 staticLibProguardFlagFiles = android.FirstUniquePaths(staticLibProguardFlagFiles)
565
566 a.Module.extraProguardFlagFiles = append(a.Module.extraProguardFlagFiles, staticLibProguardFlagFiles...)
567 a.Module.extraProguardFlagFiles = append(a.Module.extraProguardFlagFiles, a.proguardOptionsFile)
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800568}
Colin Cross66dbc0b2017-12-28 12:23:20 -0800569
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800570func (a *AndroidApp) dexBuildActions(ctx android.ModuleContext) android.Path {
Colin Cross43f08db2018-11-12 10:13:39 -0800571
572 var installDir string
573 if ctx.ModuleName() == "framework-res" {
574 // framework-res.apk is installed as system/framework/framework-res.apk
575 installDir = "framework"
Jiyong Parkf7487312019-10-17 12:54:30 +0900576 } else if a.Privileged() {
Jaewoong Jung9d22a912019-01-23 16:27:47 -0800577 installDir = filepath.Join("priv-app", a.installApkName)
Colin Cross43f08db2018-11-12 10:13:39 -0800578 } else {
Jaewoong Jung9d22a912019-01-23 16:27:47 -0800579 installDir = filepath.Join("app", a.installApkName)
Colin Cross43f08db2018-11-12 10:13:39 -0800580 }
Jaewoong Jung9d22a912019-01-23 16:27:47 -0800581 a.dexpreopter.installPath = android.PathForModuleInstall(ctx, installDir, a.installApkName+".apk")
David Srbecky98c71222020-05-20 22:20:28 +0100582 if a.deviceProperties.Uncompress_dex == nil {
583 // If the value was not force-set by the user, use reasonable default based on the module.
584 a.deviceProperties.Uncompress_dex = proptools.BoolPtr(a.shouldUncompressDex(ctx))
585 }
586 a.dexpreopter.uncompressedDex = *a.deviceProperties.Uncompress_dex
Colin Cross50ddcc42019-05-16 12:28:22 -0700587 a.dexpreopter.enforceUsesLibs = a.usesLibrary.enforceUsesLibraries()
588 a.dexpreopter.usesLibs = a.usesLibrary.usesLibraryProperties.Uses_libs
589 a.dexpreopter.optionalUsesLibs = a.usesLibrary.presentOptionalUsesLibs(ctx)
590 a.dexpreopter.libraryPaths = a.usesLibrary.usesLibraryPaths(ctx)
591 a.dexpreopter.manifestFile = a.mergedManifestFile
592
Colin Cross5ab4e6d2017-11-22 16:20:45 -0800593 if ctx.ModuleName() != "framework-res" {
594 a.Module.compile(ctx, a.aaptSrcJar)
595 }
Colin Cross30e076a2015-04-13 13:58:27 -0700596
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800597 return a.maybeStrippedDexJarFile
598}
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800599
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800600func (a *AndroidApp) jniBuildActions(jniLibs []jniLib, ctx android.ModuleContext) android.WritablePath {
Colin Crossa4f08812018-10-02 22:03:40 -0700601 var jniJarFile android.WritablePath
Colin Crossa4f08812018-10-02 22:03:40 -0700602 if len(jniLibs) > 0 {
Jaewoong Jungbc625cd2019-05-06 15:48:44 -0700603 if a.shouldEmbedJnis(ctx) {
Colin Crossa4f08812018-10-02 22:03:40 -0700604 jniJarFile = android.PathForModuleOut(ctx, "jnilibs.zip")
Sasha Smundak6ad77252019-05-01 13:16:22 -0700605 TransformJniLibsToJar(ctx, jniJarFile, jniLibs, a.useEmbeddedNativeLibs(ctx))
Jaewoong Jung37ca4a12020-03-26 14:01:48 -0700606 for _, jni := range jniLibs {
607 if jni.coverageFile.Valid() {
Jaewoong Junge62e5942020-04-07 13:07:55 -0700608 // Only collect coverage for the first target arch if this is a multilib target.
609 // TODO(jungjw): Ideally, we want to collect both reports, but that would cause coverage
610 // data file path collisions since the current coverage file path format doesn't contain
611 // arch-related strings. This is fine for now though; the code coverage team doesn't use
612 // multi-arch targets such as test_suite_* for coverage collections yet.
613 //
614 // Work with the team to come up with a new format that handles multilib modules properly
615 // and change this.
616 if len(ctx.Config().Targets[android.Android]) == 1 ||
617 ctx.Config().Targets[android.Android][0].Arch.ArchType == jni.target.Arch.ArchType {
618 a.jniCoverageOutputs = append(a.jniCoverageOutputs, jni.coverageFile.Path())
619 }
Jaewoong Jung37ca4a12020-03-26 14:01:48 -0700620 }
621 }
Colin Crossa4f08812018-10-02 22:03:40 -0700622 } else {
623 a.installJniLibs = jniLibs
624 }
625 }
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800626 return jniJarFile
627}
Colin Crossa4f08812018-10-02 22:03:40 -0700628
Jaewoong Jung0949f312019-09-11 10:25:18 -0700629func (a *AndroidApp) noticeBuildActions(ctx android.ModuleContext) {
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700630 // Collect NOTICE files from all dependencies.
631 seenModules := make(map[android.Module]bool)
632 noticePathSet := make(map[android.Path]bool)
633
634 ctx.WalkDeps(func(child android.Module, parent android.Module) bool {
635 // Have we already seen this?
636 if _, ok := seenModules[child]; ok {
637 return false
638 }
639 seenModules[child] = true
640
641 // Skip host modules.
642 if child.Target().Os.Class == android.Host || child.Target().Os.Class == android.HostCross {
643 return false
644 }
645
646 path := child.(android.Module).NoticeFile()
647 if path.Valid() {
648 noticePathSet[path.Path()] = true
649 }
650 return true
651 })
652
653 // If the app has one, add it too.
654 if a.NoticeFile().Valid() {
655 noticePathSet[a.NoticeFile().Path()] = true
656 }
657
658 if len(noticePathSet) == 0 {
Jaewoong Jung98772792019-07-01 17:15:13 -0700659 return
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700660 }
661 var noticePaths []android.Path
662 for path := range noticePathSet {
663 noticePaths = append(noticePaths, path)
664 }
665 sort.Slice(noticePaths, func(i, j int) bool {
666 return noticePaths[i].String() < noticePaths[j].String()
667 })
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700668
Jaewoong Jung0949f312019-09-11 10:25:18 -0700669 a.noticeOutputs = android.BuildNoticeOutput(ctx, a.installDir, a.installApkName+".apk", noticePaths)
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700670}
671
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700672// Reads and prepends a main cert from the default cert dir if it hasn't been set already, i.e. it
673// isn't a cert module reference. Also checks and enforces system cert restriction if applicable.
674func processMainCert(m android.ModuleBase, certPropValue string, certificates []Certificate, ctx android.ModuleContext) []Certificate {
675 if android.SrcIsModule(certPropValue) == "" {
676 var mainCert Certificate
677 if certPropValue != "" {
678 defaultDir := ctx.Config().DefaultAppCertificateDir(ctx)
679 mainCert = Certificate{
Colin Cross503c1d02020-01-28 14:00:53 -0800680 Pem: defaultDir.Join(ctx, certPropValue+".x509.pem"),
681 Key: defaultDir.Join(ctx, certPropValue+".pk8"),
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700682 }
683 } else {
684 pem, key := ctx.Config().DefaultAppCertificate(ctx)
Colin Cross503c1d02020-01-28 14:00:53 -0800685 mainCert = Certificate{
686 Pem: pem,
687 Key: key,
688 }
Colin Crossbd01e2a2018-10-04 15:21:03 -0700689 }
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700690 certificates = append([]Certificate{mainCert}, certificates...)
Colin Crossbd01e2a2018-10-04 15:21:03 -0700691 }
692
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700693 if !m.Platform() {
694 certPath := certificates[0].Pem.String()
Jeongik Chac9464142019-01-07 12:07:27 +0900695 systemCertPath := ctx.Config().DefaultAppCertificateDir(ctx).String()
696 if strings.HasPrefix(certPath, systemCertPath) {
697 enforceSystemCert := ctx.Config().EnforceSystemCertificate()
Colin Cross95f7b342020-06-11 11:32:11 -0700698 allowed := ctx.Config().EnforceSystemCertificateAllowList()
Jeongik Chac9464142019-01-07 12:07:27 +0900699
Colin Cross95f7b342020-06-11 11:32:11 -0700700 if enforceSystemCert && !inList(m.Name(), allowed) {
Jeongik Chac9464142019-01-07 12:07:27 +0900701 ctx.PropertyErrorf("certificate", "The module in product partition cannot be signed with certificate in system.")
702 }
703 }
704 }
705
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700706 return certificates
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800707}
708
Jooyung Han65cd0f02020-03-23 20:21:11 +0900709func (a *AndroidApp) InstallApkName() string {
710 return a.installApkName
711}
712
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800713func (a *AndroidApp) generateAndroidBuildActions(ctx android.ModuleContext) {
Colin Cross50ddcc42019-05-16 12:28:22 -0700714 var apkDeps android.Paths
715
Jeongik Cha538c0d02019-07-11 15:54:27 +0900716 a.aapt.useEmbeddedNativeLibs = a.useEmbeddedNativeLibs(ctx)
717 a.aapt.useEmbeddedDex = Bool(a.appProperties.Use_embedded_dex)
718
Jaewoong Jung9d22a912019-01-23 16:27:47 -0800719 // Check if the install APK name needs to be overridden.
Jaewoong Jung525443a2019-02-28 15:35:54 -0800720 a.installApkName = ctx.DeviceConfig().OverridePackageNameFor(a.Name())
Jaewoong Jung9d22a912019-01-23 16:27:47 -0800721
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700722 if ctx.ModuleName() == "framework-res" {
723 // framework-res.apk is installed as system/framework/framework-res.apk
Jaewoong Jung0949f312019-09-11 10:25:18 -0700724 a.installDir = android.PathForModuleInstall(ctx, "framework")
Jiyong Parkf7487312019-10-17 12:54:30 +0900725 } else if a.Privileged() {
Jaewoong Jung0949f312019-09-11 10:25:18 -0700726 a.installDir = android.PathForModuleInstall(ctx, "priv-app", a.installApkName)
727 } else if ctx.InstallInTestcases() {
Jaewoong Jung326a9412019-11-21 10:41:00 -0800728 a.installDir = android.PathForModuleInstall(ctx, a.installApkName, ctx.DeviceConfig().DeviceArch())
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700729 } else {
Jaewoong Jung0949f312019-09-11 10:25:18 -0700730 a.installDir = android.PathForModuleInstall(ctx, "app", a.installApkName)
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700731 }
Jaewoong Jung7dd4ae22019-09-27 17:13:15 -0700732 a.onDeviceDir = android.InstallPathToOnDevicePath(ctx, a.installDir)
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700733
Jaewoong Jung0949f312019-09-11 10:25:18 -0700734 a.noticeBuildActions(ctx)
Jaewoong Jung98772792019-07-01 17:15:13 -0700735 if Bool(a.appProperties.Embed_notices) || ctx.Config().IsEnvTrue("ALWAYS_EMBED_NOTICES") {
736 a.aapt.noticeFile = a.noticeOutputs.HtmlGzOutput
737 }
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700738
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800739 // Process all building blocks, from AAPT to certificates.
740 a.aaptBuildActions(ctx)
741
Colin Cross50ddcc42019-05-16 12:28:22 -0700742 if a.usesLibrary.enforceUsesLibraries() {
743 manifestCheckFile := a.usesLibrary.verifyUsesLibrariesManifest(ctx, a.mergedManifestFile)
744 apkDeps = append(apkDeps, manifestCheckFile)
745 }
746
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800747 a.proguardBuildActions(ctx)
748
Colin Cross1e28e3c2020-06-02 20:09:13 -0700749 a.linter.mergedManifest = a.aapt.mergedManifestFile
750 a.linter.manifest = a.aapt.manifestPath
751 a.linter.resources = a.aapt.resourceFiles
752
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800753 dexJarFile := a.dexBuildActions(ctx)
754
Colin Crosseb032962020-05-13 11:05:02 -0700755 jniLibs, certificateDeps := collectAppDeps(ctx, a, a.shouldEmbedJnis(ctx), !Bool(a.appProperties.Jni_uses_platform_apis))
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800756 jniJarFile := a.jniBuildActions(jniLibs, ctx)
757
758 if ctx.Failed() {
759 return
760 }
761
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700762 certificates := processMainCert(a.ModuleBase, a.getCertString(ctx), certificateDeps, ctx)
763 a.certificate = certificates[0]
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800764
765 // Build a final signed app package.
Jaewoong Jung5a498812019-11-07 14:14:38 -0800766 packageFile := android.PathForModuleOut(ctx, a.installApkName+".apk")
Songchun Fan688de9a2020-03-24 20:32:24 -0700767 v4SigningRequested := Bool(a.Module.deviceProperties.V4_signature)
768 var v4SignatureFile android.WritablePath = nil
769 if v4SigningRequested {
770 v4SignatureFile = android.PathForModuleOut(ctx, a.installApkName+".apk.idsig")
771 }
Liz Kammer70dd74d2020-05-07 13:24:05 -0700772 var lineageFile android.Path
773 if lineage := String(a.overridableAppProperties.Lineage); lineage != "" {
774 lineageFile = android.PathForModuleSrc(ctx, lineage)
775 }
776 CreateAndSignAppPackage(ctx, packageFile, a.exportPackage, jniJarFile, dexJarFile, certificates, apkDeps, v4SignatureFile, lineageFile)
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800777 a.outputFile = packageFile
Songchun Fan688de9a2020-03-24 20:32:24 -0700778 if v4SigningRequested {
779 a.extraOutputFiles = append(a.extraOutputFiles, v4SignatureFile)
780 }
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800781
Colin Crosse560c4a2019-03-19 16:03:11 -0700782 for _, split := range a.aapt.splits {
783 // Sign the split APKs
Jaewoong Jung5a498812019-11-07 14:14:38 -0800784 packageFile := android.PathForModuleOut(ctx, a.installApkName+"_"+split.suffix+".apk")
Songchun Fan688de9a2020-03-24 20:32:24 -0700785 if v4SigningRequested {
786 v4SignatureFile = android.PathForModuleOut(ctx, a.installApkName+"_"+split.suffix+".apk.idsig")
787 }
Liz Kammer70dd74d2020-05-07 13:24:05 -0700788 CreateAndSignAppPackage(ctx, packageFile, split.path, nil, nil, certificates, apkDeps, v4SignatureFile, lineageFile)
Colin Crosse560c4a2019-03-19 16:03:11 -0700789 a.extraOutputFiles = append(a.extraOutputFiles, packageFile)
Songchun Fan688de9a2020-03-24 20:32:24 -0700790 if v4SigningRequested {
791 a.extraOutputFiles = append(a.extraOutputFiles, v4SignatureFile)
792 }
Colin Crosse560c4a2019-03-19 16:03:11 -0700793 }
794
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800795 // Build an app bundle.
Colin Crossf6237212018-10-29 23:14:58 -0700796 bundleFile := android.PathForModuleOut(ctx, "base.zip")
797 BuildBundleModule(ctx, bundleFile, a.exportPackage, jniJarFile, dexJarFile)
798 a.bundleFile = bundleFile
799
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800800 // Install the app package.
Jiyong Park8ba50f92019-11-13 15:01:01 +0900801 if (Bool(a.Module.properties.Installable) || ctx.Host()) && a.IsForPlatform() {
802 ctx.InstallFile(a.installDir, a.outputFile.Base(), a.outputFile)
803 for _, extra := range a.extraOutputFiles {
804 ctx.InstallFile(a.installDir, extra.Base(), extra)
805 }
Colin Cross5ab4e6d2017-11-22 16:20:45 -0800806 }
Artur Satayevd9b503a2020-04-27 19:05:28 +0100807
808 a.buildAppDependencyInfo(ctx)
Colin Cross30e076a2015-04-13 13:58:27 -0700809}
810
Colin Crosseb032962020-05-13 11:05:02 -0700811type appDepsInterface interface {
812 sdkVersion() sdkSpec
813 minSdkVersion() sdkSpec
814 RequiresStableAPIs(ctx android.BaseModuleContext) bool
815}
816
817func collectAppDeps(ctx android.ModuleContext, app appDepsInterface,
818 shouldCollectRecursiveNativeDeps bool,
Colin Cross1c93c292020-02-15 10:38:00 -0800819 checkNativeSdkVersion bool) ([]jniLib, []Certificate) {
Colin Crosseb032962020-05-13 11:05:02 -0700820
Colin Crossa4f08812018-10-02 22:03:40 -0700821 var jniLibs []jniLib
Jiyong Parkc00cbd92018-10-30 21:20:05 +0900822 var certificates []Certificate
Peter Collingbournead84f972019-12-17 16:46:18 -0800823 seenModulePaths := make(map[string]bool)
Colin Crossa4f08812018-10-02 22:03:40 -0700824
Colin Crosseb032962020-05-13 11:05:02 -0700825 if checkNativeSdkVersion {
826 checkNativeSdkVersion = app.sdkVersion().specified() &&
827 app.sdkVersion().kind != sdkCorePlatform && !app.RequiresStableAPIs(ctx)
828 }
829
Peter Collingbournead84f972019-12-17 16:46:18 -0800830 ctx.WalkDeps(func(module android.Module, parent android.Module) bool {
Colin Crossa4f08812018-10-02 22:03:40 -0700831 otherName := ctx.OtherModuleName(module)
832 tag := ctx.OtherModuleDependencyTag(module)
833
Peter Collingbournead84f972019-12-17 16:46:18 -0800834 if IsJniDepTag(tag) || tag == cc.SharedDepTag {
Colin Crossa4f08812018-10-02 22:03:40 -0700835 if dep, ok := module.(*cc.Module); ok {
Peter Collingbournead84f972019-12-17 16:46:18 -0800836 if dep.IsNdk() || dep.IsStubs() {
837 return false
838 }
839
Colin Crossa4f08812018-10-02 22:03:40 -0700840 lib := dep.OutputFile()
Peter Collingbournead84f972019-12-17 16:46:18 -0800841 path := lib.Path()
842 if seenModulePaths[path.String()] {
843 return false
844 }
845 seenModulePaths[path.String()] = true
846
Colin Crosseb032962020-05-13 11:05:02 -0700847 if checkNativeSdkVersion && dep.SdkVersion() == "" {
848 ctx.PropertyErrorf("jni_libs", "JNI dependency %q uses platform APIs, but this module does not",
849 otherName)
Colin Cross1c93c292020-02-15 10:38:00 -0800850 }
851
Colin Crossa4f08812018-10-02 22:03:40 -0700852 if lib.Valid() {
853 jniLibs = append(jniLibs, jniLib{
Jaewoong Jung37ca4a12020-03-26 14:01:48 -0700854 name: ctx.OtherModuleName(module),
855 path: path,
856 target: module.Target(),
857 coverageFile: dep.CoverageOutputFile(),
Colin Crossa4f08812018-10-02 22:03:40 -0700858 })
859 } else {
860 ctx.ModuleErrorf("dependency %q missing output file", otherName)
861 }
862 } else {
863 ctx.ModuleErrorf("jni_libs dependency %q must be a cc library", otherName)
Colin Crossa4f08812018-10-02 22:03:40 -0700864 }
Peter Collingbournead84f972019-12-17 16:46:18 -0800865
866 return shouldCollectRecursiveNativeDeps
867 }
868
869 if tag == certificateTag {
Colin Crossbd01e2a2018-10-04 15:21:03 -0700870 if dep, ok := module.(*AndroidAppCertificate); ok {
Jiyong Parkc00cbd92018-10-30 21:20:05 +0900871 certificates = append(certificates, dep.Certificate)
Colin Crossbd01e2a2018-10-04 15:21:03 -0700872 } else {
873 ctx.ModuleErrorf("certificate dependency %q must be an android_app_certificate module", otherName)
874 }
Colin Crossa4f08812018-10-02 22:03:40 -0700875 }
Peter Collingbournead84f972019-12-17 16:46:18 -0800876
877 return false
Colin Crossa4f08812018-10-02 22:03:40 -0700878 })
879
Colin Crossbd01e2a2018-10-04 15:21:03 -0700880 return jniLibs, certificates
Colin Crossa4f08812018-10-02 22:03:40 -0700881}
882
Artur Satayevd9b503a2020-04-27 19:05:28 +0100883func (a *AndroidApp) walkPayloadDeps(ctx android.ModuleContext,
884 do func(ctx android.ModuleContext, from blueprint.Module, to android.ApexModule, externalDep bool)) {
885
886 ctx.WalkDeps(func(child, parent android.Module) bool {
887 isExternal := !a.DepIsInSameApex(ctx, child)
888 if am, ok := child.(android.ApexModule); ok {
889 do(ctx, parent, am, isExternal)
890 }
891 return !isExternal
892 })
893}
894
895func (a *AndroidApp) buildAppDependencyInfo(ctx android.ModuleContext) {
896 if ctx.Host() {
897 return
898 }
899
900 depsInfo := android.DepNameToDepInfoMap{}
901 a.walkPayloadDeps(ctx, func(ctx android.ModuleContext, from blueprint.Module, to android.ApexModule, externalDep bool) {
902 depName := to.Name()
903 if info, exist := depsInfo[depName]; exist {
904 info.From = append(info.From, from.Name())
905 info.IsExternal = info.IsExternal && externalDep
906 depsInfo[depName] = info
907 } else {
908 toMinSdkVersion := "(no version)"
909 if m, ok := to.(interface{ MinSdkVersion() string }); ok {
910 if v := m.MinSdkVersion(); v != "" {
911 toMinSdkVersion = v
912 }
913 }
914 depsInfo[depName] = android.ApexModuleDepInfo{
915 To: depName,
916 From: []string{from.Name()},
917 IsExternal: externalDep,
918 MinSdkVersion: toMinSdkVersion,
919 }
920 }
921 })
922
923 a.ApexBundleDepsInfo.BuildDepsInfoLists(ctx, a.MinSdkVersion(), depsInfo)
924}
925
Artur Satayev2b4b7bb2020-04-28 14:57:42 +0100926func (a *AndroidApp) Updatable() bool {
927 return Bool(a.appProperties.Updatable) || a.ApexModuleBase.Updatable()
928}
929
Colin Cross0ea8ba82019-06-06 14:33:29 -0700930func (a *AndroidApp) getCertString(ctx android.BaseModuleContext) string {
Jaewoong Jung2ad817c2019-01-18 14:27:16 -0800931 certificate, overridden := ctx.DeviceConfig().OverrideCertificateFor(ctx.ModuleName())
932 if overridden {
Jaewoong Jungacb6db32019-02-28 16:22:30 +0000933 return ":" + certificate
Jaewoong Jung2ad817c2019-01-18 14:27:16 -0800934 }
Jaewoong Jung525443a2019-02-28 15:35:54 -0800935 return String(a.overridableAppProperties.Certificate)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -0800936}
937
Jiyong Park0f80c182020-01-31 02:49:53 +0900938func (a *AndroidApp) DepIsInSameApex(ctx android.BaseModuleContext, dep android.Module) bool {
939 if IsJniDepTag(ctx.OtherModuleDependencyTag(dep)) {
940 return true
941 }
942 return a.Library.DepIsInSameApex(ctx, dep)
943}
944
Jiyong Parkb7c639e2019-08-19 14:56:02 +0900945// For OutputFileProducer interface
946func (a *AndroidApp) OutputFiles(tag string) (android.Paths, error) {
947 switch tag {
948 case ".aapt.srcjar":
949 return []android.Path{a.aaptSrcJar}, nil
950 }
951 return a.Library.OutputFiles(tag)
952}
953
Jiyong Parkf7487312019-10-17 12:54:30 +0900954func (a *AndroidApp) Privileged() bool {
955 return Bool(a.appProperties.Privileged)
956}
957
Jaewoong Jung37ca4a12020-03-26 14:01:48 -0700958func (a *AndroidApp) IsNativeCoverageNeeded(ctx android.BaseModuleContext) bool {
Colin Cross72cabc62020-06-16 17:51:46 -0700959 return ctx.Device() && ctx.DeviceConfig().NativeCoverageEnabled()
Jaewoong Jung37ca4a12020-03-26 14:01:48 -0700960}
961
962func (a *AndroidApp) PreventInstall() {
963 a.appProperties.PreventInstall = true
964}
965
966func (a *AndroidApp) HideFromMake() {
967 a.appProperties.HideFromMake = true
968}
969
970func (a *AndroidApp) MarkAsCoverageVariant(coverage bool) {
971 a.appProperties.IsCoverageVariant = coverage
972}
973
974var _ cc.Coverage = (*AndroidApp)(nil)
975
Colin Cross1b16b0e2019-02-12 14:41:32 -0800976// android_app compiles sources and Android resources into an Android application package `.apk` file.
Colin Cross36242852017-06-23 15:06:31 -0700977func AndroidAppFactory() android.Module {
Colin Cross30e076a2015-04-13 13:58:27 -0700978 module := &AndroidApp{}
979
Sasha Smundak2057f822019-04-16 17:16:58 -0700980 module.Module.deviceProperties.Optimize.EnabledByDefault = true
Colin Cross66dbc0b2017-12-28 12:23:20 -0800981 module.Module.deviceProperties.Optimize.Shrink = proptools.BoolPtr(true)
982
Colin Crossae5caf52018-05-22 11:11:52 -0700983 module.Module.properties.Instrument = true
Colin Cross9ae1b922018-06-26 17:59:05 -0700984 module.Module.properties.Installable = proptools.BoolPtr(true)
Colin Crossae5caf52018-05-22 11:11:52 -0700985
Colin Cross1c14b4e2020-06-15 16:09:53 -0700986 module.addHostAndDeviceProperties()
Colin Cross36242852017-06-23 15:06:31 -0700987 module.AddProperties(
Colin Crossa97c5d32018-03-28 14:58:31 -0700988 &module.aaptProperties,
Jaewoong Jung525443a2019-02-28 15:35:54 -0800989 &module.appProperties,
Colin Cross50ddcc42019-05-16 12:28:22 -0700990 &module.overridableAppProperties,
991 &module.usesLibrary.usesLibraryProperties)
Colin Cross36242852017-06-23 15:06:31 -0700992
Colin Crossa9d8bee2018-10-02 13:59:46 -0700993 module.Prefer32(func(ctx android.BaseModuleContext, base *android.ModuleBase, class android.OsClass) bool {
994 return class == android.Device && ctx.Config().DevicePrefer32BitApps()
995 })
996
Colin Crossa4f08812018-10-02 22:03:40 -0700997 android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibCommon)
998 android.InitDefaultableModule(module)
Jaewoong Jung525443a2019-02-28 15:35:54 -0800999 android.InitOverridableModule(module, &module.appProperties.Overrides)
Jiyong Park52cd06f2019-11-11 10:14:32 +09001000 android.InitApexModule(module)
Colin Crossa4f08812018-10-02 22:03:40 -07001001
Colin Cross36242852017-06-23 15:06:31 -07001002 return module
Colin Cross30e076a2015-04-13 13:58:27 -07001003}
Colin Crossae5caf52018-05-22 11:11:52 -07001004
1005type appTestProperties struct {
1006 Instrumentation_for *string
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001007
1008 // if specified, the instrumentation target package name in the manifest is overwritten by it.
1009 Instrumentation_target_package *string
Colin Crossae5caf52018-05-22 11:11:52 -07001010}
1011
1012type AndroidTest struct {
1013 AndroidApp
1014
1015 appTestProperties appTestProperties
1016
1017 testProperties testProperties
Colin Cross303e21f2018-08-07 16:49:25 -07001018
1019 testConfig android.Path
Colin Crossd96ca352018-08-10 16:06:24 -07001020 data android.Paths
Colin Crossae5caf52018-05-22 11:11:52 -07001021}
1022
Jaewoong Jung0949f312019-09-11 10:25:18 -07001023func (a *AndroidTest) InstallInTestcases() bool {
1024 return true
1025}
1026
Colin Crossae5caf52018-05-22 11:11:52 -07001027func (a *AndroidTest) GenerateAndroidBuildActions(ctx android.ModuleContext) {
easoncyleeba606252020-04-30 14:57:06 +08001028 var configs []tradefed.Config
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001029 if a.appTestProperties.Instrumentation_target_package != nil {
1030 a.additionalAaptFlags = append(a.additionalAaptFlags,
1031 "--rename-instrumentation-target-package "+*a.appTestProperties.Instrumentation_target_package)
1032 } else if a.appTestProperties.Instrumentation_for != nil {
1033 // Check if the instrumentation target package is overridden.
Jaewoong Jung4102e5d2019-02-27 16:26:28 -08001034 manifestPackageName, overridden := ctx.DeviceConfig().OverrideManifestPackageNameFor(*a.appTestProperties.Instrumentation_for)
1035 if overridden {
1036 a.additionalAaptFlags = append(a.additionalAaptFlags, "--rename-instrumentation-target-package "+manifestPackageName)
1037 }
1038 }
Colin Crossae5caf52018-05-22 11:11:52 -07001039 a.generateAndroidBuildActions(ctx)
Colin Cross303e21f2018-08-07 16:49:25 -07001040
easoncyleeba606252020-04-30 14:57:06 +08001041 for _, module := range a.testProperties.Test_mainline_modules {
1042 configs = append(configs, tradefed.Option{Name: "config-descriptor:metadata", Key: "mainline-param", Value: module})
1043 }
1044
Jaewoong Jung39982342020-01-14 10:27:18 -08001045 testConfig := tradefed.AutoGenInstrumentationTestConfig(ctx, a.testProperties.Test_config,
easoncyleeba606252020-04-30 14:57:06 +08001046 a.testProperties.Test_config_template, a.manifestPath, a.testProperties.Test_suites, a.testProperties.Auto_gen_config, configs)
Jaewoong Jung39982342020-01-14 10:27:18 -08001047 a.testConfig = a.FixTestConfig(ctx, testConfig)
Colin Cross8a497952019-03-05 22:25:09 -08001048 a.data = android.PathsForModuleSrc(ctx, a.testProperties.Data)
Colin Cross303e21f2018-08-07 16:49:25 -07001049}
1050
Jaewoong Jung39982342020-01-14 10:27:18 -08001051func (a *AndroidTest) FixTestConfig(ctx android.ModuleContext, testConfig android.Path) android.Path {
1052 if testConfig == nil {
1053 return nil
1054 }
1055
1056 fixedConfig := android.PathForModuleOut(ctx, "test_config_fixer", "AndroidTest.xml")
1057 rule := android.NewRuleBuilder()
1058 command := rule.Command().BuiltTool(ctx, "test_config_fixer").Input(testConfig).Output(fixedConfig)
1059 fixNeeded := false
1060
1061 if ctx.ModuleName() != a.installApkName {
1062 fixNeeded = true
1063 command.FlagWithArg("--test-file-name ", a.installApkName+".apk")
1064 }
1065
1066 if a.overridableAppProperties.Package_name != nil {
1067 fixNeeded = true
1068 command.FlagWithInput("--manifest ", a.manifestPath).
1069 FlagWithArg("--package-name ", *a.overridableAppProperties.Package_name)
1070 }
1071
1072 if fixNeeded {
1073 rule.Build(pctx, ctx, "fix_test_config", "fix test config")
1074 return fixedConfig
1075 }
1076 return testConfig
1077}
1078
Colin Cross303e21f2018-08-07 16:49:25 -07001079func (a *AndroidTest) DepsMutator(ctx android.BottomUpMutatorContext) {
Colin Cross303e21f2018-08-07 16:49:25 -07001080 a.AndroidApp.DepsMutator(ctx)
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001081}
1082
1083func (a *AndroidTest) OverridablePropertiesDepsMutator(ctx android.BottomUpMutatorContext) {
1084 a.AndroidApp.OverridablePropertiesDepsMutator(ctx)
Colin Cross4b964c02018-10-15 16:18:06 -07001085 if a.appTestProperties.Instrumentation_for != nil {
1086 // The android_app dependency listed in instrumentation_for needs to be added to the classpath for javac,
1087 // but not added to the aapt2 link includes like a normal android_app or android_library dependency, so
1088 // use instrumentationForTag instead of libTag.
1089 ctx.AddVariationDependencies(nil, instrumentationForTag, String(a.appTestProperties.Instrumentation_for))
1090 }
Colin Crossae5caf52018-05-22 11:11:52 -07001091}
1092
Colin Cross1b16b0e2019-02-12 14:41:32 -08001093// android_test compiles test sources and Android resources into an Android application package `.apk` file and
1094// creates an `AndroidTest.xml` file to allow running the test with `atest` or a `TEST_MAPPING` file.
Colin Crossae5caf52018-05-22 11:11:52 -07001095func AndroidTestFactory() android.Module {
1096 module := &AndroidTest{}
1097
Sasha Smundak2057f822019-04-16 17:16:58 -07001098 module.Module.deviceProperties.Optimize.EnabledByDefault = true
Colin Cross5067db92018-09-17 16:46:35 -07001099
1100 module.Module.properties.Instrument = true
Colin Cross9ae1b922018-06-26 17:59:05 -07001101 module.Module.properties.Installable = proptools.BoolPtr(true)
Colin Crosse4246ab2019-02-05 21:55:21 -08001102 module.appProperties.Use_embedded_native_libs = proptools.BoolPtr(true)
Colin Cross47fa9d32019-03-26 10:51:39 -07001103 module.appProperties.AlwaysPackageNativeLibs = true
Colin Cross43f08db2018-11-12 10:13:39 -08001104 module.Module.dexpreopter.isTest = true
Colin Cross1e28e3c2020-06-02 20:09:13 -07001105 module.Module.linter.test = true
Colin Crossae5caf52018-05-22 11:11:52 -07001106
Colin Cross1c14b4e2020-06-15 16:09:53 -07001107 module.addHostAndDeviceProperties()
Colin Crossae5caf52018-05-22 11:11:52 -07001108 module.AddProperties(
Colin Crossae5caf52018-05-22 11:11:52 -07001109 &module.aaptProperties,
1110 &module.appProperties,
Dan Willemsenf5531d22018-07-16 17:21:19 -07001111 &module.appTestProperties,
Jaewoong Jung525443a2019-02-28 15:35:54 -08001112 &module.overridableAppProperties,
Colin Cross50ddcc42019-05-16 12:28:22 -07001113 &module.usesLibrary.usesLibraryProperties,
Dan Willemsenf5531d22018-07-16 17:21:19 -07001114 &module.testProperties)
Colin Crossae5caf52018-05-22 11:11:52 -07001115
Colin Crossa4f08812018-10-02 22:03:40 -07001116 android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibCommon)
1117 android.InitDefaultableModule(module)
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001118 android.InitOverridableModule(module, &module.appProperties.Overrides)
Colin Crossae5caf52018-05-22 11:11:52 -07001119 return module
1120}
Colin Crossbd01e2a2018-10-04 15:21:03 -07001121
Colin Cross252fc6f2018-10-04 15:22:03 -07001122type appTestHelperAppProperties struct {
1123 // list of compatibility suites (for example "cts", "vts") that the module should be
1124 // installed into.
1125 Test_suites []string `android:"arch_variant"`
Dan Shi6ffaaa82019-09-26 11:41:36 -07001126
1127 // Flag to indicate whether or not to create test config automatically. If AndroidTest.xml
1128 // doesn't exist next to the Android.bp, this attribute doesn't need to be set to true
1129 // explicitly.
1130 Auto_gen_config *bool
Colin Cross252fc6f2018-10-04 15:22:03 -07001131}
1132
1133type AndroidTestHelperApp struct {
1134 AndroidApp
1135
1136 appTestHelperAppProperties appTestHelperAppProperties
1137}
1138
Jaewoong Jung326a9412019-11-21 10:41:00 -08001139func (a *AndroidTestHelperApp) InstallInTestcases() bool {
1140 return true
1141}
1142
Colin Cross1b16b0e2019-02-12 14:41:32 -08001143// android_test_helper_app compiles sources and Android resources into an Android application package `.apk` file that
1144// will be used by tests, but does not produce an `AndroidTest.xml` file so the module will not be run directly as a
1145// test.
Colin Cross252fc6f2018-10-04 15:22:03 -07001146func AndroidTestHelperAppFactory() android.Module {
1147 module := &AndroidTestHelperApp{}
1148
Sasha Smundak2057f822019-04-16 17:16:58 -07001149 module.Module.deviceProperties.Optimize.EnabledByDefault = true
Colin Cross252fc6f2018-10-04 15:22:03 -07001150
1151 module.Module.properties.Installable = proptools.BoolPtr(true)
Colin Crosse4246ab2019-02-05 21:55:21 -08001152 module.appProperties.Use_embedded_native_libs = proptools.BoolPtr(true)
Colin Cross47fa9d32019-03-26 10:51:39 -07001153 module.appProperties.AlwaysPackageNativeLibs = true
Colin Cross43f08db2018-11-12 10:13:39 -08001154 module.Module.dexpreopter.isTest = true
Colin Cross1e28e3c2020-06-02 20:09:13 -07001155 module.Module.linter.test = true
Colin Cross252fc6f2018-10-04 15:22:03 -07001156
Colin Cross1c14b4e2020-06-15 16:09:53 -07001157 module.addHostAndDeviceProperties()
Colin Cross252fc6f2018-10-04 15:22:03 -07001158 module.AddProperties(
Colin Cross252fc6f2018-10-04 15:22:03 -07001159 &module.aaptProperties,
1160 &module.appProperties,
Jaewoong Jung525443a2019-02-28 15:35:54 -08001161 &module.appTestHelperAppProperties,
Colin Cross50ddcc42019-05-16 12:28:22 -07001162 &module.overridableAppProperties,
1163 &module.usesLibrary.usesLibraryProperties)
Colin Cross252fc6f2018-10-04 15:22:03 -07001164
1165 android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibCommon)
1166 android.InitDefaultableModule(module)
Anton Hansson3d2b6b42020-01-10 15:06:01 +00001167 android.InitApexModule(module)
Colin Cross252fc6f2018-10-04 15:22:03 -07001168 return module
1169}
1170
Colin Crossbd01e2a2018-10-04 15:21:03 -07001171type AndroidAppCertificate struct {
1172 android.ModuleBase
1173 properties AndroidAppCertificateProperties
Jiyong Parkc00cbd92018-10-30 21:20:05 +09001174 Certificate Certificate
Colin Crossbd01e2a2018-10-04 15:21:03 -07001175}
1176
1177type AndroidAppCertificateProperties struct {
1178 // Name of the certificate files. Extensions .x509.pem and .pk8 will be added to the name.
1179 Certificate *string
1180}
1181
Colin Cross1b16b0e2019-02-12 14:41:32 -08001182// android_app_certificate modules can be referenced by the certificates property of android_app modules to select
1183// the signing key.
Colin Crossbd01e2a2018-10-04 15:21:03 -07001184func AndroidAppCertificateFactory() android.Module {
1185 module := &AndroidAppCertificate{}
1186 module.AddProperties(&module.properties)
1187 android.InitAndroidModule(module)
1188 return module
1189}
1190
Colin Crossbd01e2a2018-10-04 15:21:03 -07001191func (c *AndroidAppCertificate) GenerateAndroidBuildActions(ctx android.ModuleContext) {
1192 cert := String(c.properties.Certificate)
Jiyong Parkc00cbd92018-10-30 21:20:05 +09001193 c.Certificate = Certificate{
Colin Cross503c1d02020-01-28 14:00:53 -08001194 Pem: android.PathForModuleSrc(ctx, cert+".x509.pem"),
1195 Key: android.PathForModuleSrc(ctx, cert+".pk8"),
Colin Crossbd01e2a2018-10-04 15:21:03 -07001196 }
1197}
Jaewoong Jung525443a2019-02-28 15:35:54 -08001198
1199type OverrideAndroidApp struct {
1200 android.ModuleBase
1201 android.OverrideModuleBase
1202}
1203
Sasha Smundakc4f0ff12020-05-27 16:36:07 -07001204func (i *OverrideAndroidApp) GenerateAndroidBuildActions(_ android.ModuleContext) {
Jaewoong Jung525443a2019-02-28 15:35:54 -08001205 // All the overrides happen in the base module.
1206 // TODO(jungjw): Check the base module type.
1207}
1208
1209// override_android_app is used to create an android_app module based on another android_app by overriding
1210// some of its properties.
1211func OverrideAndroidAppModuleFactory() android.Module {
1212 m := &OverrideAndroidApp{}
1213 m.AddProperties(&overridableAppProperties{})
1214
Jaewoong Jungb639a6a2019-05-10 15:16:29 -07001215 android.InitAndroidMultiTargetsArchModule(m, android.DeviceSupported, android.MultilibCommon)
Jaewoong Jung525443a2019-02-28 15:35:54 -08001216 android.InitOverrideModule(m)
1217 return m
1218}
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001219
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001220type OverrideAndroidTest struct {
1221 android.ModuleBase
1222 android.OverrideModuleBase
1223}
1224
Sasha Smundakc4f0ff12020-05-27 16:36:07 -07001225func (i *OverrideAndroidTest) GenerateAndroidBuildActions(_ android.ModuleContext) {
Jaewoong Jung26dedd32019-06-06 08:45:58 -07001226 // All the overrides happen in the base module.
1227 // TODO(jungjw): Check the base module type.
1228}
1229
1230// override_android_test is used to create an android_app module based on another android_test by overriding
1231// some of its properties.
1232func OverrideAndroidTestModuleFactory() android.Module {
1233 m := &OverrideAndroidTest{}
1234 m.AddProperties(&overridableAppProperties{})
1235 m.AddProperties(&appTestProperties{})
1236
1237 android.InitAndroidMultiTargetsArchModule(m, android.DeviceSupported, android.MultilibCommon)
1238 android.InitOverrideModule(m)
1239 return m
1240}
1241
Roshan Piusb8307962020-04-27 09:42:27 -07001242type OverrideRuntimeResourceOverlay struct {
1243 android.ModuleBase
1244 android.OverrideModuleBase
1245}
1246
Sasha Smundakc4f0ff12020-05-27 16:36:07 -07001247func (i *OverrideRuntimeResourceOverlay) GenerateAndroidBuildActions(_ android.ModuleContext) {
Roshan Piusb8307962020-04-27 09:42:27 -07001248 // All the overrides happen in the base module.
1249 // TODO(jungjw): Check the base module type.
1250}
1251
1252// override_runtime_resource_overlay is used to create a module based on another
1253// runtime_resource_overlay module by overriding some of its properties.
1254func OverrideRuntimeResourceOverlayModuleFactory() android.Module {
1255 m := &OverrideRuntimeResourceOverlay{}
1256 m.AddProperties(&OverridableRuntimeResourceOverlayProperties{})
1257
1258 android.InitAndroidMultiTargetsArchModule(m, android.DeviceSupported, android.MultilibCommon)
1259 android.InitOverrideModule(m)
1260 return m
1261}
1262
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001263type AndroidAppImport struct {
1264 android.ModuleBase
1265 android.DefaultableModuleBase
1266 prebuilt android.Prebuilt
1267
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001268 properties AndroidAppImportProperties
1269 dpiVariants interface{}
1270 archVariants interface{}
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001271
1272 outputFile android.Path
Colin Cross503c1d02020-01-28 14:00:53 -08001273 certificate Certificate
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001274
1275 dexpreopter
Colin Cross50ddcc42019-05-16 12:28:22 -07001276
1277 usesLibrary usesLibrary
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07001278
Liz Kammer7e20dda2020-05-20 14:36:30 -07001279 preprocessed bool
1280
Colin Cross70dda7e2019-10-01 22:05:35 -07001281 installPath android.InstallPath
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001282}
1283
1284type AndroidAppImportProperties struct {
1285 // A prebuilt apk to import
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001286 Apk *string
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001287
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001288 // The name of a certificate in the default certificate directory or an android_app_certificate
1289 // module name in the form ":module". Should be empty if presigned or default_dev_cert is set.
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001290 Certificate *string
1291
1292 // Set this flag to true if the prebuilt apk is already signed. The certificate property must not
1293 // be set for presigned modules.
1294 Presigned *bool
1295
Liz Kammer2bc57f62020-05-13 15:49:21 -07001296 // Name of the signing certificate lineage file.
1297 Lineage *string
1298
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001299 // Sign with the default system dev certificate. Must be used judiciously. Most imported apps
1300 // need to either specify a specific certificate or be presigned.
1301 Default_dev_cert *bool
1302
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001303 // Specifies that this app should be installed to the priv-app directory,
1304 // where the system will grant it additional privileges not available to
1305 // normal apps.
1306 Privileged *bool
1307
1308 // Names of modules to be overridden. Listed modules can only be other binaries
1309 // (in Make or Soong).
1310 // This does not completely prevent installation of the overridden binaries, but if both
1311 // binaries would be installed by default (in PRODUCT_PACKAGES) the other binary will be removed
1312 // from PRODUCT_PACKAGES.
1313 Overrides []string
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07001314
1315 // Optional name for the installed app. If unspecified, it is derived from the module name.
1316 Filename *string
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001317}
1318
Martin Stjernholm6d415272020-01-31 17:10:36 +00001319func (a *AndroidAppImport) IsInstallable() bool {
1320 return true
1321}
1322
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001323// Updates properties with variant-specific values.
1324func (a *AndroidAppImport) processVariants(ctx android.LoadHookContext) {
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001325 config := ctx.Config()
1326
1327 dpiProps := reflect.ValueOf(a.dpiVariants).Elem().FieldByName("Dpi_variants")
1328 // Try DPI variant matches in the reverse-priority order so that the highest priority match
1329 // overwrites everything else.
1330 // TODO(jungjw): Can we optimize this by making it priority order?
1331 for i := len(config.ProductAAPTPrebuiltDPI()) - 1; i >= 0; i-- {
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001332 MergePropertiesFromVariant(ctx, &a.properties, dpiProps, config.ProductAAPTPrebuiltDPI()[i])
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001333 }
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001334 if config.ProductAAPTPreferredConfig() != "" {
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001335 MergePropertiesFromVariant(ctx, &a.properties, dpiProps, config.ProductAAPTPreferredConfig())
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001336 }
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001337
1338 archProps := reflect.ValueOf(a.archVariants).Elem().FieldByName("Arch")
1339 archType := ctx.Config().Targets[android.Android][0].Arch.ArchType
1340 MergePropertiesFromVariant(ctx, &a.properties, archProps, archType.Name)
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001341}
1342
Colin Cross1184b642019-12-30 18:43:07 -08001343func MergePropertiesFromVariant(ctx android.EarlyModuleContext,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001344 dst interface{}, variantGroup reflect.Value, variant string) {
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001345 src := variantGroup.FieldByName(proptools.FieldNameForProperty(variant))
1346 if !src.IsValid() {
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001347 return
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001348 }
1349
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001350 err := proptools.ExtendMatchingProperties([]interface{}{dst}, src.Interface(), nil, proptools.OrderAppend)
1351 if err != nil {
1352 if propertyErr, ok := err.(*proptools.ExtendPropertyError); ok {
1353 ctx.PropertyErrorf(propertyErr.Property, "%s", propertyErr.Err.Error())
1354 } else {
1355 panic(err)
1356 }
1357 }
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001358}
1359
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001360func (a *AndroidAppImport) DepsMutator(ctx android.BottomUpMutatorContext) {
1361 cert := android.SrcIsModule(String(a.properties.Certificate))
1362 if cert != "" {
1363 ctx.AddDependency(ctx.Module(), certificateTag, cert)
1364 }
Colin Cross50ddcc42019-05-16 12:28:22 -07001365
Paul Duffin250e6192019-06-07 10:44:37 +01001366 a.usesLibrary.deps(ctx, true)
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001367}
1368
1369func (a *AndroidAppImport) uncompressEmbeddedJniLibs(
1370 ctx android.ModuleContext, inputPath android.Path, outputPath android.OutputPath) {
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08001371 // Test apps don't need their JNI libraries stored uncompressed. As a matter of fact, messing
1372 // with them may invalidate pre-existing signature data.
Liz Kammer7e20dda2020-05-20 14:36:30 -07001373 if ctx.InstallInTestcases() && (Bool(a.properties.Presigned) || a.preprocessed) {
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08001374 ctx.Build(pctx, android.BuildParams{
1375 Rule: android.Cp,
1376 Output: outputPath,
1377 Input: inputPath,
1378 })
1379 return
1380 }
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001381 rule := android.NewRuleBuilder()
1382 rule.Command().
1383 Textf(`if (zipinfo %s 'lib/*.so' 2>/dev/null | grep -v ' stor ' >/dev/null) ; then`, inputPath).
Colin Crossee94d6a2019-07-08 17:08:34 -07001384 BuiltTool(ctx, "zip2zip").
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001385 FlagWithInput("-i ", inputPath).
1386 FlagWithOutput("-o ", outputPath).
1387 FlagWithArg("-0 ", "'lib/**/*.so'").
1388 Textf(`; else cp -f %s %s; fi`, inputPath, outputPath)
1389 rule.Build(pctx, ctx, "uncompress-embedded-jni-libs", "Uncompress embedded JIN libs")
1390}
1391
Jaewoong Jungacf18d72019-05-02 14:55:29 -07001392// Returns whether this module should have the dex file stored uncompressed in the APK.
1393func (a *AndroidAppImport) shouldUncompressDex(ctx android.ModuleContext) bool {
Liz Kammer7e20dda2020-05-20 14:36:30 -07001394 if ctx.Config().UnbundledBuild() || a.preprocessed {
Jaewoong Jungacf18d72019-05-02 14:55:29 -07001395 return false
1396 }
1397
1398 // Uncompress dex in APKs of privileged apps
Jiyong Parkf7487312019-10-17 12:54:30 +09001399 if ctx.Config().UncompressPrivAppDex() && a.Privileged() {
Jaewoong Jungacf18d72019-05-02 14:55:29 -07001400 return true
1401 }
1402
1403 return shouldUncompressDex(ctx, &a.dexpreopter)
1404}
1405
Jaewoong Jungea1bdb02019-05-09 14:36:34 -07001406func (a *AndroidAppImport) uncompressDex(
1407 ctx android.ModuleContext, inputPath android.Path, outputPath android.OutputPath) {
1408 rule := android.NewRuleBuilder()
1409 rule.Command().
1410 Textf(`if (zipinfo %s '*.dex' 2>/dev/null | grep -v ' stor ' >/dev/null) ; then`, inputPath).
Colin Crossee94d6a2019-07-08 17:08:34 -07001411 BuiltTool(ctx, "zip2zip").
Jaewoong Jungea1bdb02019-05-09 14:36:34 -07001412 FlagWithInput("-i ", inputPath).
1413 FlagWithOutput("-o ", outputPath).
1414 FlagWithArg("-0 ", "'classes*.dex'").
1415 Textf(`; else cp -f %s %s; fi`, inputPath, outputPath)
1416 rule.Build(pctx, ctx, "uncompress-dex", "Uncompress dex files")
1417}
1418
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001419func (a *AndroidAppImport) GenerateAndroidBuildActions(ctx android.ModuleContext) {
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07001420 a.generateAndroidBuildActions(ctx)
1421}
1422
Jooyung Han65cd0f02020-03-23 20:21:11 +09001423func (a *AndroidAppImport) InstallApkName() string {
1424 return a.BaseModuleName()
1425}
1426
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07001427func (a *AndroidAppImport) generateAndroidBuildActions(ctx android.ModuleContext) {
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001428 numCertPropsSet := 0
1429 if String(a.properties.Certificate) != "" {
1430 numCertPropsSet++
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001431 }
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001432 if Bool(a.properties.Presigned) {
1433 numCertPropsSet++
1434 }
1435 if Bool(a.properties.Default_dev_cert) {
1436 numCertPropsSet++
1437 }
1438 if numCertPropsSet != 1 {
1439 ctx.ModuleErrorf("One and only one of certficate, presigned, and default_dev_cert properties must be set")
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001440 }
1441
Colin Crosseb032962020-05-13 11:05:02 -07001442 _, certificates := collectAppDeps(ctx, a, false, false)
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001443
1444 // TODO: LOCAL_EXTRACT_APK/LOCAL_EXTRACT_DPI_APK
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001445 // TODO: LOCAL_PACKAGE_SPLITS
1446
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001447 srcApk := a.prebuilt.SingleSourcePath(ctx)
Colin Cross50ddcc42019-05-16 12:28:22 -07001448
1449 if a.usesLibrary.enforceUsesLibraries() {
1450 srcApk = a.usesLibrary.verifyUsesLibrariesAPK(ctx, srcApk)
1451 }
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001452
1453 // TODO: Install or embed JNI libraries
1454
1455 // Uncompress JNI libraries in the apk
1456 jnisUncompressed := android.PathForModuleOut(ctx, "jnis-uncompressed", ctx.ModuleName()+".apk")
1457 a.uncompressEmbeddedJniLibs(ctx, srcApk, jnisUncompressed.OutputPath)
1458
Kyeongkab.Namc4997142019-11-22 11:38:16 +09001459 var installDir android.InstallPath
1460 if Bool(a.properties.Privileged) {
1461 installDir = android.PathForModuleInstall(ctx, "priv-app", a.BaseModuleName())
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08001462 } else if ctx.InstallInTestcases() {
1463 installDir = android.PathForModuleInstall(ctx, a.BaseModuleName(), ctx.DeviceConfig().DeviceArch())
Kyeongkab.Namc4997142019-11-22 11:38:16 +09001464 } else {
1465 installDir = android.PathForModuleInstall(ctx, "app", a.BaseModuleName())
1466 }
1467
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001468 a.dexpreopter.installPath = installDir.Join(ctx, a.BaseModuleName()+".apk")
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001469 a.dexpreopter.isPresignedPrebuilt = Bool(a.properties.Presigned)
Jaewoong Jungacf18d72019-05-02 14:55:29 -07001470 a.dexpreopter.uncompressedDex = a.shouldUncompressDex(ctx)
Colin Cross50ddcc42019-05-16 12:28:22 -07001471
1472 a.dexpreopter.enforceUsesLibs = a.usesLibrary.enforceUsesLibraries()
1473 a.dexpreopter.usesLibs = a.usesLibrary.usesLibraryProperties.Uses_libs
1474 a.dexpreopter.optionalUsesLibs = a.usesLibrary.presentOptionalUsesLibs(ctx)
1475 a.dexpreopter.libraryPaths = a.usesLibrary.usesLibraryPaths(ctx)
1476
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001477 dexOutput := a.dexpreopter.dexpreopt(ctx, jnisUncompressed)
Jaewoong Jungea1bdb02019-05-09 14:36:34 -07001478 if a.dexpreopter.uncompressedDex {
1479 dexUncompressed := android.PathForModuleOut(ctx, "dex-uncompressed", ctx.ModuleName()+".apk")
1480 a.uncompressDex(ctx, dexOutput, dexUncompressed.OutputPath)
1481 dexOutput = dexUncompressed
1482 }
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001483
Jooyung Han65cd0f02020-03-23 20:21:11 +09001484 apkFilename := proptools.StringDefault(a.properties.Filename, a.BaseModuleName()+".apk")
1485
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001486 // TODO: Handle EXTERNAL
Liz Kammer7e20dda2020-05-20 14:36:30 -07001487
1488 // Sign or align the package if package has not been preprocessed
1489 if a.preprocessed {
1490 a.outputFile = srcApk
Sasha Smundakc4f0ff12020-05-27 16:36:07 -07001491 a.certificate = PresignedCertificate
Liz Kammer7e20dda2020-05-20 14:36:30 -07001492 } else if !Bool(a.properties.Presigned) {
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001493 // If the certificate property is empty at this point, default_dev_cert must be set to true.
1494 // Which makes processMainCert's behavior for the empty cert string WAI.
1495 certificates = processMainCert(a.ModuleBase, String(a.properties.Certificate), certificates, ctx)
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001496 if len(certificates) != 1 {
1497 ctx.ModuleErrorf("Unexpected number of certificates were extracted: %q", certificates)
1498 }
Colin Cross503c1d02020-01-28 14:00:53 -08001499 a.certificate = certificates[0]
Jooyung Han65cd0f02020-03-23 20:21:11 +09001500 signed := android.PathForModuleOut(ctx, "signed", apkFilename)
Liz Kammer2bc57f62020-05-13 15:49:21 -07001501 var lineageFile android.Path
1502 if lineage := String(a.properties.Lineage); lineage != "" {
1503 lineageFile = android.PathForModuleSrc(ctx, lineage)
1504 }
1505 SignAppPackage(ctx, signed, dexOutput, certificates, nil, lineageFile)
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001506 a.outputFile = signed
1507 } else {
Jooyung Han65cd0f02020-03-23 20:21:11 +09001508 alignedApk := android.PathForModuleOut(ctx, "zip-aligned", apkFilename)
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001509 TransformZipAlign(ctx, alignedApk, dexOutput)
1510 a.outputFile = alignedApk
Sasha Smundakc4f0ff12020-05-27 16:36:07 -07001511 a.certificate = PresignedCertificate
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001512 }
1513
1514 // TODO: Optionally compress the output apk.
1515
Jooyung Han65cd0f02020-03-23 20:21:11 +09001516 a.installPath = ctx.InstallFile(installDir, apkFilename, a.outputFile)
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001517
1518 // TODO: androidmk converter jni libs
1519}
1520
1521func (a *AndroidAppImport) Prebuilt() *android.Prebuilt {
1522 return &a.prebuilt
1523}
1524
1525func (a *AndroidAppImport) Name() string {
1526 return a.prebuilt.Name(a.ModuleBase.Name())
1527}
1528
Dario Frenicde2a032019-10-27 00:29:22 +01001529func (a *AndroidAppImport) OutputFile() android.Path {
1530 return a.outputFile
1531}
1532
Jiyong Park618922e2020-01-08 13:35:43 +09001533func (a *AndroidAppImport) JacocoReportClassesFile() android.Path {
1534 return nil
1535}
1536
Colin Cross503c1d02020-01-28 14:00:53 -08001537func (a *AndroidAppImport) Certificate() Certificate {
1538 return a.certificate
1539}
1540
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001541var dpiVariantGroupType reflect.Type
1542var archVariantGroupType reflect.Type
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001543
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001544func initAndroidAppImportVariantGroupTypes() {
1545 dpiVariantGroupType = createVariantGroupType(supportedDpis, "Dpi_variants")
1546
1547 archNames := make([]string, len(android.ArchTypeList()))
1548 for i, archType := range android.ArchTypeList() {
1549 archNames[i] = archType.Name
1550 }
1551 archVariantGroupType = createVariantGroupType(archNames, "Arch")
1552}
1553
1554// Populates all variant struct properties at creation time.
1555func (a *AndroidAppImport) populateAllVariantStructs() {
1556 a.dpiVariants = reflect.New(dpiVariantGroupType).Interface()
1557 a.AddProperties(a.dpiVariants)
1558
1559 a.archVariants = reflect.New(archVariantGroupType).Interface()
1560 a.AddProperties(a.archVariants)
1561}
1562
Jiyong Parkf7487312019-10-17 12:54:30 +09001563func (a *AndroidAppImport) Privileged() bool {
1564 return Bool(a.properties.Privileged)
1565}
1566
Colin Crosseb032962020-05-13 11:05:02 -07001567func (a *AndroidAppImport) sdkVersion() sdkSpec {
1568 return sdkSpecFrom("")
1569}
1570
1571func (a *AndroidAppImport) minSdkVersion() sdkSpec {
1572 return sdkSpecFrom("")
1573}
1574
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001575func createVariantGroupType(variants []string, variantGroupName string) reflect.Type {
1576 props := reflect.TypeOf((*AndroidAppImportProperties)(nil))
1577
1578 variantFields := make([]reflect.StructField, len(variants))
1579 for i, variant := range variants {
1580 variantFields[i] = reflect.StructField{
1581 Name: proptools.FieldNameForProperty(variant),
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001582 Type: props,
1583 }
1584 }
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001585
1586 variantGroupStruct := reflect.StructOf(variantFields)
1587 return reflect.StructOf([]reflect.StructField{
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001588 {
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001589 Name: variantGroupName,
1590 Type: variantGroupStruct,
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001591 },
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001592 })
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001593}
1594
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001595// android_app_import imports a prebuilt apk with additional processing specified in the module.
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001596// DPI-specific apk source files can be specified using dpi_variants. Example:
1597//
1598// android_app_import {
1599// name: "example_import",
1600// apk: "prebuilts/example.apk",
1601// dpi_variants: {
1602// mdpi: {
1603// apk: "prebuilts/example_mdpi.apk",
1604// },
1605// xhdpi: {
1606// apk: "prebuilts/example_xhdpi.apk",
1607// },
1608// },
1609// certificate: "PRESIGNED",
1610// }
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001611func AndroidAppImportFactory() android.Module {
1612 module := &AndroidAppImport{}
1613 module.AddProperties(&module.properties)
1614 module.AddProperties(&module.dexpreoptProperties)
Colin Cross50ddcc42019-05-16 12:28:22 -07001615 module.AddProperties(&module.usesLibrary.usesLibraryProperties)
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001616 module.populateAllVariantStructs()
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001617 android.AddLoadHook(module, func(ctx android.LoadHookContext) {
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001618 module.processVariants(ctx)
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001619 })
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001620
Jaewoong Jung0feed892020-05-26 20:10:08 -07001621 android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibCommon)
1622 android.InitDefaultableModule(module)
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001623 android.InitSingleSourcePrebuiltModule(module, &module.properties, "Apk")
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001624
1625 return module
1626}
Colin Cross50ddcc42019-05-16 12:28:22 -07001627
Liz Kammer7e20dda2020-05-20 14:36:30 -07001628type androidTestImportProperties struct {
1629 // Whether the prebuilt apk can be installed without additional processing. Default is false.
1630 Preprocessed *bool
1631}
1632
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07001633type AndroidTestImport struct {
1634 AndroidAppImport
1635
1636 testProperties testProperties
1637
Liz Kammer7e20dda2020-05-20 14:36:30 -07001638 testImportProperties androidTestImportProperties
1639
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07001640 data android.Paths
1641}
1642
1643func (a *AndroidTestImport) GenerateAndroidBuildActions(ctx android.ModuleContext) {
Liz Kammer7e20dda2020-05-20 14:36:30 -07001644 a.preprocessed = Bool(a.testImportProperties.Preprocessed)
1645
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07001646 a.generateAndroidBuildActions(ctx)
1647
1648 a.data = android.PathsForModuleSrc(ctx, a.testProperties.Data)
1649}
1650
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08001651func (a *AndroidTestImport) InstallInTestcases() bool {
1652 return true
1653}
1654
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07001655// android_test_import imports a prebuilt test apk with additional processing specified in the
1656// module. DPI or arch variant configurations can be made as with android_app_import.
1657func AndroidTestImportFactory() android.Module {
1658 module := &AndroidTestImport{}
1659 module.AddProperties(&module.properties)
1660 module.AddProperties(&module.dexpreoptProperties)
1661 module.AddProperties(&module.usesLibrary.usesLibraryProperties)
1662 module.AddProperties(&module.testProperties)
Liz Kammer7e20dda2020-05-20 14:36:30 -07001663 module.AddProperties(&module.testImportProperties)
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07001664 module.populateAllVariantStructs()
1665 android.AddLoadHook(module, func(ctx android.LoadHookContext) {
1666 module.processVariants(ctx)
1667 })
1668
Colin Crossf30c4532020-05-06 22:29:10 -07001669 module.dexpreopter.isTest = true
1670
Jaewoong Junga689ffe2020-05-01 15:50:08 -07001671 android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibCommon)
1672 android.InitDefaultableModule(module)
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07001673 android.InitSingleSourcePrebuiltModule(module, &module.properties, "Apk")
1674
1675 return module
1676}
1677
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001678type RuntimeResourceOverlay struct {
1679 android.ModuleBase
1680 android.DefaultableModuleBase
Roshan Piusb8307962020-04-27 09:42:27 -07001681 android.OverridableModuleBase
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001682 aapt
1683
Roshan Piusb8307962020-04-27 09:42:27 -07001684 properties RuntimeResourceOverlayProperties
1685 overridableProperties OverridableRuntimeResourceOverlayProperties
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001686
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08001687 certificate Certificate
1688
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001689 outputFile android.Path
1690 installDir android.InstallPath
1691}
1692
1693type RuntimeResourceOverlayProperties struct {
1694 // the name of a certificate in the default certificate directory or an android_app_certificate
1695 // module name in the form ":module".
1696 Certificate *string
1697
Liz Kammer7fe241f2020-05-19 16:15:25 -07001698 // Name of the signing certificate lineage file.
1699 Lineage *string
1700
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001701 // optional theme name. If specified, the overlay package will be applied
1702 // only when the ro.boot.vendor.overlay.theme system property is set to the same value.
1703 Theme *string
1704
1705 // if not blank, set to the version of the sdk to compile against.
1706 // Defaults to compiling against the current platform.
1707 Sdk_version *string
1708
1709 // if not blank, set the minimum version of the sdk that the compiled artifacts will run against.
1710 // Defaults to sdk_version if not set.
1711 Min_sdk_version *string
Jaewoong Jungca095d72020-04-09 16:15:30 -07001712
1713 // list of android_library modules whose resources are extracted and linked against statically
1714 Static_libs []string
1715
1716 // list of android_app modules whose resources are extracted and linked against
1717 Resource_libs []string
Jaewoong Jungbfc6ac02020-04-24 15:22:40 -07001718
1719 // Names of modules to be overridden. Listed modules can only be other overlays
1720 // (in Make or Soong).
1721 // This does not completely prevent installation of the overridden overlays, but if both
1722 // overlays would be installed by default (in PRODUCT_PACKAGES) the other overlay will be removed
1723 // from PRODUCT_PACKAGES.
1724 Overrides []string
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001725}
1726
1727func (r *RuntimeResourceOverlay) DepsMutator(ctx android.BottomUpMutatorContext) {
1728 sdkDep := decodeSdkDep(ctx, sdkContext(r))
1729 if sdkDep.hasFrameworkLibs() {
1730 r.aapt.deps(ctx, sdkDep)
1731 }
1732
1733 cert := android.SrcIsModule(String(r.properties.Certificate))
1734 if cert != "" {
1735 ctx.AddDependency(ctx.Module(), certificateTag, cert)
1736 }
Jaewoong Jungca095d72020-04-09 16:15:30 -07001737
1738 ctx.AddVariationDependencies(nil, staticLibTag, r.properties.Static_libs...)
1739 ctx.AddVariationDependencies(nil, libTag, r.properties.Resource_libs...)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001740}
1741
1742func (r *RuntimeResourceOverlay) GenerateAndroidBuildActions(ctx android.ModuleContext) {
1743 // Compile and link resources
1744 r.aapt.hasNoCode = true
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08001745 // Do not remove resources without default values nor dedupe resource configurations with the same value
Roshan Piusb8307962020-04-27 09:42:27 -07001746 aaptLinkFlags := []string{"--no-resource-deduping", "--no-resource-removal"}
1747 // Allow the override of "package name" and "overlay target package name"
1748 manifestPackageName, overridden := ctx.DeviceConfig().OverrideManifestPackageNameFor(ctx.ModuleName())
1749 if overridden || r.overridableProperties.Package_name != nil {
1750 // The product override variable has a priority over the package_name property.
1751 if !overridden {
1752 manifestPackageName = *r.overridableProperties.Package_name
1753 }
1754 aaptLinkFlags = append(aaptLinkFlags, "--rename-manifest-package "+manifestPackageName)
1755 }
1756 if r.overridableProperties.Target_package_name != nil {
1757 aaptLinkFlags = append(aaptLinkFlags,
1758 "--rename-overlay-target-package "+*r.overridableProperties.Target_package_name)
1759 }
1760 r.aapt.buildActions(ctx, r, aaptLinkFlags...)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001761
1762 // Sign the built package
Colin Crosseb032962020-05-13 11:05:02 -07001763 _, certificates := collectAppDeps(ctx, r, false, false)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001764 certificates = processMainCert(r.ModuleBase, String(r.properties.Certificate), certificates, ctx)
1765 signed := android.PathForModuleOut(ctx, "signed", r.Name()+".apk")
Liz Kammer7fe241f2020-05-19 16:15:25 -07001766 var lineageFile android.Path
1767 if lineage := String(r.properties.Lineage); lineage != "" {
1768 lineageFile = android.PathForModuleSrc(ctx, lineage)
1769 }
1770 SignAppPackage(ctx, signed, r.aapt.exportPackage, certificates, nil, lineageFile)
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08001771 r.certificate = certificates[0]
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001772
1773 r.outputFile = signed
1774 r.installDir = android.PathForModuleInstall(ctx, "overlay", String(r.properties.Theme))
1775 ctx.InstallFile(r.installDir, r.outputFile.Base(), r.outputFile)
1776}
1777
Jiyong Park6a927c42020-01-21 02:03:43 +09001778func (r *RuntimeResourceOverlay) sdkVersion() sdkSpec {
1779 return sdkSpecFrom(String(r.properties.Sdk_version))
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001780}
1781
1782func (r *RuntimeResourceOverlay) systemModules() string {
1783 return ""
1784}
1785
Jiyong Park6a927c42020-01-21 02:03:43 +09001786func (r *RuntimeResourceOverlay) minSdkVersion() sdkSpec {
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001787 if r.properties.Min_sdk_version != nil {
Jiyong Park6a927c42020-01-21 02:03:43 +09001788 return sdkSpecFrom(*r.properties.Min_sdk_version)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001789 }
1790 return r.sdkVersion()
1791}
1792
Jiyong Park6a927c42020-01-21 02:03:43 +09001793func (r *RuntimeResourceOverlay) targetSdkVersion() sdkSpec {
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001794 return r.sdkVersion()
1795}
1796
1797// runtime_resource_overlay generates a resource-only apk file that can overlay application and
1798// system resources at run time.
1799func RuntimeResourceOverlayFactory() android.Module {
1800 module := &RuntimeResourceOverlay{}
1801 module.AddProperties(
1802 &module.properties,
Roshan Piusb8307962020-04-27 09:42:27 -07001803 &module.aaptProperties,
1804 &module.overridableProperties)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001805
Roshan Piusb8307962020-04-27 09:42:27 -07001806 android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibCommon)
1807 android.InitDefaultableModule(module)
1808 android.InitOverridableModule(module, &module.properties.Overrides)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001809 return module
1810}
1811
Colin Cross50ddcc42019-05-16 12:28:22 -07001812type UsesLibraryProperties struct {
1813 // A list of shared library modules that will be listed in uses-library tags in the AndroidManifest.xml file.
1814 Uses_libs []string
1815
1816 // A list of shared library modules that will be listed in uses-library tags in the AndroidManifest.xml file with
1817 // required=false.
1818 Optional_uses_libs []string
1819
1820 // If true, the list of uses_libs and optional_uses_libs modules must match the AndroidManifest.xml file. Defaults
1821 // to true if either uses_libs or optional_uses_libs is set. Will unconditionally default to true in the future.
1822 Enforce_uses_libs *bool
1823}
1824
1825// usesLibrary provides properties and helper functions for AndroidApp and AndroidAppImport to verify that the
1826// <uses-library> tags that end up in the manifest of an APK match the ones known to the build system through the
1827// uses_libs and optional_uses_libs properties. The build system's values are used by dexpreopt to preopt apps
1828// with knowledge of their shared libraries.
1829type usesLibrary struct {
1830 usesLibraryProperties UsesLibraryProperties
1831}
1832
Paul Duffin250e6192019-06-07 10:44:37 +01001833func (u *usesLibrary) deps(ctx android.BottomUpMutatorContext, hasFrameworkLibs bool) {
Colin Cross3245b2c2019-06-07 13:18:09 -07001834 if !ctx.Config().UnbundledBuild() {
1835 ctx.AddVariationDependencies(nil, usesLibTag, u.usesLibraryProperties.Uses_libs...)
1836 ctx.AddVariationDependencies(nil, usesLibTag, u.presentOptionalUsesLibs(ctx)...)
Paul Duffin250e6192019-06-07 10:44:37 +01001837 // Only add these extra dependencies if the module depends on framework libs. This avoids
1838 // creating a cyclic dependency:
1839 // e.g. framework-res -> org.apache.http.legacy -> ... -> framework-res.
1840 if hasFrameworkLibs {
Colin Cross3245b2c2019-06-07 13:18:09 -07001841 // dexpreopt/dexpreopt.go needs the paths to the dex jars of these libraries in case construct_context.sh needs
1842 // to pass them to dex2oat. Add them as a dependency so we can determine the path to the dex jar of each
1843 // library to dexpreopt.
1844 ctx.AddVariationDependencies(nil, usesLibTag,
1845 "org.apache.http.legacy",
1846 "android.hidl.base-V1.0-java",
1847 "android.hidl.manager-V1.0-java")
1848 }
Colin Cross50ddcc42019-05-16 12:28:22 -07001849 }
1850}
1851
1852// presentOptionalUsesLibs returns optional_uses_libs after filtering out MissingUsesLibraries, which don't exist in the
1853// build.
1854func (u *usesLibrary) presentOptionalUsesLibs(ctx android.BaseModuleContext) []string {
1855 optionalUsesLibs, _ := android.FilterList(u.usesLibraryProperties.Optional_uses_libs, ctx.Config().MissingUsesLibraries())
1856 return optionalUsesLibs
1857}
1858
1859// usesLibraryPaths returns a map of module names of shared library dependencies to the paths to their dex jars.
1860func (u *usesLibrary) usesLibraryPaths(ctx android.ModuleContext) map[string]android.Path {
1861 usesLibPaths := make(map[string]android.Path)
1862
1863 if !ctx.Config().UnbundledBuild() {
1864 ctx.VisitDirectDepsWithTag(usesLibTag, func(m android.Module) {
1865 if lib, ok := m.(Dependency); ok {
1866 if dexJar := lib.DexJar(); dexJar != nil {
1867 usesLibPaths[ctx.OtherModuleName(m)] = dexJar
1868 } else {
1869 ctx.ModuleErrorf("module %q in uses_libs or optional_uses_libs must produce a dex jar, does it have installable: true?",
1870 ctx.OtherModuleName(m))
1871 }
1872 } else if ctx.Config().AllowMissingDependencies() {
1873 ctx.AddMissingDependencies([]string{ctx.OtherModuleName(m)})
1874 } else {
1875 ctx.ModuleErrorf("module %q in uses_libs or optional_uses_libs must be a java library",
1876 ctx.OtherModuleName(m))
1877 }
1878 })
1879 }
1880
1881 return usesLibPaths
1882}
1883
1884// enforceUsesLibraries returns true of <uses-library> tags should be checked against uses_libs and optional_uses_libs
1885// properties. Defaults to true if either of uses_libs or optional_uses_libs is specified. Will default to true
1886// unconditionally in the future.
1887func (u *usesLibrary) enforceUsesLibraries() bool {
1888 defaultEnforceUsesLibs := len(u.usesLibraryProperties.Uses_libs) > 0 ||
1889 len(u.usesLibraryProperties.Optional_uses_libs) > 0
1890 return BoolDefault(u.usesLibraryProperties.Enforce_uses_libs, defaultEnforceUsesLibs)
1891}
1892
1893// verifyUsesLibrariesManifest checks the <uses-library> tags in an AndroidManifest.xml against the ones specified
1894// in the uses_libs and optional_uses_libs properties. It returns the path to a copy of the manifest.
1895func (u *usesLibrary) verifyUsesLibrariesManifest(ctx android.ModuleContext, manifest android.Path) android.Path {
1896 outputFile := android.PathForModuleOut(ctx, "manifest_check", "AndroidManifest.xml")
1897
1898 rule := android.NewRuleBuilder()
Colin Crossee94d6a2019-07-08 17:08:34 -07001899 cmd := rule.Command().BuiltTool(ctx, "manifest_check").
Colin Cross50ddcc42019-05-16 12:28:22 -07001900 Flag("--enforce-uses-libraries").
1901 Input(manifest).
1902 FlagWithOutput("-o ", outputFile)
1903
1904 for _, lib := range u.usesLibraryProperties.Uses_libs {
1905 cmd.FlagWithArg("--uses-library ", lib)
1906 }
1907
1908 for _, lib := range u.usesLibraryProperties.Optional_uses_libs {
1909 cmd.FlagWithArg("--optional-uses-library ", lib)
1910 }
1911
1912 rule.Build(pctx, ctx, "verify_uses_libraries", "verify <uses-library>")
1913
1914 return outputFile
1915}
1916
1917// verifyUsesLibrariesAPK checks the <uses-library> tags in the manifest of an APK against the ones specified
1918// in the uses_libs and optional_uses_libs properties. It returns the path to a copy of the APK.
1919func (u *usesLibrary) verifyUsesLibrariesAPK(ctx android.ModuleContext, apk android.Path) android.Path {
1920 outputFile := android.PathForModuleOut(ctx, "verify_uses_libraries", apk.Base())
1921
1922 rule := android.NewRuleBuilder()
1923 aapt := ctx.Config().HostToolPath(ctx, "aapt")
1924 rule.Command().
1925 Textf("aapt_binary=%s", aapt.String()).Implicit(aapt).
1926 Textf(`uses_library_names="%s"`, strings.Join(u.usesLibraryProperties.Uses_libs, " ")).
1927 Textf(`optional_uses_library_names="%s"`, strings.Join(u.usesLibraryProperties.Optional_uses_libs, " ")).
1928 Tool(android.PathForSource(ctx, "build/make/core/verify_uses_libraries.sh")).Input(apk)
1929 rule.Command().Text("cp -f").Input(apk).Output(outputFile)
1930
1931 rule.Build(pctx, ctx, "verify_uses_libraries", "verify <uses-library>")
1932
1933 return outputFile
1934}