blob: 52b9d94e4ed1204ddbb501da82d0727240aabd73 [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 Kang0a6315b2025-01-30 01:14:49 +000018 "fmt"
Spandan Das258c08f2025-01-08 23:30:45 +000019 "strings"
Cole Faustb55a41c2025-01-09 16:53:58 -080020 "sync/atomic"
Spandan Das258c08f2025-01-08 23:30:45 +000021
Jihoon Kangf2c53982024-10-09 17:32:52 +000022 "android/soong/android"
23
24 "github.com/google/blueprint"
25 "github.com/google/blueprint/proptools"
26)
27
28type PartitionNameProperties struct {
Cole Faust2bdc5e52025-01-10 10:29:36 -080029 // Name of the super partition filesystem module
30 Super_partition_name *string
Jihoon Kange7e3ec82025-01-02 21:29:14 +000031 // Name of the boot partition filesystem module
Jihoon Kangf2c53982024-10-09 17:32:52 +000032 Boot_partition_name *string
Jihoon Kange7e3ec82025-01-02 21:29:14 +000033 // Name of the vendor boot partition filesystem module
34 Vendor_boot_partition_name *string
35 // Name of the init boot partition filesystem module
36 Init_boot_partition_name *string
37 // Name of the system partition filesystem module
Jihoon Kangf2c53982024-10-09 17:32:52 +000038 System_partition_name *string
Jihoon Kange7e3ec82025-01-02 21:29:14 +000039 // Name of the system_ext partition filesystem module
Jihoon Kangf2c53982024-10-09 17:32:52 +000040 System_ext_partition_name *string
Jihoon Kange7e3ec82025-01-02 21:29:14 +000041 // Name of the product partition filesystem module
Jihoon Kangf2c53982024-10-09 17:32:52 +000042 Product_partition_name *string
Jihoon Kange7e3ec82025-01-02 21:29:14 +000043 // Name of the vendor partition filesystem module
Jihoon Kangf2c53982024-10-09 17:32:52 +000044 Vendor_partition_name *string
Jihoon Kange7e3ec82025-01-02 21:29:14 +000045 // Name of the odm partition filesystem module
Spandan Dasc5717162024-11-01 18:33:57 +000046 Odm_partition_name *string
Jihoon Kange7e3ec82025-01-02 21:29:14 +000047 // Name of the recovery partition filesystem module
48 Recovery_partition_name *string
Cole Faust3552eb62024-11-06 18:07:26 -080049 // The vbmeta partition and its "chained" partitions
50 Vbmeta_partitions []string
Jihoon Kange7e3ec82025-01-02 21:29:14 +000051 // Name of the userdata partition filesystem module
mrziwang23ba8762024-11-07 16:21:53 -080052 Userdata_partition_name *string
Spandan Dasa0394002025-01-07 18:38:34 +000053 // Name of the system_dlkm partition filesystem module
54 System_dlkm_partition_name *string
55 // Name of the vendor_dlkm partition filesystem module
56 Vendor_dlkm_partition_name *string
57 // Name of the odm_dlkm partition filesystem module
58 Odm_dlkm_partition_name *string
Jihoon Kangf2c53982024-10-09 17:32:52 +000059}
60
Jihoon Kang3be17162025-01-09 20:51:54 +000061type DeviceProperties struct {
62 // Path to the prebuilt bootloader that would be copied to PRODUCT_OUT
63 Bootloader *string `android:"path"`
Spandan Dase51ff952025-01-09 18:11:59 +000064 // Path to android-info.txt file containing board specific info.
65 Android_info *string `android:"path"`
Cole Faust11fda332025-01-14 16:47:19 -080066 // If this is the "main" android_device target for the build, i.e. the one that gets built
67 // when running a plain `m` command. Currently, this is the autogenerated android_device module
68 // in soong-only builds, but in the future when we check in android_device modules, the main
69 // one will be determined based on the lunch product. TODO: Figure out how to make this
70 // blueprint:"mutated" and still set it from filesystem_creator
71 Main_device *bool
Spandan Das29d44882025-01-15 21:12:36 +000072
73 Ab_ota_updater *bool
Jihoon Kang3be17162025-01-09 20:51:54 +000074}
75
Jihoon Kangf2c53982024-10-09 17:32:52 +000076type androidDevice struct {
77 android.ModuleBase
78
79 partitionProps PartitionNameProperties
Jihoon Kang3be17162025-01-09 20:51:54 +000080
81 deviceProps DeviceProperties
Cole Fausta472a6f2025-02-10 16:10:04 -080082
83 allImagesZip android.Path
Jihoon Kangf2c53982024-10-09 17:32:52 +000084}
85
86func AndroidDeviceFactory() android.Module {
87 module := &androidDevice{}
Jihoon Kang3be17162025-01-09 20:51:54 +000088 module.AddProperties(&module.partitionProps, &module.deviceProps)
Cole Faust341d5f12025-01-07 15:32:38 -080089 android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibFirst)
Jihoon Kangf2c53982024-10-09 17:32:52 +000090 return module
91}
92
Cole Faust11fda332025-01-14 16:47:19 -080093var numMainAndroidDevicesOnceKey android.OnceKey = android.NewOnceKey("num_auto_generated_anroid_devices")
Cole Faustb55a41c2025-01-09 16:53:58 -080094
Jihoon Kangf2c53982024-10-09 17:32:52 +000095type partitionDepTagType struct {
96 blueprint.BaseDependencyTag
97}
98
Cole Faust2bdc5e52025-01-10 10:29:36 -080099type superPartitionDepTagType struct {
100 blueprint.BaseDependencyTag
101}
Spandan Das29d44882025-01-15 21:12:36 +0000102type targetFilesMetadataDepTagType struct {
103 blueprint.BaseDependencyTag
104}
Cole Faust2bdc5e52025-01-10 10:29:36 -0800105
106var superPartitionDepTag superPartitionDepTagType
Jihoon Kangf2c53982024-10-09 17:32:52 +0000107var filesystemDepTag partitionDepTagType
Spandan Das29d44882025-01-15 21:12:36 +0000108var targetFilesMetadataDepTag targetFilesMetadataDepTagType
Jihoon Kangf2c53982024-10-09 17:32:52 +0000109
110func (a *androidDevice) DepsMutator(ctx android.BottomUpMutatorContext) {
111 addDependencyIfDefined := func(dep *string) {
112 if dep != nil {
Cole Faust341d5f12025-01-07 15:32:38 -0800113 ctx.AddDependency(ctx.Module(), filesystemDepTag, proptools.String(dep))
Jihoon Kangf2c53982024-10-09 17:32:52 +0000114 }
115 }
116
Cole Faust2bdc5e52025-01-10 10:29:36 -0800117 if a.partitionProps.Super_partition_name != nil {
118 ctx.AddDependency(ctx.Module(), superPartitionDepTag, *a.partitionProps.Super_partition_name)
119 }
Jihoon Kangf2c53982024-10-09 17:32:52 +0000120 addDependencyIfDefined(a.partitionProps.Boot_partition_name)
Jihoon Kang9e087002025-01-08 19:12:23 +0000121 addDependencyIfDefined(a.partitionProps.Init_boot_partition_name)
Spandan Dasef200ac2025-01-08 01:42:45 +0000122 addDependencyIfDefined(a.partitionProps.Vendor_boot_partition_name)
Jihoon Kangf2c53982024-10-09 17:32:52 +0000123 addDependencyIfDefined(a.partitionProps.System_partition_name)
124 addDependencyIfDefined(a.partitionProps.System_ext_partition_name)
125 addDependencyIfDefined(a.partitionProps.Product_partition_name)
126 addDependencyIfDefined(a.partitionProps.Vendor_partition_name)
Spandan Dasc5717162024-11-01 18:33:57 +0000127 addDependencyIfDefined(a.partitionProps.Odm_partition_name)
mrziwang23ba8762024-11-07 16:21:53 -0800128 addDependencyIfDefined(a.partitionProps.Userdata_partition_name)
Spandan Dasa0394002025-01-07 18:38:34 +0000129 addDependencyIfDefined(a.partitionProps.System_dlkm_partition_name)
130 addDependencyIfDefined(a.partitionProps.Vendor_dlkm_partition_name)
131 addDependencyIfDefined(a.partitionProps.Odm_dlkm_partition_name)
Spandan Dasef200ac2025-01-08 01:42:45 +0000132 addDependencyIfDefined(a.partitionProps.Recovery_partition_name)
Cole Faust3552eb62024-11-06 18:07:26 -0800133 for _, vbmetaPartition := range a.partitionProps.Vbmeta_partitions {
134 ctx.AddDependency(ctx.Module(), filesystemDepTag, vbmetaPartition)
135 }
Spandan Das29d44882025-01-15 21:12:36 +0000136 a.addDepsForTargetFilesMetadata(ctx)
137}
138
139func (a *androidDevice) addDepsForTargetFilesMetadata(ctx android.BottomUpMutatorContext) {
140 ctx.AddFarVariationDependencies(ctx.Config().BuildOSTarget.Variations(), targetFilesMetadataDepTag, "liblz4") // host variant
Jihoon Kangf2c53982024-10-09 17:32:52 +0000141}
142
Cole Faust11fda332025-01-14 16:47:19 -0800143func (a *androidDevice) GenerateAndroidBuildActions(ctx android.ModuleContext) {
144 if proptools.Bool(a.deviceProps.Main_device) {
145 numMainAndroidDevices := ctx.Config().Once(numMainAndroidDevicesOnceKey, func() interface{} {
146 return &atomic.Int32{}
147 }).(*atomic.Int32)
148 total := numMainAndroidDevices.Add(1)
149 if total > 1 {
150 // There should only be 1 main android_device module. That one will be
151 // made the default thing to build in soong-only builds.
152 ctx.ModuleErrorf("There cannot be more than 1 main android_device module")
153 }
Jihoon Kang3be17162025-01-09 20:51:54 +0000154 }
155
Spandan Das3ccda6e2025-01-30 00:22:05 +0000156 //a.buildTargetFilesZip(ctx) TODO(b/393203512): re-enable target_files.zip
mrziwang2fd33a72025-01-08 12:22:08 -0800157 var deps []android.Path
Cole Faust2bdc5e52025-01-10 10:29:36 -0800158 if proptools.String(a.partitionProps.Super_partition_name) != "" {
Cole Faust19eb09d2025-01-14 13:27:00 -0800159 superImage := ctx.GetDirectDepProxyWithTag(*a.partitionProps.Super_partition_name, superPartitionDepTag)
Cole Faust2bdc5e52025-01-10 10:29:36 -0800160 if info, ok := android.OtherModuleProvider(ctx, superImage, SuperImageProvider); ok {
161 assertUnset := func(prop *string, propName string) {
162 if prop != nil && *prop != "" {
163 ctx.PropertyErrorf(propName, "Cannot be set because it's already part of the super image")
164 }
165 }
166 for _, subPartitionType := range android.SortedKeys(info.SubImageInfo) {
167 switch subPartitionType {
168 case "system":
169 assertUnset(a.partitionProps.System_partition_name, "system_partition_name")
170 case "system_ext":
171 assertUnset(a.partitionProps.System_ext_partition_name, "system_ext_partition_name")
172 case "system_dlkm":
173 assertUnset(a.partitionProps.System_dlkm_partition_name, "system_dlkm_partition_name")
174 case "system_other":
175 // TODO
176 case "product":
177 assertUnset(a.partitionProps.Product_partition_name, "product_partition_name")
178 case "vendor":
179 assertUnset(a.partitionProps.Vendor_partition_name, "vendor_partition_name")
180 case "vendor_dlkm":
181 assertUnset(a.partitionProps.Vendor_dlkm_partition_name, "vendor_dlkm_partition_name")
182 case "odm":
183 assertUnset(a.partitionProps.Odm_partition_name, "odm_partition_name")
184 case "odm_dlkm":
185 assertUnset(a.partitionProps.Odm_dlkm_partition_name, "odm_dlkm_partition_name")
186 default:
187 ctx.ModuleErrorf("Unsupported sub-partition of super partition: %q", subPartitionType)
188 }
189 }
190
191 deps = append(deps, info.SuperImage)
192 } else {
193 ctx.ModuleErrorf("Expected super image dep to provide SuperImageProvider")
194 }
195 }
Cole Faust19eb09d2025-01-14 13:27:00 -0800196 ctx.VisitDirectDepsProxyWithTag(filesystemDepTag, func(m android.ModuleProxy) {
mrziwang2fd33a72025-01-08 12:22:08 -0800197 imageOutput, ok := android.OtherModuleProvider(ctx, m, android.OutputFilesProvider)
198 if !ok {
199 ctx.ModuleErrorf("Partition module %s doesn't set OutputfilesProvider", m.Name())
200 }
201 if len(imageOutput.DefaultOutputFiles) != 1 {
202 ctx.ModuleErrorf("Partition module %s should provide exact 1 output file", m.Name())
203 }
204 deps = append(deps, imageOutput.DefaultOutputFiles[0])
205 })
Jihoon Kang3be17162025-01-09 20:51:54 +0000206
Cole Fausta472a6f2025-02-10 16:10:04 -0800207 allImagesZip := android.PathForModuleOut(ctx, "all_images.zip")
208 allImagesZipBuilder := android.NewRuleBuilder(pctx, ctx)
209 cmd := allImagesZipBuilder.Command().BuiltTool("soong_zip").Flag("--sort_entries")
210 for _, dep := range deps {
211 cmd.FlagWithArg("-e ", dep.Base())
212 cmd.FlagWithInput("-f ", dep)
213 }
214 cmd.FlagWithOutput("-o ", allImagesZip)
215 allImagesZipBuilder.Build("soong_all_images_zip", "all_images.zip")
216 a.allImagesZip = allImagesZip
217
Cole Faustb55a41c2025-01-09 16:53:58 -0800218 allImagesStamp := android.PathForModuleOut(ctx, "all_images_stamp")
Cole Faust11fda332025-01-14 16:47:19 -0800219 var validations android.Paths
220 if !ctx.Config().KatiEnabled() && proptools.Bool(a.deviceProps.Main_device) {
Cole Faustb55a41c2025-01-09 16:53:58 -0800221 // In soong-only builds, build this module by default.
222 // This is the analogue to this make code:
223 // https://cs.android.com/android/platform/superproject/main/+/main:build/make/core/main.mk;l=1396;drc=6595459cdd8164a6008335f6372c9f97b9094060
224 ctx.Phony("droidcore-unbundled", allImagesStamp)
Cole Faust11fda332025-01-14 16:47:19 -0800225
Cole Faust19fbb072025-01-30 18:19:29 -0800226 deps = append(deps, a.copyFilesToProductOutForSoongOnly(ctx))
Cole Faustb55a41c2025-01-09 16:53:58 -0800227 }
Cole Faust11fda332025-01-14 16:47:19 -0800228
229 ctx.Build(pctx, android.BuildParams{
230 Rule: android.Touch,
231 Output: allImagesStamp,
232 Implicits: deps,
233 Validations: validations,
234 })
235
236 // Checkbuilding it causes soong to make a phony, so you can say `m <module name>`
237 ctx.CheckbuildFile(allImagesStamp)
Jihoon Kang0a6315b2025-01-30 01:14:49 +0000238
239 a.setVbmetaPhonyTargets(ctx)
Jihoon Kangf67b7de2025-02-12 01:01:09 +0000240
241 a.distFiles(ctx)
242}
243
244func (a *androidDevice) distFiles(ctx android.ModuleContext) {
245 if !ctx.Config().KatiEnabled() {
246 if proptools.Bool(a.deviceProps.Main_device) {
247 fsInfoMap := a.getFsInfos(ctx)
248 for _, partition := range android.SortedKeys(fsInfoMap) {
249 fsInfo := fsInfoMap[partition]
250 if fsInfo.InstalledFiles.Json != nil {
251 ctx.DistForGoal("droidcore-unbundled", fsInfo.InstalledFiles.Json)
252 }
253 if fsInfo.InstalledFiles.Txt != nil {
254 ctx.DistForGoal("droidcore-unbundled", fsInfo.InstalledFiles.Txt)
255 }
256 }
257 }
258 }
259
Cole Faust44080412024-12-20 14:17:07 -0800260}
Jihoon Kangf2c53982024-10-09 17:32:52 +0000261
Cole Fausta472a6f2025-02-10 16:10:04 -0800262func (a *androidDevice) MakeVars(ctx android.MakeVarsModuleContext) {
263 if proptools.Bool(a.deviceProps.Main_device) {
264 ctx.StrictRaw("SOONG_ONLY_ALL_IMAGES_ZIP", a.allImagesZip.String())
265 }
266}
267
Spandan Dasef775742025-01-13 22:17:40 +0000268// Helper structs for target_files.zip creation
Spandan Dasef200ac2025-01-08 01:42:45 +0000269type targetFilesZipCopy struct {
270 srcModule *string
271 destSubdir string
272}
273
Spandan Dasef775742025-01-13 22:17:40 +0000274type targetFilesystemZipCopy struct {
275 fsInfo FilesystemInfo
276 destSubdir string
277}
278
Cole Faust44080412024-12-20 14:17:07 -0800279func (a *androidDevice) buildTargetFilesZip(ctx android.ModuleContext) {
280 targetFilesDir := android.PathForModuleOut(ctx, "target_files_dir")
281 targetFilesZip := android.PathForModuleOut(ctx, "target_files.zip")
282
283 builder := android.NewRuleBuilder(pctx, ctx)
284 builder.Command().Textf("rm -rf %s", targetFilesDir.String())
285 builder.Command().Textf("mkdir -p %s", targetFilesDir.String())
Spandan Dasef200ac2025-01-08 01:42:45 +0000286 toCopy := []targetFilesZipCopy{
287 targetFilesZipCopy{a.partitionProps.System_partition_name, "SYSTEM"},
288 targetFilesZipCopy{a.partitionProps.System_ext_partition_name, "SYSTEM_EXT"},
289 targetFilesZipCopy{a.partitionProps.Product_partition_name, "PRODUCT"},
290 targetFilesZipCopy{a.partitionProps.Vendor_partition_name, "VENDOR"},
291 targetFilesZipCopy{a.partitionProps.Odm_partition_name, "ODM"},
292 targetFilesZipCopy{a.partitionProps.System_dlkm_partition_name, "SYSTEM_DLKM"},
293 targetFilesZipCopy{a.partitionProps.Vendor_dlkm_partition_name, "VENDOR_DLKM"},
294 targetFilesZipCopy{a.partitionProps.Odm_dlkm_partition_name, "ODM_DLKM"},
295 targetFilesZipCopy{a.partitionProps.Init_boot_partition_name, "BOOT/RAMDISK"},
296 targetFilesZipCopy{a.partitionProps.Init_boot_partition_name, "INIT_BOOT/RAMDISK"},
297 targetFilesZipCopy{a.partitionProps.Vendor_boot_partition_name, "VENDOR_BOOT/RAMDISK"},
Spandan Das25649f52025-01-07 18:09:22 +0000298 }
Spandan Dasef200ac2025-01-08 01:42:45 +0000299 // TODO: Handle cases where recovery files are copied to BOOT/ or RECOVERY/
300 // https://cs.android.com/android/platform/superproject/main/+/main:build/make/core/Makefile;l=6211-6219?q=core%2FMakefile&ss=android%2Fplatform%2Fsuperproject%2Fmain
301 if ctx.DeviceConfig().BoardMoveRecoveryResourcesToVendorBoot() {
302 toCopy = append(toCopy, targetFilesZipCopy{a.partitionProps.Recovery_partition_name, "VENDOR_BOOT/RAMDISK"})
303 }
304
Spandan Dasef775742025-01-13 22:17:40 +0000305 filesystemsToCopy := []targetFilesystemZipCopy{}
Spandan Dasef200ac2025-01-08 01:42:45 +0000306 for _, zipCopy := range toCopy {
307 if zipCopy.srcModule == nil {
Spandan Das25649f52025-01-07 18:09:22 +0000308 continue
309 }
Spandan Dasef775742025-01-13 22:17:40 +0000310 filesystemsToCopy = append(
311 filesystemsToCopy,
312 targetFilesystemZipCopy{a.getFilesystemInfo(ctx, *zipCopy.srcModule), zipCopy.destSubdir},
313 )
314 }
315 // Get additional filesystems from super_partition dependency
316 if a.partitionProps.Super_partition_name != nil {
Cole Faust19eb09d2025-01-14 13:27:00 -0800317 superPartition := ctx.GetDirectDepProxyWithTag(*a.partitionProps.Super_partition_name, superPartitionDepTag)
Spandan Dasef775742025-01-13 22:17:40 +0000318 if info, ok := android.OtherModuleProvider(ctx, superPartition, SuperImageProvider); ok {
319 for _, partition := range android.SortedStringKeys(info.SubImageInfo) {
320 filesystemsToCopy = append(
321 filesystemsToCopy,
322 targetFilesystemZipCopy{info.SubImageInfo[partition], strings.ToUpper(partition)},
323 )
324 }
325 } else {
326 ctx.ModuleErrorf("Super partition %s does set SuperImageProvider\n", superPartition.Name())
327 }
328 }
329
330 for _, toCopy := range filesystemsToCopy {
331 rootDirString := toCopy.fsInfo.RootDir.String()
332 if toCopy.destSubdir == "SYSTEM" {
Spandan Dasef200ac2025-01-08 01:42:45 +0000333 rootDirString = rootDirString + "/system"
334 }
Spandan Dasef775742025-01-13 22:17:40 +0000335 builder.Command().Textf("mkdir -p %s/%s", targetFilesDir.String(), toCopy.destSubdir)
Cole Faust44080412024-12-20 14:17:07 -0800336 builder.Command().
337 BuiltTool("acp").
Spandan Dasef775742025-01-13 22:17:40 +0000338 Textf("-rd %s/. %s/%s", rootDirString, targetFilesDir, toCopy.destSubdir).
339 Implicit(toCopy.fsInfo.Output) // so that the staging dir is built
Spandan Dasef200ac2025-01-08 01:42:45 +0000340
Spandan Dasef775742025-01-13 22:17:40 +0000341 if toCopy.destSubdir == "SYSTEM" {
Spandan Das3ec6d062025-01-09 19:37:47 +0000342 // Create the ROOT partition in target_files.zip
Spandan Dasef775742025-01-13 22:17:40 +0000343 builder.Command().Textf("rsync --links --exclude=system/* %s/ -r %s/ROOT", toCopy.fsInfo.RootDir, targetFilesDir.String())
Spandan Das3ec6d062025-01-09 19:37:47 +0000344 }
Cole Faust44080412024-12-20 14:17:07 -0800345 }
Spandan Das9b17df22025-01-08 23:30:45 +0000346 // Copy cmdline, kernel etc. files of boot images
Spandan Das258c08f2025-01-08 23:30:45 +0000347 if a.partitionProps.Vendor_boot_partition_name != nil {
Cole Faust19eb09d2025-01-14 13:27:00 -0800348 bootImg := ctx.GetDirectDepProxyWithTag(proptools.String(a.partitionProps.Vendor_boot_partition_name), filesystemDepTag)
Spandan Das258c08f2025-01-08 23:30:45 +0000349 bootImgInfo, _ := android.OtherModuleProvider(ctx, bootImg, BootimgInfoProvider)
Spandan Das9b17df22025-01-08 23:30:45 +0000350 builder.Command().Textf("echo %s > %s/VENDOR_BOOT/cmdline", proptools.ShellEscape(strings.Join(bootImgInfo.Cmdline, " ")), targetFilesDir)
351 builder.Command().Textf("echo %s > %s/VENDOR_BOOT/vendor_cmdline", proptools.ShellEscape(strings.Join(bootImgInfo.Cmdline, " ")), targetFilesDir)
352 if bootImgInfo.Dtb != nil {
Spandan Dasfed3d042025-01-13 21:38:47 +0000353 builder.Command().Textf("cp ").Input(bootImgInfo.Dtb).Textf(" %s/VENDOR_BOOT/dtb", targetFilesDir)
Spandan Das9b17df22025-01-08 23:30:45 +0000354 }
Spandan Das23511372025-01-08 23:30:45 +0000355 if bootImgInfo.Bootconfig != nil {
Spandan Dasfed3d042025-01-13 21:38:47 +0000356 builder.Command().Textf("cp ").Input(bootImgInfo.Bootconfig).Textf(" %s/VENDOR_BOOT/vendor_bootconfig", targetFilesDir)
Spandan Das23511372025-01-08 23:30:45 +0000357 }
Spandan Das258c08f2025-01-08 23:30:45 +0000358 }
359 if a.partitionProps.Boot_partition_name != nil {
Cole Faust19eb09d2025-01-14 13:27:00 -0800360 bootImg := ctx.GetDirectDepProxyWithTag(proptools.String(a.partitionProps.Boot_partition_name), filesystemDepTag)
Spandan Das258c08f2025-01-08 23:30:45 +0000361 bootImgInfo, _ := android.OtherModuleProvider(ctx, bootImg, BootimgInfoProvider)
Spandan Das9b17df22025-01-08 23:30:45 +0000362 builder.Command().Textf("echo %s > %s/BOOT/cmdline", proptools.ShellEscape(strings.Join(bootImgInfo.Cmdline, " ")), targetFilesDir)
363 if bootImgInfo.Dtb != nil {
Spandan Dasfed3d042025-01-13 21:38:47 +0000364 builder.Command().Textf("cp ").Input(bootImgInfo.Dtb).Textf(" %s/BOOT/dtb", targetFilesDir)
Spandan Das9b17df22025-01-08 23:30:45 +0000365 }
366 if bootImgInfo.Kernel != nil {
Spandan Dasfed3d042025-01-13 21:38:47 +0000367 builder.Command().Textf("cp ").Input(bootImgInfo.Kernel).Textf(" %s/BOOT/kernel", targetFilesDir)
Spandan Das9b17df22025-01-08 23:30:45 +0000368 // 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 +0000369 builder.Command().Textf("cp ").Input(bootImgInfo.Kernel).Textf(" %s/VENDOR_BOOT/kernel", targetFilesDir)
Spandan Das9b17df22025-01-08 23:30:45 +0000370 }
Spandan Das23511372025-01-08 23:30:45 +0000371 if bootImgInfo.Bootconfig != nil {
Spandan Dasfed3d042025-01-13 21:38:47 +0000372 builder.Command().Textf("cp ").Input(bootImgInfo.Bootconfig).Textf(" %s/BOOT/bootconfig", targetFilesDir)
Spandan Das23511372025-01-08 23:30:45 +0000373 }
Spandan Das258c08f2025-01-08 23:30:45 +0000374 }
375
Spandan Dase51ff952025-01-09 18:11:59 +0000376 if a.deviceProps.Android_info != nil {
377 builder.Command().Textf("mkdir -p %s/OTA", targetFilesDir)
Cole Faust11fda332025-01-14 16:47:19 -0800378 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 +0000379 }
380
Spandan Das0036fa32025-01-10 23:40:45 +0000381 a.copyImagesToTargetZip(ctx, builder, targetFilesDir)
Spandan Das29d44882025-01-15 21:12:36 +0000382 a.copyMetadataToTargetZip(ctx, builder, targetFilesDir)
Spandan Das0036fa32025-01-10 23:40:45 +0000383
Cole Faust44080412024-12-20 14:17:07 -0800384 builder.Command().
385 BuiltTool("soong_zip").
386 Text("-d").
387 FlagWithOutput("-o ", targetFilesZip).
388 FlagWithArg("-C ", targetFilesDir.String()).
389 FlagWithArg("-D ", targetFilesDir.String()).
390 Text("-sha256")
391 builder.Build("target_files_"+ctx.ModuleName(), "Build target_files.zip")
392}
393
Spandan Das0036fa32025-01-10 23:40:45 +0000394func (a *androidDevice) copyImagesToTargetZip(ctx android.ModuleContext, builder *android.RuleBuilder, targetFilesDir android.WritablePath) {
395 // Create an IMAGES/ subdirectory
396 builder.Command().Textf("mkdir -p %s/IMAGES", targetFilesDir.String())
397 if a.deviceProps.Bootloader != nil {
398 builder.Command().Textf("cp ").Input(android.PathForModuleSrc(ctx, proptools.String(a.deviceProps.Bootloader))).Textf(" %s/IMAGES/bootloader", targetFilesDir.String())
399 }
400 // Copy the filesystem ,boot and vbmeta img files to IMAGES/
401 ctx.VisitDirectDepsProxyWithTag(filesystemDepTag, func(child android.ModuleProxy) {
Spandan Dasa9db76d2025-01-14 01:34:43 +0000402 if strings.Contains(child.Name(), "recovery") {
403 return // skip recovery.img to match the make packaging behavior
404 }
Spandan Dasef775742025-01-13 22:17:40 +0000405 if info, ok := android.OtherModuleProvider(ctx, child, BootimgInfoProvider); ok {
406 // Check Boot img first so that the boot.img is copied and not its dep ramdisk.img
Spandan Das0036fa32025-01-10 23:40:45 +0000407 builder.Command().Textf("cp ").Input(info.Output).Textf(" %s/IMAGES/", targetFilesDir.String())
Spandan Dasef775742025-01-13 22:17:40 +0000408 } else if info, ok := android.OtherModuleProvider(ctx, child, FilesystemProvider); ok {
Spandan Das0036fa32025-01-10 23:40:45 +0000409 builder.Command().Textf("cp ").Input(info.Output).Textf(" %s/IMAGES/", targetFilesDir.String())
410 } else if info, ok := android.OtherModuleProvider(ctx, child, vbmetaPartitionProvider); ok {
411 builder.Command().Textf("cp ").Input(info.Output).Textf(" %s/IMAGES/", targetFilesDir.String())
412 } else {
413 ctx.ModuleErrorf("Module %s does not provide an .img file output for target_files.zip", child.Name())
414 }
415 })
Spandan Dasef775742025-01-13 22:17:40 +0000416
417 if a.partitionProps.Super_partition_name != nil {
Cole Faust19eb09d2025-01-14 13:27:00 -0800418 superPartition := ctx.GetDirectDepProxyWithTag(*a.partitionProps.Super_partition_name, superPartitionDepTag)
Spandan Dasef775742025-01-13 22:17:40 +0000419 if info, ok := android.OtherModuleProvider(ctx, superPartition, SuperImageProvider); ok {
Cole Faust19eb09d2025-01-14 13:27:00 -0800420 for _, partition := range android.SortedKeys(info.SubImageInfo) {
Spandan Das1f0a5a12025-01-15 00:53:15 +0000421 builder.Command().Textf("cp ").Input(info.SubImageInfo[partition].OutputHermetic).Textf(" %s/IMAGES/", targetFilesDir.String())
Spandan Das33c9c472025-01-14 19:26:23 +0000422 builder.Command().Textf("cp ").Input(info.SubImageInfo[partition].MapFile).Textf(" %s/IMAGES/", targetFilesDir.String())
Spandan Dasef775742025-01-13 22:17:40 +0000423 }
424 } else {
425 ctx.ModuleErrorf("Super partition %s does set SuperImageProvider\n", superPartition.Name())
426 }
427 }
Spandan Das0036fa32025-01-10 23:40:45 +0000428}
429
Spandan Das29d44882025-01-15 21:12:36 +0000430func (a *androidDevice) copyMetadataToTargetZip(ctx android.ModuleContext, builder *android.RuleBuilder, targetFilesDir android.WritablePath) {
431 // Create a META/ subdirectory
432 builder.Command().Textf("mkdir -p %s/META", targetFilesDir.String())
433 if proptools.Bool(a.deviceProps.Ab_ota_updater) {
434 ctx.VisitDirectDepsProxyWithTag(targetFilesMetadataDepTag, func(child android.ModuleProxy) {
435 info, _ := android.OtherModuleProvider(ctx, child, android.OutputFilesProvider)
436 builder.Command().Textf("cp").Inputs(info.DefaultOutputFiles).Textf(" %s/META/", targetFilesDir.String())
437 })
438 }
439 builder.Command().Textf("cp").Input(android.PathForSource(ctx, "external/zucchini/version_info.h")).Textf(" %s/META/zucchini_config.txt", targetFilesDir.String())
440 builder.Command().Textf("cp").Input(android.PathForSource(ctx, "system/update_engine/update_engine.conf")).Textf(" %s/META/update_engine_config.txt", targetFilesDir.String())
441}
442
Cole Faust44080412024-12-20 14:17:07 -0800443func (a *androidDevice) getFilesystemInfo(ctx android.ModuleContext, depName string) FilesystemInfo {
Cole Faust19eb09d2025-01-14 13:27:00 -0800444 fsMod := ctx.GetDirectDepProxyWithTag(depName, filesystemDepTag)
Cole Faust44080412024-12-20 14:17:07 -0800445 fsInfo, ok := android.OtherModuleProvider(ctx, fsMod, FilesystemProvider)
446 if !ok {
447 ctx.ModuleErrorf("Expected dependency %s to be a filesystem", depName)
448 }
449 return fsInfo
Jihoon Kangf2c53982024-10-09 17:32:52 +0000450}
Jihoon Kang0a6315b2025-01-30 01:14:49 +0000451
452func (a *androidDevice) setVbmetaPhonyTargets(ctx android.ModuleContext) {
453 if !proptools.Bool(a.deviceProps.Main_device) {
454 return
455 }
456
457 if !ctx.Config().KatiEnabled() {
458 for _, vbmetaPartitionName := range a.partitionProps.Vbmeta_partitions {
459 img := ctx.GetDirectDepProxyWithTag(vbmetaPartitionName, filesystemDepTag)
460 if provider, ok := android.OtherModuleProvider(ctx, img, vbmetaPartitionProvider); ok {
461 // make generates `vbmetasystemimage` phony target instead of `vbmeta_systemimage` phony target.
462 partitionName := strings.ReplaceAll(provider.Name, "_", "")
463 ctx.Phony(fmt.Sprintf("%simage", partitionName), provider.Output)
464 }
465 }
466 }
467}