// Copyright (C) 2024 The Android Open Source Project // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package filesystem import ( "cmp" "fmt" "path/filepath" "slices" "strings" "sync/atomic" "android/soong/android" "android/soong/java" "github.com/google/blueprint" "github.com/google/blueprint/proptools" ) var proguardDictToProto = pctx.AndroidStaticRule("proguard_dict_to_proto", blueprint.RuleParams{ Command: `${symbols_map} -r8 $in -location $location -write_if_changed $out`, Restat: true, CommandDeps: []string{"${symbols_map}"}, }, "location") type PartitionNameProperties struct { // Name of the super partition filesystem module Super_partition_name *string // Name of the boot partition filesystem module Boot_partition_name *string // Name of the vendor boot partition filesystem module Vendor_boot_partition_name *string // Name of the init boot partition filesystem module Init_boot_partition_name *string // Name of the system partition filesystem module System_partition_name *string // Name of the system_ext partition filesystem module System_ext_partition_name *string // Name of the product partition filesystem module Product_partition_name *string // Name of the vendor partition filesystem module Vendor_partition_name *string // Name of the odm partition filesystem module Odm_partition_name *string // Name of the recovery partition filesystem module Recovery_partition_name *string // The vbmeta partition and its "chained" partitions Vbmeta_partitions []string // Name of the userdata partition filesystem module Userdata_partition_name *string // Name of the system_dlkm partition filesystem module System_dlkm_partition_name *string // Name of the vendor_dlkm partition filesystem module Vendor_dlkm_partition_name *string // Name of the odm_dlkm partition filesystem module Odm_dlkm_partition_name *string } type DeviceProperties struct { // Path to the prebuilt bootloader that would be copied to PRODUCT_OUT Bootloader *string `android:"path"` // Path to android-info.txt file containing board specific info. Android_info *string `android:"path"` // If this is the "main" android_device target for the build, i.e. the one that gets built // when running a plain `m` command. Currently, this is the autogenerated android_device module // in soong-only builds, but in the future when we check in android_device modules, the main // one will be determined based on the lunch product. TODO: Figure out how to make this // blueprint:"mutated" and still set it from filesystem_creator Main_device *bool Ab_ota_updater *bool Ab_ota_partitions []string Ab_ota_keys []string Ab_ota_postinstall_config []string Ramdisk_node_list *string `android:"path"` Releasetools_extension *string `android:"path"` FastbootInfo *string `android:"path"` } type androidDevice struct { android.ModuleBase partitionProps PartitionNameProperties deviceProps DeviceProperties allImagesZip android.Path proguardDictZip android.Path proguardDictMapping android.Path proguardUsageZip android.Path } func AndroidDeviceFactory() android.Module { module := &androidDevice{} module.AddProperties(&module.partitionProps, &module.deviceProps) android.InitAndroidMultiTargetsArchModule(module, android.DeviceSupported, android.MultilibFirst) return module } var numMainAndroidDevicesOnceKey android.OnceKey = android.NewOnceKey("num_auto_generated_anroid_devices") type partitionDepTagType struct { blueprint.BaseDependencyTag } type superPartitionDepTagType struct { blueprint.BaseDependencyTag } type targetFilesMetadataDepTagType struct { blueprint.BaseDependencyTag } var superPartitionDepTag superPartitionDepTagType var filesystemDepTag partitionDepTagType var targetFilesMetadataDepTag targetFilesMetadataDepTagType func (a *androidDevice) DepsMutator(ctx android.BottomUpMutatorContext) { addDependencyIfDefined := func(dep *string) { if dep != nil { ctx.AddDependency(ctx.Module(), filesystemDepTag, proptools.String(dep)) } } if a.partitionProps.Super_partition_name != nil { ctx.AddDependency(ctx.Module(), superPartitionDepTag, *a.partitionProps.Super_partition_name) } addDependencyIfDefined(a.partitionProps.Boot_partition_name) addDependencyIfDefined(a.partitionProps.Init_boot_partition_name) addDependencyIfDefined(a.partitionProps.Vendor_boot_partition_name) addDependencyIfDefined(a.partitionProps.System_partition_name) addDependencyIfDefined(a.partitionProps.System_ext_partition_name) addDependencyIfDefined(a.partitionProps.Product_partition_name) addDependencyIfDefined(a.partitionProps.Vendor_partition_name) addDependencyIfDefined(a.partitionProps.Odm_partition_name) addDependencyIfDefined(a.partitionProps.Userdata_partition_name) addDependencyIfDefined(a.partitionProps.System_dlkm_partition_name) addDependencyIfDefined(a.partitionProps.Vendor_dlkm_partition_name) addDependencyIfDefined(a.partitionProps.Odm_dlkm_partition_name) addDependencyIfDefined(a.partitionProps.Recovery_partition_name) for _, vbmetaPartition := range a.partitionProps.Vbmeta_partitions { ctx.AddDependency(ctx.Module(), filesystemDepTag, vbmetaPartition) } a.addDepsForTargetFilesMetadata(ctx) } func (a *androidDevice) addDepsForTargetFilesMetadata(ctx android.BottomUpMutatorContext) { ctx.AddFarVariationDependencies(ctx.Config().BuildOSTarget.Variations(), targetFilesMetadataDepTag, "liblz4") // host variant } func (a *androidDevice) GenerateAndroidBuildActions(ctx android.ModuleContext) { if proptools.Bool(a.deviceProps.Main_device) { numMainAndroidDevices := ctx.Config().Once(numMainAndroidDevicesOnceKey, func() interface{} { return &atomic.Int32{} }).(*atomic.Int32) total := numMainAndroidDevices.Add(1) if total > 1 { // There should only be 1 main android_device module. That one will be // made the default thing to build in soong-only builds. ctx.ModuleErrorf("There cannot be more than 1 main android_device module") } } allInstalledModules := a.allInstalledModules(ctx) a.buildTargetFilesZip(ctx, allInstalledModules) a.buildProguardZips(ctx, allInstalledModules) var deps []android.Path if proptools.String(a.partitionProps.Super_partition_name) != "" { superImage := ctx.GetDirectDepProxyWithTag(*a.partitionProps.Super_partition_name, superPartitionDepTag) if info, ok := android.OtherModuleProvider(ctx, superImage, SuperImageProvider); ok { assertUnset := func(prop *string, propName string) { if prop != nil && *prop != "" { ctx.PropertyErrorf(propName, "Cannot be set because it's already part of the super image") } } for _, subPartitionType := range android.SortedKeys(info.SubImageInfo) { switch subPartitionType { case "system": assertUnset(a.partitionProps.System_partition_name, "system_partition_name") case "system_ext": assertUnset(a.partitionProps.System_ext_partition_name, "system_ext_partition_name") case "system_dlkm": assertUnset(a.partitionProps.System_dlkm_partition_name, "system_dlkm_partition_name") case "system_other": // TODO case "product": assertUnset(a.partitionProps.Product_partition_name, "product_partition_name") case "vendor": assertUnset(a.partitionProps.Vendor_partition_name, "vendor_partition_name") case "vendor_dlkm": assertUnset(a.partitionProps.Vendor_dlkm_partition_name, "vendor_dlkm_partition_name") case "odm": assertUnset(a.partitionProps.Odm_partition_name, "odm_partition_name") case "odm_dlkm": assertUnset(a.partitionProps.Odm_dlkm_partition_name, "odm_dlkm_partition_name") default: ctx.ModuleErrorf("Unsupported sub-partition of super partition: %q", subPartitionType) } } deps = append(deps, info.SuperImage) } else { ctx.ModuleErrorf("Expected super image dep to provide SuperImageProvider") } } ctx.VisitDirectDepsProxyWithTag(filesystemDepTag, func(m android.ModuleProxy) { imageOutput, ok := android.OtherModuleProvider(ctx, m, android.OutputFilesProvider) if !ok { ctx.ModuleErrorf("Partition module %s doesn't set OutputfilesProvider", m.Name()) } if len(imageOutput.DefaultOutputFiles) != 1 { ctx.ModuleErrorf("Partition module %s should provide exact 1 output file", m.Name()) } deps = append(deps, imageOutput.DefaultOutputFiles[0]) }) allImagesZip := android.PathForModuleOut(ctx, "all_images.zip") allImagesZipBuilder := android.NewRuleBuilder(pctx, ctx) cmd := allImagesZipBuilder.Command().BuiltTool("soong_zip") for _, dep := range deps { cmd.FlagWithArg("-e ", dep.Base()) cmd.FlagWithInput("-f ", dep) } cmd.FlagWithOutput("-o ", allImagesZip) allImagesZipBuilder.Build("soong_all_images_zip", "all_images.zip") a.allImagesZip = allImagesZip allImagesStamp := android.PathForModuleOut(ctx, "all_images_stamp") var validations android.Paths if !ctx.Config().KatiEnabled() && proptools.Bool(a.deviceProps.Main_device) { // In soong-only builds, build this module by default. // This is the analogue to this make code: // https://cs.android.com/android/platform/superproject/main/+/main:build/make/core/main.mk;l=1396;drc=6595459cdd8164a6008335f6372c9f97b9094060 ctx.Phony("droidcore-unbundled", allImagesStamp) deps = append(deps, a.copyFilesToProductOutForSoongOnly(ctx)) } ctx.Build(pctx, android.BuildParams{ Rule: android.Touch, Output: allImagesStamp, Implicits: deps, Validations: validations, }) // Checkbuilding it causes soong to make a phony, so you can say `m ` ctx.CheckbuildFile(allImagesStamp) a.setVbmetaPhonyTargets(ctx) a.distFiles(ctx) } // Returns a list of modules that are installed, which are collected from the dependency // filesystem and super_image modules. func (a *androidDevice) allInstalledModules(ctx android.ModuleContext) []android.Module { fsInfoMap := a.getFsInfos(ctx) allOwners := make(map[string][]string) for _, partition := range android.SortedKeys(fsInfoMap) { fsInfo := fsInfoMap[partition] for _, owner := range fsInfo.Owners { allOwners[owner.Name] = append(allOwners[owner.Name], owner.Variation) } } ret := []android.Module{} ctx.WalkDepsProxy(func(mod, _ android.ModuleProxy) bool { if variations, ok := allOwners[ctx.OtherModuleName(mod)]; ok && android.InList(ctx.OtherModuleSubDir(mod), variations) { ret = append(ret, mod) } return true }) // Remove duplicates ret = android.FirstUniqueFunc(ret, func(a, b android.Module) bool { return a.String() == b.String() }) // Sort the modules by their names and variants slices.SortFunc(ret, func(a, b android.Module) int { return cmp.Compare(a.String(), b.String()) }) return ret } func insertBeforeExtension(file, insertion string) string { ext := filepath.Ext(file) return strings.TrimSuffix(file, ext) + insertion + ext } func (a *androidDevice) distFiles(ctx android.ModuleContext) { if !ctx.Config().KatiEnabled() && proptools.Bool(a.deviceProps.Main_device) { fsInfoMap := a.getFsInfos(ctx) for _, partition := range android.SortedKeys(fsInfoMap) { fsInfo := fsInfoMap[partition] if fsInfo.InstalledFiles.Json != nil { ctx.DistForGoal("droidcore-unbundled", fsInfo.InstalledFiles.Json) } if fsInfo.InstalledFiles.Txt != nil { ctx.DistForGoal("droidcore-unbundled", fsInfo.InstalledFiles.Txt) } } namePrefix := "" if ctx.Config().HasDeviceProduct() { namePrefix = ctx.Config().DeviceProduct() + "-" } ctx.DistForGoalWithFilename("droidcore-unbundled", a.proguardDictZip, namePrefix+insertBeforeExtension(a.proguardDictZip.Base(), "-FILE_NAME_TAG_PLACEHOLDER")) ctx.DistForGoalWithFilename("droidcore-unbundled", a.proguardDictMapping, namePrefix+insertBeforeExtension(a.proguardDictMapping.Base(), "-FILE_NAME_TAG_PLACEHOLDER")) ctx.DistForGoalWithFilename("droidcore-unbundled", a.proguardUsageZip, namePrefix+insertBeforeExtension(a.proguardUsageZip.Base(), "-FILE_NAME_TAG_PLACEHOLDER")) } } func (a *androidDevice) MakeVars(_ android.MakeVarsModuleContext) []android.ModuleMakeVarsValue { if proptools.Bool(a.deviceProps.Main_device) { return []android.ModuleMakeVarsValue{{"SOONG_ONLY_ALL_IMAGES_ZIP", a.allImagesZip.String()}} } return nil } func (a *androidDevice) buildProguardZips(ctx android.ModuleContext, allInstalledModules []android.Module) { dictZip := android.PathForModuleOut(ctx, "proguard-dict.zip") dictZipBuilder := android.NewRuleBuilder(pctx, ctx) dictZipCmd := dictZipBuilder.Command().BuiltTool("soong_zip").Flag("-d").FlagWithOutput("-o ", dictZip) dictMapping := android.PathForModuleOut(ctx, "proguard-dict-mapping.textproto") dictMappingBuilder := android.NewRuleBuilder(pctx, ctx) dictMappingCmd := dictMappingBuilder.Command().BuiltTool("symbols_map").Flag("-merge").Output(dictMapping) protosDir := android.PathForModuleOut(ctx, "proguard_mapping_protos") usageZip := android.PathForModuleOut(ctx, "proguard-usage.zip") usageZipBuilder := android.NewRuleBuilder(pctx, ctx) usageZipCmd := usageZipBuilder.Command().BuiltTool("merge_zips").Output(usageZip) for _, mod := range allInstalledModules { if proguardInfo, ok := android.OtherModuleProvider(ctx, mod, java.ProguardProvider); ok { // Maintain these out/target/common paths for backwards compatibility. They may be able // to be changed if tools look up file locations from the protobuf, but I'm not // exactly sure how that works. dictionaryFakePath := fmt.Sprintf("out/target/common/obj/%s/%s_intermediates/proguard_dictionary", proguardInfo.Class, proguardInfo.ModuleName) dictZipCmd.FlagWithArg("-e ", dictionaryFakePath) dictZipCmd.FlagWithInput("-f ", proguardInfo.ProguardDictionary) dictZipCmd.Textf("-e out/target/common/obj/%s/%s_intermediates/classes.jar", proguardInfo.Class, proguardInfo.ModuleName) dictZipCmd.FlagWithInput("-f ", proguardInfo.ClassesJar) protoFile := protosDir.Join(ctx, filepath.Dir(dictionaryFakePath), "proguard_dictionary.textproto") ctx.Build(pctx, android.BuildParams{ Rule: proguardDictToProto, Input: proguardInfo.ProguardDictionary, Output: protoFile, Args: map[string]string{ "location": dictionaryFakePath, }, }) dictMappingCmd.Input(protoFile) usageZipCmd.Input(proguardInfo.ProguardUsageZip) } } dictZipBuilder.Build("proguard_dict_zip", "Building proguard dictionary zip") dictMappingBuilder.Build("proguard_dict_mapping_proto", "Building proguard mapping proto") usageZipBuilder.Build("proguard_usage_zip", "Building proguard usage zip") a.proguardDictZip = dictZip a.proguardDictMapping = dictMapping a.proguardUsageZip = usageZip } // Helper structs for target_files.zip creation type targetFilesZipCopy struct { srcModule *string destSubdir string } type targetFilesystemZipCopy struct { fsInfo FilesystemInfo destSubdir string } func (a *androidDevice) buildTargetFilesZip(ctx android.ModuleContext, allInstalledModules []android.Module) { targetFilesDir := android.PathForModuleOut(ctx, "target_files_dir") targetFilesZip := android.PathForModuleOut(ctx, "target_files.zip") builder := android.NewRuleBuilder(pctx, ctx) builder.Command().Textf("rm -rf %s", targetFilesDir.String()) builder.Command().Textf("mkdir -p %s", targetFilesDir.String()) toCopy := []targetFilesZipCopy{ targetFilesZipCopy{a.partitionProps.System_partition_name, "SYSTEM"}, targetFilesZipCopy{a.partitionProps.System_ext_partition_name, "SYSTEM_EXT"}, targetFilesZipCopy{a.partitionProps.Product_partition_name, "PRODUCT"}, targetFilesZipCopy{a.partitionProps.Vendor_partition_name, "VENDOR"}, targetFilesZipCopy{a.partitionProps.Odm_partition_name, "ODM"}, targetFilesZipCopy{a.partitionProps.System_dlkm_partition_name, "SYSTEM_DLKM"}, targetFilesZipCopy{a.partitionProps.Vendor_dlkm_partition_name, "VENDOR_DLKM"}, targetFilesZipCopy{a.partitionProps.Odm_dlkm_partition_name, "ODM_DLKM"}, targetFilesZipCopy{a.partitionProps.Init_boot_partition_name, "BOOT/RAMDISK"}, targetFilesZipCopy{a.partitionProps.Init_boot_partition_name, "INIT_BOOT/RAMDISK"}, targetFilesZipCopy{a.partitionProps.Vendor_boot_partition_name, "VENDOR_BOOT/RAMDISK"}, } filesystemsToCopy := []targetFilesystemZipCopy{} for _, zipCopy := range toCopy { if zipCopy.srcModule == nil { continue } filesystemsToCopy = append( filesystemsToCopy, targetFilesystemZipCopy{a.getFilesystemInfo(ctx, *zipCopy.srcModule), zipCopy.destSubdir}, ) } // Get additional filesystems from super_partition dependency if a.partitionProps.Super_partition_name != nil { superPartition := ctx.GetDirectDepProxyWithTag(*a.partitionProps.Super_partition_name, superPartitionDepTag) if info, ok := android.OtherModuleProvider(ctx, superPartition, SuperImageProvider); ok { for _, partition := range android.SortedKeys(info.SubImageInfo) { filesystemsToCopy = append( filesystemsToCopy, targetFilesystemZipCopy{info.SubImageInfo[partition], strings.ToUpper(partition)}, ) } } else { ctx.ModuleErrorf("Super partition %s does set SuperImageProvider\n", superPartition.Name()) } } for _, toCopy := range filesystemsToCopy { rootDirString := toCopy.fsInfo.RootDir.String() if toCopy.destSubdir == "SYSTEM" { rootDirString = rootDirString + "/system" } builder.Command().Textf("mkdir -p %s/%s", targetFilesDir.String(), toCopy.destSubdir) builder.Command(). BuiltTool("acp"). Textf("-rd %s/. %s/%s", rootDirString, targetFilesDir, toCopy.destSubdir). Implicit(toCopy.fsInfo.Output) // so that the staging dir is built for _, extraRootDir := range toCopy.fsInfo.ExtraRootDirs { builder.Command(). BuiltTool("acp"). Textf("-rd %s/. %s/%s", extraRootDir, targetFilesDir, toCopy.destSubdir). Implicit(toCopy.fsInfo.Output) // so that the staging dir is built } if toCopy.destSubdir == "SYSTEM" { // Create the ROOT partition in target_files.zip builder.Command().Textf("rsync --links --exclude=system/* %s/ -r %s/ROOT", toCopy.fsInfo.RootDir, targetFilesDir.String()) } } // Copy cmdline, kernel etc. files of boot images if a.partitionProps.Vendor_boot_partition_name != nil { bootImg := ctx.GetDirectDepProxyWithTag(proptools.String(a.partitionProps.Vendor_boot_partition_name), filesystemDepTag) bootImgInfo, _ := android.OtherModuleProvider(ctx, bootImg, BootimgInfoProvider) builder.Command().Textf("echo %s > %s/VENDOR_BOOT/cmdline", proptools.ShellEscape(strings.Join(bootImgInfo.Cmdline, " ")), targetFilesDir) builder.Command().Textf("echo %s > %s/VENDOR_BOOT/vendor_cmdline", proptools.ShellEscape(strings.Join(bootImgInfo.Cmdline, " ")), targetFilesDir) if bootImgInfo.Dtb != nil { builder.Command().Textf("cp ").Input(bootImgInfo.Dtb).Textf(" %s/VENDOR_BOOT/dtb", targetFilesDir) } if bootImgInfo.Bootconfig != nil { builder.Command().Textf("cp ").Input(bootImgInfo.Bootconfig).Textf(" %s/VENDOR_BOOT/vendor_bootconfig", targetFilesDir) } } if a.partitionProps.Boot_partition_name != nil { bootImg := ctx.GetDirectDepProxyWithTag(proptools.String(a.partitionProps.Boot_partition_name), filesystemDepTag) bootImgInfo, _ := android.OtherModuleProvider(ctx, bootImg, BootimgInfoProvider) builder.Command().Textf("echo %s > %s/BOOT/cmdline", proptools.ShellEscape(strings.Join(bootImgInfo.Cmdline, " ")), targetFilesDir) if bootImgInfo.Dtb != nil { builder.Command().Textf("cp ").Input(bootImgInfo.Dtb).Textf(" %s/BOOT/dtb", targetFilesDir) } if bootImgInfo.Kernel != nil { builder.Command().Textf("cp ").Input(bootImgInfo.Kernel).Textf(" %s/BOOT/kernel", targetFilesDir) // Even though kernel is not used to build vendor_boot, copy the kernel to VENDOR_BOOT to match the behavior of make packaging. builder.Command().Textf("cp ").Input(bootImgInfo.Kernel).Textf(" %s/VENDOR_BOOT/kernel", targetFilesDir) } if bootImgInfo.Bootconfig != nil { builder.Command().Textf("cp ").Input(bootImgInfo.Bootconfig).Textf(" %s/BOOT/bootconfig", targetFilesDir) } } if a.deviceProps.Android_info != nil { builder.Command().Textf("mkdir -p %s/OTA", targetFilesDir) builder.Command().Textf("cp ").Input(android.PathForModuleSrc(ctx, *a.deviceProps.Android_info)).Textf(" %s/OTA/android-info.txt", targetFilesDir) } a.copyImagesToTargetZip(ctx, builder, targetFilesDir) a.copyMetadataToTargetZip(ctx, builder, targetFilesDir, allInstalledModules) builder.Command(). BuiltTool("soong_zip"). Text("-d"). FlagWithOutput("-o ", targetFilesZip). FlagWithArg("-C ", targetFilesDir.String()). FlagWithArg("-D ", targetFilesDir.String()). Text("-sha256") builder.Build("target_files_"+ctx.ModuleName(), "Build target_files.zip") } func (a *androidDevice) copyImagesToTargetZip(ctx android.ModuleContext, builder *android.RuleBuilder, targetFilesDir android.WritablePath) { // Create an IMAGES/ subdirectory builder.Command().Textf("mkdir -p %s/IMAGES", targetFilesDir.String()) if a.deviceProps.Bootloader != nil { builder.Command().Textf("cp ").Input(android.PathForModuleSrc(ctx, proptools.String(a.deviceProps.Bootloader))).Textf(" %s/IMAGES/bootloader", targetFilesDir.String()) } // Copy the filesystem ,boot and vbmeta img files to IMAGES/ ctx.VisitDirectDepsProxyWithTag(filesystemDepTag, func(child android.ModuleProxy) { if strings.Contains(child.Name(), "recovery") { return // skip recovery.img to match the make packaging behavior } if info, ok := android.OtherModuleProvider(ctx, child, BootimgInfoProvider); ok { // Check Boot img first so that the boot.img is copied and not its dep ramdisk.img builder.Command().Textf("cp ").Input(info.Output).Textf(" %s/IMAGES/", targetFilesDir.String()) } else if info, ok := android.OtherModuleProvider(ctx, child, FilesystemProvider); ok { builder.Command().Textf("cp ").Input(info.Output).Textf(" %s/IMAGES/", targetFilesDir.String()) } else if info, ok := android.OtherModuleProvider(ctx, child, vbmetaPartitionProvider); ok { builder.Command().Textf("cp ").Input(info.Output).Textf(" %s/IMAGES/", targetFilesDir.String()) } else { ctx.ModuleErrorf("Module %s does not provide an .img file output for target_files.zip", child.Name()) } }) if a.partitionProps.Super_partition_name != nil { superPartition := ctx.GetDirectDepProxyWithTag(*a.partitionProps.Super_partition_name, superPartitionDepTag) if info, ok := android.OtherModuleProvider(ctx, superPartition, SuperImageProvider); ok { for _, partition := range android.SortedKeys(info.SubImageInfo) { if info.SubImageInfo[partition].OutputHermetic != nil { builder.Command().Textf("cp ").Input(info.SubImageInfo[partition].OutputHermetic).Textf(" %s/IMAGES/", targetFilesDir.String()) } if info.SubImageInfo[partition].MapFile != nil { builder.Command().Textf("cp ").Input(info.SubImageInfo[partition].MapFile).Textf(" %s/IMAGES/", targetFilesDir.String()) } } } else { ctx.ModuleErrorf("Super partition %s does set SuperImageProvider\n", superPartition.Name()) } } } func (a *androidDevice) copyMetadataToTargetZip(ctx android.ModuleContext, builder *android.RuleBuilder, targetFilesDir android.WritablePath, allInstalledModules []android.Module) { // Create a META/ subdirectory builder.Command().Textf("mkdir -p %s/META", targetFilesDir.String()) if proptools.Bool(a.deviceProps.Ab_ota_updater) { ctx.VisitDirectDepsProxyWithTag(targetFilesMetadataDepTag, func(child android.ModuleProxy) { info, _ := android.OtherModuleProvider(ctx, child, android.OutputFilesProvider) builder.Command().Textf("cp").Inputs(info.DefaultOutputFiles).Textf(" %s/META/", targetFilesDir.String()) }) builder.Command().Textf("cp").Input(android.PathForSource(ctx, "external/zucchini/version_info.h")).Textf(" %s/META/zucchini_config.txt", targetFilesDir.String()) builder.Command().Textf("cp").Input(android.PathForSource(ctx, "system/update_engine/update_engine.conf")).Textf(" %s/META/update_engine_config.txt", targetFilesDir.String()) if a.getFsInfos(ctx)["system"].ErofsCompressHints != nil { builder.Command().Textf("cp").Input(a.getFsInfos(ctx)["system"].ErofsCompressHints).Textf(" %s/META/erofs_default_compress_hints.txt", targetFilesDir.String()) } // ab_partitions.txt abPartitionsSorted := android.SortedUniqueStrings(a.deviceProps.Ab_ota_partitions) abPartitionsSortedString := proptools.ShellEscape(strings.Join(abPartitionsSorted, "\\n")) builder.Command().Textf("echo -e").Flag(abPartitionsSortedString).Textf(" > %s/META/ab_partitions.txt", targetFilesDir.String()) // otakeys.txt abOtaKeysSorted := android.SortedUniqueStrings(a.deviceProps.Ab_ota_keys) abOtaKeysSortedString := proptools.ShellEscape(strings.Join(abOtaKeysSorted, "\\n")) builder.Command().Textf("echo -e").Flag(abOtaKeysSortedString).Textf(" > %s/META/otakeys.txt", targetFilesDir.String()) // postinstall_config.txt abOtaPostInstallConfigString := proptools.ShellEscape(strings.Join(a.deviceProps.Ab_ota_postinstall_config, "\\n")) builder.Command().Textf("echo -e").Flag(abOtaPostInstallConfigString).Textf(" > %s/META/postinstall_config.txt", targetFilesDir.String()) // selinuxfc if a.getFsInfos(ctx)["system"].SelinuxFc != nil { builder.Command().Textf("cp").Input(a.getFsInfos(ctx)["system"].SelinuxFc).Textf(" %s/META/file_contexts.bin", targetFilesDir.String()) } } // Copy $partition_filesystem_config.txt fsInfos := a.getFsInfos(ctx) for _, partition := range android.SortedKeys(fsInfos) { if fsInfos[partition].FilesystemConfig == nil { continue } if android.InList(partition, []string{"userdata"}) { continue } if partition != "vendor_ramdisk" { // vendor_ramdisk will be handled separately. builder.Command().Textf("cp").Input(fsInfos[partition].FilesystemConfig).Textf(" %s/META/%s", targetFilesDir.String(), a.filesystemConfigNameForTargetFiles(partition)) } if partition == "ramdisk" { // Create an additional copy at boot_filesystem_config.txt builder.Command().Textf("cp").Input(fsInfos[partition].FilesystemConfig).Textf(" %s/META/boot_filesystem_config.txt", targetFilesDir.String()) } if partition == "system" { // Create root_filesystem_config from the assembled ROOT/ intermediates directory a.generateFilesystemConfigForTargetFiles(ctx, builder, targetFilesDir.String(), targetFilesDir.String()+"/ROOT", "root_filesystem_config.txt") } if partition == "vendor_ramdisk" { // Create vendor_boot_filesystem_config from the assembled VENDOR_BOOT/RAMDISK intermediates directory a.generateFilesystemConfigForTargetFiles(ctx, builder, targetFilesDir.String(), targetFilesDir.String()+"/VENDOR_BOOT/RAMDISK", "vendor_boot_filesystem_config.txt") } } // Copy ramdisk_node_list if ramdiskNodeList := android.PathForModuleSrc(ctx, proptools.String(a.deviceProps.Ramdisk_node_list)); ramdiskNodeList != nil { builder.Command().Textf("cp").Input(ramdiskNodeList).Textf(" %s/META/", targetFilesDir.String()) } // Copy releasetools.py if releaseTools := android.PathForModuleSrc(ctx, proptools.String(a.deviceProps.Releasetools_extension)); releaseTools != nil { builder.Command().Textf("cp").Input(releaseTools).Textf(" %s/META/", targetFilesDir.String()) } // apexkeys.txt var installedApexKeys []android.Path for _, installedModule := range allInstalledModules { if info, ok := android.OtherModuleProvider(ctx, installedModule, ApexKeyPathInfoProvider); ok { installedApexKeys = append(installedApexKeys, info.ApexKeyPath) } } installedApexKeys = android.SortedUniquePaths(installedApexKeys) // Sort by keypath to match make builder.Command().Text("cat").Inputs(installedApexKeys).Textf(" >> %s/META/apexkeys.txt", targetFilesDir.String()) // Copy fastboot-info.txt if fastbootInfo := android.PathForModuleSrc(ctx, proptools.String(a.deviceProps.FastbootInfo)); fastbootInfo != nil { // TODO (b/399788523): Autogenerate fastboot-info.txt if there is no source fastboot-info.txt // https://cs.android.com/android/_/android/platform/build/+/80b9546f8f69e78b8fe1870e0e745d70fc18dfcd:core/Makefile;l=5831-5893;drc=077490384423dff9eac954da5c001c6f0be3fa6e;bpv=0;bpt=0 builder.Command().Textf("cp").Input(fastbootInfo).Textf(" %s/META/fastboot-info.txt", targetFilesDir.String()) } if a.partitionProps.Super_partition_name != nil { superPartition := ctx.GetDirectDepProxyWithTag(*a.partitionProps.Super_partition_name, superPartitionDepTag) if info, ok := android.OtherModuleProvider(ctx, superPartition, SuperImageProvider); ok { // dynamic_partitions_info.txt // TODO (b/390192334): Add `building_super_empty_partition=true` builder.Command().Text("cp").Input(info.DynamicPartitionsInfo).Textf(" %s/META/", targetFilesDir.String()) } else { ctx.ModuleErrorf("Super partition %s does set SuperImageProvider\n", superPartition.Name()) } } } type ApexKeyPathInfo struct { ApexKeyPath android.Path } var ApexKeyPathInfoProvider = blueprint.NewProvider[ApexKeyPathInfo]() func (a *androidDevice) generateFilesystemConfigForTargetFiles(ctx android.ModuleContext, builder *android.RuleBuilder, targetFilesDir, stagingDir, filename string) { fsConfigOut := android.PathForModuleOut(ctx, filename) ctx.Build(pctx, android.BuildParams{ Rule: fsConfigRule, Output: fsConfigOut, Args: map[string]string{ "rootDir": stagingDir, "prefix": "", }, }) builder.Command().Textf("cp").Input(fsConfigOut).Textf(" %s/META/", targetFilesDir) } // Filenames for the partition specific fs_config files. // Hardcode the ramdisk files to their boot image prefix func (a *androidDevice) filesystemConfigNameForTargetFiles(partition string) string { name := partition + "_filesystem_config.txt" if partition == "system" { name = "filesystem_config.txt" } else if partition == "ramdisk" { name = "init_boot_filesystem_config.txt" } return name } func (a *androidDevice) getFilesystemInfo(ctx android.ModuleContext, depName string) FilesystemInfo { fsMod := ctx.GetDirectDepProxyWithTag(depName, filesystemDepTag) fsInfo, ok := android.OtherModuleProvider(ctx, fsMod, FilesystemProvider) if !ok { ctx.ModuleErrorf("Expected dependency %s to be a filesystem", depName) } return fsInfo } func (a *androidDevice) setVbmetaPhonyTargets(ctx android.ModuleContext) { if !proptools.Bool(a.deviceProps.Main_device) { return } if !ctx.Config().KatiEnabled() { for _, vbmetaPartitionName := range a.partitionProps.Vbmeta_partitions { img := ctx.GetDirectDepProxyWithTag(vbmetaPartitionName, filesystemDepTag) if provider, ok := android.OtherModuleProvider(ctx, img, vbmetaPartitionProvider); ok { // make generates `vbmetasystemimage` phony target instead of `vbmeta_systemimage` phony target. partitionName := strings.ReplaceAll(provider.Name, "_", "") ctx.Phony(fmt.Sprintf("%simage", partitionName), provider.Output) } } } }