blob: 449558033042a11fa0e6d38dc03bdec888367e85 [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"
Jaewoong Junga5e5abc2019-04-26 14:31:50 -070023 "strings"
Colin Cross30e076a2015-04-13 13:58:27 -070024
Colin Cross50ddcc42019-05-16 12:28:22 -070025 "github.com/google/blueprint"
26 "github.com/google/blueprint/proptools"
27
Colin Cross635c3b02016-05-18 15:37:25 -070028 "android/soong/android"
Colin Crossa4f08812018-10-02 22:03:40 -070029 "android/soong/cc"
Colin Cross303e21f2018-08-07 16:49:25 -070030 "android/soong/tradefed"
Colin Cross30e076a2015-04-13 13:58:27 -070031)
32
Jaewoong Jung3e18b192019-06-11 12:25:34 -070033var supportedDpis = []string{"ldpi", "mdpi", "hdpi", "xhdpi", "xxhdpi", "xxxhdpi"}
Jaewoong Junga5e5abc2019-04-26 14:31:50 -070034
Colin Cross3bc7ffa2017-11-22 16:19:37 -080035func init() {
Paul Duffinf9b1da02019-12-18 19:51:55 +000036 RegisterAppBuildComponents(android.InitRegistrationContext)
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -070037
38 initAndroidAppImportVariantGroupTypes()
Colin Cross3bc7ffa2017-11-22 16:19:37 -080039}
40
Paul Duffinf9b1da02019-12-18 19:51:55 +000041func RegisterAppBuildComponents(ctx android.RegistrationContext) {
42 ctx.RegisterModuleType("android_app", AndroidAppFactory)
43 ctx.RegisterModuleType("android_test", AndroidTestFactory)
44 ctx.RegisterModuleType("android_test_helper_app", AndroidTestHelperAppFactory)
45 ctx.RegisterModuleType("android_app_certificate", AndroidAppCertificateFactory)
46 ctx.RegisterModuleType("override_android_app", OverrideAndroidAppModuleFactory)
47 ctx.RegisterModuleType("override_android_test", OverrideAndroidTestModuleFactory)
48 ctx.RegisterModuleType("android_app_import", AndroidAppImportFactory)
49 ctx.RegisterModuleType("android_test_import", AndroidTestImportFactory)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -080050 ctx.RegisterModuleType("runtime_resource_overlay", RuntimeResourceOverlayFactory)
Paul Duffinf9b1da02019-12-18 19:51:55 +000051}
52
Colin Cross30e076a2015-04-13 13:58:27 -070053// AndroidManifest.xml merging
54// package splits
55
Colin Crossfabb6082018-02-20 17:22:23 -080056type appProperties struct {
Colin Crossbd01e2a2018-10-04 15:21:03 -070057 // Names of extra android_app_certificate modules to sign the apk with in the form ":module".
Colin Cross7d5136f2015-05-11 13:39:40 -070058 Additional_certificates []string
59
60 // If set, create package-export.apk, which other packages can
61 // use to get PRODUCT-agnostic resource data like IDs and type definitions.
Nan Zhangea568a42017-11-08 21:20:04 -080062 Export_package_resources *bool
Colin Cross7d5136f2015-05-11 13:39:40 -070063
Colin Cross16056062017-12-13 22:46:28 -080064 // Specifies that this app should be installed to the priv-app directory,
65 // where the system will grant it additional privileges not available to
66 // normal apps.
67 Privileged *bool
Colin Crossa97c5d32018-03-28 14:58:31 -070068
69 // list of resource labels to generate individual resource packages
70 Package_splits []string
Jason Monkd4122be2018-08-10 09:33:36 -040071
72 // Names of modules to be overridden. Listed modules can only be other binaries
73 // (in Make or Soong).
74 // This does not completely prevent installation of the overridden binaries, but if both
75 // binaries would be installed by default (in PRODUCT_PACKAGES) the other binary will be removed
76 // from PRODUCT_PACKAGES.
77 Overrides []string
Colin Crossa4f08812018-10-02 22:03:40 -070078
79 // list of native libraries that will be provided in or alongside the resulting jar
80 Jni_libs []string `android:"arch_variant"`
81
Colin Crossee87c602020-02-19 16:57:15 -080082 // if true, allow JNI libraries that link against platform APIs even if this module sets
83 // sdk_version.
84 Jni_uses_platform_apis *bool
85
Jaewoong Jungbc625cd2019-05-06 15:48:44 -070086 // STL library to use for JNI libraries.
87 Stl *string `android:"arch_variant"`
88
Colin Crosse4246ab2019-02-05 21:55:21 -080089 // Store native libraries uncompressed in the APK and set the android:extractNativeLibs="false" manifest
90 // 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 +090091 // sdk_version or min_sdk_version is set to a version that doesn't support it (<23), defaults to true for
92 // android_app modules that are embedded to APEXes, defaults to false for other module types where the native
93 // libraries are generally preinstalled outside the APK.
Colin Crosse4246ab2019-02-05 21:55:21 -080094 Use_embedded_native_libs *bool
Colin Cross46abdad2019-02-07 13:07:08 -080095
96 // Store dex files uncompressed in the APK and set the android:useEmbeddedDex="true" manifest attribute so that
97 // they are used from inside the APK at runtime.
98 Use_embedded_dex *bool
Colin Cross47fa9d32019-03-26 10:51:39 -070099
100 // Forces native libraries to always be packaged into the APK,
101 // Use_embedded_native_libs still selects whether they are stored uncompressed and aligned or compressed.
102 // True for android_test* modules.
103 AlwaysPackageNativeLibs bool `blueprint:"mutated"`
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700104
105 // If set, find and merge all NOTICE files that this module and its dependencies have and store
106 // it in the APK as an asset.
107 Embed_notices *bool
Jaewoong Jung87a33e72020-03-26 14:01:48 -0700108
109 // cc.Coverage related properties
110 PreventInstall bool `blueprint:"mutated"`
111 HideFromMake bool `blueprint:"mutated"`
112 IsCoverageVariant bool `blueprint:"mutated"`
Colin Cross7d5136f2015-05-11 13:39:40 -0700113}
114
Jaewoong Jung525443a2019-02-28 15:35:54 -0800115// android_app properties that can be overridden by override_android_app
116type overridableAppProperties struct {
117 // The name of a certificate in the default certificate directory, blank to use the default product certificate,
118 // or an android_app_certificate module name in the form ":module".
119 Certificate *string
Jaewoong Jung6f373f62019-03-13 10:13:24 -0700120
121 // the package name of this app. The package name in the manifest file is used if one was not given.
122 Package_name *string
Baligh Uddin5b16dfb2020-02-11 17:27:19 -0800123
124 // the logging parent of this app.
125 Logging_parent *string
Jaewoong Jung525443a2019-02-28 15:35:54 -0800126}
127
Colin Cross30e076a2015-04-13 13:58:27 -0700128type AndroidApp struct {
Colin Crossa97c5d32018-03-28 14:58:31 -0700129 Library
130 aapt
Jaewoong Jung525443a2019-02-28 15:35:54 -0800131 android.OverridableModuleBase
Colin Crossa97c5d32018-03-28 14:58:31 -0700132
Colin Cross50ddcc42019-05-16 12:28:22 -0700133 usesLibrary usesLibrary
134
Jiyong Parkc00cbd92018-10-30 21:20:05 +0900135 certificate Certificate
Colin Cross30e076a2015-04-13 13:58:27 -0700136
Colin Crossfabb6082018-02-20 17:22:23 -0800137 appProperties appProperties
Colin Crossae5caf52018-05-22 11:11:52 -0700138
Jaewoong Jung525443a2019-02-28 15:35:54 -0800139 overridableAppProperties overridableAppProperties
140
Jaewoong Jung87a33e72020-03-26 14:01:48 -0700141 installJniLibs []jniLib
142 jniCoverageOutputs android.Paths
Colin Crossf6237212018-10-29 23:14:58 -0700143
144 bundleFile android.Path
Jaewoong Jung9d22a912019-01-23 16:27:47 -0800145
146 // the install APK name is normally the same as the module name, but can be overridden with PRODUCT_PACKAGE_NAME_OVERRIDES.
147 installApkName string
Jaewoong Jung4102e5d2019-02-27 16:26:28 -0800148
Colin Cross70dda7e2019-10-01 22:05:35 -0700149 installDir android.InstallPath
Jaewoong Jung0949f312019-09-11 10:25:18 -0700150
Jaewoong Jung7dd4ae22019-09-27 17:13:15 -0700151 onDeviceDir string
152
Jaewoong Jung4102e5d2019-02-27 16:26:28 -0800153 additionalAaptFlags []string
Jaewoong Jung98772792019-07-01 17:15:13 -0700154
155 noticeOutputs android.NoticeOutputs
Jiyong Parkcfaa1642020-02-28 16:51:07 +0900156
157 overriddenManifestPackageName string
Colin Crosse1731a52017-12-14 11:22:55 -0800158}
159
Martin Stjernholm6d415272020-01-31 17:10:36 +0000160func (a *AndroidApp) IsInstallable() bool {
161 return Bool(a.properties.Installable)
162}
163
Colin Cross89c31582018-04-30 15:55:11 -0700164func (a *AndroidApp) ExportedProguardFlagFiles() android.Paths {
165 return nil
166}
167
Colin Cross66f78822018-05-02 12:58:28 -0700168func (a *AndroidApp) ExportedStaticPackages() android.Paths {
169 return nil
170}
171
Sundong Ahne1f05aa2019-08-27 13:55:42 +0900172func (a *AndroidApp) OutputFile() android.Path {
173 return a.outputFile
174}
175
Colin Cross503c1d02020-01-28 14:00:53 -0800176func (a *AndroidApp) Certificate() Certificate {
177 return a.certificate
178}
179
Jaewoong Jung87a33e72020-03-26 14:01:48 -0700180func (a *AndroidApp) JniCoverageOutputs() android.Paths {
181 return a.jniCoverageOutputs
182}
183
Colin Crossa97c5d32018-03-28 14:58:31 -0700184var _ AndroidLibraryDependency = (*AndroidApp)(nil)
185
Jiyong Parkc00cbd92018-10-30 21:20:05 +0900186type Certificate struct {
Colin Cross503c1d02020-01-28 14:00:53 -0800187 Pem, Key android.Path
188 presigned bool
189}
190
191var presignedCertificate = Certificate{presigned: true}
192
193func (c Certificate) AndroidMkString() string {
194 if c.presigned {
195 return "PRESIGNED"
196 } else {
197 return c.Pem.String()
198 }
Colin Cross30e076a2015-04-13 13:58:27 -0700199}
200
Colin Cross46c9b8b2017-06-22 16:51:17 -0700201func (a *AndroidApp) DepsMutator(ctx android.BottomUpMutatorContext) {
202 a.Module.deps(ctx)
Colin Crossa4f08812018-10-02 22:03:40 -0700203
Jiyong Park6a927c42020-01-21 02:03:43 +0900204 if String(a.appProperties.Stl) == "c++_shared" && !a.sdkVersion().specified() {
Jaewoong Jungbc625cd2019-05-06 15:48:44 -0700205 ctx.PropertyErrorf("stl", "sdk_version must be set in order to use c++_shared")
206 }
207
Paul Duffin250e6192019-06-07 10:44:37 +0100208 sdkDep := decodeSdkDep(ctx, sdkContext(a))
209 if sdkDep.hasFrameworkLibs() {
210 a.aapt.deps(ctx, sdkDep)
Colin Cross30e076a2015-04-13 13:58:27 -0700211 }
Colin Crossa4f08812018-10-02 22:03:40 -0700212
Peter Collingbournead84f972019-12-17 16:46:18 -0800213 tag := &jniDependencyTag{}
Colin Crossa4f08812018-10-02 22:03:40 -0700214 for _, jniTarget := range ctx.MultiTargets() {
Colin Cross0f7d2ef2019-10-16 11:03:10 -0700215 variation := append(jniTarget.Variations(),
216 blueprint.Variation{Mutator: "link", Variation: "shared"})
Colin Crossa4f08812018-10-02 22:03:40 -0700217 ctx.AddFarVariationDependencies(variation, tag, a.appProperties.Jni_libs...)
218 }
Colin Cross50ddcc42019-05-16 12:28:22 -0700219
Paul Duffin250e6192019-06-07 10:44:37 +0100220 a.usesLibrary.deps(ctx, sdkDep.hasFrameworkLibs())
Jaewoong Jungb639a6a2019-05-10 15:16:29 -0700221}
Colin Crossbd01e2a2018-10-04 15:21:03 -0700222
Jaewoong Jungb639a6a2019-05-10 15:16:29 -0700223func (a *AndroidApp) OverridablePropertiesDepsMutator(ctx android.BottomUpMutatorContext) {
Jaewoong Jung2ad817c2019-01-18 14:27:16 -0800224 cert := android.SrcIsModule(a.getCertString(ctx))
Colin Crossbd01e2a2018-10-04 15:21:03 -0700225 if cert != "" {
226 ctx.AddDependency(ctx.Module(), certificateTag, cert)
227 }
228
229 for _, cert := range a.appProperties.Additional_certificates {
230 cert = android.SrcIsModule(cert)
231 if cert != "" {
232 ctx.AddDependency(ctx.Module(), certificateTag, cert)
233 } else {
234 ctx.PropertyErrorf("additional_certificates",
235 `must be names of android_app_certificate modules in the form ":module"`)
236 }
237 }
Colin Cross30e076a2015-04-13 13:58:27 -0700238}
239
Jeongik Cha538c0d02019-07-11 15:54:27 +0900240func (a *AndroidTestHelperApp) GenerateAndroidBuildActions(ctx android.ModuleContext) {
241 a.generateAndroidBuildActions(ctx)
242}
243
Colin Cross46c9b8b2017-06-22 16:51:17 -0700244func (a *AndroidApp) GenerateAndroidBuildActions(ctx android.ModuleContext) {
Jeongik Cha538c0d02019-07-11 15:54:27 +0900245 a.checkPlatformAPI(ctx)
Jeongik Cha2cc570d2019-10-29 15:44:45 +0900246 a.checkSdkVersion(ctx)
Colin Crossae5caf52018-05-22 11:11:52 -0700247 a.generateAndroidBuildActions(ctx)
248}
249
Sasha Smundak6ad77252019-05-01 13:16:22 -0700250// Returns true if the native libraries should be stored in the APK uncompressed and the
Colin Crosse4246ab2019-02-05 21:55:21 -0800251// extractNativeLibs application flag should be set to false in the manifest.
Sasha Smundak6ad77252019-05-01 13:16:22 -0700252func (a *AndroidApp) useEmbeddedNativeLibs(ctx android.ModuleContext) bool {
Jiyong Park6a927c42020-01-21 02:03:43 +0900253 minSdkVersion, err := a.minSdkVersion().effectiveVersion(ctx)
Colin Crosse4246ab2019-02-05 21:55:21 -0800254 if err != nil {
255 ctx.PropertyErrorf("min_sdk_version", "invalid value %q: %s", a.minSdkVersion(), err)
256 }
257
Jiyong Park52cd06f2019-11-11 10:14:32 +0900258 return (minSdkVersion >= 23 && Bool(a.appProperties.Use_embedded_native_libs)) ||
259 !a.IsForPlatform()
Colin Crosse4246ab2019-02-05 21:55:21 -0800260}
261
Colin Cross43f08db2018-11-12 10:13:39 -0800262// Returns whether this module should have the dex file stored uncompressed in the APK.
263func (a *AndroidApp) shouldUncompressDex(ctx android.ModuleContext) bool {
Colin Cross46abdad2019-02-07 13:07:08 -0800264 if Bool(a.appProperties.Use_embedded_dex) {
265 return true
266 }
267
Colin Cross53a87f52019-06-25 13:35:30 -0700268 // Uncompress dex in APKs of privileged apps (even for unbundled builds, they may
269 // be preinstalled as prebuilts).
Jiyong Parkf7487312019-10-17 12:54:30 +0900270 if ctx.Config().UncompressPrivAppDex() && a.Privileged() {
Nicolas Geoffrayfa6e9ec2019-02-12 13:12:16 +0000271 return true
272 }
273
Colin Cross53a87f52019-06-25 13:35:30 -0700274 if ctx.Config().UnbundledBuild() {
275 return false
276 }
277
Jaewoong Jungacf18d72019-05-02 14:55:29 -0700278 return shouldUncompressDex(ctx, &a.dexpreopter)
Colin Cross5a0dcd52018-10-05 14:20:06 -0700279}
280
Jaewoong Jungbc625cd2019-05-06 15:48:44 -0700281func (a *AndroidApp) shouldEmbedJnis(ctx android.BaseModuleContext) bool {
282 return ctx.Config().UnbundledBuild() || Bool(a.appProperties.Use_embedded_native_libs) ||
Jiyong Park52cd06f2019-11-11 10:14:32 +0900283 !a.IsForPlatform() || a.appProperties.AlwaysPackageNativeLibs
Jaewoong Jungbc625cd2019-05-06 15:48:44 -0700284}
285
Jiyong Parkcfaa1642020-02-28 16:51:07 +0900286func (a *AndroidApp) OverriddenManifestPackageName() string {
287 return a.overriddenManifestPackageName
288}
289
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800290func (a *AndroidApp) aaptBuildActions(ctx android.ModuleContext) {
David Brazdild25060a2019-02-18 18:24:16 +0000291 a.aapt.usesNonSdkApis = Bool(a.Module.deviceProperties.Platform_apis)
292
Jaewoong Jungc27ab662019-05-30 15:51:14 -0700293 // Ask manifest_fixer to add or update the application element indicating this app has no code.
294 a.aapt.hasNoCode = !a.hasCode(ctx)
295
Jaewoong Jungde4c02f2019-01-22 11:19:56 -0800296 aaptLinkFlags := []string{}
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800297
Jaewoong Jungde4c02f2019-01-22 11:19:56 -0800298 // 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 -0800299 hasProduct := android.PrefixInList(a.aaptProperties.Aaptflags, "--product")
Colin Crosse78dcd32018-04-19 15:25:19 -0700300 if !hasProduct && len(ctx.Config().ProductAAPTCharacteristics()) > 0 {
Jaewoong Jungde4c02f2019-01-22 11:19:56 -0800301 aaptLinkFlags = append(aaptLinkFlags, "--product", ctx.Config().ProductAAPTCharacteristics())
Colin Crosse78dcd32018-04-19 15:25:19 -0700302 }
303
Dan Willemsen72be5902018-10-24 20:24:57 -0700304 if !Bool(a.aaptProperties.Aapt_include_all_resources) {
305 // Product AAPT config
306 for _, aaptConfig := range ctx.Config().ProductAAPTConfig() {
Jaewoong Jungde4c02f2019-01-22 11:19:56 -0800307 aaptLinkFlags = append(aaptLinkFlags, "-c", aaptConfig)
Dan Willemsen72be5902018-10-24 20:24:57 -0700308 }
Colin Crosse78dcd32018-04-19 15:25:19 -0700309
Dan Willemsen72be5902018-10-24 20:24:57 -0700310 // Product AAPT preferred config
311 if len(ctx.Config().ProductAAPTPreferredConfig()) > 0 {
Jaewoong Jungde4c02f2019-01-22 11:19:56 -0800312 aaptLinkFlags = append(aaptLinkFlags, "--preferred-density", ctx.Config().ProductAAPTPreferredConfig())
Dan Willemsen72be5902018-10-24 20:24:57 -0700313 }
Colin Crosse78dcd32018-04-19 15:25:19 -0700314 }
315
Jiyong Park7f67f482019-01-05 12:57:48 +0900316 manifestPackageName, overridden := ctx.DeviceConfig().OverrideManifestPackageNameFor(ctx.ModuleName())
Jaewoong Jung6f373f62019-03-13 10:13:24 -0700317 if overridden || a.overridableAppProperties.Package_name != nil {
318 // The product override variable has a priority over the package_name property.
319 if !overridden {
320 manifestPackageName = *a.overridableAppProperties.Package_name
321 }
Jaewoong Jungde4c02f2019-01-22 11:19:56 -0800322 aaptLinkFlags = append(aaptLinkFlags, "--rename-manifest-package "+manifestPackageName)
Jiyong Parkcfaa1642020-02-28 16:51:07 +0900323 a.overriddenManifestPackageName = manifestPackageName
Jiyong Park7f67f482019-01-05 12:57:48 +0900324 }
325
Jaewoong Jung4102e5d2019-02-27 16:26:28 -0800326 aaptLinkFlags = append(aaptLinkFlags, a.additionalAaptFlags...)
327
Colin Crosse560c4a2019-03-19 16:03:11 -0700328 a.aapt.splitNames = a.appProperties.Package_splits
Colin Cross50ddcc42019-05-16 12:28:22 -0700329 a.aapt.sdkLibraries = a.exportedSdkLibs
Baligh Uddin5b16dfb2020-02-11 17:27:19 -0800330 a.aapt.LoggingParent = String(a.overridableAppProperties.Logging_parent)
Jaewoong Jungde4c02f2019-01-22 11:19:56 -0800331 a.aapt.buildActions(ctx, sdkContext(a), aaptLinkFlags...)
Colin Cross30e076a2015-04-13 13:58:27 -0700332
Colin Cross46c9b8b2017-06-22 16:51:17 -0700333 // apps manifests are handled by aapt, don't let Module see them
Dan Willemsen34cc69e2015-09-23 15:26:20 -0700334 a.properties.Manifest = nil
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800335}
Colin Cross30e076a2015-04-13 13:58:27 -0700336
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800337func (a *AndroidApp) proguardBuildActions(ctx android.ModuleContext) {
Colin Cross89c31582018-04-30 15:55:11 -0700338 var staticLibProguardFlagFiles android.Paths
339 ctx.VisitDirectDeps(func(m android.Module) {
340 if lib, ok := m.(AndroidLibraryDependency); ok && ctx.OtherModuleDependencyTag(m) == staticLibTag {
341 staticLibProguardFlagFiles = append(staticLibProguardFlagFiles, lib.ExportedProguardFlagFiles()...)
342 }
343 })
344
345 staticLibProguardFlagFiles = android.FirstUniquePaths(staticLibProguardFlagFiles)
346
347 a.Module.extraProguardFlagFiles = append(a.Module.extraProguardFlagFiles, staticLibProguardFlagFiles...)
348 a.Module.extraProguardFlagFiles = append(a.Module.extraProguardFlagFiles, a.proguardOptionsFile)
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800349}
Colin Cross66dbc0b2017-12-28 12:23:20 -0800350
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800351func (a *AndroidApp) dexBuildActions(ctx android.ModuleContext) android.Path {
Colin Cross43f08db2018-11-12 10:13:39 -0800352
353 var installDir string
354 if ctx.ModuleName() == "framework-res" {
355 // framework-res.apk is installed as system/framework/framework-res.apk
356 installDir = "framework"
Jiyong Parkf7487312019-10-17 12:54:30 +0900357 } else if a.Privileged() {
Jaewoong Jung9d22a912019-01-23 16:27:47 -0800358 installDir = filepath.Join("priv-app", a.installApkName)
Colin Cross43f08db2018-11-12 10:13:39 -0800359 } else {
Jaewoong Jung9d22a912019-01-23 16:27:47 -0800360 installDir = filepath.Join("app", a.installApkName)
Colin Cross43f08db2018-11-12 10:13:39 -0800361 }
Jaewoong Jung9d22a912019-01-23 16:27:47 -0800362 a.dexpreopter.installPath = android.PathForModuleInstall(ctx, installDir, a.installApkName+".apk")
Nicolas Geoffrayfa6e9ec2019-02-12 13:12:16 +0000363 a.dexpreopter.uncompressedDex = a.shouldUncompressDex(ctx)
Colin Cross50ddcc42019-05-16 12:28:22 -0700364
365 a.dexpreopter.enforceUsesLibs = a.usesLibrary.enforceUsesLibraries()
366 a.dexpreopter.usesLibs = a.usesLibrary.usesLibraryProperties.Uses_libs
367 a.dexpreopter.optionalUsesLibs = a.usesLibrary.presentOptionalUsesLibs(ctx)
368 a.dexpreopter.libraryPaths = a.usesLibrary.usesLibraryPaths(ctx)
369 a.dexpreopter.manifestFile = a.mergedManifestFile
370
Nicolas Geoffrayfa6e9ec2019-02-12 13:12:16 +0000371 a.deviceProperties.UncompressDex = a.dexpreopter.uncompressedDex
Colin Cross5a0dcd52018-10-05 14:20:06 -0700372
Colin Cross5ab4e6d2017-11-22 16:20:45 -0800373 if ctx.ModuleName() != "framework-res" {
374 a.Module.compile(ctx, a.aaptSrcJar)
375 }
Colin Cross30e076a2015-04-13 13:58:27 -0700376
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800377 return a.maybeStrippedDexJarFile
378}
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800379
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800380func (a *AndroidApp) jniBuildActions(jniLibs []jniLib, ctx android.ModuleContext) android.WritablePath {
Colin Crossa4f08812018-10-02 22:03:40 -0700381 var jniJarFile android.WritablePath
Colin Crossa4f08812018-10-02 22:03:40 -0700382 if len(jniLibs) > 0 {
Jaewoong Jungbc625cd2019-05-06 15:48:44 -0700383 if a.shouldEmbedJnis(ctx) {
Colin Crossa4f08812018-10-02 22:03:40 -0700384 jniJarFile = android.PathForModuleOut(ctx, "jnilibs.zip")
Sasha Smundak6ad77252019-05-01 13:16:22 -0700385 TransformJniLibsToJar(ctx, jniJarFile, jniLibs, a.useEmbeddedNativeLibs(ctx))
Jaewoong Jung87a33e72020-03-26 14:01:48 -0700386 for _, jni := range jniLibs {
387 if jni.coverageFile.Valid() {
388 a.jniCoverageOutputs = append(a.jniCoverageOutputs, jni.coverageFile.Path())
389 }
390 }
Colin Crossa4f08812018-10-02 22:03:40 -0700391 } else {
392 a.installJniLibs = jniLibs
393 }
394 }
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800395 return jniJarFile
396}
Colin Crossa4f08812018-10-02 22:03:40 -0700397
Jaewoong Jung0949f312019-09-11 10:25:18 -0700398func (a *AndroidApp) noticeBuildActions(ctx android.ModuleContext) {
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700399 // Collect NOTICE files from all dependencies.
400 seenModules := make(map[android.Module]bool)
401 noticePathSet := make(map[android.Path]bool)
402
403 ctx.WalkDeps(func(child android.Module, parent android.Module) bool {
404 // Have we already seen this?
405 if _, ok := seenModules[child]; ok {
406 return false
407 }
408 seenModules[child] = true
409
410 // Skip host modules.
411 if child.Target().Os.Class == android.Host || child.Target().Os.Class == android.HostCross {
412 return false
413 }
414
Bob Badoura75b0572020-02-18 20:21:55 -0800415 paths := child.(android.Module).NoticeFiles()
416 if len(paths) > 0 {
417 for _, path := range paths {
418 noticePathSet[path] = true
419 }
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700420 }
421 return true
422 })
423
424 // If the app has one, add it too.
Bob Badoura75b0572020-02-18 20:21:55 -0800425 if len(a.NoticeFiles()) > 0 {
426 for _, path := range a.NoticeFiles() {
427 noticePathSet[path] = true
428 }
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700429 }
430
431 if len(noticePathSet) == 0 {
Jaewoong Jung98772792019-07-01 17:15:13 -0700432 return
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700433 }
434 var noticePaths []android.Path
435 for path := range noticePathSet {
436 noticePaths = append(noticePaths, path)
437 }
438 sort.Slice(noticePaths, func(i, j int) bool {
439 return noticePaths[i].String() < noticePaths[j].String()
440 })
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700441
Jaewoong Jung0949f312019-09-11 10:25:18 -0700442 a.noticeOutputs = android.BuildNoticeOutput(ctx, a.installDir, a.installApkName+".apk", noticePaths)
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700443}
444
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700445// Reads and prepends a main cert from the default cert dir if it hasn't been set already, i.e. it
446// isn't a cert module reference. Also checks and enforces system cert restriction if applicable.
447func processMainCert(m android.ModuleBase, certPropValue string, certificates []Certificate, ctx android.ModuleContext) []Certificate {
448 if android.SrcIsModule(certPropValue) == "" {
449 var mainCert Certificate
450 if certPropValue != "" {
451 defaultDir := ctx.Config().DefaultAppCertificateDir(ctx)
452 mainCert = Certificate{
Colin Cross503c1d02020-01-28 14:00:53 -0800453 Pem: defaultDir.Join(ctx, certPropValue+".x509.pem"),
454 Key: defaultDir.Join(ctx, certPropValue+".pk8"),
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700455 }
456 } else {
457 pem, key := ctx.Config().DefaultAppCertificate(ctx)
Colin Cross503c1d02020-01-28 14:00:53 -0800458 mainCert = Certificate{
459 Pem: pem,
460 Key: key,
461 }
Colin Crossbd01e2a2018-10-04 15:21:03 -0700462 }
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700463 certificates = append([]Certificate{mainCert}, certificates...)
Colin Crossbd01e2a2018-10-04 15:21:03 -0700464 }
465
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700466 if !m.Platform() {
467 certPath := certificates[0].Pem.String()
Jeongik Chac9464142019-01-07 12:07:27 +0900468 systemCertPath := ctx.Config().DefaultAppCertificateDir(ctx).String()
469 if strings.HasPrefix(certPath, systemCertPath) {
470 enforceSystemCert := ctx.Config().EnforceSystemCertificate()
471 whitelist := ctx.Config().EnforceSystemCertificateWhitelist()
472
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700473 if enforceSystemCert && !inList(m.Name(), whitelist) {
Jeongik Chac9464142019-01-07 12:07:27 +0900474 ctx.PropertyErrorf("certificate", "The module in product partition cannot be signed with certificate in system.")
475 }
476 }
477 }
478
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700479 return certificates
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800480}
481
Jooyung Han39ee1192020-03-23 20:21:11 +0900482func (a *AndroidApp) InstallApkName() string {
483 return a.installApkName
484}
485
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800486func (a *AndroidApp) generateAndroidBuildActions(ctx android.ModuleContext) {
Colin Cross50ddcc42019-05-16 12:28:22 -0700487 var apkDeps android.Paths
488
Jeongik Cha538c0d02019-07-11 15:54:27 +0900489 a.aapt.useEmbeddedNativeLibs = a.useEmbeddedNativeLibs(ctx)
490 a.aapt.useEmbeddedDex = Bool(a.appProperties.Use_embedded_dex)
491
Jaewoong Jung9d22a912019-01-23 16:27:47 -0800492 // Check if the install APK name needs to be overridden.
Jaewoong Jung525443a2019-02-28 15:35:54 -0800493 a.installApkName = ctx.DeviceConfig().OverridePackageNameFor(a.Name())
Jaewoong Jung9d22a912019-01-23 16:27:47 -0800494
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700495 if ctx.ModuleName() == "framework-res" {
496 // framework-res.apk is installed as system/framework/framework-res.apk
Jaewoong Jung0949f312019-09-11 10:25:18 -0700497 a.installDir = android.PathForModuleInstall(ctx, "framework")
Jiyong Parkf7487312019-10-17 12:54:30 +0900498 } else if a.Privileged() {
Jaewoong Jung0949f312019-09-11 10:25:18 -0700499 a.installDir = android.PathForModuleInstall(ctx, "priv-app", a.installApkName)
500 } else if ctx.InstallInTestcases() {
Jaewoong Jung326a9412019-11-21 10:41:00 -0800501 a.installDir = android.PathForModuleInstall(ctx, a.installApkName, ctx.DeviceConfig().DeviceArch())
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700502 } else {
Jaewoong Jung0949f312019-09-11 10:25:18 -0700503 a.installDir = android.PathForModuleInstall(ctx, "app", a.installApkName)
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700504 }
Jaewoong Jung7dd4ae22019-09-27 17:13:15 -0700505 a.onDeviceDir = android.InstallPathToOnDevicePath(ctx, a.installDir)
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700506
Jaewoong Jung0949f312019-09-11 10:25:18 -0700507 a.noticeBuildActions(ctx)
Jaewoong Jung98772792019-07-01 17:15:13 -0700508 if Bool(a.appProperties.Embed_notices) || ctx.Config().IsEnvTrue("ALWAYS_EMBED_NOTICES") {
509 a.aapt.noticeFile = a.noticeOutputs.HtmlGzOutput
510 }
Jaewoong Jung5b425e22019-06-17 17:40:56 -0700511
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800512 // Process all building blocks, from AAPT to certificates.
513 a.aaptBuildActions(ctx)
514
Colin Cross50ddcc42019-05-16 12:28:22 -0700515 if a.usesLibrary.enforceUsesLibraries() {
516 manifestCheckFile := a.usesLibrary.verifyUsesLibrariesManifest(ctx, a.mergedManifestFile)
517 apkDeps = append(apkDeps, manifestCheckFile)
518 }
519
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800520 a.proguardBuildActions(ctx)
521
522 dexJarFile := a.dexBuildActions(ctx)
523
Colin Cross094cde42020-02-15 10:38:00 -0800524 jniLibs, certificateDeps := collectAppDeps(ctx, a.shouldEmbedJnis(ctx), !Bool(a.appProperties.Jni_uses_platform_apis))
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800525 jniJarFile := a.jniBuildActions(jniLibs, ctx)
526
527 if ctx.Failed() {
528 return
529 }
530
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700531 certificates := processMainCert(a.ModuleBase, a.getCertString(ctx), certificateDeps, ctx)
532 a.certificate = certificates[0]
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800533
534 // Build a final signed app package.
Jaewoong Jung5a498812019-11-07 14:14:38 -0800535 packageFile := android.PathForModuleOut(ctx, a.installApkName+".apk")
Songchun Fan17d69e32020-03-24 20:32:24 -0700536 v4SigningRequested := Bool(a.Module.deviceProperties.V4_signature)
537 var v4SignatureFile android.WritablePath = nil
538 if v4SigningRequested {
539 v4SignatureFile = android.PathForModuleOut(ctx, a.installApkName+".apk.idsig")
540 }
541 CreateAndSignAppPackage(ctx, packageFile, a.exportPackage, jniJarFile, dexJarFile, certificates, apkDeps, v4SignatureFile)
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800542 a.outputFile = packageFile
Songchun Fan17d69e32020-03-24 20:32:24 -0700543 if v4SigningRequested {
544 a.extraOutputFiles = append(a.extraOutputFiles, v4SignatureFile)
545 }
Colin Cross3bc7ffa2017-11-22 16:19:37 -0800546
Colin Crosse560c4a2019-03-19 16:03:11 -0700547 for _, split := range a.aapt.splits {
548 // Sign the split APKs
Jaewoong Jung5a498812019-11-07 14:14:38 -0800549 packageFile := android.PathForModuleOut(ctx, a.installApkName+"_"+split.suffix+".apk")
Songchun Fan17d69e32020-03-24 20:32:24 -0700550 if v4SigningRequested {
551 v4SignatureFile = android.PathForModuleOut(ctx, a.installApkName+"_"+split.suffix+".apk.idsig")
552 }
553 CreateAndSignAppPackage(ctx, packageFile, split.path, nil, nil, certificates, apkDeps, v4SignatureFile)
Colin Crosse560c4a2019-03-19 16:03:11 -0700554 a.extraOutputFiles = append(a.extraOutputFiles, packageFile)
Songchun Fan17d69e32020-03-24 20:32:24 -0700555 if v4SigningRequested {
556 a.extraOutputFiles = append(a.extraOutputFiles, v4SignatureFile)
557 }
Colin Crosse560c4a2019-03-19 16:03:11 -0700558 }
559
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800560 // Build an app bundle.
Colin Crossf6237212018-10-29 23:14:58 -0700561 bundleFile := android.PathForModuleOut(ctx, "base.zip")
562 BuildBundleModule(ctx, bundleFile, a.exportPackage, jniJarFile, dexJarFile)
563 a.bundleFile = bundleFile
564
Jaewoong Jung590b1ae2019-01-22 16:40:58 -0800565 // Install the app package.
Jiyong Park8ba50f92019-11-13 15:01:01 +0900566 if (Bool(a.Module.properties.Installable) || ctx.Host()) && a.IsForPlatform() {
567 ctx.InstallFile(a.installDir, a.outputFile.Base(), a.outputFile)
568 for _, extra := range a.extraOutputFiles {
569 ctx.InstallFile(a.installDir, extra.Base(), extra)
570 }
Colin Cross5ab4e6d2017-11-22 16:20:45 -0800571 }
Colin Cross30e076a2015-04-13 13:58:27 -0700572}
573
Colin Cross094cde42020-02-15 10:38:00 -0800574func collectAppDeps(ctx android.ModuleContext, shouldCollectRecursiveNativeDeps bool,
575 checkNativeSdkVersion bool) ([]jniLib, []Certificate) {
Colin Crossa4f08812018-10-02 22:03:40 -0700576 var jniLibs []jniLib
Jiyong Parkc00cbd92018-10-30 21:20:05 +0900577 var certificates []Certificate
Peter Collingbournead84f972019-12-17 16:46:18 -0800578 seenModulePaths := make(map[string]bool)
Colin Crossa4f08812018-10-02 22:03:40 -0700579
Peter Collingbournead84f972019-12-17 16:46:18 -0800580 ctx.WalkDeps(func(module android.Module, parent android.Module) bool {
Colin Crossa4f08812018-10-02 22:03:40 -0700581 otherName := ctx.OtherModuleName(module)
582 tag := ctx.OtherModuleDependencyTag(module)
583
Peter Collingbournead84f972019-12-17 16:46:18 -0800584 if IsJniDepTag(tag) || tag == cc.SharedDepTag {
Colin Crossa4f08812018-10-02 22:03:40 -0700585 if dep, ok := module.(*cc.Module); ok {
Peter Collingbournead84f972019-12-17 16:46:18 -0800586 if dep.IsNdk() || dep.IsStubs() {
587 return false
588 }
589
Colin Crossa4f08812018-10-02 22:03:40 -0700590 lib := dep.OutputFile()
Peter Collingbournead84f972019-12-17 16:46:18 -0800591 path := lib.Path()
592 if seenModulePaths[path.String()] {
593 return false
594 }
595 seenModulePaths[path.String()] = true
596
Colin Cross094cde42020-02-15 10:38:00 -0800597 if checkNativeSdkVersion {
598 if app, ok := ctx.Module().(interface{ sdkVersion() sdkSpec }); ok {
599 if app.sdkVersion().specified() &&
600 app.sdkVersion().kind != sdkCorePlatform &&
601 dep.SdkVersion() == "" {
602 ctx.PropertyErrorf("jni_libs",
603 "JNI dependency %q uses platform APIs, but this module does not",
604 otherName)
605 }
606 }
607 }
608
Colin Crossa4f08812018-10-02 22:03:40 -0700609 if lib.Valid() {
610 jniLibs = append(jniLibs, jniLib{
Jaewoong Jung87a33e72020-03-26 14:01:48 -0700611 name: ctx.OtherModuleName(module),
612 path: path,
613 target: module.Target(),
614 coverageFile: dep.CoverageOutputFile(),
Colin Crossa4f08812018-10-02 22:03:40 -0700615 })
616 } else {
617 ctx.ModuleErrorf("dependency %q missing output file", otherName)
618 }
619 } else {
620 ctx.ModuleErrorf("jni_libs dependency %q must be a cc library", otherName)
Colin Crossa4f08812018-10-02 22:03:40 -0700621 }
Peter Collingbournead84f972019-12-17 16:46:18 -0800622
623 return shouldCollectRecursiveNativeDeps
624 }
625
626 if tag == certificateTag {
Colin Crossbd01e2a2018-10-04 15:21:03 -0700627 if dep, ok := module.(*AndroidAppCertificate); ok {
Jiyong Parkc00cbd92018-10-30 21:20:05 +0900628 certificates = append(certificates, dep.Certificate)
Colin Crossbd01e2a2018-10-04 15:21:03 -0700629 } else {
630 ctx.ModuleErrorf("certificate dependency %q must be an android_app_certificate module", otherName)
631 }
Colin Crossa4f08812018-10-02 22:03:40 -0700632 }
Peter Collingbournead84f972019-12-17 16:46:18 -0800633
634 return false
Colin Crossa4f08812018-10-02 22:03:40 -0700635 })
636
Colin Crossbd01e2a2018-10-04 15:21:03 -0700637 return jniLibs, certificates
Colin Crossa4f08812018-10-02 22:03:40 -0700638}
639
Colin Cross0ea8ba82019-06-06 14:33:29 -0700640func (a *AndroidApp) getCertString(ctx android.BaseModuleContext) string {
Jaewoong Jung2ad817c2019-01-18 14:27:16 -0800641 certificate, overridden := ctx.DeviceConfig().OverrideCertificateFor(ctx.ModuleName())
642 if overridden {
Jaewoong Jungacb6db32019-02-28 16:22:30 +0000643 return ":" + certificate
Jaewoong Jung2ad817c2019-01-18 14:27:16 -0800644 }
Jaewoong Jung525443a2019-02-28 15:35:54 -0800645 return String(a.overridableAppProperties.Certificate)
Jaewoong Jung2ad817c2019-01-18 14:27:16 -0800646}
647
Jiyong Park0f80c182020-01-31 02:49:53 +0900648func (a *AndroidApp) DepIsInSameApex(ctx android.BaseModuleContext, dep android.Module) bool {
649 if IsJniDepTag(ctx.OtherModuleDependencyTag(dep)) {
650 return true
651 }
652 return a.Library.DepIsInSameApex(ctx, dep)
653}
654
Jiyong Parkb7c639e2019-08-19 14:56:02 +0900655// For OutputFileProducer interface
656func (a *AndroidApp) OutputFiles(tag string) (android.Paths, error) {
657 switch tag {
658 case ".aapt.srcjar":
659 return []android.Path{a.aaptSrcJar}, nil
660 }
661 return a.Library.OutputFiles(tag)
662}
663
Jiyong Parkf7487312019-10-17 12:54:30 +0900664func (a *AndroidApp) Privileged() bool {
665 return Bool(a.appProperties.Privileged)
666}
667
Jaewoong Jung87a33e72020-03-26 14:01:48 -0700668func (a *AndroidApp) IsNativeCoverageNeeded(ctx android.BaseModuleContext) bool {
669 return ctx.Device() && (ctx.DeviceConfig().NativeCoverageEnabled() || ctx.DeviceConfig().ClangCoverageEnabled())
670}
671
672func (a *AndroidApp) PreventInstall() {
673 a.appProperties.PreventInstall = true
674}
675
676func (a *AndroidApp) HideFromMake() {
677 a.appProperties.HideFromMake = true
678}
679
680func (a *AndroidApp) MarkAsCoverageVariant(coverage bool) {
681 a.appProperties.IsCoverageVariant = coverage
682}
683
684var _ cc.Coverage = (*AndroidApp)(nil)
685
Colin Cross1b16b0e2019-02-12 14:41:32 -0800686// android_app compiles sources and Android resources into an Android application package `.apk` file.
Colin Cross36242852017-06-23 15:06:31 -0700687func AndroidAppFactory() android.Module {
Colin Cross30e076a2015-04-13 13:58:27 -0700688 module := &AndroidApp{}
689
Sasha Smundak2057f822019-04-16 17:16:58 -0700690 module.Module.deviceProperties.Optimize.EnabledByDefault = true
Colin Cross66dbc0b2017-12-28 12:23:20 -0800691 module.Module.deviceProperties.Optimize.Shrink = proptools.BoolPtr(true)
692
Colin Crossae5caf52018-05-22 11:11:52 -0700693 module.Module.properties.Instrument = true
Colin Cross9ae1b922018-06-26 17:59:05 -0700694 module.Module.properties.Installable = proptools.BoolPtr(true)
Colin Crossae5caf52018-05-22 11:11:52 -0700695
Colin Cross36242852017-06-23 15:06:31 -0700696 module.AddProperties(
Colin Cross540eff82017-06-22 17:01:52 -0700697 &module.Module.properties,
698 &module.Module.deviceProperties,
Colin Cross43f08db2018-11-12 10:13:39 -0800699 &module.Module.dexpreoptProperties,
Colin Crossa97c5d32018-03-28 14:58:31 -0700700 &module.Module.protoProperties,
701 &module.aaptProperties,
Jaewoong Jung525443a2019-02-28 15:35:54 -0800702 &module.appProperties,
Colin Cross50ddcc42019-05-16 12:28:22 -0700703 &module.overridableAppProperties,
704 &module.usesLibrary.usesLibraryProperties)
Colin Cross36242852017-06-23 15:06:31 -0700705
Colin Crossa9d8bee2018-10-02 13:59:46 -0700706 module.Prefer32(func(ctx android.BaseModuleContext, base *android.ModuleBase, class android.OsClass) bool {
707 return class == android.Device && ctx.Config().DevicePrefer32BitApps()
708 })
709
Colin Crossa4f08812018-10-02 22:03:40 -0700710 android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibCommon)
711 android.InitDefaultableModule(module)
Jaewoong Jung525443a2019-02-28 15:35:54 -0800712 android.InitOverridableModule(module, &module.appProperties.Overrides)
Jiyong Park52cd06f2019-11-11 10:14:32 +0900713 android.InitApexModule(module)
Colin Crossa4f08812018-10-02 22:03:40 -0700714
Colin Cross36242852017-06-23 15:06:31 -0700715 return module
Colin Cross30e076a2015-04-13 13:58:27 -0700716}
Colin Crossae5caf52018-05-22 11:11:52 -0700717
718type appTestProperties struct {
719 Instrumentation_for *string
Jaewoong Jung26dedd32019-06-06 08:45:58 -0700720
721 // if specified, the instrumentation target package name in the manifest is overwritten by it.
722 Instrumentation_target_package *string
Colin Crossae5caf52018-05-22 11:11:52 -0700723}
724
725type AndroidTest struct {
726 AndroidApp
727
728 appTestProperties appTestProperties
729
730 testProperties testProperties
Colin Cross303e21f2018-08-07 16:49:25 -0700731
732 testConfig android.Path
Colin Crossd96ca352018-08-10 16:06:24 -0700733 data android.Paths
Colin Crossae5caf52018-05-22 11:11:52 -0700734}
735
Jaewoong Jung0949f312019-09-11 10:25:18 -0700736func (a *AndroidTest) InstallInTestcases() bool {
737 return true
738}
739
Colin Crossae5caf52018-05-22 11:11:52 -0700740func (a *AndroidTest) GenerateAndroidBuildActions(ctx android.ModuleContext) {
Jaewoong Jung26dedd32019-06-06 08:45:58 -0700741 if a.appTestProperties.Instrumentation_target_package != nil {
742 a.additionalAaptFlags = append(a.additionalAaptFlags,
743 "--rename-instrumentation-target-package "+*a.appTestProperties.Instrumentation_target_package)
744 } else if a.appTestProperties.Instrumentation_for != nil {
745 // Check if the instrumentation target package is overridden.
Jaewoong Jung4102e5d2019-02-27 16:26:28 -0800746 manifestPackageName, overridden := ctx.DeviceConfig().OverrideManifestPackageNameFor(*a.appTestProperties.Instrumentation_for)
747 if overridden {
748 a.additionalAaptFlags = append(a.additionalAaptFlags, "--rename-instrumentation-target-package "+manifestPackageName)
749 }
750 }
Colin Crossae5caf52018-05-22 11:11:52 -0700751 a.generateAndroidBuildActions(ctx)
Colin Cross303e21f2018-08-07 16:49:25 -0700752
Jaewoong Jung39982342020-01-14 10:27:18 -0800753 testConfig := tradefed.AutoGenInstrumentationTestConfig(ctx, a.testProperties.Test_config,
Dan Shi6ffaaa82019-09-26 11:41:36 -0700754 a.testProperties.Test_config_template, a.manifestPath, a.testProperties.Test_suites, a.testProperties.Auto_gen_config)
Jaewoong Jung39982342020-01-14 10:27:18 -0800755 a.testConfig = a.FixTestConfig(ctx, testConfig)
Colin Cross8a497952019-03-05 22:25:09 -0800756 a.data = android.PathsForModuleSrc(ctx, a.testProperties.Data)
Colin Cross303e21f2018-08-07 16:49:25 -0700757}
758
Jaewoong Jung39982342020-01-14 10:27:18 -0800759func (a *AndroidTest) FixTestConfig(ctx android.ModuleContext, testConfig android.Path) android.Path {
760 if testConfig == nil {
761 return nil
762 }
763
764 fixedConfig := android.PathForModuleOut(ctx, "test_config_fixer", "AndroidTest.xml")
765 rule := android.NewRuleBuilder()
766 command := rule.Command().BuiltTool(ctx, "test_config_fixer").Input(testConfig).Output(fixedConfig)
767 fixNeeded := false
768
769 if ctx.ModuleName() != a.installApkName {
770 fixNeeded = true
771 command.FlagWithArg("--test-file-name ", a.installApkName+".apk")
772 }
773
774 if a.overridableAppProperties.Package_name != nil {
775 fixNeeded = true
776 command.FlagWithInput("--manifest ", a.manifestPath).
777 FlagWithArg("--package-name ", *a.overridableAppProperties.Package_name)
778 }
779
780 if fixNeeded {
781 rule.Build(pctx, ctx, "fix_test_config", "fix test config")
782 return fixedConfig
783 }
784 return testConfig
785}
786
Colin Cross303e21f2018-08-07 16:49:25 -0700787func (a *AndroidTest) DepsMutator(ctx android.BottomUpMutatorContext) {
Colin Cross303e21f2018-08-07 16:49:25 -0700788 a.AndroidApp.DepsMutator(ctx)
Jaewoong Jung26dedd32019-06-06 08:45:58 -0700789}
790
791func (a *AndroidTest) OverridablePropertiesDepsMutator(ctx android.BottomUpMutatorContext) {
792 a.AndroidApp.OverridablePropertiesDepsMutator(ctx)
Colin Cross4b964c02018-10-15 16:18:06 -0700793 if a.appTestProperties.Instrumentation_for != nil {
794 // The android_app dependency listed in instrumentation_for needs to be added to the classpath for javac,
795 // but not added to the aapt2 link includes like a normal android_app or android_library dependency, so
796 // use instrumentationForTag instead of libTag.
797 ctx.AddVariationDependencies(nil, instrumentationForTag, String(a.appTestProperties.Instrumentation_for))
798 }
Colin Crossae5caf52018-05-22 11:11:52 -0700799}
800
Colin Cross1b16b0e2019-02-12 14:41:32 -0800801// android_test compiles test sources and Android resources into an Android application package `.apk` file and
802// creates an `AndroidTest.xml` file to allow running the test with `atest` or a `TEST_MAPPING` file.
Colin Crossae5caf52018-05-22 11:11:52 -0700803func AndroidTestFactory() android.Module {
804 module := &AndroidTest{}
805
Sasha Smundak2057f822019-04-16 17:16:58 -0700806 module.Module.deviceProperties.Optimize.EnabledByDefault = true
Colin Cross5067db92018-09-17 16:46:35 -0700807
808 module.Module.properties.Instrument = true
Colin Cross9ae1b922018-06-26 17:59:05 -0700809 module.Module.properties.Installable = proptools.BoolPtr(true)
Colin Crosse4246ab2019-02-05 21:55:21 -0800810 module.appProperties.Use_embedded_native_libs = proptools.BoolPtr(true)
Colin Cross47fa9d32019-03-26 10:51:39 -0700811 module.appProperties.AlwaysPackageNativeLibs = true
Colin Cross43f08db2018-11-12 10:13:39 -0800812 module.Module.dexpreopter.isTest = true
Colin Crossae5caf52018-05-22 11:11:52 -0700813
814 module.AddProperties(
815 &module.Module.properties,
816 &module.Module.deviceProperties,
Colin Cross43f08db2018-11-12 10:13:39 -0800817 &module.Module.dexpreoptProperties,
Colin Crossae5caf52018-05-22 11:11:52 -0700818 &module.Module.protoProperties,
819 &module.aaptProperties,
820 &module.appProperties,
Dan Willemsenf5531d22018-07-16 17:21:19 -0700821 &module.appTestProperties,
Jaewoong Jung525443a2019-02-28 15:35:54 -0800822 &module.overridableAppProperties,
Colin Cross50ddcc42019-05-16 12:28:22 -0700823 &module.usesLibrary.usesLibraryProperties,
Dan Willemsenf5531d22018-07-16 17:21:19 -0700824 &module.testProperties)
Colin Crossae5caf52018-05-22 11:11:52 -0700825
Colin Crossa4f08812018-10-02 22:03:40 -0700826 android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibCommon)
827 android.InitDefaultableModule(module)
Jaewoong Jung26dedd32019-06-06 08:45:58 -0700828 android.InitOverridableModule(module, &module.appProperties.Overrides)
Colin Crossae5caf52018-05-22 11:11:52 -0700829 return module
830}
Colin Crossbd01e2a2018-10-04 15:21:03 -0700831
Colin Cross252fc6f2018-10-04 15:22:03 -0700832type appTestHelperAppProperties struct {
833 // list of compatibility suites (for example "cts", "vts") that the module should be
834 // installed into.
835 Test_suites []string `android:"arch_variant"`
Dan Shi6ffaaa82019-09-26 11:41:36 -0700836
837 // Flag to indicate whether or not to create test config automatically. If AndroidTest.xml
838 // doesn't exist next to the Android.bp, this attribute doesn't need to be set to true
839 // explicitly.
840 Auto_gen_config *bool
Colin Cross252fc6f2018-10-04 15:22:03 -0700841}
842
843type AndroidTestHelperApp struct {
844 AndroidApp
845
846 appTestHelperAppProperties appTestHelperAppProperties
847}
848
Jaewoong Jung326a9412019-11-21 10:41:00 -0800849func (a *AndroidTestHelperApp) InstallInTestcases() bool {
850 return true
851}
852
Colin Cross1b16b0e2019-02-12 14:41:32 -0800853// android_test_helper_app compiles sources and Android resources into an Android application package `.apk` file that
854// will be used by tests, but does not produce an `AndroidTest.xml` file so the module will not be run directly as a
855// test.
Colin Cross252fc6f2018-10-04 15:22:03 -0700856func AndroidTestHelperAppFactory() android.Module {
857 module := &AndroidTestHelperApp{}
858
Sasha Smundak2057f822019-04-16 17:16:58 -0700859 module.Module.deviceProperties.Optimize.EnabledByDefault = true
Colin Cross252fc6f2018-10-04 15:22:03 -0700860
861 module.Module.properties.Installable = proptools.BoolPtr(true)
Colin Crosse4246ab2019-02-05 21:55:21 -0800862 module.appProperties.Use_embedded_native_libs = proptools.BoolPtr(true)
Colin Cross47fa9d32019-03-26 10:51:39 -0700863 module.appProperties.AlwaysPackageNativeLibs = true
Colin Cross43f08db2018-11-12 10:13:39 -0800864 module.Module.dexpreopter.isTest = true
Colin Cross252fc6f2018-10-04 15:22:03 -0700865
866 module.AddProperties(
867 &module.Module.properties,
868 &module.Module.deviceProperties,
Colin Cross43f08db2018-11-12 10:13:39 -0800869 &module.Module.dexpreoptProperties,
Colin Cross252fc6f2018-10-04 15:22:03 -0700870 &module.Module.protoProperties,
871 &module.aaptProperties,
872 &module.appProperties,
Jaewoong Jung525443a2019-02-28 15:35:54 -0800873 &module.appTestHelperAppProperties,
Colin Cross50ddcc42019-05-16 12:28:22 -0700874 &module.overridableAppProperties,
875 &module.usesLibrary.usesLibraryProperties)
Colin Cross252fc6f2018-10-04 15:22:03 -0700876
877 android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibCommon)
878 android.InitDefaultableModule(module)
Anton Hansson3d2b6b42020-01-10 15:06:01 +0000879 android.InitApexModule(module)
Colin Cross252fc6f2018-10-04 15:22:03 -0700880 return module
881}
882
Colin Crossbd01e2a2018-10-04 15:21:03 -0700883type AndroidAppCertificate struct {
884 android.ModuleBase
885 properties AndroidAppCertificateProperties
Jiyong Parkc00cbd92018-10-30 21:20:05 +0900886 Certificate Certificate
Colin Crossbd01e2a2018-10-04 15:21:03 -0700887}
888
889type AndroidAppCertificateProperties struct {
890 // Name of the certificate files. Extensions .x509.pem and .pk8 will be added to the name.
891 Certificate *string
892}
893
Colin Cross1b16b0e2019-02-12 14:41:32 -0800894// android_app_certificate modules can be referenced by the certificates property of android_app modules to select
895// the signing key.
Colin Crossbd01e2a2018-10-04 15:21:03 -0700896func AndroidAppCertificateFactory() android.Module {
897 module := &AndroidAppCertificate{}
898 module.AddProperties(&module.properties)
899 android.InitAndroidModule(module)
900 return module
901}
902
Colin Crossbd01e2a2018-10-04 15:21:03 -0700903func (c *AndroidAppCertificate) GenerateAndroidBuildActions(ctx android.ModuleContext) {
904 cert := String(c.properties.Certificate)
Jiyong Parkc00cbd92018-10-30 21:20:05 +0900905 c.Certificate = Certificate{
Colin Cross503c1d02020-01-28 14:00:53 -0800906 Pem: android.PathForModuleSrc(ctx, cert+".x509.pem"),
907 Key: android.PathForModuleSrc(ctx, cert+".pk8"),
Colin Crossbd01e2a2018-10-04 15:21:03 -0700908 }
909}
Jaewoong Jung525443a2019-02-28 15:35:54 -0800910
911type OverrideAndroidApp struct {
912 android.ModuleBase
913 android.OverrideModuleBase
914}
915
916func (i *OverrideAndroidApp) GenerateAndroidBuildActions(ctx android.ModuleContext) {
917 // All the overrides happen in the base module.
918 // TODO(jungjw): Check the base module type.
919}
920
921// override_android_app is used to create an android_app module based on another android_app by overriding
922// some of its properties.
923func OverrideAndroidAppModuleFactory() android.Module {
924 m := &OverrideAndroidApp{}
925 m.AddProperties(&overridableAppProperties{})
926
Jaewoong Jungb639a6a2019-05-10 15:16:29 -0700927 android.InitAndroidMultiTargetsArchModule(m, android.DeviceSupported, android.MultilibCommon)
Jaewoong Jung525443a2019-02-28 15:35:54 -0800928 android.InitOverrideModule(m)
929 return m
930}
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700931
Jaewoong Jung26dedd32019-06-06 08:45:58 -0700932type OverrideAndroidTest struct {
933 android.ModuleBase
934 android.OverrideModuleBase
935}
936
937func (i *OverrideAndroidTest) GenerateAndroidBuildActions(ctx android.ModuleContext) {
938 // All the overrides happen in the base module.
939 // TODO(jungjw): Check the base module type.
940}
941
942// override_android_test is used to create an android_app module based on another android_test by overriding
943// some of its properties.
944func OverrideAndroidTestModuleFactory() android.Module {
945 m := &OverrideAndroidTest{}
946 m.AddProperties(&overridableAppProperties{})
947 m.AddProperties(&appTestProperties{})
948
949 android.InitAndroidMultiTargetsArchModule(m, android.DeviceSupported, android.MultilibCommon)
950 android.InitOverrideModule(m)
951 return m
952}
953
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700954type AndroidAppImport struct {
955 android.ModuleBase
956 android.DefaultableModuleBase
957 prebuilt android.Prebuilt
958
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -0700959 properties AndroidAppImportProperties
960 dpiVariants interface{}
961 archVariants interface{}
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700962
963 outputFile android.Path
Colin Cross503c1d02020-01-28 14:00:53 -0800964 certificate Certificate
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700965
966 dexpreopter
Colin Cross50ddcc42019-05-16 12:28:22 -0700967
968 usesLibrary usesLibrary
Jaewoong Jung8aae22e2019-07-17 10:21:49 -0700969
Colin Cross70dda7e2019-10-01 22:05:35 -0700970 installPath android.InstallPath
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700971}
972
973type AndroidAppImportProperties struct {
974 // A prebuilt apk to import
Jaewoong Jung3e18b192019-06-11 12:25:34 -0700975 Apk *string
Jaewoong Junga5e5abc2019-04-26 14:31:50 -0700976
Jaewoong Jung961d4fd2019-08-22 14:25:58 -0700977 // The name of a certificate in the default certificate directory or an android_app_certificate
978 // module name in the form ":module". Should be empty if presigned or default_dev_cert is set.
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700979 Certificate *string
980
981 // Set this flag to true if the prebuilt apk is already signed. The certificate property must not
982 // be set for presigned modules.
983 Presigned *bool
984
Jaewoong Jung961d4fd2019-08-22 14:25:58 -0700985 // Sign with the default system dev certificate. Must be used judiciously. Most imported apps
986 // need to either specify a specific certificate or be presigned.
987 Default_dev_cert *bool
988
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700989 // Specifies that this app should be installed to the priv-app directory,
990 // where the system will grant it additional privileges not available to
991 // normal apps.
992 Privileged *bool
993
994 // Names of modules to be overridden. Listed modules can only be other binaries
995 // (in Make or Soong).
996 // This does not completely prevent installation of the overridden binaries, but if both
997 // binaries would be installed by default (in PRODUCT_PACKAGES) the other binary will be removed
998 // from PRODUCT_PACKAGES.
999 Overrides []string
Jaewoong Jung8aae22e2019-07-17 10:21:49 -07001000
1001 // Optional name for the installed app. If unspecified, it is derived from the module name.
1002 Filename *string
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001003}
1004
Martin Stjernholm6d415272020-01-31 17:10:36 +00001005func (a *AndroidAppImport) IsInstallable() bool {
1006 return true
1007}
1008
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001009// Updates properties with variant-specific values.
1010func (a *AndroidAppImport) processVariants(ctx android.LoadHookContext) {
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001011 config := ctx.Config()
1012
1013 dpiProps := reflect.ValueOf(a.dpiVariants).Elem().FieldByName("Dpi_variants")
1014 // Try DPI variant matches in the reverse-priority order so that the highest priority match
1015 // overwrites everything else.
1016 // TODO(jungjw): Can we optimize this by making it priority order?
1017 for i := len(config.ProductAAPTPrebuiltDPI()) - 1; i >= 0; i-- {
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001018 MergePropertiesFromVariant(ctx, &a.properties, dpiProps, config.ProductAAPTPrebuiltDPI()[i])
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001019 }
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001020 if config.ProductAAPTPreferredConfig() != "" {
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001021 MergePropertiesFromVariant(ctx, &a.properties, dpiProps, config.ProductAAPTPreferredConfig())
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001022 }
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001023
1024 archProps := reflect.ValueOf(a.archVariants).Elem().FieldByName("Arch")
1025 archType := ctx.Config().Targets[android.Android][0].Arch.ArchType
1026 MergePropertiesFromVariant(ctx, &a.properties, archProps, archType.Name)
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001027}
1028
Colin Cross1184b642019-12-30 18:43:07 -08001029func MergePropertiesFromVariant(ctx android.EarlyModuleContext,
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001030 dst interface{}, variantGroup reflect.Value, variant string) {
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001031 src := variantGroup.FieldByName(proptools.FieldNameForProperty(variant))
1032 if !src.IsValid() {
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001033 return
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001034 }
1035
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001036 err := proptools.ExtendMatchingProperties([]interface{}{dst}, src.Interface(), nil, proptools.OrderAppend)
1037 if err != nil {
1038 if propertyErr, ok := err.(*proptools.ExtendPropertyError); ok {
1039 ctx.PropertyErrorf(propertyErr.Property, "%s", propertyErr.Err.Error())
1040 } else {
1041 panic(err)
1042 }
1043 }
Jaewoong Junga5e5abc2019-04-26 14:31:50 -07001044}
1045
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001046func (a *AndroidAppImport) DepsMutator(ctx android.BottomUpMutatorContext) {
1047 cert := android.SrcIsModule(String(a.properties.Certificate))
1048 if cert != "" {
1049 ctx.AddDependency(ctx.Module(), certificateTag, cert)
1050 }
Colin Cross50ddcc42019-05-16 12:28:22 -07001051
Paul Duffin250e6192019-06-07 10:44:37 +01001052 a.usesLibrary.deps(ctx, true)
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001053}
1054
1055func (a *AndroidAppImport) uncompressEmbeddedJniLibs(
1056 ctx android.ModuleContext, inputPath android.Path, outputPath android.OutputPath) {
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08001057 // Test apps don't need their JNI libraries stored uncompressed. As a matter of fact, messing
1058 // with them may invalidate pre-existing signature data.
1059 if ctx.InstallInTestcases() && Bool(a.properties.Presigned) {
1060 ctx.Build(pctx, android.BuildParams{
1061 Rule: android.Cp,
1062 Output: outputPath,
1063 Input: inputPath,
1064 })
1065 return
1066 }
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001067 rule := android.NewRuleBuilder()
1068 rule.Command().
1069 Textf(`if (zipinfo %s 'lib/*.so' 2>/dev/null | grep -v ' stor ' >/dev/null) ; then`, inputPath).
Colin Crossee94d6a2019-07-08 17:08:34 -07001070 BuiltTool(ctx, "zip2zip").
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001071 FlagWithInput("-i ", inputPath).
1072 FlagWithOutput("-o ", outputPath).
1073 FlagWithArg("-0 ", "'lib/**/*.so'").
1074 Textf(`; else cp -f %s %s; fi`, inputPath, outputPath)
1075 rule.Build(pctx, ctx, "uncompress-embedded-jni-libs", "Uncompress embedded JIN libs")
1076}
1077
Jaewoong Jungacf18d72019-05-02 14:55:29 -07001078// Returns whether this module should have the dex file stored uncompressed in the APK.
1079func (a *AndroidAppImport) shouldUncompressDex(ctx android.ModuleContext) bool {
1080 if ctx.Config().UnbundledBuild() {
1081 return false
1082 }
1083
1084 // Uncompress dex in APKs of privileged apps
Jiyong Parkf7487312019-10-17 12:54:30 +09001085 if ctx.Config().UncompressPrivAppDex() && a.Privileged() {
Jaewoong Jungacf18d72019-05-02 14:55:29 -07001086 return true
1087 }
1088
1089 return shouldUncompressDex(ctx, &a.dexpreopter)
1090}
1091
Jaewoong Jungea1bdb02019-05-09 14:36:34 -07001092func (a *AndroidAppImport) uncompressDex(
1093 ctx android.ModuleContext, inputPath android.Path, outputPath android.OutputPath) {
1094 rule := android.NewRuleBuilder()
1095 rule.Command().
1096 Textf(`if (zipinfo %s '*.dex' 2>/dev/null | grep -v ' stor ' >/dev/null) ; then`, inputPath).
Colin Crossee94d6a2019-07-08 17:08:34 -07001097 BuiltTool(ctx, "zip2zip").
Jaewoong Jungea1bdb02019-05-09 14:36:34 -07001098 FlagWithInput("-i ", inputPath).
1099 FlagWithOutput("-o ", outputPath).
1100 FlagWithArg("-0 ", "'classes*.dex'").
1101 Textf(`; else cp -f %s %s; fi`, inputPath, outputPath)
1102 rule.Build(pctx, ctx, "uncompress-dex", "Uncompress dex files")
1103}
1104
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001105func (a *AndroidAppImport) GenerateAndroidBuildActions(ctx android.ModuleContext) {
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07001106 a.generateAndroidBuildActions(ctx)
1107}
1108
Jooyung Han39ee1192020-03-23 20:21:11 +09001109func (a *AndroidAppImport) InstallApkName() string {
1110 return a.BaseModuleName()
1111}
1112
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07001113func (a *AndroidAppImport) generateAndroidBuildActions(ctx android.ModuleContext) {
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001114 numCertPropsSet := 0
1115 if String(a.properties.Certificate) != "" {
1116 numCertPropsSet++
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001117 }
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001118 if Bool(a.properties.Presigned) {
1119 numCertPropsSet++
1120 }
1121 if Bool(a.properties.Default_dev_cert) {
1122 numCertPropsSet++
1123 }
1124 if numCertPropsSet != 1 {
1125 ctx.ModuleErrorf("One and only one of certficate, presigned, and default_dev_cert properties must be set")
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001126 }
1127
Colin Cross094cde42020-02-15 10:38:00 -08001128 _, certificates := collectAppDeps(ctx, false, false)
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001129
1130 // TODO: LOCAL_EXTRACT_APK/LOCAL_EXTRACT_DPI_APK
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001131 // TODO: LOCAL_PACKAGE_SPLITS
1132
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001133 srcApk := a.prebuilt.SingleSourcePath(ctx)
Colin Cross50ddcc42019-05-16 12:28:22 -07001134
1135 if a.usesLibrary.enforceUsesLibraries() {
1136 srcApk = a.usesLibrary.verifyUsesLibrariesAPK(ctx, srcApk)
1137 }
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001138
1139 // TODO: Install or embed JNI libraries
1140
1141 // Uncompress JNI libraries in the apk
1142 jnisUncompressed := android.PathForModuleOut(ctx, "jnis-uncompressed", ctx.ModuleName()+".apk")
1143 a.uncompressEmbeddedJniLibs(ctx, srcApk, jnisUncompressed.OutputPath)
1144
Kyeongkab.Namc4997142019-11-22 11:38:16 +09001145 var installDir android.InstallPath
1146 if Bool(a.properties.Privileged) {
1147 installDir = android.PathForModuleInstall(ctx, "priv-app", a.BaseModuleName())
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08001148 } else if ctx.InstallInTestcases() {
1149 installDir = android.PathForModuleInstall(ctx, a.BaseModuleName(), ctx.DeviceConfig().DeviceArch())
Kyeongkab.Namc4997142019-11-22 11:38:16 +09001150 } else {
1151 installDir = android.PathForModuleInstall(ctx, "app", a.BaseModuleName())
1152 }
1153
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001154 a.dexpreopter.installPath = installDir.Join(ctx, a.BaseModuleName()+".apk")
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001155 a.dexpreopter.isPresignedPrebuilt = Bool(a.properties.Presigned)
Jaewoong Jungacf18d72019-05-02 14:55:29 -07001156 a.dexpreopter.uncompressedDex = a.shouldUncompressDex(ctx)
Colin Cross50ddcc42019-05-16 12:28:22 -07001157
1158 a.dexpreopter.enforceUsesLibs = a.usesLibrary.enforceUsesLibraries()
1159 a.dexpreopter.usesLibs = a.usesLibrary.usesLibraryProperties.Uses_libs
1160 a.dexpreopter.optionalUsesLibs = a.usesLibrary.presentOptionalUsesLibs(ctx)
1161 a.dexpreopter.libraryPaths = a.usesLibrary.usesLibraryPaths(ctx)
1162
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001163 dexOutput := a.dexpreopter.dexpreopt(ctx, jnisUncompressed)
Jaewoong Jungea1bdb02019-05-09 14:36:34 -07001164 if a.dexpreopter.uncompressedDex {
1165 dexUncompressed := android.PathForModuleOut(ctx, "dex-uncompressed", ctx.ModuleName()+".apk")
1166 a.uncompressDex(ctx, dexOutput, dexUncompressed.OutputPath)
1167 dexOutput = dexUncompressed
1168 }
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001169
Jooyung Han39ee1192020-03-23 20:21:11 +09001170 apkFilename := proptools.StringDefault(a.properties.Filename, a.BaseModuleName()+".apk")
1171
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001172 // Sign or align the package
1173 // TODO: Handle EXTERNAL
1174 if !Bool(a.properties.Presigned) {
Jaewoong Jung961d4fd2019-08-22 14:25:58 -07001175 // If the certificate property is empty at this point, default_dev_cert must be set to true.
1176 // Which makes processMainCert's behavior for the empty cert string WAI.
1177 certificates = processMainCert(a.ModuleBase, String(a.properties.Certificate), certificates, ctx)
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001178 if len(certificates) != 1 {
1179 ctx.ModuleErrorf("Unexpected number of certificates were extracted: %q", certificates)
1180 }
Colin Cross503c1d02020-01-28 14:00:53 -08001181 a.certificate = certificates[0]
Jooyung Han39ee1192020-03-23 20:21:11 +09001182 signed := android.PathForModuleOut(ctx, "signed", apkFilename)
Songchun Fan17d69e32020-03-24 20:32:24 -07001183 SignAppPackage(ctx, signed, dexOutput, certificates, nil)
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001184 a.outputFile = signed
1185 } else {
Jooyung Han39ee1192020-03-23 20:21:11 +09001186 alignedApk := android.PathForModuleOut(ctx, "zip-aligned", apkFilename)
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001187 TransformZipAlign(ctx, alignedApk, dexOutput)
1188 a.outputFile = alignedApk
Colin Cross503c1d02020-01-28 14:00:53 -08001189 a.certificate = presignedCertificate
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001190 }
1191
1192 // TODO: Optionally compress the output apk.
1193
Jooyung Han39ee1192020-03-23 20:21:11 +09001194 a.installPath = ctx.InstallFile(installDir, apkFilename, a.outputFile)
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001195
1196 // TODO: androidmk converter jni libs
1197}
1198
1199func (a *AndroidAppImport) Prebuilt() *android.Prebuilt {
1200 return &a.prebuilt
1201}
1202
1203func (a *AndroidAppImport) Name() string {
1204 return a.prebuilt.Name(a.ModuleBase.Name())
1205}
1206
Dario Frenicde2a032019-10-27 00:29:22 +01001207func (a *AndroidAppImport) OutputFile() android.Path {
1208 return a.outputFile
1209}
1210
Jiyong Park618922e2020-01-08 13:35:43 +09001211func (a *AndroidAppImport) JacocoReportClassesFile() android.Path {
1212 return nil
1213}
1214
Colin Cross503c1d02020-01-28 14:00:53 -08001215func (a *AndroidAppImport) Certificate() Certificate {
1216 return a.certificate
1217}
1218
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001219var dpiVariantGroupType reflect.Type
1220var archVariantGroupType reflect.Type
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001221
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001222func initAndroidAppImportVariantGroupTypes() {
1223 dpiVariantGroupType = createVariantGroupType(supportedDpis, "Dpi_variants")
1224
1225 archNames := make([]string, len(android.ArchTypeList()))
1226 for i, archType := range android.ArchTypeList() {
1227 archNames[i] = archType.Name
1228 }
1229 archVariantGroupType = createVariantGroupType(archNames, "Arch")
1230}
1231
1232// Populates all variant struct properties at creation time.
1233func (a *AndroidAppImport) populateAllVariantStructs() {
1234 a.dpiVariants = reflect.New(dpiVariantGroupType).Interface()
1235 a.AddProperties(a.dpiVariants)
1236
1237 a.archVariants = reflect.New(archVariantGroupType).Interface()
1238 a.AddProperties(a.archVariants)
1239}
1240
Jiyong Parkf7487312019-10-17 12:54:30 +09001241func (a *AndroidAppImport) Privileged() bool {
1242 return Bool(a.properties.Privileged)
1243}
1244
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001245func createVariantGroupType(variants []string, variantGroupName string) reflect.Type {
1246 props := reflect.TypeOf((*AndroidAppImportProperties)(nil))
1247
1248 variantFields := make([]reflect.StructField, len(variants))
1249 for i, variant := range variants {
1250 variantFields[i] = reflect.StructField{
1251 Name: proptools.FieldNameForProperty(variant),
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001252 Type: props,
1253 }
1254 }
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001255
1256 variantGroupStruct := reflect.StructOf(variantFields)
1257 return reflect.StructOf([]reflect.StructField{
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001258 {
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001259 Name: variantGroupName,
1260 Type: variantGroupStruct,
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001261 },
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001262 })
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001263}
1264
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001265// android_app_import imports a prebuilt apk with additional processing specified in the module.
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001266// DPI-specific apk source files can be specified using dpi_variants. Example:
1267//
1268// android_app_import {
1269// name: "example_import",
1270// apk: "prebuilts/example.apk",
1271// dpi_variants: {
1272// mdpi: {
1273// apk: "prebuilts/example_mdpi.apk",
1274// },
1275// xhdpi: {
1276// apk: "prebuilts/example_xhdpi.apk",
1277// },
1278// },
1279// certificate: "PRESIGNED",
1280// }
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001281func AndroidAppImportFactory() android.Module {
1282 module := &AndroidAppImport{}
1283 module.AddProperties(&module.properties)
1284 module.AddProperties(&module.dexpreoptProperties)
Colin Cross50ddcc42019-05-16 12:28:22 -07001285 module.AddProperties(&module.usesLibrary.usesLibraryProperties)
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001286 module.populateAllVariantStructs()
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001287 android.AddLoadHook(module, func(ctx android.LoadHookContext) {
Jaewoong Jung1ce9ac62019-08-13 14:11:33 -07001288 module.processVariants(ctx)
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001289 })
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001290
1291 InitJavaModule(module, android.DeviceSupported)
Jaewoong Jung3e18b192019-06-11 12:25:34 -07001292 android.InitSingleSourcePrebuiltModule(module, &module.properties, "Apk")
Jaewoong Jungccbb3932019-04-15 09:48:31 -07001293
1294 return module
1295}
Colin Cross50ddcc42019-05-16 12:28:22 -07001296
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07001297type AndroidTestImport struct {
1298 AndroidAppImport
1299
1300 testProperties testProperties
1301
1302 data android.Paths
1303}
1304
1305func (a *AndroidTestImport) GenerateAndroidBuildActions(ctx android.ModuleContext) {
1306 a.generateAndroidBuildActions(ctx)
1307
1308 a.data = android.PathsForModuleSrc(ctx, a.testProperties.Data)
1309}
1310
Jaewoong Jung7c5bd832020-01-13 09:55:39 -08001311func (a *AndroidTestImport) InstallInTestcases() bool {
1312 return true
1313}
1314
Jaewoong Jungb28eb5f2019-08-27 15:01:50 -07001315// android_test_import imports a prebuilt test apk with additional processing specified in the
1316// module. DPI or arch variant configurations can be made as with android_app_import.
1317func AndroidTestImportFactory() android.Module {
1318 module := &AndroidTestImport{}
1319 module.AddProperties(&module.properties)
1320 module.AddProperties(&module.dexpreoptProperties)
1321 module.AddProperties(&module.usesLibrary.usesLibraryProperties)
1322 module.AddProperties(&module.testProperties)
1323 module.populateAllVariantStructs()
1324 android.AddLoadHook(module, func(ctx android.LoadHookContext) {
1325 module.processVariants(ctx)
1326 })
1327
1328 InitJavaModule(module, android.DeviceSupported)
1329 android.InitSingleSourcePrebuiltModule(module, &module.properties, "Apk")
1330
1331 return module
1332}
1333
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001334type RuntimeResourceOverlay struct {
1335 android.ModuleBase
1336 android.DefaultableModuleBase
1337 aapt
1338
1339 properties RuntimeResourceOverlayProperties
1340
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08001341 certificate Certificate
1342
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001343 outputFile android.Path
1344 installDir android.InstallPath
1345}
1346
1347type RuntimeResourceOverlayProperties struct {
1348 // the name of a certificate in the default certificate directory or an android_app_certificate
1349 // module name in the form ":module".
1350 Certificate *string
1351
1352 // optional theme name. If specified, the overlay package will be applied
1353 // only when the ro.boot.vendor.overlay.theme system property is set to the same value.
1354 Theme *string
1355
1356 // if not blank, set to the version of the sdk to compile against.
1357 // Defaults to compiling against the current platform.
1358 Sdk_version *string
1359
1360 // if not blank, set the minimum version of the sdk that the compiled artifacts will run against.
1361 // Defaults to sdk_version if not set.
1362 Min_sdk_version *string
1363}
1364
1365func (r *RuntimeResourceOverlay) DepsMutator(ctx android.BottomUpMutatorContext) {
1366 sdkDep := decodeSdkDep(ctx, sdkContext(r))
1367 if sdkDep.hasFrameworkLibs() {
1368 r.aapt.deps(ctx, sdkDep)
1369 }
1370
1371 cert := android.SrcIsModule(String(r.properties.Certificate))
1372 if cert != "" {
1373 ctx.AddDependency(ctx.Module(), certificateTag, cert)
1374 }
1375}
1376
1377func (r *RuntimeResourceOverlay) GenerateAndroidBuildActions(ctx android.ModuleContext) {
1378 // Compile and link resources
1379 r.aapt.hasNoCode = true
Jaewoong Jungf0f747c2020-01-24 10:30:02 -08001380 // Do not remove resources without default values nor dedupe resource configurations with the same value
1381 r.aapt.buildActions(ctx, r, "--no-resource-deduping", "--no-resource-removal")
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001382
1383 // Sign the built package
Colin Cross094cde42020-02-15 10:38:00 -08001384 _, certificates := collectAppDeps(ctx, false, false)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001385 certificates = processMainCert(r.ModuleBase, String(r.properties.Certificate), certificates, ctx)
1386 signed := android.PathForModuleOut(ctx, "signed", r.Name()+".apk")
Songchun Fan17d69e32020-03-24 20:32:24 -07001387 SignAppPackage(ctx, signed, r.aapt.exportPackage, certificates, nil)
Jaewoong Jung78ec5d82020-01-31 10:11:47 -08001388 r.certificate = certificates[0]
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001389
1390 r.outputFile = signed
1391 r.installDir = android.PathForModuleInstall(ctx, "overlay", String(r.properties.Theme))
1392 ctx.InstallFile(r.installDir, r.outputFile.Base(), r.outputFile)
1393}
1394
Jiyong Park6a927c42020-01-21 02:03:43 +09001395func (r *RuntimeResourceOverlay) sdkVersion() sdkSpec {
1396 return sdkSpecFrom(String(r.properties.Sdk_version))
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001397}
1398
1399func (r *RuntimeResourceOverlay) systemModules() string {
1400 return ""
1401}
1402
Jiyong Park6a927c42020-01-21 02:03:43 +09001403func (r *RuntimeResourceOverlay) minSdkVersion() sdkSpec {
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001404 if r.properties.Min_sdk_version != nil {
Jiyong Park6a927c42020-01-21 02:03:43 +09001405 return sdkSpecFrom(*r.properties.Min_sdk_version)
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001406 }
1407 return r.sdkVersion()
1408}
1409
Jiyong Park6a927c42020-01-21 02:03:43 +09001410func (r *RuntimeResourceOverlay) targetSdkVersion() sdkSpec {
Jaewoong Jung9befb0c2020-01-18 10:33:43 -08001411 return r.sdkVersion()
1412}
1413
1414// runtime_resource_overlay generates a resource-only apk file that can overlay application and
1415// system resources at run time.
1416func RuntimeResourceOverlayFactory() android.Module {
1417 module := &RuntimeResourceOverlay{}
1418 module.AddProperties(
1419 &module.properties,
1420 &module.aaptProperties)
1421
1422 InitJavaModule(module, android.DeviceSupported)
1423
1424 return module
1425}
1426
Colin Cross50ddcc42019-05-16 12:28:22 -07001427type UsesLibraryProperties struct {
1428 // A list of shared library modules that will be listed in uses-library tags in the AndroidManifest.xml file.
1429 Uses_libs []string
1430
1431 // A list of shared library modules that will be listed in uses-library tags in the AndroidManifest.xml file with
1432 // required=false.
1433 Optional_uses_libs []string
1434
1435 // If true, the list of uses_libs and optional_uses_libs modules must match the AndroidManifest.xml file. Defaults
1436 // to true if either uses_libs or optional_uses_libs is set. Will unconditionally default to true in the future.
1437 Enforce_uses_libs *bool
1438}
1439
1440// usesLibrary provides properties and helper functions for AndroidApp and AndroidAppImport to verify that the
1441// <uses-library> tags that end up in the manifest of an APK match the ones known to the build system through the
1442// uses_libs and optional_uses_libs properties. The build system's values are used by dexpreopt to preopt apps
1443// with knowledge of their shared libraries.
1444type usesLibrary struct {
1445 usesLibraryProperties UsesLibraryProperties
1446}
1447
Paul Duffin250e6192019-06-07 10:44:37 +01001448func (u *usesLibrary) deps(ctx android.BottomUpMutatorContext, hasFrameworkLibs bool) {
Colin Cross3245b2c2019-06-07 13:18:09 -07001449 if !ctx.Config().UnbundledBuild() {
1450 ctx.AddVariationDependencies(nil, usesLibTag, u.usesLibraryProperties.Uses_libs...)
1451 ctx.AddVariationDependencies(nil, usesLibTag, u.presentOptionalUsesLibs(ctx)...)
Paul Duffin250e6192019-06-07 10:44:37 +01001452 // Only add these extra dependencies if the module depends on framework libs. This avoids
1453 // creating a cyclic dependency:
1454 // e.g. framework-res -> org.apache.http.legacy -> ... -> framework-res.
1455 if hasFrameworkLibs {
Colin Cross3245b2c2019-06-07 13:18:09 -07001456 // dexpreopt/dexpreopt.go needs the paths to the dex jars of these libraries in case construct_context.sh needs
1457 // to pass them to dex2oat. Add them as a dependency so we can determine the path to the dex jar of each
1458 // library to dexpreopt.
1459 ctx.AddVariationDependencies(nil, usesLibTag,
1460 "org.apache.http.legacy",
1461 "android.hidl.base-V1.0-java",
1462 "android.hidl.manager-V1.0-java")
1463 }
Colin Cross50ddcc42019-05-16 12:28:22 -07001464 }
1465}
1466
1467// presentOptionalUsesLibs returns optional_uses_libs after filtering out MissingUsesLibraries, which don't exist in the
1468// build.
1469func (u *usesLibrary) presentOptionalUsesLibs(ctx android.BaseModuleContext) []string {
1470 optionalUsesLibs, _ := android.FilterList(u.usesLibraryProperties.Optional_uses_libs, ctx.Config().MissingUsesLibraries())
1471 return optionalUsesLibs
1472}
1473
1474// usesLibraryPaths returns a map of module names of shared library dependencies to the paths to their dex jars.
1475func (u *usesLibrary) usesLibraryPaths(ctx android.ModuleContext) map[string]android.Path {
1476 usesLibPaths := make(map[string]android.Path)
1477
1478 if !ctx.Config().UnbundledBuild() {
1479 ctx.VisitDirectDepsWithTag(usesLibTag, func(m android.Module) {
1480 if lib, ok := m.(Dependency); ok {
1481 if dexJar := lib.DexJar(); dexJar != nil {
1482 usesLibPaths[ctx.OtherModuleName(m)] = dexJar
1483 } else {
1484 ctx.ModuleErrorf("module %q in uses_libs or optional_uses_libs must produce a dex jar, does it have installable: true?",
1485 ctx.OtherModuleName(m))
1486 }
1487 } else if ctx.Config().AllowMissingDependencies() {
1488 ctx.AddMissingDependencies([]string{ctx.OtherModuleName(m)})
1489 } else {
1490 ctx.ModuleErrorf("module %q in uses_libs or optional_uses_libs must be a java library",
1491 ctx.OtherModuleName(m))
1492 }
1493 })
1494 }
1495
1496 return usesLibPaths
1497}
1498
1499// enforceUsesLibraries returns true of <uses-library> tags should be checked against uses_libs and optional_uses_libs
1500// properties. Defaults to true if either of uses_libs or optional_uses_libs is specified. Will default to true
1501// unconditionally in the future.
1502func (u *usesLibrary) enforceUsesLibraries() bool {
1503 defaultEnforceUsesLibs := len(u.usesLibraryProperties.Uses_libs) > 0 ||
1504 len(u.usesLibraryProperties.Optional_uses_libs) > 0
1505 return BoolDefault(u.usesLibraryProperties.Enforce_uses_libs, defaultEnforceUsesLibs)
1506}
1507
1508// verifyUsesLibrariesManifest checks the <uses-library> tags in an AndroidManifest.xml against the ones specified
1509// in the uses_libs and optional_uses_libs properties. It returns the path to a copy of the manifest.
1510func (u *usesLibrary) verifyUsesLibrariesManifest(ctx android.ModuleContext, manifest android.Path) android.Path {
1511 outputFile := android.PathForModuleOut(ctx, "manifest_check", "AndroidManifest.xml")
1512
1513 rule := android.NewRuleBuilder()
Colin Crossee94d6a2019-07-08 17:08:34 -07001514 cmd := rule.Command().BuiltTool(ctx, "manifest_check").
Colin Cross50ddcc42019-05-16 12:28:22 -07001515 Flag("--enforce-uses-libraries").
1516 Input(manifest).
1517 FlagWithOutput("-o ", outputFile)
1518
1519 for _, lib := range u.usesLibraryProperties.Uses_libs {
1520 cmd.FlagWithArg("--uses-library ", lib)
1521 }
1522
1523 for _, lib := range u.usesLibraryProperties.Optional_uses_libs {
1524 cmd.FlagWithArg("--optional-uses-library ", lib)
1525 }
1526
1527 rule.Build(pctx, ctx, "verify_uses_libraries", "verify <uses-library>")
1528
1529 return outputFile
1530}
1531
1532// verifyUsesLibrariesAPK checks the <uses-library> tags in the manifest of an APK against the ones specified
1533// in the uses_libs and optional_uses_libs properties. It returns the path to a copy of the APK.
1534func (u *usesLibrary) verifyUsesLibrariesAPK(ctx android.ModuleContext, apk android.Path) android.Path {
1535 outputFile := android.PathForModuleOut(ctx, "verify_uses_libraries", apk.Base())
1536
1537 rule := android.NewRuleBuilder()
1538 aapt := ctx.Config().HostToolPath(ctx, "aapt")
1539 rule.Command().
1540 Textf("aapt_binary=%s", aapt.String()).Implicit(aapt).
1541 Textf(`uses_library_names="%s"`, strings.Join(u.usesLibraryProperties.Uses_libs, " ")).
1542 Textf(`optional_uses_library_names="%s"`, strings.Join(u.usesLibraryProperties.Optional_uses_libs, " ")).
1543 Tool(android.PathForSource(ctx, "build/make/core/verify_uses_libraries.sh")).Input(apk)
1544 rule.Command().Text("cp -f").Input(apk).Output(outputFile)
1545
1546 rule.Build(pctx, ctx, "verify_uses_libraries", "verify <uses-library>")
1547
1548 return outputFile
1549}