blob: a26ed432e409aeab6b27a342d7c1e9e608a31092 [file] [log] [blame]
Jihoon Kangf2c53982024-10-09 17:32:52 +00001// Copyright (C) 2024 The Android Open Source Project
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 filesystem
16
17import (
Jihoon Kangabec3ec2025-02-19 00:55:10 +000018 "cmp"
Jihoon Kang0a6315b2025-01-30 01:14:49 +000019 "fmt"
Cole Faust1dcf9e42025-02-19 17:23:34 -080020 "path/filepath"
Jihoon Kangabec3ec2025-02-19 00:55:10 +000021 "slices"
Spandan Das258c08f2025-01-08 23:30:45 +000022 "strings"
Cole Faustb55a41c2025-01-09 16:53:58 -080023 "sync/atomic"
Spandan Das258c08f2025-01-08 23:30:45 +000024
Jihoon Kangf2c53982024-10-09 17:32:52 +000025 "android/soong/android"
Cole Faust1dcf9e42025-02-19 17:23:34 -080026 "android/soong/java"
Jihoon Kangf2c53982024-10-09 17:32:52 +000027
28 "github.com/google/blueprint"
29 "github.com/google/blueprint/proptools"
30)
31
Cole Faust1dcf9e42025-02-19 17:23:34 -080032var proguardDictToProto = pctx.AndroidStaticRule("proguard_dict_to_proto", blueprint.RuleParams{
33 Command: `${symbols_map} -r8 $in -location $location -write_if_changed $out`,
34 Restat: true,
35 CommandDeps: []string{"${symbols_map}"},
36}, "location")
37
Jihoon Kangf2c53982024-10-09 17:32:52 +000038type PartitionNameProperties struct {
Cole Faust2bdc5e52025-01-10 10:29:36 -080039 // Name of the super partition filesystem module
40 Super_partition_name *string
Jihoon Kange7e3ec82025-01-02 21:29:14 +000041 // Name of the boot partition filesystem module
Jihoon Kangf2c53982024-10-09 17:32:52 +000042 Boot_partition_name *string
Jihoon Kange7e3ec82025-01-02 21:29:14 +000043 // Name of the vendor boot partition filesystem module
44 Vendor_boot_partition_name *string
45 // Name of the init boot partition filesystem module
46 Init_boot_partition_name *string
47 // Name of the system partition filesystem module
Jihoon Kangf2c53982024-10-09 17:32:52 +000048 System_partition_name *string
Jihoon Kange7e3ec82025-01-02 21:29:14 +000049 // Name of the system_ext partition filesystem module
Jihoon Kangf2c53982024-10-09 17:32:52 +000050 System_ext_partition_name *string
Jihoon Kange7e3ec82025-01-02 21:29:14 +000051 // Name of the product partition filesystem module
Jihoon Kangf2c53982024-10-09 17:32:52 +000052 Product_partition_name *string
Jihoon Kange7e3ec82025-01-02 21:29:14 +000053 // Name of the vendor partition filesystem module
Jihoon Kangf2c53982024-10-09 17:32:52 +000054 Vendor_partition_name *string
Jihoon Kange7e3ec82025-01-02 21:29:14 +000055 // Name of the odm partition filesystem module
Spandan Dasc5717162024-11-01 18:33:57 +000056 Odm_partition_name *string
Jihoon Kange7e3ec82025-01-02 21:29:14 +000057 // Name of the recovery partition filesystem module
58 Recovery_partition_name *string
Cole Faust3552eb62024-11-06 18:07:26 -080059 // The vbmeta partition and its "chained" partitions
60 Vbmeta_partitions []string
Jihoon Kange7e3ec82025-01-02 21:29:14 +000061 // Name of the userdata partition filesystem module
mrziwang23ba8762024-11-07 16:21:53 -080062 Userdata_partition_name *string
Spandan Dasa0394002025-01-07 18:38:34 +000063 // Name of the system_dlkm partition filesystem module
64 System_dlkm_partition_name *string
65 // Name of the vendor_dlkm partition filesystem module
66 Vendor_dlkm_partition_name *string
67 // Name of the odm_dlkm partition filesystem module
68 Odm_dlkm_partition_name *string
Jihoon Kangf2c53982024-10-09 17:32:52 +000069}
70
Jihoon Kang3be17162025-01-09 20:51:54 +000071type DeviceProperties struct {
72 // Path to the prebuilt bootloader that would be copied to PRODUCT_OUT
73 Bootloader *string `android:"path"`
Spandan Dase51ff952025-01-09 18:11:59 +000074 // Path to android-info.txt file containing board specific info.
75 Android_info *string `android:"path"`
Cole Faust11fda332025-01-14 16:47:19 -080076 // If this is the "main" android_device target for the build, i.e. the one that gets built
77 // when running a plain `m` command. Currently, this is the autogenerated android_device module
78 // in soong-only builds, but in the future when we check in android_device modules, the main
79 // one will be determined based on the lunch product. TODO: Figure out how to make this
80 // blueprint:"mutated" and still set it from filesystem_creator
81 Main_device *bool
Spandan Das29d44882025-01-15 21:12:36 +000082
Spandan Das00948072025-02-12 19:36:03 +000083 Ab_ota_updater *bool
84 Ab_ota_partitions []string
85 Ab_ota_keys []string
86 Ab_ota_postinstall_config []string
Spandan Das75955b12025-02-13 22:12:52 +000087
Spandan Das37240d92025-02-14 00:18:41 +000088 Ramdisk_node_list *string `android:"path"`
89 Releasetools_extension *string `android:"path"`
Spandan Das3dfa17f2025-02-28 09:48:28 +000090 FastbootInfo *string `android:"path"`
Cole Faust21c11462025-03-03 14:13:17 -080091
92 // The kernel version in the build. Will be verified against the actual kernel.
93 // If not provided, will attempt to extract it from the loose kernel or the kernel inside
94 // the boot image. The version is later used to decide whether or not to enable uffd_gc
95 // when dexpreopting apps. So setting this doesn't really do anything except enforce that the
96 // actual kernel version is as specified here.
97 Kernel_version *string
Jihoon Kang3be17162025-01-09 20:51:54 +000098}
99
Jihoon Kangf2c53982024-10-09 17:32:52 +0000100type androidDevice struct {
101 android.ModuleBase
102
103 partitionProps PartitionNameProperties
Jihoon Kang3be17162025-01-09 20:51:54 +0000104
105 deviceProps DeviceProperties
Cole Fausta472a6f2025-02-10 16:10:04 -0800106
107 allImagesZip android.Path
Cole Faust54124c82025-02-21 14:46:02 -0800108
Spandan Dase06dd112025-03-05 01:29:04 +0000109 proguardDictZip android.Path
110 proguardDictMapping android.Path
111 proguardUsageZip android.Path
112 kernelConfig android.Path
113 kernelVersion android.Path
114 miscInfo android.Path
115 rootDirForFsConfig string
116 rootDirForFsConfigTimestamp android.Path
Spandan Das38afe712025-03-05 23:27:55 +0000117 apkCertsInfo android.Path
Jihoon Kangf2c53982024-10-09 17:32:52 +0000118}
119
120func AndroidDeviceFactory() android.Module {
121 module := &androidDevice{}
Jihoon Kang3be17162025-01-09 20:51:54 +0000122 module.AddProperties(&module.partitionProps, &module.deviceProps)
Cole Faust341d5f12025-01-07 15:32:38 -0800123 android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibFirst)
Jihoon Kangf2c53982024-10-09 17:32:52 +0000124 return module
125}
126
Cole Faust11fda332025-01-14 16:47:19 -0800127var numMainAndroidDevicesOnceKey android.OnceKey = android.NewOnceKey("num_auto_generated_anroid_devices")
Cole Faustb55a41c2025-01-09 16:53:58 -0800128
Jihoon Kangf2c53982024-10-09 17:32:52 +0000129type partitionDepTagType struct {
130 blueprint.BaseDependencyTag
131}
132
Cole Faust2bdc5e52025-01-10 10:29:36 -0800133type superPartitionDepTagType struct {
134 blueprint.BaseDependencyTag
135}
Spandan Das29d44882025-01-15 21:12:36 +0000136type targetFilesMetadataDepTagType struct {
137 blueprint.BaseDependencyTag
138}
Cole Faust2bdc5e52025-01-10 10:29:36 -0800139
140var superPartitionDepTag superPartitionDepTagType
Jihoon Kangf2c53982024-10-09 17:32:52 +0000141var filesystemDepTag partitionDepTagType
Spandan Das29d44882025-01-15 21:12:36 +0000142var targetFilesMetadataDepTag targetFilesMetadataDepTagType
Jihoon Kangf2c53982024-10-09 17:32:52 +0000143
144func (a *androidDevice) DepsMutator(ctx android.BottomUpMutatorContext) {
145 addDependencyIfDefined := func(dep *string) {
146 if dep != nil {
Cole Faust341d5f12025-01-07 15:32:38 -0800147 ctx.AddDependency(ctx.Module(), filesystemDepTag, proptools.String(dep))
Jihoon Kangf2c53982024-10-09 17:32:52 +0000148 }
149 }
150
Cole Faust2bdc5e52025-01-10 10:29:36 -0800151 if a.partitionProps.Super_partition_name != nil {
152 ctx.AddDependency(ctx.Module(), superPartitionDepTag, *a.partitionProps.Super_partition_name)
153 }
Jihoon Kangf2c53982024-10-09 17:32:52 +0000154 addDependencyIfDefined(a.partitionProps.Boot_partition_name)
Jihoon Kang9e087002025-01-08 19:12:23 +0000155 addDependencyIfDefined(a.partitionProps.Init_boot_partition_name)
Spandan Dasef200ac2025-01-08 01:42:45 +0000156 addDependencyIfDefined(a.partitionProps.Vendor_boot_partition_name)
Jihoon Kangf2c53982024-10-09 17:32:52 +0000157 addDependencyIfDefined(a.partitionProps.System_partition_name)
158 addDependencyIfDefined(a.partitionProps.System_ext_partition_name)
159 addDependencyIfDefined(a.partitionProps.Product_partition_name)
160 addDependencyIfDefined(a.partitionProps.Vendor_partition_name)
Spandan Dasc5717162024-11-01 18:33:57 +0000161 addDependencyIfDefined(a.partitionProps.Odm_partition_name)
mrziwang23ba8762024-11-07 16:21:53 -0800162 addDependencyIfDefined(a.partitionProps.Userdata_partition_name)
Spandan Dasa0394002025-01-07 18:38:34 +0000163 addDependencyIfDefined(a.partitionProps.System_dlkm_partition_name)
164 addDependencyIfDefined(a.partitionProps.Vendor_dlkm_partition_name)
165 addDependencyIfDefined(a.partitionProps.Odm_dlkm_partition_name)
Spandan Dasef200ac2025-01-08 01:42:45 +0000166 addDependencyIfDefined(a.partitionProps.Recovery_partition_name)
Cole Faust3552eb62024-11-06 18:07:26 -0800167 for _, vbmetaPartition := range a.partitionProps.Vbmeta_partitions {
168 ctx.AddDependency(ctx.Module(), filesystemDepTag, vbmetaPartition)
169 }
Spandan Das29d44882025-01-15 21:12:36 +0000170 a.addDepsForTargetFilesMetadata(ctx)
171}
172
173func (a *androidDevice) addDepsForTargetFilesMetadata(ctx android.BottomUpMutatorContext) {
174 ctx.AddFarVariationDependencies(ctx.Config().BuildOSTarget.Variations(), targetFilesMetadataDepTag, "liblz4") // host variant
Jihoon Kangf2c53982024-10-09 17:32:52 +0000175}
176
Cole Faust11fda332025-01-14 16:47:19 -0800177func (a *androidDevice) GenerateAndroidBuildActions(ctx android.ModuleContext) {
178 if proptools.Bool(a.deviceProps.Main_device) {
179 numMainAndroidDevices := ctx.Config().Once(numMainAndroidDevicesOnceKey, func() interface{} {
180 return &atomic.Int32{}
181 }).(*atomic.Int32)
182 total := numMainAndroidDevices.Add(1)
183 if total > 1 {
184 // There should only be 1 main android_device module. That one will be
185 // made the default thing to build in soong-only builds.
186 ctx.ModuleErrorf("There cannot be more than 1 main android_device module")
187 }
Jihoon Kang3be17162025-01-09 20:51:54 +0000188 }
189
Cole Faust1dcf9e42025-02-19 17:23:34 -0800190 allInstalledModules := a.allInstalledModules(ctx)
191
Spandan Das38afe712025-03-05 23:27:55 +0000192 a.apkCertsInfo = a.buildApkCertsInfo(ctx, allInstalledModules)
Spandan Dasc09353c2025-03-04 00:11:21 +0000193 a.kernelConfig, a.kernelVersion = a.extractKernelVersionAndConfigs(ctx)
Spandan Das447a0ab2025-03-04 23:10:19 +0000194 a.miscInfo = a.addMiscInfo(ctx)
Spandan Dasdad98702025-02-26 14:32:28 +0000195 a.buildTargetFilesZip(ctx, allInstalledModules)
Cole Faust1dcf9e42025-02-19 17:23:34 -0800196 a.buildProguardZips(ctx, allInstalledModules)
197
mrziwang2fd33a72025-01-08 12:22:08 -0800198 var deps []android.Path
Cole Faust2bdc5e52025-01-10 10:29:36 -0800199 if proptools.String(a.partitionProps.Super_partition_name) != "" {
Cole Faust19eb09d2025-01-14 13:27:00 -0800200 superImage := ctx.GetDirectDepProxyWithTag(*a.partitionProps.Super_partition_name, superPartitionDepTag)
Cole Faust2bdc5e52025-01-10 10:29:36 -0800201 if info, ok := android.OtherModuleProvider(ctx, superImage, SuperImageProvider); ok {
202 assertUnset := func(prop *string, propName string) {
203 if prop != nil && *prop != "" {
204 ctx.PropertyErrorf(propName, "Cannot be set because it's already part of the super image")
205 }
206 }
207 for _, subPartitionType := range android.SortedKeys(info.SubImageInfo) {
208 switch subPartitionType {
209 case "system":
210 assertUnset(a.partitionProps.System_partition_name, "system_partition_name")
211 case "system_ext":
212 assertUnset(a.partitionProps.System_ext_partition_name, "system_ext_partition_name")
213 case "system_dlkm":
214 assertUnset(a.partitionProps.System_dlkm_partition_name, "system_dlkm_partition_name")
215 case "system_other":
216 // TODO
217 case "product":
218 assertUnset(a.partitionProps.Product_partition_name, "product_partition_name")
219 case "vendor":
220 assertUnset(a.partitionProps.Vendor_partition_name, "vendor_partition_name")
221 case "vendor_dlkm":
222 assertUnset(a.partitionProps.Vendor_dlkm_partition_name, "vendor_dlkm_partition_name")
223 case "odm":
224 assertUnset(a.partitionProps.Odm_partition_name, "odm_partition_name")
225 case "odm_dlkm":
226 assertUnset(a.partitionProps.Odm_dlkm_partition_name, "odm_dlkm_partition_name")
227 default:
228 ctx.ModuleErrorf("Unsupported sub-partition of super partition: %q", subPartitionType)
229 }
230 }
231
232 deps = append(deps, info.SuperImage)
233 } else {
234 ctx.ModuleErrorf("Expected super image dep to provide SuperImageProvider")
235 }
236 }
Cole Faust19eb09d2025-01-14 13:27:00 -0800237 ctx.VisitDirectDepsProxyWithTag(filesystemDepTag, func(m android.ModuleProxy) {
mrziwang2fd33a72025-01-08 12:22:08 -0800238 imageOutput, ok := android.OtherModuleProvider(ctx, m, android.OutputFilesProvider)
239 if !ok {
240 ctx.ModuleErrorf("Partition module %s doesn't set OutputfilesProvider", m.Name())
241 }
242 if len(imageOutput.DefaultOutputFiles) != 1 {
243 ctx.ModuleErrorf("Partition module %s should provide exact 1 output file", m.Name())
244 }
245 deps = append(deps, imageOutput.DefaultOutputFiles[0])
246 })
Jihoon Kang3be17162025-01-09 20:51:54 +0000247
Cole Fausta472a6f2025-02-10 16:10:04 -0800248 allImagesZip := android.PathForModuleOut(ctx, "all_images.zip")
249 allImagesZipBuilder := android.NewRuleBuilder(pctx, ctx)
Cole Faust8967d752025-02-19 17:27:29 -0800250 cmd := allImagesZipBuilder.Command().BuiltTool("soong_zip")
Cole Fausta472a6f2025-02-10 16:10:04 -0800251 for _, dep := range deps {
252 cmd.FlagWithArg("-e ", dep.Base())
253 cmd.FlagWithInput("-f ", dep)
254 }
255 cmd.FlagWithOutput("-o ", allImagesZip)
256 allImagesZipBuilder.Build("soong_all_images_zip", "all_images.zip")
257 a.allImagesZip = allImagesZip
258
Cole Faustb55a41c2025-01-09 16:53:58 -0800259 allImagesStamp := android.PathForModuleOut(ctx, "all_images_stamp")
Cole Faust11fda332025-01-14 16:47:19 -0800260 var validations android.Paths
261 if !ctx.Config().KatiEnabled() && proptools.Bool(a.deviceProps.Main_device) {
Cole Faustb55a41c2025-01-09 16:53:58 -0800262 // In soong-only builds, build this module by default.
263 // This is the analogue to this make code:
264 // https://cs.android.com/android/platform/superproject/main/+/main:build/make/core/main.mk;l=1396;drc=6595459cdd8164a6008335f6372c9f97b9094060
265 ctx.Phony("droidcore-unbundled", allImagesStamp)
Cole Faust11fda332025-01-14 16:47:19 -0800266
Cole Faust19fbb072025-01-30 18:19:29 -0800267 deps = append(deps, a.copyFilesToProductOutForSoongOnly(ctx))
Cole Faustb55a41c2025-01-09 16:53:58 -0800268 }
Cole Faust11fda332025-01-14 16:47:19 -0800269
270 ctx.Build(pctx, android.BuildParams{
271 Rule: android.Touch,
272 Output: allImagesStamp,
273 Implicits: deps,
274 Validations: validations,
275 })
276
277 // Checkbuilding it causes soong to make a phony, so you can say `m <module name>`
278 ctx.CheckbuildFile(allImagesStamp)
Jihoon Kang0a6315b2025-01-30 01:14:49 +0000279
280 a.setVbmetaPhonyTargets(ctx)
Jihoon Kangf67b7de2025-02-12 01:01:09 +0000281
282 a.distFiles(ctx)
283}
284
Jihoon Kangabec3ec2025-02-19 00:55:10 +0000285// Returns a list of modules that are installed, which are collected from the dependency
286// filesystem and super_image modules.
287func (a *androidDevice) allInstalledModules(ctx android.ModuleContext) []android.Module {
288 fsInfoMap := a.getFsInfos(ctx)
289 allOwners := make(map[string][]string)
290 for _, partition := range android.SortedKeys(fsInfoMap) {
291 fsInfo := fsInfoMap[partition]
292 for _, owner := range fsInfo.Owners {
293 allOwners[owner.Name] = append(allOwners[owner.Name], owner.Variation)
294 }
295 }
296
297 ret := []android.Module{}
298 ctx.WalkDepsProxy(func(mod, _ android.ModuleProxy) bool {
Spandan Das5c78fe92025-02-26 14:25:56 +0000299 if variations, ok := allOwners[ctx.OtherModuleName(mod)]; ok && android.InList(ctx.OtherModuleSubDir(mod), variations) {
Jihoon Kangabec3ec2025-02-19 00:55:10 +0000300 ret = append(ret, mod)
301 }
302 return true
303 })
304
305 // Remove duplicates
306 ret = android.FirstUniqueFunc(ret, func(a, b android.Module) bool {
307 return a.String() == b.String()
308 })
309
310 // Sort the modules by their names and variants
311 slices.SortFunc(ret, func(a, b android.Module) int {
312 return cmp.Compare(a.String(), b.String())
313 })
314 return ret
315}
316
Cole Faust54124c82025-02-21 14:46:02 -0800317func insertBeforeExtension(file, insertion string) string {
318 ext := filepath.Ext(file)
319 return strings.TrimSuffix(file, ext) + insertion + ext
320}
321
Jihoon Kangdd49f412025-03-07 01:30:43 +0000322func (a *androidDevice) distInstalledFiles(ctx android.ModuleContext) {
323 distInstalledFilesJsonAndTxt := func(installedFiles InstalledFilesStruct) {
324 if installedFiles.Json != nil {
325 ctx.DistForGoal("droidcore-unbundled", installedFiles.Json)
326 }
327 if installedFiles.Txt != nil {
328 ctx.DistForGoal("droidcore-unbundled", installedFiles.Txt)
329 }
330 }
331
332 fsInfoMap := a.getFsInfos(ctx)
333 for _, partition := range android.SortedKeys(fsInfoMap) {
334 // installed-files-*{.txt | .json} is not disted for userdata partition
335 if partition == "userdata" {
336 continue
337 }
338 fsInfo := fsInfoMap[partition]
339 for _, installedFiles := range fsInfo.InstalledFilesDepSet.ToList() {
340 distInstalledFilesJsonAndTxt(installedFiles)
341 }
342 }
343}
344
Jihoon Kangf67b7de2025-02-12 01:01:09 +0000345func (a *androidDevice) distFiles(ctx android.ModuleContext) {
Cole Faust54124c82025-02-21 14:46:02 -0800346 if !ctx.Config().KatiEnabled() && proptools.Bool(a.deviceProps.Main_device) {
Jihoon Kangdd49f412025-03-07 01:30:43 +0000347 a.distInstalledFiles(ctx)
Jihoon Kangf67b7de2025-02-12 01:01:09 +0000348
Cole Faust54124c82025-02-21 14:46:02 -0800349 namePrefix := ""
350 if ctx.Config().HasDeviceProduct() {
351 namePrefix = ctx.Config().DeviceProduct() + "-"
352 }
353 ctx.DistForGoalWithFilename("droidcore-unbundled", a.proguardDictZip, namePrefix+insertBeforeExtension(a.proguardDictZip.Base(), "-FILE_NAME_TAG_PLACEHOLDER"))
354 ctx.DistForGoalWithFilename("droidcore-unbundled", a.proguardDictMapping, namePrefix+insertBeforeExtension(a.proguardDictMapping.Base(), "-FILE_NAME_TAG_PLACEHOLDER"))
355 ctx.DistForGoalWithFilename("droidcore-unbundled", a.proguardUsageZip, namePrefix+insertBeforeExtension(a.proguardUsageZip.Base(), "-FILE_NAME_TAG_PLACEHOLDER"))
Cole Faustd8d570c2025-03-04 14:20:16 -0800356
357 if a.deviceProps.Android_info != nil {
358 ctx.DistForGoal("droidcore-unbundled", android.PathForModuleSrc(ctx, *a.deviceProps.Android_info))
359 }
Cole Faust54124c82025-02-21 14:46:02 -0800360 }
Cole Faust44080412024-12-20 14:17:07 -0800361}
Jihoon Kangf2c53982024-10-09 17:32:52 +0000362
Yu Liu2a815b62025-02-21 20:46:25 +0000363func (a *androidDevice) MakeVars(_ android.MakeVarsModuleContext) []android.ModuleMakeVarsValue {
Cole Fausta472a6f2025-02-10 16:10:04 -0800364 if proptools.Bool(a.deviceProps.Main_device) {
Yu Liu2a815b62025-02-21 20:46:25 +0000365 return []android.ModuleMakeVarsValue{{"SOONG_ONLY_ALL_IMAGES_ZIP", a.allImagesZip.String()}}
Cole Fausta472a6f2025-02-10 16:10:04 -0800366 }
Yu Liu2a815b62025-02-21 20:46:25 +0000367 return nil
Cole Fausta472a6f2025-02-10 16:10:04 -0800368}
369
Cole Faust1dcf9e42025-02-19 17:23:34 -0800370func (a *androidDevice) buildProguardZips(ctx android.ModuleContext, allInstalledModules []android.Module) {
371 dictZip := android.PathForModuleOut(ctx, "proguard-dict.zip")
372 dictZipBuilder := android.NewRuleBuilder(pctx, ctx)
373 dictZipCmd := dictZipBuilder.Command().BuiltTool("soong_zip").Flag("-d").FlagWithOutput("-o ", dictZip)
374
375 dictMapping := android.PathForModuleOut(ctx, "proguard-dict-mapping.textproto")
376 dictMappingBuilder := android.NewRuleBuilder(pctx, ctx)
377 dictMappingCmd := dictMappingBuilder.Command().BuiltTool("symbols_map").Flag("-merge").Output(dictMapping)
378
379 protosDir := android.PathForModuleOut(ctx, "proguard_mapping_protos")
380
381 usageZip := android.PathForModuleOut(ctx, "proguard-usage.zip")
382 usageZipBuilder := android.NewRuleBuilder(pctx, ctx)
383 usageZipCmd := usageZipBuilder.Command().BuiltTool("merge_zips").Output(usageZip)
384
385 for _, mod := range allInstalledModules {
386 if proguardInfo, ok := android.OtherModuleProvider(ctx, mod, java.ProguardProvider); ok {
387 // Maintain these out/target/common paths for backwards compatibility. They may be able
388 // to be changed if tools look up file locations from the protobuf, but I'm not
389 // exactly sure how that works.
390 dictionaryFakePath := fmt.Sprintf("out/target/common/obj/%s/%s_intermediates/proguard_dictionary", proguardInfo.Class, proguardInfo.ModuleName)
391 dictZipCmd.FlagWithArg("-e ", dictionaryFakePath)
392 dictZipCmd.FlagWithInput("-f ", proguardInfo.ProguardDictionary)
393 dictZipCmd.Textf("-e out/target/common/obj/%s/%s_intermediates/classes.jar", proguardInfo.Class, proguardInfo.ModuleName)
394 dictZipCmd.FlagWithInput("-f ", proguardInfo.ClassesJar)
395
396 protoFile := protosDir.Join(ctx, filepath.Dir(dictionaryFakePath), "proguard_dictionary.textproto")
397 ctx.Build(pctx, android.BuildParams{
398 Rule: proguardDictToProto,
399 Input: proguardInfo.ProguardDictionary,
400 Output: protoFile,
401 Args: map[string]string{
402 "location": dictionaryFakePath,
403 },
404 })
405 dictMappingCmd.Input(protoFile)
406
407 usageZipCmd.Input(proguardInfo.ProguardUsageZip)
408 }
409 }
410
411 dictZipBuilder.Build("proguard_dict_zip", "Building proguard dictionary zip")
412 dictMappingBuilder.Build("proguard_dict_mapping_proto", "Building proguard mapping proto")
413 usageZipBuilder.Build("proguard_usage_zip", "Building proguard usage zip")
Cole Faust54124c82025-02-21 14:46:02 -0800414
415 a.proguardDictZip = dictZip
416 a.proguardDictMapping = dictMapping
417 a.proguardUsageZip = usageZip
Cole Faust1dcf9e42025-02-19 17:23:34 -0800418}
419
Spandan Dasef775742025-01-13 22:17:40 +0000420// Helper structs for target_files.zip creation
Spandan Dasef200ac2025-01-08 01:42:45 +0000421type targetFilesZipCopy struct {
422 srcModule *string
423 destSubdir string
424}
425
Spandan Dasef775742025-01-13 22:17:40 +0000426type targetFilesystemZipCopy struct {
427 fsInfo FilesystemInfo
428 destSubdir string
429}
430
Spandan Dasdad98702025-02-26 14:32:28 +0000431func (a *androidDevice) buildTargetFilesZip(ctx android.ModuleContext, allInstalledModules []android.Module) {
Cole Faust44080412024-12-20 14:17:07 -0800432 targetFilesDir := android.PathForModuleOut(ctx, "target_files_dir")
433 targetFilesZip := android.PathForModuleOut(ctx, "target_files.zip")
434
435 builder := android.NewRuleBuilder(pctx, ctx)
436 builder.Command().Textf("rm -rf %s", targetFilesDir.String())
437 builder.Command().Textf("mkdir -p %s", targetFilesDir.String())
Spandan Dasef200ac2025-01-08 01:42:45 +0000438 toCopy := []targetFilesZipCopy{
439 targetFilesZipCopy{a.partitionProps.System_partition_name, "SYSTEM"},
440 targetFilesZipCopy{a.partitionProps.System_ext_partition_name, "SYSTEM_EXT"},
441 targetFilesZipCopy{a.partitionProps.Product_partition_name, "PRODUCT"},
442 targetFilesZipCopy{a.partitionProps.Vendor_partition_name, "VENDOR"},
443 targetFilesZipCopy{a.partitionProps.Odm_partition_name, "ODM"},
444 targetFilesZipCopy{a.partitionProps.System_dlkm_partition_name, "SYSTEM_DLKM"},
445 targetFilesZipCopy{a.partitionProps.Vendor_dlkm_partition_name, "VENDOR_DLKM"},
446 targetFilesZipCopy{a.partitionProps.Odm_dlkm_partition_name, "ODM_DLKM"},
447 targetFilesZipCopy{a.partitionProps.Init_boot_partition_name, "BOOT/RAMDISK"},
448 targetFilesZipCopy{a.partitionProps.Init_boot_partition_name, "INIT_BOOT/RAMDISK"},
449 targetFilesZipCopy{a.partitionProps.Vendor_boot_partition_name, "VENDOR_BOOT/RAMDISK"},
Spandan Das25649f52025-01-07 18:09:22 +0000450 }
Spandan Dasef200ac2025-01-08 01:42:45 +0000451
Spandan Dasef775742025-01-13 22:17:40 +0000452 filesystemsToCopy := []targetFilesystemZipCopy{}
Spandan Dasef200ac2025-01-08 01:42:45 +0000453 for _, zipCopy := range toCopy {
454 if zipCopy.srcModule == nil {
Spandan Das25649f52025-01-07 18:09:22 +0000455 continue
456 }
Spandan Dasef775742025-01-13 22:17:40 +0000457 filesystemsToCopy = append(
458 filesystemsToCopy,
459 targetFilesystemZipCopy{a.getFilesystemInfo(ctx, *zipCopy.srcModule), zipCopy.destSubdir},
460 )
461 }
462 // Get additional filesystems from super_partition dependency
463 if a.partitionProps.Super_partition_name != nil {
Cole Faust19eb09d2025-01-14 13:27:00 -0800464 superPartition := ctx.GetDirectDepProxyWithTag(*a.partitionProps.Super_partition_name, superPartitionDepTag)
Spandan Dasef775742025-01-13 22:17:40 +0000465 if info, ok := android.OtherModuleProvider(ctx, superPartition, SuperImageProvider); ok {
Cole Faust310de662025-02-19 16:15:07 -0800466 for _, partition := range android.SortedKeys(info.SubImageInfo) {
Spandan Dasef775742025-01-13 22:17:40 +0000467 filesystemsToCopy = append(
468 filesystemsToCopy,
469 targetFilesystemZipCopy{info.SubImageInfo[partition], strings.ToUpper(partition)},
470 )
471 }
472 } else {
473 ctx.ModuleErrorf("Super partition %s does set SuperImageProvider\n", superPartition.Name())
474 }
475 }
476
477 for _, toCopy := range filesystemsToCopy {
478 rootDirString := toCopy.fsInfo.RootDir.String()
479 if toCopy.destSubdir == "SYSTEM" {
Spandan Dasef200ac2025-01-08 01:42:45 +0000480 rootDirString = rootDirString + "/system"
481 }
Spandan Dasef775742025-01-13 22:17:40 +0000482 builder.Command().Textf("mkdir -p %s/%s", targetFilesDir.String(), toCopy.destSubdir)
Cole Faust44080412024-12-20 14:17:07 -0800483 builder.Command().
484 BuiltTool("acp").
Spandan Dasef775742025-01-13 22:17:40 +0000485 Textf("-rd %s/. %s/%s", rootDirString, targetFilesDir, toCopy.destSubdir).
486 Implicit(toCopy.fsInfo.Output) // so that the staging dir is built
Cole Faustb36763e2025-02-18 15:21:44 -0800487 for _, extraRootDir := range toCopy.fsInfo.ExtraRootDirs {
488 builder.Command().
489 BuiltTool("acp").
490 Textf("-rd %s/. %s/%s", extraRootDir, targetFilesDir, toCopy.destSubdir).
491 Implicit(toCopy.fsInfo.Output) // so that the staging dir is built
492 }
Spandan Dasef200ac2025-01-08 01:42:45 +0000493
Spandan Dasef775742025-01-13 22:17:40 +0000494 if toCopy.destSubdir == "SYSTEM" {
Spandan Das3ec6d062025-01-09 19:37:47 +0000495 // Create the ROOT partition in target_files.zip
Spandan Dasef775742025-01-13 22:17:40 +0000496 builder.Command().Textf("rsync --links --exclude=system/* %s/ -r %s/ROOT", toCopy.fsInfo.RootDir, targetFilesDir.String())
Spandan Dase06dd112025-03-05 01:29:04 +0000497 // Add a duplicate rule to assemble the ROOT/ directory in separate intermediates.
498 // The output timestamp will be an input to a separate fs_config call.
499 a.rootDirForFsConfig = android.PathForModuleOut(ctx, "root_dir_for_fs_config").String()
500 rootDirBuilder := android.NewRuleBuilder(pctx, ctx)
501 rootDirForFsConfigTimestamp := android.PathForModuleOut(ctx, "root_dir_for_fs_config.timestamp")
502 rootDirBuilder.Command().Textf("rsync --links --exclude=system/* %s/ -r %s", toCopy.fsInfo.RootDir, a.rootDirForFsConfig).
503 Implicit(toCopy.fsInfo.Output).
504 Text("&& touch").
505 Output(rootDirForFsConfigTimestamp)
506 rootDirBuilder.Build("assemble_root_dir_for_fs_config", "Assemble ROOT/ for fs_config")
507 a.rootDirForFsConfigTimestamp = rootDirForFsConfigTimestamp
Spandan Das3ec6d062025-01-09 19:37:47 +0000508 }
Cole Faust44080412024-12-20 14:17:07 -0800509 }
Spandan Das9b17df22025-01-08 23:30:45 +0000510 // Copy cmdline, kernel etc. files of boot images
Spandan Das258c08f2025-01-08 23:30:45 +0000511 if a.partitionProps.Vendor_boot_partition_name != nil {
Cole Faust19eb09d2025-01-14 13:27:00 -0800512 bootImg := ctx.GetDirectDepProxyWithTag(proptools.String(a.partitionProps.Vendor_boot_partition_name), filesystemDepTag)
Spandan Das258c08f2025-01-08 23:30:45 +0000513 bootImgInfo, _ := android.OtherModuleProvider(ctx, bootImg, BootimgInfoProvider)
Spandan Das9b17df22025-01-08 23:30:45 +0000514 builder.Command().Textf("echo %s > %s/VENDOR_BOOT/cmdline", proptools.ShellEscape(strings.Join(bootImgInfo.Cmdline, " ")), targetFilesDir)
515 builder.Command().Textf("echo %s > %s/VENDOR_BOOT/vendor_cmdline", proptools.ShellEscape(strings.Join(bootImgInfo.Cmdline, " ")), targetFilesDir)
516 if bootImgInfo.Dtb != nil {
Spandan Dasfed3d042025-01-13 21:38:47 +0000517 builder.Command().Textf("cp ").Input(bootImgInfo.Dtb).Textf(" %s/VENDOR_BOOT/dtb", targetFilesDir)
Spandan Das9b17df22025-01-08 23:30:45 +0000518 }
Spandan Das23511372025-01-08 23:30:45 +0000519 if bootImgInfo.Bootconfig != nil {
Spandan Dasfed3d042025-01-13 21:38:47 +0000520 builder.Command().Textf("cp ").Input(bootImgInfo.Bootconfig).Textf(" %s/VENDOR_BOOT/vendor_bootconfig", targetFilesDir)
Spandan Das23511372025-01-08 23:30:45 +0000521 }
Spandan Das258c08f2025-01-08 23:30:45 +0000522 }
523 if a.partitionProps.Boot_partition_name != nil {
Cole Faust19eb09d2025-01-14 13:27:00 -0800524 bootImg := ctx.GetDirectDepProxyWithTag(proptools.String(a.partitionProps.Boot_partition_name), filesystemDepTag)
Spandan Das258c08f2025-01-08 23:30:45 +0000525 bootImgInfo, _ := android.OtherModuleProvider(ctx, bootImg, BootimgInfoProvider)
Spandan Das9b17df22025-01-08 23:30:45 +0000526 builder.Command().Textf("echo %s > %s/BOOT/cmdline", proptools.ShellEscape(strings.Join(bootImgInfo.Cmdline, " ")), targetFilesDir)
527 if bootImgInfo.Dtb != nil {
Spandan Dasfed3d042025-01-13 21:38:47 +0000528 builder.Command().Textf("cp ").Input(bootImgInfo.Dtb).Textf(" %s/BOOT/dtb", targetFilesDir)
Spandan Das9b17df22025-01-08 23:30:45 +0000529 }
530 if bootImgInfo.Kernel != nil {
Spandan Dasfed3d042025-01-13 21:38:47 +0000531 builder.Command().Textf("cp ").Input(bootImgInfo.Kernel).Textf(" %s/BOOT/kernel", targetFilesDir)
Spandan Das9b17df22025-01-08 23:30:45 +0000532 // Even though kernel is not used to build vendor_boot, copy the kernel to VENDOR_BOOT to match the behavior of make packaging.
Spandan Dasfed3d042025-01-13 21:38:47 +0000533 builder.Command().Textf("cp ").Input(bootImgInfo.Kernel).Textf(" %s/VENDOR_BOOT/kernel", targetFilesDir)
Spandan Das9b17df22025-01-08 23:30:45 +0000534 }
Spandan Das23511372025-01-08 23:30:45 +0000535 if bootImgInfo.Bootconfig != nil {
Spandan Dasfed3d042025-01-13 21:38:47 +0000536 builder.Command().Textf("cp ").Input(bootImgInfo.Bootconfig).Textf(" %s/BOOT/bootconfig", targetFilesDir)
Spandan Das23511372025-01-08 23:30:45 +0000537 }
Spandan Das258c08f2025-01-08 23:30:45 +0000538 }
539
Spandan Dase51ff952025-01-09 18:11:59 +0000540 if a.deviceProps.Android_info != nil {
541 builder.Command().Textf("mkdir -p %s/OTA", targetFilesDir)
Cole Faust11fda332025-01-14 16:47:19 -0800542 builder.Command().Textf("cp ").Input(android.PathForModuleSrc(ctx, *a.deviceProps.Android_info)).Textf(" %s/OTA/android-info.txt", targetFilesDir)
Spandan Dase51ff952025-01-09 18:11:59 +0000543 }
544
Spandan Das0036fa32025-01-10 23:40:45 +0000545 a.copyImagesToTargetZip(ctx, builder, targetFilesDir)
Spandan Dasdad98702025-02-26 14:32:28 +0000546 a.copyMetadataToTargetZip(ctx, builder, targetFilesDir, allInstalledModules)
Spandan Das0036fa32025-01-10 23:40:45 +0000547
Cole Faust44080412024-12-20 14:17:07 -0800548 builder.Command().
549 BuiltTool("soong_zip").
550 Text("-d").
551 FlagWithOutput("-o ", targetFilesZip).
552 FlagWithArg("-C ", targetFilesDir.String()).
553 FlagWithArg("-D ", targetFilesDir.String()).
554 Text("-sha256")
555 builder.Build("target_files_"+ctx.ModuleName(), "Build target_files.zip")
556}
557
Spandan Das0036fa32025-01-10 23:40:45 +0000558func (a *androidDevice) copyImagesToTargetZip(ctx android.ModuleContext, builder *android.RuleBuilder, targetFilesDir android.WritablePath) {
559 // Create an IMAGES/ subdirectory
560 builder.Command().Textf("mkdir -p %s/IMAGES", targetFilesDir.String())
561 if a.deviceProps.Bootloader != nil {
562 builder.Command().Textf("cp ").Input(android.PathForModuleSrc(ctx, proptools.String(a.deviceProps.Bootloader))).Textf(" %s/IMAGES/bootloader", targetFilesDir.String())
563 }
564 // Copy the filesystem ,boot and vbmeta img files to IMAGES/
565 ctx.VisitDirectDepsProxyWithTag(filesystemDepTag, func(child android.ModuleProxy) {
Spandan Dasa9db76d2025-01-14 01:34:43 +0000566 if strings.Contains(child.Name(), "recovery") {
567 return // skip recovery.img to match the make packaging behavior
568 }
Spandan Dasef775742025-01-13 22:17:40 +0000569 if info, ok := android.OtherModuleProvider(ctx, child, BootimgInfoProvider); ok {
570 // Check Boot img first so that the boot.img is copied and not its dep ramdisk.img
Spandan Das0036fa32025-01-10 23:40:45 +0000571 builder.Command().Textf("cp ").Input(info.Output).Textf(" %s/IMAGES/", targetFilesDir.String())
Spandan Dasef775742025-01-13 22:17:40 +0000572 } else if info, ok := android.OtherModuleProvider(ctx, child, FilesystemProvider); ok {
Spandan Das0036fa32025-01-10 23:40:45 +0000573 builder.Command().Textf("cp ").Input(info.Output).Textf(" %s/IMAGES/", targetFilesDir.String())
574 } else if info, ok := android.OtherModuleProvider(ctx, child, vbmetaPartitionProvider); ok {
575 builder.Command().Textf("cp ").Input(info.Output).Textf(" %s/IMAGES/", targetFilesDir.String())
576 } else {
577 ctx.ModuleErrorf("Module %s does not provide an .img file output for target_files.zip", child.Name())
578 }
579 })
Spandan Dasef775742025-01-13 22:17:40 +0000580
581 if a.partitionProps.Super_partition_name != nil {
Cole Faust19eb09d2025-01-14 13:27:00 -0800582 superPartition := ctx.GetDirectDepProxyWithTag(*a.partitionProps.Super_partition_name, superPartitionDepTag)
Spandan Dasef775742025-01-13 22:17:40 +0000583 if info, ok := android.OtherModuleProvider(ctx, superPartition, SuperImageProvider); ok {
Cole Faust19eb09d2025-01-14 13:27:00 -0800584 for _, partition := range android.SortedKeys(info.SubImageInfo) {
Spandan Das7a42d1c2025-02-12 01:32:21 +0000585 if info.SubImageInfo[partition].OutputHermetic != nil {
586 builder.Command().Textf("cp ").Input(info.SubImageInfo[partition].OutputHermetic).Textf(" %s/IMAGES/", targetFilesDir.String())
587 }
588 if info.SubImageInfo[partition].MapFile != nil {
589 builder.Command().Textf("cp ").Input(info.SubImageInfo[partition].MapFile).Textf(" %s/IMAGES/", targetFilesDir.String())
590 }
Spandan Dasef775742025-01-13 22:17:40 +0000591 }
592 } else {
593 ctx.ModuleErrorf("Super partition %s does set SuperImageProvider\n", superPartition.Name())
594 }
595 }
Spandan Das0036fa32025-01-10 23:40:45 +0000596}
597
Spandan Dasdad98702025-02-26 14:32:28 +0000598func (a *androidDevice) copyMetadataToTargetZip(ctx android.ModuleContext, builder *android.RuleBuilder, targetFilesDir android.WritablePath, allInstalledModules []android.Module) {
Spandan Das29d44882025-01-15 21:12:36 +0000599 // Create a META/ subdirectory
600 builder.Command().Textf("mkdir -p %s/META", targetFilesDir.String())
601 if proptools.Bool(a.deviceProps.Ab_ota_updater) {
602 ctx.VisitDirectDepsProxyWithTag(targetFilesMetadataDepTag, func(child android.ModuleProxy) {
603 info, _ := android.OtherModuleProvider(ctx, child, android.OutputFilesProvider)
604 builder.Command().Textf("cp").Inputs(info.DefaultOutputFiles).Textf(" %s/META/", targetFilesDir.String())
605 })
Spandan Dasb5f40cf2025-02-12 19:36:03 +0000606 builder.Command().Textf("cp").Input(android.PathForSource(ctx, "external/zucchini/version_info.h")).Textf(" %s/META/zucchini_config.txt", targetFilesDir.String())
607 builder.Command().Textf("cp").Input(android.PathForSource(ctx, "system/update_engine/update_engine.conf")).Textf(" %s/META/update_engine_config.txt", targetFilesDir.String())
608 if a.getFsInfos(ctx)["system"].ErofsCompressHints != nil {
609 builder.Command().Textf("cp").Input(a.getFsInfos(ctx)["system"].ErofsCompressHints).Textf(" %s/META/erofs_default_compress_hints.txt", targetFilesDir.String())
610 }
611 // ab_partitions.txt
612 abPartitionsSorted := android.SortedUniqueStrings(a.deviceProps.Ab_ota_partitions)
613 abPartitionsSortedString := proptools.ShellEscape(strings.Join(abPartitionsSorted, "\\n"))
614 builder.Command().Textf("echo -e").Flag(abPartitionsSortedString).Textf(" > %s/META/ab_partitions.txt", targetFilesDir.String())
Spandan Das35b78742025-02-12 19:36:03 +0000615 // otakeys.txt
616 abOtaKeysSorted := android.SortedUniqueStrings(a.deviceProps.Ab_ota_keys)
617 abOtaKeysSortedString := proptools.ShellEscape(strings.Join(abOtaKeysSorted, "\\n"))
618 builder.Command().Textf("echo -e").Flag(abOtaKeysSortedString).Textf(" > %s/META/otakeys.txt", targetFilesDir.String())
Spandan Das00948072025-02-12 19:36:03 +0000619 // postinstall_config.txt
620 abOtaPostInstallConfigString := proptools.ShellEscape(strings.Join(a.deviceProps.Ab_ota_postinstall_config, "\\n"))
621 builder.Command().Textf("echo -e").Flag(abOtaPostInstallConfigString).Textf(" > %s/META/postinstall_config.txt", targetFilesDir.String())
Spandan Dasf12ff9b2025-02-12 22:27:43 +0000622 // selinuxfc
623 if a.getFsInfos(ctx)["system"].SelinuxFc != nil {
624 builder.Command().Textf("cp").Input(a.getFsInfos(ctx)["system"].SelinuxFc).Textf(" %s/META/file_contexts.bin", targetFilesDir.String())
625 }
Spandan Dasd71af182025-02-12 18:03:29 +0000626 }
Spandan Dasdd262fb2025-02-13 00:15:59 +0000627 // Copy $partition_filesystem_config.txt
628 fsInfos := a.getFsInfos(ctx)
629 for _, partition := range android.SortedKeys(fsInfos) {
630 if fsInfos[partition].FilesystemConfig == nil {
631 continue
632 }
633 if android.InList(partition, []string{"userdata"}) {
634 continue
635 }
Spandan Das853a7a42025-02-27 07:44:51 +0000636 if partition != "vendor_ramdisk" {
637 // vendor_ramdisk will be handled separately.
638 builder.Command().Textf("cp").Input(fsInfos[partition].FilesystemConfig).Textf(" %s/META/%s", targetFilesDir.String(), a.filesystemConfigNameForTargetFiles(partition))
639 }
Spandan Das54af3572025-02-27 07:42:08 +0000640 if partition == "ramdisk" {
641 // Create an additional copy at boot_filesystem_config.txt
642 builder.Command().Textf("cp").Input(fsInfos[partition].FilesystemConfig).Textf(" %s/META/boot_filesystem_config.txt", targetFilesDir.String())
643 }
Spandan Das853a7a42025-02-27 07:44:51 +0000644 if partition == "system" {
645 // Create root_filesystem_config from the assembled ROOT/ intermediates directory
Spandan Dase06dd112025-03-05 01:29:04 +0000646 a.generateFilesystemConfigForTargetFiles(ctx, builder, a.rootDirForFsConfigTimestamp, targetFilesDir.String(), a.rootDirForFsConfig, "root_filesystem_config.txt")
Spandan Das853a7a42025-02-27 07:44:51 +0000647 }
648 if partition == "vendor_ramdisk" {
649 // Create vendor_boot_filesystem_config from the assembled VENDOR_BOOT/RAMDISK intermediates directory
Spandan Dase06dd112025-03-05 01:29:04 +0000650 a.generateFilesystemConfigForTargetFiles(ctx, builder, nil, targetFilesDir.String(), targetFilesDir.String()+"/VENDOR_BOOT/RAMDISK", "vendor_boot_filesystem_config.txt")
Spandan Das853a7a42025-02-27 07:44:51 +0000651 }
Spandan Dasdd262fb2025-02-13 00:15:59 +0000652 }
Spandan Das75955b12025-02-13 22:12:52 +0000653 // Copy ramdisk_node_list
Spandan Das37240d92025-02-14 00:18:41 +0000654 if ramdiskNodeList := android.PathForModuleSrc(ctx, proptools.String(a.deviceProps.Ramdisk_node_list)); ramdiskNodeList != nil {
655 builder.Command().Textf("cp").Input(ramdiskNodeList).Textf(" %s/META/", targetFilesDir.String())
656 }
657 // Copy releasetools.py
658 if releaseTools := android.PathForModuleSrc(ctx, proptools.String(a.deviceProps.Releasetools_extension)); releaseTools != nil {
659 builder.Command().Textf("cp").Input(releaseTools).Textf(" %s/META/", targetFilesDir.String())
660 }
Spandan Dasdad98702025-02-26 14:32:28 +0000661 // apexkeys.txt
662 var installedApexKeys []android.Path
663 for _, installedModule := range allInstalledModules {
664 if info, ok := android.OtherModuleProvider(ctx, installedModule, ApexKeyPathInfoProvider); ok {
665 installedApexKeys = append(installedApexKeys, info.ApexKeyPath)
666 }
667 }
668 installedApexKeys = android.SortedUniquePaths(installedApexKeys) // Sort by keypath to match make
669 builder.Command().Text("cat").Inputs(installedApexKeys).Textf(" >> %s/META/apexkeys.txt", targetFilesDir.String())
Spandan Das38afe712025-03-05 23:27:55 +0000670 // apkcerts.txt
671 builder.Command().Textf("cp").Input(a.apkCertsInfo).Textf(" %s/META/", targetFilesDir.String())
672
Spandan Das3dfa17f2025-02-28 09:48:28 +0000673 // Copy fastboot-info.txt
674 if fastbootInfo := android.PathForModuleSrc(ctx, proptools.String(a.deviceProps.FastbootInfo)); fastbootInfo != nil {
675 // TODO (b/399788523): Autogenerate fastboot-info.txt if there is no source fastboot-info.txt
676 // https://cs.android.com/android/_/android/platform/build/+/80b9546f8f69e78b8fe1870e0e745d70fc18dfcd:core/Makefile;l=5831-5893;drc=077490384423dff9eac954da5c001c6f0be3fa6e;bpv=0;bpt=0
677 builder.Command().Textf("cp").Input(fastbootInfo).Textf(" %s/META/fastboot-info.txt", targetFilesDir.String())
678 }
Spandan Dasdad98702025-02-26 14:32:28 +0000679
Spandan Dasc09353c2025-03-04 00:11:21 +0000680 // kernel_configs.txt and kernel_version.txt
681 if a.kernelConfig != nil {
682 builder.Command().Textf("cp").Input(a.kernelConfig).Textf(" %s/META/", targetFilesDir.String())
683 }
684 if a.kernelVersion != nil {
685 builder.Command().Textf("cp").Input(a.kernelVersion).Textf(" %s/META/", targetFilesDir.String())
686 }
Spandan Das447a0ab2025-03-04 23:10:19 +0000687 // misc_info.txt
688 if a.miscInfo != nil {
689 builder.Command().Textf("cp").Input(a.miscInfo).Textf(" %s/META/", targetFilesDir.String())
690 }
Spandan Das4100dd02025-03-04 23:14:34 +0000691 // apex_info.pb, care_map.pb, vbmeta_digest.txt
692 a.addImgToTargetFiles(ctx, builder, targetFilesDir.String())
Spandan Dasc09353c2025-03-04 00:11:21 +0000693
Spandan Dasf079f0d2025-02-27 13:13:50 +0000694 if a.partitionProps.Super_partition_name != nil {
695 superPartition := ctx.GetDirectDepProxyWithTag(*a.partitionProps.Super_partition_name, superPartitionDepTag)
696 if info, ok := android.OtherModuleProvider(ctx, superPartition, SuperImageProvider); ok {
697 // dynamic_partitions_info.txt
698 // TODO (b/390192334): Add `building_super_empty_partition=true`
699 builder.Command().Text("cp").Input(info.DynamicPartitionsInfo).Textf(" %s/META/", targetFilesDir.String())
700 } else {
701 ctx.ModuleErrorf("Super partition %s does set SuperImageProvider\n", superPartition.Name())
702 }
703 }
704
Spandan Dasdd262fb2025-02-13 00:15:59 +0000705}
706
Spandan Das447a0ab2025-03-04 23:10:19 +0000707// A partial implementation of make's $PRODUCT_OUT/misc_info.txt
708// https://cs.android.com/android/platform/superproject/main/+/main:build/make/core/Makefile;l=5894?q=misc_info.txt%20f:build%2Fmake%2Fcore%2FMakefile&ss=android%2Fplatform%2Fsuperproject%2Fmain
709// This file is subsequently used by add_img_to_target_files to create additioanl metadata files like apex_info.pb
710// TODO (b/399788119): Complete the migration of misc_info.txt
711func (a *androidDevice) addMiscInfo(ctx android.ModuleContext) android.Path {
712 builder := android.NewRuleBuilder(pctx, ctx)
713 miscInfo := android.PathForModuleOut(ctx, "misc_info.txt")
714 builder.Command().
715 Textf("rm -f %s", miscInfo).
716 Textf("&& echo recovery_api_version=%s >> %s", ctx.Config().VendorConfig("recovery").String("recovery_api_version"), miscInfo).
717 Textf("&& echo fstab_version=%s >> %s", ctx.Config().VendorConfig("recovery").String("recovery_fstab_version"), miscInfo).
718 ImplicitOutput(miscInfo)
719
720 if a.partitionProps.Recovery_partition_name == nil {
721 builder.Command().Textf("echo no_recovery=true >> %s", miscInfo)
722 }
723 fsInfos := a.getFsInfos(ctx)
724 for _, partition := range android.SortedKeys(fsInfos) {
725 if fsInfos[partition].UseAvb {
726 builder.Command().Textf("echo 'avb_%s_hashtree_enable=true' >> %s", partition, miscInfo)
727 }
728 }
729 if len(a.partitionProps.Vbmeta_partitions) > 0 {
730 builder.Command().
731 Textf("echo avb_enable=true >> %s", miscInfo).
732 Textf("&& echo avb_building_vbmeta_image=true >> %s", miscInfo).
733 Textf("&& echo avb_avbtool=avbtool >> %s", miscInfo)
734 }
735 if a.partitionProps.Boot_partition_name != nil {
736 builder.Command().Textf("echo boot_images=boot.img >> %s", miscInfo)
737 }
738
739 builder.Build("misc_info", "Building misc_info")
740
741 return miscInfo
742}
743
Spandan Das4100dd02025-03-04 23:14:34 +0000744// addImgToTargetFiles invokes `add_img_to_target_files` and creates the following files in META/
745// - apex_info.pb
746// - care_map.pb
747// - vbmeta_digest.txt
748func (a *androidDevice) addImgToTargetFiles(ctx android.ModuleContext, builder *android.RuleBuilder, targetFilesDir string) {
749 mkbootimg := ctx.Config().HostToolPath(ctx, "mkbootimg")
750 builder.Command().
751 Textf("PATH=%s:$PATH", ctx.Config().HostToolDir()).
752 Textf("MKBOOTIMG=%s", mkbootimg).
753 Implicit(mkbootimg).
754 BuiltTool("add_img_to_target_files").
755 Flag("-a -v -p").
756 Flag(ctx.Config().HostToolDir()).
757 Text(targetFilesDir)
758}
759
Spandan Dasdad98702025-02-26 14:32:28 +0000760type ApexKeyPathInfo struct {
761 ApexKeyPath android.Path
762}
763
764var ApexKeyPathInfoProvider = blueprint.NewProvider[ApexKeyPathInfo]()
765
Spandan Dase06dd112025-03-05 01:29:04 +0000766func (a *androidDevice) generateFilesystemConfigForTargetFiles(ctx android.ModuleContext, builder *android.RuleBuilder, stagingDirTimestamp android.Path, targetFilesDir, stagingDir, filename string) {
Spandan Das853a7a42025-02-27 07:44:51 +0000767 fsConfigOut := android.PathForModuleOut(ctx, filename)
768 ctx.Build(pctx, android.BuildParams{
Spandan Dase06dd112025-03-05 01:29:04 +0000769 Rule: fsConfigRule,
770 Implicit: stagingDirTimestamp,
771 Output: fsConfigOut,
Spandan Das853a7a42025-02-27 07:44:51 +0000772 Args: map[string]string{
773 "rootDir": stagingDir,
774 "prefix": "",
775 },
776 })
777 builder.Command().Textf("cp").Input(fsConfigOut).Textf(" %s/META/", targetFilesDir)
778}
779
Spandan Dasdd262fb2025-02-13 00:15:59 +0000780// Filenames for the partition specific fs_config files.
781// Hardcode the ramdisk files to their boot image prefix
782func (a *androidDevice) filesystemConfigNameForTargetFiles(partition string) string {
783 name := partition + "_filesystem_config.txt"
784 if partition == "system" {
785 name = "filesystem_config.txt"
786 } else if partition == "ramdisk" {
787 name = "init_boot_filesystem_config.txt"
788 }
789 return name
Spandan Das29d44882025-01-15 21:12:36 +0000790}
791
Cole Faust44080412024-12-20 14:17:07 -0800792func (a *androidDevice) getFilesystemInfo(ctx android.ModuleContext, depName string) FilesystemInfo {
Cole Faust19eb09d2025-01-14 13:27:00 -0800793 fsMod := ctx.GetDirectDepProxyWithTag(depName, filesystemDepTag)
Cole Faust44080412024-12-20 14:17:07 -0800794 fsInfo, ok := android.OtherModuleProvider(ctx, fsMod, FilesystemProvider)
795 if !ok {
796 ctx.ModuleErrorf("Expected dependency %s to be a filesystem", depName)
797 }
798 return fsInfo
Jihoon Kangf2c53982024-10-09 17:32:52 +0000799}
Jihoon Kang0a6315b2025-01-30 01:14:49 +0000800
801func (a *androidDevice) setVbmetaPhonyTargets(ctx android.ModuleContext) {
802 if !proptools.Bool(a.deviceProps.Main_device) {
803 return
804 }
805
806 if !ctx.Config().KatiEnabled() {
807 for _, vbmetaPartitionName := range a.partitionProps.Vbmeta_partitions {
808 img := ctx.GetDirectDepProxyWithTag(vbmetaPartitionName, filesystemDepTag)
809 if provider, ok := android.OtherModuleProvider(ctx, img, vbmetaPartitionProvider); ok {
810 // make generates `vbmetasystemimage` phony target instead of `vbmeta_systemimage` phony target.
811 partitionName := strings.ReplaceAll(provider.Name, "_", "")
812 ctx.Phony(fmt.Sprintf("%simage", partitionName), provider.Output)
813 }
814 }
815 }
816}
Cole Faust21c11462025-03-03 14:13:17 -0800817
818func (a *androidDevice) getKernel(ctx android.ModuleContext) android.Path {
819 if a.partitionProps.Boot_partition_name != nil {
820 bootImg := ctx.GetDirectDepProxyWithTag(proptools.String(a.partitionProps.Boot_partition_name), filesystemDepTag)
821 bootImgInfo, _ := android.OtherModuleProvider(ctx, bootImg, BootimgInfoProvider)
822 return bootImgInfo.Kernel
823 }
824 return nil
825}
826
827// Gets the kernel version and configs from the actual kernel file itself. Roughly equivalent to
828// this make code: https://cs.android.com/android/platform/superproject/main/+/main:build/make/core/Makefile;l=5443;drc=c0b66fc59de069e06ce0ffd703d4d21613be30c6
829// However, it is a simplified version of that make code. Differences include:
830// - Not handling BOARD_KERNEL_CONFIG_FILE because BOARD_KERNEL_CONFIG_FILE was never used.
831// - Not unpacking the bootimage, as we should be able to just always export the kernel directly
832// in the BootimgInfo. We don't currently support prebuilt boot images, but even if we add that
833// in the future, it can be done in a prebuilt_bootimage module type that still exports the same
834// BootimgInfo.
835// - We don't print a warning and output '<unknown-kernel>' to kernel_version_for_uffd_gc.txt
836// because we expect the kernel to always be present. If it's not, we will get an error that
837// kernel_version_for_uffd_gc.txt doesn't exist. This may require later tweaking to the
838// dexpreopt rules so that they don't attempt to access that file in builds that don't have
839// a kernel.
840func (a *androidDevice) extractKernelVersionAndConfigs(ctx android.ModuleContext) (android.Path, android.Path) {
841 kernel := a.getKernel(ctx)
842 // If there's no kernel, don't create kernel version / kernel config files. Reverse dependencies
843 // on those files have to account for this, for example by disabling dexpreopt in unbundled
844 // builds.
845 if kernel == nil {
846 return nil, nil
847 }
848
849 lz4tool := ctx.Config().HostToolPath(ctx, "lz4")
850
851 extractedVersionFile := android.PathForModuleOut(ctx, "kernel_version.txt")
852 extractedConfigsFile := android.PathForModuleOut(ctx, "kernel_configs.txt")
853 builder := android.NewRuleBuilder(pctx, ctx)
854 builder.Command().BuiltTool("extract_kernel").
855 Flag("--tools lz4:"+lz4tool.String()).Implicit(lz4tool).
856 FlagWithInput("--input ", kernel).
857 FlagWithOutput("--output-release ", extractedVersionFile).
858 FlagWithOutput("--output-configs ", extractedConfigsFile)
859
860 if specifiedVersion := proptools.String(a.deviceProps.Kernel_version); specifiedVersion != "" {
861 specifiedVersionFile := android.PathForModuleOut(ctx, "specified_kernel_version.txt")
862 android.WriteFileRuleVerbatim(ctx, specifiedVersionFile, specifiedVersion)
863 builder.Command().Text("diff -q").
864 Input(specifiedVersionFile).
865 Input(extractedVersionFile).
866 Textf(`|| (echo "Specified kernel version '$(cat %s)' does not match actual kernel version '$(cat %s)'"; exit 1)`, specifiedVersionFile, extractedVersionFile)
867 }
868
869 builder.Build("extract_kernel_info", "Extract kernel version and configs")
870
871 if proptools.Bool(a.deviceProps.Main_device) && !ctx.Config().KatiEnabled() {
872 if ctx.Config().EnableUffdGc() == "default" {
873 kernelVersionFile := android.PathForOutput(ctx, "dexpreopt/kernel_version_for_uffd_gc.txt")
874 ctx.Build(pctx, android.BuildParams{
875 Rule: android.CpIfChanged,
876 Input: extractedVersionFile,
877 Output: kernelVersionFile,
878 })
879 }
880
881 ctx.DistForGoal("droid_targets", extractedVersionFile)
882 }
883
884 return extractedVersionFile, extractedConfigsFile
885}
Spandan Das38afe712025-03-05 23:27:55 +0000886
887func (a *androidDevice) buildApkCertsInfo(ctx android.ModuleContext, allInstalledModules []android.Module) android.Path {
888 // TODO (spandandas): Add compressed
889 formatLine := func(cert java.Certificate, name, partition string) string {
890 pem := cert.AndroidMkString()
891 var key string
892 if cert.Key == nil {
893 key = ""
894 } else {
895 key = cert.Key.String()
896 }
897 return fmt.Sprintf(`name="%s" certificate="%s" private_key="%s" partition="%s"`, name, pem, key, partition)
898 }
899
900 apkCerts := []string{}
901 for _, installedModule := range allInstalledModules {
902 partition := ""
903 if commonInfo, ok := android.OtherModuleProvider(ctx, installedModule, android.CommonModuleInfoKey); ok {
904 partition = commonInfo.PartitionTag
905 } else {
906 ctx.ModuleErrorf("%s does not set CommonModuleInfoKey", installedModule.Name())
907 }
908 if info, ok := android.OtherModuleProvider(ctx, installedModule, java.AppInfoProvider); ok {
909 apkCerts = append(apkCerts, formatLine(info.Certificate, info.InstallApkName+".apk", partition))
910 } else if info, ok := android.OtherModuleProvider(ctx, installedModule, java.AppInfosProvider); ok {
911 for _, certInfo := range info {
912 apkCerts = append(apkCerts, formatLine(certInfo.Certificate, certInfo.InstallApkName+".apk", partition))
913 }
914 } else if info, ok := android.OtherModuleProvider(ctx, installedModule, java.RuntimeResourceOverlayInfoProvider); ok {
915 apkCerts = append(apkCerts, formatLine(info.Certificate, info.OutputFile.Base(), partition))
916 }
917 }
918 slices.Sort(apkCerts) // sort by name
919 fsInfos := a.getFsInfos(ctx)
920 if fsInfos["system"].HasFsverity {
921 defaultPem, defaultKey := ctx.Config().DefaultAppCertificate(ctx)
922 apkCerts = append(apkCerts, formatLine(java.Certificate{Pem: defaultPem, Key: defaultKey}, "BuildManifest.apk", "system"))
923 if info, ok := fsInfos["system_ext"]; ok && info.HasFsverity {
924 apkCerts = append(apkCerts, formatLine(java.Certificate{Pem: defaultPem, Key: defaultKey}, "BuildManifestSystemExt.apk", "system_ext"))
925 }
926 }
927
928 apkCertsInfo := android.PathForModuleOut(ctx, "apkcerts.txt")
929 android.WriteFileRuleVerbatim(ctx, apkCertsInfo, strings.Join(apkCerts, "\n")+"\n")
930 return apkCertsInfo
931}