| Jaewoong Jung | f9b4465 | 2020-12-21 12:29:12 -0800 | [diff] [blame] | 1 | // Copyright 2020 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 |  | 
|  | 15 | package java | 
|  | 16 |  | 
|  | 17 | // This file contains the module implementations for android_app_import and android_test_import. | 
|  | 18 |  | 
|  | 19 | import ( | 
|  | 20 | "reflect" | 
|  | 21 |  | 
|  | 22 | "github.com/google/blueprint/proptools" | 
|  | 23 |  | 
|  | 24 | "android/soong/android" | 
|  | 25 | ) | 
|  | 26 |  | 
|  | 27 | func init() { | 
|  | 28 | RegisterAppImportBuildComponents(android.InitRegistrationContext) | 
|  | 29 |  | 
|  | 30 | initAndroidAppImportVariantGroupTypes() | 
|  | 31 | } | 
|  | 32 |  | 
|  | 33 | func RegisterAppImportBuildComponents(ctx android.RegistrationContext) { | 
|  | 34 | ctx.RegisterModuleType("android_app_import", AndroidAppImportFactory) | 
|  | 35 | ctx.RegisterModuleType("android_test_import", AndroidTestImportFactory) | 
|  | 36 | } | 
|  | 37 |  | 
|  | 38 | type AndroidAppImport struct { | 
|  | 39 | android.ModuleBase | 
|  | 40 | android.DefaultableModuleBase | 
|  | 41 | android.ApexModuleBase | 
|  | 42 | prebuilt android.Prebuilt | 
|  | 43 |  | 
|  | 44 | properties   AndroidAppImportProperties | 
|  | 45 | dpiVariants  interface{} | 
|  | 46 | archVariants interface{} | 
|  | 47 |  | 
|  | 48 | outputFile  android.Path | 
|  | 49 | certificate Certificate | 
|  | 50 |  | 
|  | 51 | dexpreopter | 
|  | 52 |  | 
|  | 53 | usesLibrary usesLibrary | 
|  | 54 |  | 
|  | 55 | preprocessed bool | 
|  | 56 |  | 
|  | 57 | installPath android.InstallPath | 
|  | 58 |  | 
|  | 59 | hideApexVariantFromMake bool | 
|  | 60 | } | 
|  | 61 |  | 
|  | 62 | type AndroidAppImportProperties struct { | 
|  | 63 | // A prebuilt apk to import | 
|  | 64 | Apk *string | 
|  | 65 |  | 
|  | 66 | // The name of a certificate in the default certificate directory or an android_app_certificate | 
|  | 67 | // module name in the form ":module". Should be empty if presigned or default_dev_cert is set. | 
|  | 68 | Certificate *string | 
|  | 69 |  | 
|  | 70 | // Set this flag to true if the prebuilt apk is already signed. The certificate property must not | 
|  | 71 | // be set for presigned modules. | 
|  | 72 | Presigned *bool | 
|  | 73 |  | 
|  | 74 | // Name of the signing certificate lineage file. | 
|  | 75 | Lineage *string | 
|  | 76 |  | 
|  | 77 | // Sign with the default system dev certificate. Must be used judiciously. Most imported apps | 
|  | 78 | // need to either specify a specific certificate or be presigned. | 
|  | 79 | Default_dev_cert *bool | 
|  | 80 |  | 
|  | 81 | // Specifies that this app should be installed to the priv-app directory, | 
|  | 82 | // where the system will grant it additional privileges not available to | 
|  | 83 | // normal apps. | 
|  | 84 | Privileged *bool | 
|  | 85 |  | 
|  | 86 | // Names of modules to be overridden. Listed modules can only be other binaries | 
|  | 87 | // (in Make or Soong). | 
|  | 88 | // This does not completely prevent installation of the overridden binaries, but if both | 
|  | 89 | // binaries would be installed by default (in PRODUCT_PACKAGES) the other binary will be removed | 
|  | 90 | // from PRODUCT_PACKAGES. | 
|  | 91 | Overrides []string | 
|  | 92 |  | 
|  | 93 | // Optional name for the installed app. If unspecified, it is derived from the module name. | 
|  | 94 | Filename *string | 
|  | 95 | } | 
|  | 96 |  | 
|  | 97 | func (a *AndroidAppImport) IsInstallable() bool { | 
|  | 98 | return true | 
|  | 99 | } | 
|  | 100 |  | 
|  | 101 | // Updates properties with variant-specific values. | 
|  | 102 | func (a *AndroidAppImport) processVariants(ctx android.LoadHookContext) { | 
|  | 103 | config := ctx.Config() | 
|  | 104 |  | 
|  | 105 | dpiProps := reflect.ValueOf(a.dpiVariants).Elem().FieldByName("Dpi_variants") | 
|  | 106 | // Try DPI variant matches in the reverse-priority order so that the highest priority match | 
|  | 107 | // overwrites everything else. | 
|  | 108 | // TODO(jungjw): Can we optimize this by making it priority order? | 
|  | 109 | for i := len(config.ProductAAPTPrebuiltDPI()) - 1; i >= 0; i-- { | 
|  | 110 | MergePropertiesFromVariant(ctx, &a.properties, dpiProps, config.ProductAAPTPrebuiltDPI()[i]) | 
|  | 111 | } | 
|  | 112 | if config.ProductAAPTPreferredConfig() != "" { | 
|  | 113 | MergePropertiesFromVariant(ctx, &a.properties, dpiProps, config.ProductAAPTPreferredConfig()) | 
|  | 114 | } | 
|  | 115 |  | 
|  | 116 | archProps := reflect.ValueOf(a.archVariants).Elem().FieldByName("Arch") | 
|  | 117 | archType := ctx.Config().AndroidFirstDeviceTarget.Arch.ArchType | 
|  | 118 | MergePropertiesFromVariant(ctx, &a.properties, archProps, archType.Name) | 
|  | 119 |  | 
|  | 120 | if String(a.properties.Apk) == "" { | 
|  | 121 | // Disable this module since the apk property is still empty after processing all matching | 
|  | 122 | // variants. This likely means there is no matching variant, and the default variant doesn't | 
|  | 123 | // have an apk property value either. | 
|  | 124 | a.Disable() | 
|  | 125 | } | 
|  | 126 | } | 
|  | 127 |  | 
|  | 128 | func MergePropertiesFromVariant(ctx android.EarlyModuleContext, | 
|  | 129 | dst interface{}, variantGroup reflect.Value, variant string) { | 
|  | 130 | src := variantGroup.FieldByName(proptools.FieldNameForProperty(variant)) | 
|  | 131 | if !src.IsValid() { | 
|  | 132 | return | 
|  | 133 | } | 
|  | 134 |  | 
|  | 135 | err := proptools.ExtendMatchingProperties([]interface{}{dst}, src.Interface(), nil, proptools.OrderAppend) | 
|  | 136 | if err != nil { | 
|  | 137 | if propertyErr, ok := err.(*proptools.ExtendPropertyError); ok { | 
|  | 138 | ctx.PropertyErrorf(propertyErr.Property, "%s", propertyErr.Err.Error()) | 
|  | 139 | } else { | 
|  | 140 | panic(err) | 
|  | 141 | } | 
|  | 142 | } | 
|  | 143 | } | 
|  | 144 |  | 
|  | 145 | func (a *AndroidAppImport) DepsMutator(ctx android.BottomUpMutatorContext) { | 
|  | 146 | cert := android.SrcIsModule(String(a.properties.Certificate)) | 
|  | 147 | if cert != "" { | 
|  | 148 | ctx.AddDependency(ctx.Module(), certificateTag, cert) | 
|  | 149 | } | 
|  | 150 |  | 
|  | 151 | a.usesLibrary.deps(ctx, true) | 
|  | 152 | } | 
|  | 153 |  | 
|  | 154 | func (a *AndroidAppImport) uncompressEmbeddedJniLibs( | 
|  | 155 | ctx android.ModuleContext, inputPath android.Path, outputPath android.OutputPath) { | 
|  | 156 | // Test apps don't need their JNI libraries stored uncompressed. As a matter of fact, messing | 
|  | 157 | // with them may invalidate pre-existing signature data. | 
|  | 158 | if ctx.InstallInTestcases() && (Bool(a.properties.Presigned) || a.preprocessed) { | 
|  | 159 | ctx.Build(pctx, android.BuildParams{ | 
|  | 160 | Rule:   android.Cp, | 
|  | 161 | Output: outputPath, | 
|  | 162 | Input:  inputPath, | 
|  | 163 | }) | 
|  | 164 | return | 
|  | 165 | } | 
|  | 166 | rule := android.NewRuleBuilder(pctx, ctx) | 
|  | 167 | rule.Command(). | 
|  | 168 | Textf(`if (zipinfo %s 'lib/*.so' 2>/dev/null | grep -v ' stor ' >/dev/null) ; then`, inputPath). | 
|  | 169 | BuiltTool("zip2zip"). | 
|  | 170 | FlagWithInput("-i ", inputPath). | 
|  | 171 | FlagWithOutput("-o ", outputPath). | 
|  | 172 | FlagWithArg("-0 ", "'lib/**/*.so'"). | 
|  | 173 | Textf(`; else cp -f %s %s; fi`, inputPath, outputPath) | 
|  | 174 | rule.Build("uncompress-embedded-jni-libs", "Uncompress embedded JIN libs") | 
|  | 175 | } | 
|  | 176 |  | 
|  | 177 | // Returns whether this module should have the dex file stored uncompressed in the APK. | 
|  | 178 | func (a *AndroidAppImport) shouldUncompressDex(ctx android.ModuleContext) bool { | 
|  | 179 | if ctx.Config().UnbundledBuild() || a.preprocessed { | 
|  | 180 | return false | 
|  | 181 | } | 
|  | 182 |  | 
|  | 183 | // Uncompress dex in APKs of privileged apps | 
|  | 184 | if ctx.Config().UncompressPrivAppDex() && a.Privileged() { | 
|  | 185 | return true | 
|  | 186 | } | 
|  | 187 |  | 
|  | 188 | return shouldUncompressDex(ctx, &a.dexpreopter) | 
|  | 189 | } | 
|  | 190 |  | 
|  | 191 | func (a *AndroidAppImport) uncompressDex( | 
|  | 192 | ctx android.ModuleContext, inputPath android.Path, outputPath android.OutputPath) { | 
|  | 193 | rule := android.NewRuleBuilder(pctx, ctx) | 
|  | 194 | rule.Command(). | 
|  | 195 | Textf(`if (zipinfo %s '*.dex' 2>/dev/null | grep -v ' stor ' >/dev/null) ; then`, inputPath). | 
|  | 196 | BuiltTool("zip2zip"). | 
|  | 197 | FlagWithInput("-i ", inputPath). | 
|  | 198 | FlagWithOutput("-o ", outputPath). | 
|  | 199 | FlagWithArg("-0 ", "'classes*.dex'"). | 
|  | 200 | Textf(`; else cp -f %s %s; fi`, inputPath, outputPath) | 
|  | 201 | rule.Build("uncompress-dex", "Uncompress dex files") | 
|  | 202 | } | 
|  | 203 |  | 
|  | 204 | func (a *AndroidAppImport) GenerateAndroidBuildActions(ctx android.ModuleContext) { | 
|  | 205 | a.generateAndroidBuildActions(ctx) | 
|  | 206 | } | 
|  | 207 |  | 
|  | 208 | func (a *AndroidAppImport) InstallApkName() string { | 
|  | 209 | return a.BaseModuleName() | 
|  | 210 | } | 
|  | 211 |  | 
|  | 212 | func (a *AndroidAppImport) generateAndroidBuildActions(ctx android.ModuleContext) { | 
|  | 213 | apexInfo := ctx.Provider(android.ApexInfoProvider).(android.ApexInfo) | 
|  | 214 | if !apexInfo.IsForPlatform() { | 
|  | 215 | a.hideApexVariantFromMake = true | 
|  | 216 | } | 
|  | 217 |  | 
|  | 218 | numCertPropsSet := 0 | 
|  | 219 | if String(a.properties.Certificate) != "" { | 
|  | 220 | numCertPropsSet++ | 
|  | 221 | } | 
|  | 222 | if Bool(a.properties.Presigned) { | 
|  | 223 | numCertPropsSet++ | 
|  | 224 | } | 
|  | 225 | if Bool(a.properties.Default_dev_cert) { | 
|  | 226 | numCertPropsSet++ | 
|  | 227 | } | 
|  | 228 | if numCertPropsSet != 1 { | 
|  | 229 | ctx.ModuleErrorf("One and only one of certficate, presigned, and default_dev_cert properties must be set") | 
|  | 230 | } | 
|  | 231 |  | 
|  | 232 | _, certificates := collectAppDeps(ctx, a, false, false) | 
|  | 233 |  | 
|  | 234 | // TODO: LOCAL_EXTRACT_APK/LOCAL_EXTRACT_DPI_APK | 
|  | 235 | // TODO: LOCAL_PACKAGE_SPLITS | 
|  | 236 |  | 
|  | 237 | srcApk := a.prebuilt.SingleSourcePath(ctx) | 
|  | 238 |  | 
|  | 239 | if a.usesLibrary.enforceUsesLibraries() { | 
|  | 240 | srcApk = a.usesLibrary.verifyUsesLibrariesAPK(ctx, srcApk) | 
|  | 241 | } | 
|  | 242 |  | 
|  | 243 | // TODO: Install or embed JNI libraries | 
|  | 244 |  | 
|  | 245 | // Uncompress JNI libraries in the apk | 
|  | 246 | jnisUncompressed := android.PathForModuleOut(ctx, "jnis-uncompressed", ctx.ModuleName()+".apk") | 
|  | 247 | a.uncompressEmbeddedJniLibs(ctx, srcApk, jnisUncompressed.OutputPath) | 
|  | 248 |  | 
|  | 249 | var installDir android.InstallPath | 
|  | 250 | if Bool(a.properties.Privileged) { | 
|  | 251 | installDir = android.PathForModuleInstall(ctx, "priv-app", a.BaseModuleName()) | 
|  | 252 | } else if ctx.InstallInTestcases() { | 
|  | 253 | installDir = android.PathForModuleInstall(ctx, a.BaseModuleName(), ctx.DeviceConfig().DeviceArch()) | 
|  | 254 | } else { | 
|  | 255 | installDir = android.PathForModuleInstall(ctx, "app", a.BaseModuleName()) | 
|  | 256 | } | 
|  | 257 |  | 
| Ulya Trafimovich | 76b0852 | 2021-01-14 17:52:43 +0000 | [diff] [blame] | 258 | a.dexpreopter.isApp = true | 
| Jaewoong Jung | f9b4465 | 2020-12-21 12:29:12 -0800 | [diff] [blame] | 259 | a.dexpreopter.installPath = installDir.Join(ctx, a.BaseModuleName()+".apk") | 
|  | 260 | a.dexpreopter.isPresignedPrebuilt = Bool(a.properties.Presigned) | 
|  | 261 | a.dexpreopter.uncompressedDex = a.shouldUncompressDex(ctx) | 
|  | 262 |  | 
|  | 263 | a.dexpreopter.enforceUsesLibs = a.usesLibrary.enforceUsesLibraries() | 
|  | 264 | a.dexpreopter.classLoaderContexts = a.usesLibrary.classLoaderContextForUsesLibDeps(ctx) | 
|  | 265 |  | 
|  | 266 | a.dexpreopter.dexpreopt(ctx, jnisUncompressed) | 
|  | 267 | if a.dexpreopter.uncompressedDex { | 
|  | 268 | dexUncompressed := android.PathForModuleOut(ctx, "dex-uncompressed", ctx.ModuleName()+".apk") | 
|  | 269 | a.uncompressDex(ctx, jnisUncompressed, dexUncompressed.OutputPath) | 
|  | 270 | jnisUncompressed = dexUncompressed | 
|  | 271 | } | 
|  | 272 |  | 
|  | 273 | apkFilename := proptools.StringDefault(a.properties.Filename, a.BaseModuleName()+".apk") | 
|  | 274 |  | 
|  | 275 | // TODO: Handle EXTERNAL | 
|  | 276 |  | 
|  | 277 | // Sign or align the package if package has not been preprocessed | 
|  | 278 | if a.preprocessed { | 
|  | 279 | a.outputFile = srcApk | 
|  | 280 | a.certificate = PresignedCertificate | 
|  | 281 | } else if !Bool(a.properties.Presigned) { | 
|  | 282 | // If the certificate property is empty at this point, default_dev_cert must be set to true. | 
|  | 283 | // Which makes processMainCert's behavior for the empty cert string WAI. | 
|  | 284 | certificates = processMainCert(a.ModuleBase, String(a.properties.Certificate), certificates, ctx) | 
|  | 285 | if len(certificates) != 1 { | 
|  | 286 | ctx.ModuleErrorf("Unexpected number of certificates were extracted: %q", certificates) | 
|  | 287 | } | 
|  | 288 | a.certificate = certificates[0] | 
|  | 289 | signed := android.PathForModuleOut(ctx, "signed", apkFilename) | 
|  | 290 | var lineageFile android.Path | 
|  | 291 | if lineage := String(a.properties.Lineage); lineage != "" { | 
|  | 292 | lineageFile = android.PathForModuleSrc(ctx, lineage) | 
|  | 293 | } | 
|  | 294 | SignAppPackage(ctx, signed, jnisUncompressed, certificates, nil, lineageFile) | 
|  | 295 | a.outputFile = signed | 
|  | 296 | } else { | 
|  | 297 | alignedApk := android.PathForModuleOut(ctx, "zip-aligned", apkFilename) | 
|  | 298 | TransformZipAlign(ctx, alignedApk, jnisUncompressed) | 
|  | 299 | a.outputFile = alignedApk | 
|  | 300 | a.certificate = PresignedCertificate | 
|  | 301 | } | 
|  | 302 |  | 
|  | 303 | // TODO: Optionally compress the output apk. | 
|  | 304 |  | 
|  | 305 | if apexInfo.IsForPlatform() { | 
|  | 306 | a.installPath = ctx.InstallFile(installDir, apkFilename, a.outputFile) | 
|  | 307 | } | 
|  | 308 |  | 
|  | 309 | // TODO: androidmk converter jni libs | 
|  | 310 | } | 
|  | 311 |  | 
|  | 312 | func (a *AndroidAppImport) Prebuilt() *android.Prebuilt { | 
|  | 313 | return &a.prebuilt | 
|  | 314 | } | 
|  | 315 |  | 
|  | 316 | func (a *AndroidAppImport) Name() string { | 
|  | 317 | return a.prebuilt.Name(a.ModuleBase.Name()) | 
|  | 318 | } | 
|  | 319 |  | 
|  | 320 | func (a *AndroidAppImport) OutputFile() android.Path { | 
|  | 321 | return a.outputFile | 
|  | 322 | } | 
|  | 323 |  | 
|  | 324 | func (a *AndroidAppImport) JacocoReportClassesFile() android.Path { | 
|  | 325 | return nil | 
|  | 326 | } | 
|  | 327 |  | 
|  | 328 | func (a *AndroidAppImport) Certificate() Certificate { | 
|  | 329 | return a.certificate | 
|  | 330 | } | 
|  | 331 |  | 
|  | 332 | var dpiVariantGroupType reflect.Type | 
|  | 333 | var archVariantGroupType reflect.Type | 
|  | 334 | var supportedDpis = []string{"ldpi", "mdpi", "hdpi", "xhdpi", "xxhdpi", "xxxhdpi"} | 
|  | 335 |  | 
|  | 336 | func initAndroidAppImportVariantGroupTypes() { | 
|  | 337 | dpiVariantGroupType = createVariantGroupType(supportedDpis, "Dpi_variants") | 
|  | 338 |  | 
|  | 339 | archNames := make([]string, len(android.ArchTypeList())) | 
|  | 340 | for i, archType := range android.ArchTypeList() { | 
|  | 341 | archNames[i] = archType.Name | 
|  | 342 | } | 
|  | 343 | archVariantGroupType = createVariantGroupType(archNames, "Arch") | 
|  | 344 | } | 
|  | 345 |  | 
|  | 346 | // Populates all variant struct properties at creation time. | 
|  | 347 | func (a *AndroidAppImport) populateAllVariantStructs() { | 
|  | 348 | a.dpiVariants = reflect.New(dpiVariantGroupType).Interface() | 
|  | 349 | a.AddProperties(a.dpiVariants) | 
|  | 350 |  | 
|  | 351 | a.archVariants = reflect.New(archVariantGroupType).Interface() | 
|  | 352 | a.AddProperties(a.archVariants) | 
|  | 353 | } | 
|  | 354 |  | 
|  | 355 | func (a *AndroidAppImport) Privileged() bool { | 
|  | 356 | return Bool(a.properties.Privileged) | 
|  | 357 | } | 
|  | 358 |  | 
|  | 359 | func (a *AndroidAppImport) DepIsInSameApex(_ android.BaseModuleContext, _ android.Module) bool { | 
|  | 360 | // android_app_import might have extra dependencies via uses_libs property. | 
|  | 361 | // Don't track the dependency as we don't automatically add those libraries | 
|  | 362 | // to the classpath. It should be explicitly added to java_libs property of APEX | 
|  | 363 | return false | 
|  | 364 | } | 
|  | 365 |  | 
|  | 366 | func (a *AndroidAppImport) sdkVersion() sdkSpec { | 
|  | 367 | return sdkSpecFrom("") | 
|  | 368 | } | 
|  | 369 |  | 
|  | 370 | func (a *AndroidAppImport) minSdkVersion() sdkSpec { | 
|  | 371 | return sdkSpecFrom("") | 
|  | 372 | } | 
|  | 373 |  | 
|  | 374 | var _ android.ApexModule = (*AndroidAppImport)(nil) | 
|  | 375 |  | 
|  | 376 | // Implements android.ApexModule | 
|  | 377 | func (j *AndroidAppImport) ShouldSupportSdkVersion(ctx android.BaseModuleContext, | 
|  | 378 | sdkVersion android.ApiLevel) error { | 
|  | 379 | // Do not check for prebuilts against the min_sdk_version of enclosing APEX | 
|  | 380 | return nil | 
|  | 381 | } | 
|  | 382 |  | 
|  | 383 | func createVariantGroupType(variants []string, variantGroupName string) reflect.Type { | 
|  | 384 | props := reflect.TypeOf((*AndroidAppImportProperties)(nil)) | 
|  | 385 |  | 
|  | 386 | variantFields := make([]reflect.StructField, len(variants)) | 
|  | 387 | for i, variant := range variants { | 
|  | 388 | variantFields[i] = reflect.StructField{ | 
|  | 389 | Name: proptools.FieldNameForProperty(variant), | 
|  | 390 | Type: props, | 
|  | 391 | } | 
|  | 392 | } | 
|  | 393 |  | 
|  | 394 | variantGroupStruct := reflect.StructOf(variantFields) | 
|  | 395 | return reflect.StructOf([]reflect.StructField{ | 
|  | 396 | { | 
|  | 397 | Name: variantGroupName, | 
|  | 398 | Type: variantGroupStruct, | 
|  | 399 | }, | 
|  | 400 | }) | 
|  | 401 | } | 
|  | 402 |  | 
|  | 403 | // android_app_import imports a prebuilt apk with additional processing specified in the module. | 
|  | 404 | // DPI-specific apk source files can be specified using dpi_variants. Example: | 
|  | 405 | // | 
|  | 406 | //     android_app_import { | 
|  | 407 | //         name: "example_import", | 
|  | 408 | //         apk: "prebuilts/example.apk", | 
|  | 409 | //         dpi_variants: { | 
|  | 410 | //             mdpi: { | 
|  | 411 | //                 apk: "prebuilts/example_mdpi.apk", | 
|  | 412 | //             }, | 
|  | 413 | //             xhdpi: { | 
|  | 414 | //                 apk: "prebuilts/example_xhdpi.apk", | 
|  | 415 | //             }, | 
|  | 416 | //         }, | 
|  | 417 | //         certificate: "PRESIGNED", | 
|  | 418 | //     } | 
|  | 419 | func AndroidAppImportFactory() android.Module { | 
|  | 420 | module := &AndroidAppImport{} | 
|  | 421 | module.AddProperties(&module.properties) | 
|  | 422 | module.AddProperties(&module.dexpreoptProperties) | 
|  | 423 | module.AddProperties(&module.usesLibrary.usesLibraryProperties) | 
|  | 424 | module.populateAllVariantStructs() | 
|  | 425 | android.AddLoadHook(module, func(ctx android.LoadHookContext) { | 
|  | 426 | module.processVariants(ctx) | 
|  | 427 | }) | 
|  | 428 |  | 
|  | 429 | android.InitApexModule(module) | 
|  | 430 | android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibCommon) | 
|  | 431 | android.InitDefaultableModule(module) | 
|  | 432 | android.InitSingleSourcePrebuiltModule(module, &module.properties, "Apk") | 
|  | 433 |  | 
| Ulya Trafimovich | 22890c4 | 2021-01-05 12:04:17 +0000 | [diff] [blame] | 434 | module.usesLibrary.enforce = true | 
|  | 435 |  | 
| Jaewoong Jung | f9b4465 | 2020-12-21 12:29:12 -0800 | [diff] [blame] | 436 | return module | 
|  | 437 | } | 
|  | 438 |  | 
|  | 439 | type androidTestImportProperties struct { | 
|  | 440 | // Whether the prebuilt apk can be installed without additional processing. Default is false. | 
|  | 441 | Preprocessed *bool | 
|  | 442 | } | 
|  | 443 |  | 
|  | 444 | type AndroidTestImport struct { | 
|  | 445 | AndroidAppImport | 
|  | 446 |  | 
|  | 447 | testProperties testProperties | 
|  | 448 |  | 
|  | 449 | testImportProperties androidTestImportProperties | 
|  | 450 |  | 
|  | 451 | data android.Paths | 
|  | 452 | } | 
|  | 453 |  | 
|  | 454 | func (a *AndroidTestImport) GenerateAndroidBuildActions(ctx android.ModuleContext) { | 
|  | 455 | a.preprocessed = Bool(a.testImportProperties.Preprocessed) | 
|  | 456 |  | 
|  | 457 | a.generateAndroidBuildActions(ctx) | 
|  | 458 |  | 
|  | 459 | a.data = android.PathsForModuleSrc(ctx, a.testProperties.Data) | 
|  | 460 | } | 
|  | 461 |  | 
|  | 462 | func (a *AndroidTestImport) InstallInTestcases() bool { | 
|  | 463 | return true | 
|  | 464 | } | 
|  | 465 |  | 
|  | 466 | // android_test_import imports a prebuilt test apk with additional processing specified in the | 
|  | 467 | // module. DPI or arch variant configurations can be made as with android_app_import. | 
|  | 468 | func AndroidTestImportFactory() android.Module { | 
|  | 469 | module := &AndroidTestImport{} | 
|  | 470 | module.AddProperties(&module.properties) | 
|  | 471 | module.AddProperties(&module.dexpreoptProperties) | 
|  | 472 | module.AddProperties(&module.testProperties) | 
|  | 473 | module.AddProperties(&module.testImportProperties) | 
|  | 474 | module.populateAllVariantStructs() | 
|  | 475 | android.AddLoadHook(module, func(ctx android.LoadHookContext) { | 
|  | 476 | module.processVariants(ctx) | 
|  | 477 | }) | 
|  | 478 |  | 
|  | 479 | module.dexpreopter.isTest = true | 
|  | 480 |  | 
|  | 481 | android.InitApexModule(module) | 
|  | 482 | android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibCommon) | 
|  | 483 | android.InitDefaultableModule(module) | 
|  | 484 | android.InitSingleSourcePrebuiltModule(module, &module.properties, "Apk") | 
|  | 485 |  | 
|  | 486 | return module | 
|  | 487 | } |