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 implementation for android_app_set. |
| 18 | |
| 19 | import ( |
| 20 | "strconv" |
| 21 | "strings" |
| 22 | |
| 23 | "github.com/google/blueprint/proptools" |
| 24 | |
| 25 | "android/soong/android" |
| 26 | ) |
| 27 | |
| 28 | func init() { |
| 29 | RegisterAppSetBuildComponents(android.InitRegistrationContext) |
| 30 | } |
| 31 | |
| 32 | func RegisterAppSetBuildComponents(ctx android.RegistrationContext) { |
| 33 | ctx.RegisterModuleType("android_app_set", AndroidAppSetFactory) |
| 34 | } |
| 35 | |
| 36 | type AndroidAppSetProperties struct { |
| 37 | // APK Set path |
| 38 | Set *string |
| 39 | |
| 40 | // Specifies that this app should be installed to the priv-app directory, |
| 41 | // where the system will grant it additional privileges not available to |
| 42 | // normal apps. |
| 43 | Privileged *bool |
| 44 | |
| 45 | // APKs in this set use prerelease SDK version |
| 46 | Prerelease *bool |
| 47 | |
| 48 | // Names of modules to be overridden. Listed modules can only be other apps |
| 49 | // (in Make or Soong). |
| 50 | Overrides []string |
Spandan Das | 3490dfd | 2024-03-11 21:37:25 +0000 | [diff] [blame] | 51 | |
| 52 | // Path to the .prebuilt_info file of the prebuilt app. |
| 53 | // In case of mainline modules, the .prebuilt_info file contains the build_id that was used |
| 54 | // to generate the prebuilt. |
| 55 | Prebuilt_info *string `android:"path"` |
Jaewoong Jung | f9b4465 | 2020-12-21 12:29:12 -0800 | [diff] [blame] | 56 | } |
| 57 | |
| 58 | type AndroidAppSet struct { |
| 59 | android.ModuleBase |
| 60 | android.DefaultableModuleBase |
| 61 | prebuilt android.Prebuilt |
| 62 | |
Colin Cross | ffbcd1d | 2021-11-12 12:19:42 -0800 | [diff] [blame] | 63 | properties AndroidAppSetProperties |
| 64 | packedOutput android.WritablePath |
| 65 | primaryOutput android.WritablePath |
| 66 | apkcertsFile android.ModuleOutPath |
Jaewoong Jung | f9b4465 | 2020-12-21 12:29:12 -0800 | [diff] [blame] | 67 | } |
| 68 | |
| 69 | func (as *AndroidAppSet) Name() string { |
| 70 | return as.prebuilt.Name(as.ModuleBase.Name()) |
| 71 | } |
| 72 | |
| 73 | func (as *AndroidAppSet) IsInstallable() bool { |
| 74 | return true |
| 75 | } |
| 76 | |
| 77 | func (as *AndroidAppSet) Prebuilt() *android.Prebuilt { |
| 78 | return &as.prebuilt |
| 79 | } |
| 80 | |
| 81 | func (as *AndroidAppSet) Privileged() bool { |
| 82 | return Bool(as.properties.Privileged) |
| 83 | } |
| 84 | |
| 85 | func (as *AndroidAppSet) OutputFile() android.Path { |
Colin Cross | ffbcd1d | 2021-11-12 12:19:42 -0800 | [diff] [blame] | 86 | return as.primaryOutput |
Jaewoong Jung | f9b4465 | 2020-12-21 12:29:12 -0800 | [diff] [blame] | 87 | } |
| 88 | |
Colin Cross | ffbcd1d | 2021-11-12 12:19:42 -0800 | [diff] [blame] | 89 | func (as *AndroidAppSet) PackedAdditionalOutputs() android.Path { |
| 90 | return as.packedOutput |
Jaewoong Jung | f9b4465 | 2020-12-21 12:29:12 -0800 | [diff] [blame] | 91 | } |
| 92 | |
| 93 | func (as *AndroidAppSet) APKCertsFile() android.Path { |
| 94 | return as.apkcertsFile |
| 95 | } |
| 96 | |
| 97 | var TargetCpuAbi = map[string]string{ |
Colin Cross | 9f09316 | 2022-11-01 16:14:33 -0700 | [diff] [blame] | 98 | "arm": "ARMEABI_V7A", |
| 99 | "arm64": "ARM64_V8A", |
| 100 | // TODO: use "RISCV64" when that is supported in bundles |
| 101 | "riscv64": "ARM64_V8A", |
Colin Cross | a2aaa2f | 2022-10-03 12:41:50 -0700 | [diff] [blame] | 102 | "x86": "X86", |
| 103 | "x86_64": "X86_64", |
Jaewoong Jung | f9b4465 | 2020-12-21 12:29:12 -0800 | [diff] [blame] | 104 | } |
| 105 | |
Anton Hansson | 805e0a5 | 2022-11-25 14:06:46 +0000 | [diff] [blame] | 106 | func SupportedAbis(ctx android.ModuleContext, excludeNativeBridgeAbis bool) []string { |
Jaewoong Jung | f9b4465 | 2020-12-21 12:29:12 -0800 | [diff] [blame] | 107 | abiName := func(targetIdx int, deviceArch string) string { |
| 108 | if abi, found := TargetCpuAbi[deviceArch]; found { |
| 109 | return abi |
| 110 | } |
| 111 | ctx.ModuleErrorf("Target %d has invalid Arch: %s", targetIdx, deviceArch) |
| 112 | return "BAD_ABI" |
| 113 | } |
| 114 | |
| 115 | var result []string |
| 116 | for i, target := range ctx.Config().Targets[android.Android] { |
Anton Hansson | 805e0a5 | 2022-11-25 14:06:46 +0000 | [diff] [blame] | 117 | if target.NativeBridge == android.NativeBridgeEnabled && excludeNativeBridgeAbis { |
| 118 | continue |
| 119 | } |
Jaewoong Jung | f9b4465 | 2020-12-21 12:29:12 -0800 | [diff] [blame] | 120 | result = append(result, abiName(i, target.Arch.ArchType.String())) |
| 121 | } |
| 122 | return result |
| 123 | } |
| 124 | |
Spandan Das | 3490dfd | 2024-03-11 21:37:25 +0000 | [diff] [blame] | 125 | type prebuiltInfoProps struct { |
| 126 | baseModuleName string |
| 127 | isPrebuilt bool |
| 128 | prebuiltInfo *string |
| 129 | } |
| 130 | |
| 131 | // Set prebuiltInfoProvider. This will be used by `apex_prebuiltinfo_singleton` to print out a metadata file |
| 132 | // with information about whether source or prebuilt of an apex was used during the build. |
| 133 | func providePrebuiltInfo(ctx android.ModuleContext, p prebuiltInfoProps) { |
| 134 | info := android.PrebuiltInfo{ |
| 135 | Name: p.baseModuleName, |
| 136 | Is_prebuilt: p.isPrebuilt, |
| 137 | } |
| 138 | // If Prebuilt_info information is available in the soong module definition, add it to prebuilt_info.json. |
| 139 | if p.prebuiltInfo != nil { |
| 140 | prebuiltInfoFile := android.PathForModuleSrc(ctx, *p.prebuiltInfo) |
| 141 | info.Prebuilt_info_file_path = prebuiltInfoFile.String() |
| 142 | } |
| 143 | android.SetProvider(ctx, android.PrebuiltInfoProvider, info) |
| 144 | } |
| 145 | |
Jaewoong Jung | f9b4465 | 2020-12-21 12:29:12 -0800 | [diff] [blame] | 146 | func (as *AndroidAppSet) GenerateAndroidBuildActions(ctx android.ModuleContext) { |
| 147 | as.packedOutput = android.PathForModuleOut(ctx, ctx.ModuleName()+".zip") |
Colin Cross | ffbcd1d | 2021-11-12 12:19:42 -0800 | [diff] [blame] | 148 | as.primaryOutput = android.PathForModuleOut(ctx, as.BaseModuleName()+".apk") |
Jaewoong Jung | f9b4465 | 2020-12-21 12:29:12 -0800 | [diff] [blame] | 149 | as.apkcertsFile = android.PathForModuleOut(ctx, "apkcerts.txt") |
| 150 | // We are assuming here that the install file in the APK |
| 151 | // set has `.apk` suffix. If it doesn't the build will fail. |
| 152 | // APK sets containing APEX files are handled elsewhere. |
Jaewoong Jung | f9b4465 | 2020-12-21 12:29:12 -0800 | [diff] [blame] | 153 | screenDensities := "all" |
| 154 | if dpis := ctx.Config().ProductAAPTPrebuiltDPI(); len(dpis) > 0 { |
| 155 | screenDensities = strings.ToUpper(strings.Join(dpis, ",")) |
| 156 | } |
| 157 | // TODO(asmundak): handle locales. |
| 158 | // TODO(asmundak): do we support device features |
| 159 | ctx.Build(pctx, |
| 160 | android.BuildParams{ |
Colin Cross | ffbcd1d | 2021-11-12 12:19:42 -0800 | [diff] [blame] | 161 | Rule: extractMatchingApks, |
| 162 | Description: "Extract APKs from APK set", |
| 163 | Output: as.primaryOutput, |
| 164 | ImplicitOutputs: android.WritablePaths{as.packedOutput, as.apkcertsFile}, |
| 165 | Inputs: android.Paths{as.prebuilt.SingleSourcePath(ctx)}, |
Jaewoong Jung | f9b4465 | 2020-12-21 12:29:12 -0800 | [diff] [blame] | 166 | Args: map[string]string{ |
Anton Hansson | 805e0a5 | 2022-11-25 14:06:46 +0000 | [diff] [blame] | 167 | "abis": strings.Join(SupportedAbis(ctx, false), ","), |
Jaewoong Jung | f9b4465 | 2020-12-21 12:29:12 -0800 | [diff] [blame] | 168 | "allow-prereleased": strconv.FormatBool(proptools.Bool(as.properties.Prerelease)), |
| 169 | "screen-densities": screenDensities, |
| 170 | "sdk-version": ctx.Config().PlatformSdkVersion().String(), |
Pranav Gupta | 51645ff | 2023-03-20 16:19:53 -0700 | [diff] [blame] | 171 | "skip-sdk-check": strconv.FormatBool(ctx.Config().IsEnvTrue("SOONG_SKIP_APPSET_SDK_CHECK")), |
Jaewoong Jung | f9b4465 | 2020-12-21 12:29:12 -0800 | [diff] [blame] | 172 | "stem": as.BaseModuleName(), |
| 173 | "apkcerts": as.apkcertsFile.String(), |
| 174 | "partition": as.PartitionTag(ctx.DeviceConfig()), |
Colin Cross | ffbcd1d | 2021-11-12 12:19:42 -0800 | [diff] [blame] | 175 | "zip": as.packedOutput.String(), |
Jaewoong Jung | f9b4465 | 2020-12-21 12:29:12 -0800 | [diff] [blame] | 176 | }, |
| 177 | }) |
Colin Cross | 50ed1f9 | 2021-11-12 17:41:02 -0800 | [diff] [blame] | 178 | |
| 179 | var installDir android.InstallPath |
| 180 | if as.Privileged() { |
| 181 | installDir = android.PathForModuleInstall(ctx, "priv-app", as.BaseModuleName()) |
| 182 | } else { |
| 183 | installDir = android.PathForModuleInstall(ctx, "app", as.BaseModuleName()) |
| 184 | } |
| 185 | ctx.InstallFileWithExtraFilesZip(installDir, as.BaseModuleName()+".apk", as.primaryOutput, as.packedOutput) |
Spandan Das | 3490dfd | 2024-03-11 21:37:25 +0000 | [diff] [blame] | 186 | |
| 187 | providePrebuiltInfo(ctx, |
| 188 | prebuiltInfoProps{ |
| 189 | baseModuleName: as.BaseModuleName(), |
| 190 | isPrebuilt: true, |
| 191 | prebuiltInfo: as.properties.Prebuilt_info, |
| 192 | }, |
| 193 | ) |
| 194 | |
Jaewoong Jung | f9b4465 | 2020-12-21 12:29:12 -0800 | [diff] [blame] | 195 | } |
| 196 | |
Colin Cross | 50ed1f9 | 2021-11-12 17:41:02 -0800 | [diff] [blame] | 197 | func (as *AndroidAppSet) InstallBypassMake() bool { return true } |
| 198 | |
Jaewoong Jung | f9b4465 | 2020-12-21 12:29:12 -0800 | [diff] [blame] | 199 | // android_app_set extracts a set of APKs based on the target device |
| 200 | // configuration and installs this set as "split APKs". |
| 201 | // The extracted set always contains an APK whose name is |
| 202 | // _module_name_.apk and every split APK matching target device. |
| 203 | // The extraction of the density-specific splits depends on |
| 204 | // PRODUCT_AAPT_PREBUILT_DPI variable. If present (its value should |
| 205 | // be a list density names: LDPI, MDPI, HDPI, etc.), only listed |
| 206 | // splits will be extracted. Otherwise all density-specific splits |
| 207 | // will be extracted. |
| 208 | func AndroidAppSetFactory() android.Module { |
| 209 | module := &AndroidAppSet{} |
| 210 | module.AddProperties(&module.properties) |
| 211 | InitJavaModule(module, android.DeviceSupported) |
| 212 | android.InitSingleSourcePrebuiltModule(module, &module.properties, "Set") |
| 213 | return module |
| 214 | } |