blob: 5282636384b1a3d434aaa5473570c23d1df11be4 [file] [log] [blame]
Jihoon Kang98047cf2024-10-02 17:13:54 +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 fsgen
16
17import (
Cole Faust92ccbe22024-10-03 14:38:37 -070018 "crypto/sha256"
Jihoon Kang98047cf2024-10-02 17:13:54 +000019 "fmt"
Spandan Das312cc412024-10-29 18:20:11 +000020 "path/filepath"
Jihoon Kang98047cf2024-10-02 17:13:54 +000021 "strconv"
mrziwang8f86c882024-10-03 12:34:33 -070022 "strings"
mrziwang8f86c882024-10-03 12:34:33 -070023
24 "android/soong/android"
25 "android/soong/filesystem"
Spandan Das5e336422024-11-01 22:31:20 +000026 "android/soong/kernel"
Jihoon Kang98047cf2024-10-02 17:13:54 +000027
Cole Faust92ccbe22024-10-03 14:38:37 -070028 "github.com/google/blueprint"
mrziwang8f86c882024-10-03 12:34:33 -070029 "github.com/google/blueprint/parser"
Jihoon Kang98047cf2024-10-02 17:13:54 +000030 "github.com/google/blueprint/proptools"
31)
32
Cole Faust92ccbe22024-10-03 14:38:37 -070033var pctx = android.NewPackageContext("android/soong/fsgen")
34
Jihoon Kang98047cf2024-10-02 17:13:54 +000035func init() {
36 registerBuildComponents(android.InitRegistrationContext)
37}
38
39func registerBuildComponents(ctx android.RegistrationContext) {
40 ctx.RegisterModuleType("soong_filesystem_creator", filesystemCreatorFactory)
mrziwang8f86c882024-10-03 12:34:33 -070041 ctx.PreDepsMutators(RegisterCollectFileSystemDepsMutators)
42}
43
Cole Faust92ccbe22024-10-03 14:38:37 -070044type filesystemCreatorProps struct {
45 Generated_partition_types []string `blueprint:"mutated"`
46 Unsupported_partition_types []string `blueprint:"mutated"`
47}
48
Jihoon Kang98047cf2024-10-02 17:13:54 +000049type filesystemCreator struct {
50 android.ModuleBase
Cole Faust92ccbe22024-10-03 14:38:37 -070051
52 properties filesystemCreatorProps
Jihoon Kang98047cf2024-10-02 17:13:54 +000053}
54
55func filesystemCreatorFactory() android.Module {
56 module := &filesystemCreator{}
57
Cole Faust69788792024-10-10 11:00:36 -070058 android.InitAndroidArchModule(module, android.DeviceSupported, android.MultilibCommon)
Cole Faust92ccbe22024-10-03 14:38:37 -070059 module.AddProperties(&module.properties)
Jihoon Kang98047cf2024-10-02 17:13:54 +000060 android.AddLoadHook(module, func(ctx android.LoadHookContext) {
Jihoon Kang675d4682024-10-24 23:45:11 +000061 generatedPrebuiltEtcModuleNames := createPrebuiltEtcModules(ctx)
62 createFsGenState(ctx, generatedPrebuiltEtcModuleNames)
Jihoon Kang98047cf2024-10-02 17:13:54 +000063 module.createInternalModules(ctx)
64 })
65
66 return module
67}
68
69func (f *filesystemCreator) createInternalModules(ctx android.LoadHookContext) {
Jihoon Kang0d545b82024-10-11 00:21:57 +000070 soongGeneratedPartitions := &ctx.Config().Get(fsGenStateOnceKey).(*FsGenState).soongGeneratedPartitions
71 for _, partitionType := range *soongGeneratedPartitions {
Cole Faust92ccbe22024-10-03 14:38:37 -070072 if f.createPartition(ctx, partitionType) {
73 f.properties.Generated_partition_types = append(f.properties.Generated_partition_types, partitionType)
74 } else {
75 f.properties.Unsupported_partition_types = append(f.properties.Unsupported_partition_types, partitionType)
Jihoon Kang0d545b82024-10-11 00:21:57 +000076 _, *soongGeneratedPartitions = android.RemoveFromList(partitionType, *soongGeneratedPartitions)
Cole Faust92ccbe22024-10-03 14:38:37 -070077 }
78 }
Jihoon Kangf1c79ca2024-10-09 20:18:38 +000079 f.createDeviceModule(ctx)
Jihoon Kang98047cf2024-10-02 17:13:54 +000080}
81
Jihoon Kang0d545b82024-10-11 00:21:57 +000082func generatedModuleName(cfg android.Config, suffix string) string {
Cole Faust92ccbe22024-10-03 14:38:37 -070083 prefix := "soong"
84 if cfg.HasDeviceProduct() {
85 prefix = cfg.DeviceProduct()
86 }
Jihoon Kangf1c79ca2024-10-09 20:18:38 +000087 return fmt.Sprintf("%s_generated_%s", prefix, suffix)
88}
89
Jihoon Kang0d545b82024-10-11 00:21:57 +000090func generatedModuleNameForPartition(cfg android.Config, partitionType string) string {
91 return generatedModuleName(cfg, fmt.Sprintf("%s_image", partitionType))
Jihoon Kangf1c79ca2024-10-09 20:18:38 +000092}
93
94func (f *filesystemCreator) createDeviceModule(ctx android.LoadHookContext) {
95 baseProps := &struct {
96 Name *string
97 }{
Jihoon Kang0d545b82024-10-11 00:21:57 +000098 Name: proptools.StringPtr(generatedModuleName(ctx.Config(), "device")),
Jihoon Kangf1c79ca2024-10-09 20:18:38 +000099 }
100
Priyanka Advani (xWF)dafaa7f2024-10-21 22:55:13 +0000101 // Currently, only the system and system_ext partition module is created.
Jihoon Kangf1c79ca2024-10-09 20:18:38 +0000102 partitionProps := &filesystem.PartitionNameProperties{}
103 if android.InList("system", f.properties.Generated_partition_types) {
Jihoon Kang0d545b82024-10-11 00:21:57 +0000104 partitionProps.System_partition_name = proptools.StringPtr(generatedModuleNameForPartition(ctx.Config(), "system"))
Jihoon Kangf1c79ca2024-10-09 20:18:38 +0000105 }
Spandan Das7a46f6c2024-10-14 18:41:18 +0000106 if android.InList("system_ext", f.properties.Generated_partition_types) {
Jihoon Kang0d545b82024-10-11 00:21:57 +0000107 partitionProps.System_ext_partition_name = proptools.StringPtr(generatedModuleNameForPartition(ctx.Config(), "system_ext"))
Spandan Das7a46f6c2024-10-14 18:41:18 +0000108 }
Spandan Dase3b65312024-10-22 00:27:27 +0000109 if android.InList("vendor", f.properties.Generated_partition_types) {
110 partitionProps.Vendor_partition_name = proptools.StringPtr(generatedModuleNameForPartition(ctx.Config(), "vendor"))
111 }
Jihoon Kang6dd13b62024-10-22 23:21:02 +0000112 if android.InList("product", f.properties.Generated_partition_types) {
113 partitionProps.Product_partition_name = proptools.StringPtr(generatedModuleNameForPartition(ctx.Config(), "product"))
114 }
Spandan Dasc5717162024-11-01 18:33:57 +0000115 if android.InList("odm", f.properties.Generated_partition_types) {
116 partitionProps.Odm_partition_name = proptools.StringPtr(generatedModuleNameForPartition(ctx.Config(), "odm"))
117 }
Jihoon Kangf1c79ca2024-10-09 20:18:38 +0000118
119 ctx.CreateModule(filesystem.AndroidDeviceFactory, baseProps, partitionProps)
Cole Faust92ccbe22024-10-03 14:38:37 -0700120}
121
Jihoon Kang6850d8f2024-10-17 20:45:58 +0000122func partitionSpecificFsProps(fsProps *filesystem.FilesystemProperties, partitionType string) {
123 switch partitionType {
124 case "system":
125 fsProps.Build_logtags = proptools.BoolPtr(true)
126 // https://source.corp.google.com/h/googleplex-android/platform/build//639d79f5012a6542ab1f733b0697db45761ab0f3:core/packaging/flags.mk;l=21;drc=5ba8a8b77507f93aa48cc61c5ba3f31a4d0cbf37;bpv=1;bpt=0
127 fsProps.Gen_aconfig_flags_pb = proptools.BoolPtr(true)
Spandan Dasa8fa6b42024-10-23 00:45:29 +0000128 // Identical to that of the generic_system_image
129 fsProps.Fsverity.Inputs = []string{
130 "etc/boot-image.prof",
131 "etc/dirty-image-objects",
132 "etc/preloaded-classes",
133 "etc/classpaths/*.pb",
134 "framework/*",
135 "framework/*/*", // framework/{arch}
136 "framework/oat/*/*", // framework/oat/{arch}
137 }
138 fsProps.Fsverity.Libs = []string{":framework-res{.export-package.apk}"}
mrziwang9afc2982024-11-05 14:29:48 -0800139 // TODO(b/377734331): only generate the symlinks if the relevant partitions exist
140 fsProps.Symlinks = []filesystem.SymlinkDefinition{
141 filesystem.SymlinkDefinition{
142 Target: proptools.StringPtr("/product"),
143 Name: proptools.StringPtr("system/product"),
144 },
145 filesystem.SymlinkDefinition{
146 Target: proptools.StringPtr("/system_ext"),
147 Name: proptools.StringPtr("system/system_ext"),
148 },
149 filesystem.SymlinkDefinition{
150 Target: proptools.StringPtr("/vendor"),
151 Name: proptools.StringPtr("system/vendor"),
152 },
153 filesystem.SymlinkDefinition{
154 Target: proptools.StringPtr("/system_dlkm/lib/modules"),
155 Name: proptools.StringPtr("system/lib/modules"),
156 },
157 }
158 fsProps.Base_dir = proptools.StringPtr("system")
Spandan Dasa8fa6b42024-10-23 00:45:29 +0000159 case "system_ext":
160 fsProps.Fsverity.Inputs = []string{
161 "framework/*",
162 "framework/*/*", // framework/{arch}
163 "framework/oat/*/*", // framework/oat/{arch}
164 }
165 fsProps.Fsverity.Libs = []string{":framework-res{.export-package.apk}"}
Jihoon Kang6850d8f2024-10-17 20:45:58 +0000166 case "product":
167 fsProps.Gen_aconfig_flags_pb = proptools.BoolPtr(true)
168 case "vendor":
169 fsProps.Gen_aconfig_flags_pb = proptools.BoolPtr(true)
Spandan Das69464c32024-10-25 20:08:06 +0000170 fsProps.Symlinks = []filesystem.SymlinkDefinition{
171 filesystem.SymlinkDefinition{
172 Target: proptools.StringPtr("/odm"),
173 Name: proptools.StringPtr("vendor/odm"),
174 },
175 filesystem.SymlinkDefinition{
176 Target: proptools.StringPtr("/vendor_dlkm/lib/modules"),
177 Name: proptools.StringPtr("vendor/lib/modules"),
178 },
179 }
180 fsProps.Base_dir = proptools.StringPtr("vendor")
Spandan Dasc5717162024-11-01 18:33:57 +0000181 case "odm":
182 fsProps.Symlinks = []filesystem.SymlinkDefinition{
183 filesystem.SymlinkDefinition{
184 Target: proptools.StringPtr("/odm_dlkm/lib/modules"),
185 Name: proptools.StringPtr("odm/lib/modules"),
186 },
187 }
188 fsProps.Base_dir = proptools.StringPtr("odm")
189
Jihoon Kang6850d8f2024-10-17 20:45:58 +0000190 }
191}
Spandan Dascbe641a2024-10-14 21:07:34 +0000192
Cole Faust92ccbe22024-10-03 14:38:37 -0700193// Creates a soong module to build the given partition. Returns false if we can't support building
194// it.
195func (f *filesystemCreator) createPartition(ctx android.LoadHookContext, partitionType string) bool {
mrziwang4b0ca972024-10-17 14:56:19 -0700196 baseProps := generateBaseProps(proptools.StringPtr(generatedModuleNameForPartition(ctx.Config(), partitionType)))
197
198 fsProps, supported := generateFsProps(ctx, partitionType)
199 if !supported {
200 return false
mrziwanga077b942024-10-16 16:00:06 -0700201 }
mrziwanga077b942024-10-16 16:00:06 -0700202
Spandan Das8fe68dc2024-10-29 18:20:11 +0000203 if partitionType == "vendor" || partitionType == "product" {
Spandan Das173256b2024-10-31 19:59:30 +0000204 fsProps.Linkerconfig.Gen_linker_config = proptools.BoolPtr(true)
205 fsProps.Linkerconfig.Linker_config_srcs = f.createLinkerConfigSourceFilegroups(ctx, partitionType)
Spandan Das312cc412024-10-29 18:20:11 +0000206 }
207
Spandan Das5e336422024-11-01 22:31:20 +0000208 if partitionType == "system_dlkm" {
209 kernelModules := ctx.Config().ProductVariables().PartitionVarsForSoongMigrationOnlyDoNotUse.SystemKernelModules
210 f.createPrebuiltKernelModules(ctx, partitionType, kernelModules)
211 }
212
mrziwang4b0ca972024-10-17 14:56:19 -0700213 var module android.Module
214 if partitionType == "system" {
215 module = ctx.CreateModule(filesystem.SystemImageFactory, baseProps, fsProps)
Spandan Das5e336422024-11-01 22:31:20 +0000216 } else if partitionType == "system_dlkm" {
217 // Do not set partition_type. build/soong/android/paths#modulePartition currently does not support dlkm
218 // partitions. Since `android_filesystem` uses a partition based filter, setting the partition here
219 // would result in missing in entries.
220 module = ctx.CreateModule(filesystem.FilesystemFactory, baseProps, fsProps)
mrziwang4b0ca972024-10-17 14:56:19 -0700221 } else {
222 // Explicitly set the partition.
223 fsProps.Partition_type = proptools.StringPtr(partitionType)
224 module = ctx.CreateModule(filesystem.FilesystemFactory, baseProps, fsProps)
225 }
226 module.HideFromMake()
Spandan Das168098c2024-10-28 19:44:34 +0000227 if partitionType == "vendor" {
Spandan Das4cd93b52024-11-05 23:27:03 +0000228 f.createVendorBuildProp(ctx)
Spandan Das168098c2024-10-28 19:44:34 +0000229 }
mrziwang4b0ca972024-10-17 14:56:19 -0700230 return true
231}
232
Spandan Das5e336422024-11-01 22:31:20 +0000233// createPrebuiltKernelModules creates `prebuilt_kernel_modules`. These modules will be added to deps of the
234// autogenerated *_dlkm filsystem modules.
235// The input `kernelModules` is a space separated list of .ko files in the workspace. This will be partitioned per directory
236// and a `prebuilt_kernel_modules` will be created per partition.
237// These autogenerated modules will be subsequently added to the deps of the top level *_dlkm android_filesystem
238func (f *filesystemCreator) createPrebuiltKernelModules(ctx android.LoadHookContext, partitionType string, kernelModules []string) {
239 // Partition the files per directory
240 dirToFiles := map[string][]string{}
241 for _, kernelModule := range kernelModules {
242 dir := filepath.Dir(kernelModule)
243 base := filepath.Base(kernelModule)
244 dirToFiles[dir] = append(dirToFiles[dir], base)
245 }
246 // Create a prebuilt_kernel_modules module per partition
247 fsGenState := ctx.Config().Get(fsGenStateOnceKey).(*FsGenState)
248 for index, dir := range android.SortedKeys(dirToFiles) {
249 name := generatedModuleName(ctx.Config(), fmt.Sprintf("%s-kernel-modules-%s", partitionType, strconv.Itoa(index)))
250 props := &struct {
251 Name *string
252 Srcs []string
253 }{
254 Name: proptools.StringPtr(name),
255 Srcs: dirToFiles[dir],
256 }
257 kernelModule := ctx.CreateModuleInDirectory(
258 kernel.PrebuiltKernelModulesFactory,
259 dir,
260 props,
261 )
262 kernelModule.HideFromMake()
263 // Add to deps
264 (*fsGenState.fsDeps[partitionType])[name] = defaultDepCandidateProps(ctx.Config())
265 }
266}
267
Spandan Das4cd93b52024-11-05 23:27:03 +0000268// Create a build_prop and android_info module. This will be used to create /vendor/build.prop
269func (f *filesystemCreator) createVendorBuildProp(ctx android.LoadHookContext) {
270 // Create a android_info for vendor
271 // The board info files might be in a directory outside the root soong namespace, so create
272 // the module in "."
273 partitionVars := ctx.Config().ProductVariables().PartitionVarsForSoongMigrationOnlyDoNotUse
274 androidInfoProps := &struct {
275 Name *string
276 Board_info_files []string
277 Bootloader_board_name *string
278 }{
279 Name: proptools.StringPtr(generatedModuleName(ctx.Config(), "android-info.prop")),
280 Board_info_files: partitionVars.BoardInfoFiles,
281 }
282 if len(androidInfoProps.Board_info_files) == 0 {
283 androidInfoProps.Bootloader_board_name = proptools.StringPtr(partitionVars.BootLoaderBoardName)
284 }
285 androidInfoProp := ctx.CreateModuleInDirectory(
286 android.AndroidInfoFactory,
287 ".",
288 androidInfoProps,
289 )
290 androidInfoProp.HideFromMake()
291 // Create a build prop for vendor
292 vendorBuildProps := &struct {
293 Name *string
294 Vendor *bool
295 Stem *string
296 Product_config *string
297 Android_info *string
298 }{
299 Name: proptools.StringPtr(generatedModuleName(ctx.Config(), "vendor-build.prop")),
300 Vendor: proptools.BoolPtr(true),
301 Stem: proptools.StringPtr("build.prop"),
302 Product_config: proptools.StringPtr(":product_config"),
303 Android_info: proptools.StringPtr(":" + androidInfoProp.Name()),
304 }
305 vendorBuildProp := ctx.CreateModule(
306 android.BuildPropFactory,
307 vendorBuildProps,
308 )
309 vendorBuildProp.HideFromMake()
310}
311
Spandan Das8fe68dc2024-10-29 18:20:11 +0000312// createLinkerConfigSourceFilegroups creates filegroup modules to generate linker.config.pb for the following partitions
313// 1. vendor: Using PRODUCT_VENDOR_LINKER_CONFIG_FRAGMENTS (space separated file list)
314// 1. product: Using PRODUCT_PRODUCT_LINKER_CONFIG_FRAGMENTS (space separated file list)
315// It creates a filegroup for each file in the fragment list
Spandan Das312cc412024-10-29 18:20:11 +0000316// The filegroup modules are then added to `linker_config_srcs` of the autogenerated vendor `android_filesystem`.
Spandan Das8fe68dc2024-10-29 18:20:11 +0000317func (f *filesystemCreator) createLinkerConfigSourceFilegroups(ctx android.LoadHookContext, partitionType string) []string {
Spandan Das312cc412024-10-29 18:20:11 +0000318 ret := []string{}
319 partitionVars := ctx.Config().ProductVariables().PartitionVarsForSoongMigrationOnlyDoNotUse
Spandan Das8fe68dc2024-10-29 18:20:11 +0000320 var linkerConfigSrcs []string
321 if partitionType == "vendor" {
322 linkerConfigSrcs = android.FirstUniqueStrings(partitionVars.VendorLinkerConfigSrcs)
323 } else if partitionType == "product" {
324 linkerConfigSrcs = android.FirstUniqueStrings(partitionVars.ProductLinkerConfigSrcs)
325 } else {
326 ctx.ModuleErrorf("linker.config.pb is only supported for vendor and product partitions. For system partition, use `android_system_image`")
327 }
328
329 if len(linkerConfigSrcs) > 0 {
Spandan Das312cc412024-10-29 18:20:11 +0000330 // Create a filegroup, and add `:<filegroup_name>` to ret.
331 for index, linkerConfigSrc := range linkerConfigSrcs {
332 dir := filepath.Dir(linkerConfigSrc)
333 base := filepath.Base(linkerConfigSrc)
Spandan Das8fe68dc2024-10-29 18:20:11 +0000334 fgName := generatedModuleName(ctx.Config(), fmt.Sprintf("%s-linker-config-src%s", partitionType, strconv.Itoa(index)))
Spandan Das312cc412024-10-29 18:20:11 +0000335 srcs := []string{base}
336 fgProps := &struct {
337 Name *string
338 Srcs proptools.Configurable[[]string]
339 }{
340 Name: proptools.StringPtr(fgName),
341 Srcs: proptools.NewSimpleConfigurable(srcs),
342 }
343 ctx.CreateModuleInDirectory(
344 android.FileGroupFactory,
345 dir,
346 fgProps,
347 )
348 ret = append(ret, ":"+fgName)
349 }
350 }
351 return ret
352}
353
mrziwang4b0ca972024-10-17 14:56:19 -0700354type filesystemBaseProperty struct {
355 Name *string
356 Compile_multilib *string
357}
358
359func generateBaseProps(namePtr *string) *filesystemBaseProperty {
360 return &filesystemBaseProperty{
361 Name: namePtr,
362 Compile_multilib: proptools.StringPtr("both"),
363 }
364}
365
366func generateFsProps(ctx android.EarlyModuleContext, partitionType string) (*filesystem.FilesystemProperties, bool) {
Cole Faust92ccbe22024-10-03 14:38:37 -0700367 fsProps := &filesystem.FilesystemProperties{}
368
mrziwang4b0ca972024-10-17 14:56:19 -0700369 partitionVars := ctx.Config().ProductVariables().PartitionVarsForSoongMigrationOnlyDoNotUse
370 specificPartitionVars := partitionVars.PartitionQualifiedVariables[partitionType]
371
372 // BOARD_SYSTEMIMAGE_FILE_SYSTEM_TYPE
373 fsType := specificPartitionVars.BoardFileSystemType
374 if fsType == "" {
375 fsType = "ext4" //default
376 }
377 fsProps.Type = proptools.StringPtr(fsType)
378 if filesystem.GetFsTypeFromString(ctx, *fsProps.Type).IsUnknown() {
379 // Currently the android_filesystem module type only supports a handful of FS types like ext4, erofs
380 return nil, false
381 }
382
Cole Faust92ccbe22024-10-03 14:38:37 -0700383 // Don't build this module on checkbuilds, the soong-built partitions are still in-progress
384 // and sometimes don't build.
385 fsProps.Unchecked_module = proptools.BoolPtr(true)
386
Jihoon Kang98047cf2024-10-02 17:13:54 +0000387 // BOARD_AVB_ENABLE
388 fsProps.Use_avb = proptools.BoolPtr(partitionVars.BoardAvbEnable)
389 // BOARD_AVB_KEY_PATH
Cole Faust92ccbe22024-10-03 14:38:37 -0700390 fsProps.Avb_private_key = proptools.StringPtr(specificPartitionVars.BoardAvbKeyPath)
Jihoon Kang98047cf2024-10-02 17:13:54 +0000391 // BOARD_AVB_ALGORITHM
Cole Faust92ccbe22024-10-03 14:38:37 -0700392 fsProps.Avb_algorithm = proptools.StringPtr(specificPartitionVars.BoardAvbAlgorithm)
Jihoon Kang98047cf2024-10-02 17:13:54 +0000393 // BOARD_AVB_SYSTEM_ROLLBACK_INDEX
Cole Faust92ccbe22024-10-03 14:38:37 -0700394 if rollbackIndex, err := strconv.ParseInt(specificPartitionVars.BoardAvbRollbackIndex, 10, 64); err == nil {
Jihoon Kang98047cf2024-10-02 17:13:54 +0000395 fsProps.Rollback_index = proptools.Int64Ptr(rollbackIndex)
396 }
397
Cole Faust92ccbe22024-10-03 14:38:37 -0700398 fsProps.Partition_name = proptools.StringPtr(partitionType)
Jihoon Kang98047cf2024-10-02 17:13:54 +0000399
Cole Faust92ccbe22024-10-03 14:38:37 -0700400 fsProps.Base_dir = proptools.StringPtr(partitionType)
Jihoon Kang98047cf2024-10-02 17:13:54 +0000401
Jihoon Kang0d545b82024-10-11 00:21:57 +0000402 fsProps.Is_auto_generated = proptools.BoolPtr(true)
403
Jihoon Kang6850d8f2024-10-17 20:45:58 +0000404 partitionSpecificFsProps(fsProps, partitionType)
405
Jihoon Kang98047cf2024-10-02 17:13:54 +0000406 // system_image properties that are not set:
407 // - filesystemProperties.Avb_hash_algorithm
408 // - filesystemProperties.File_contexts
409 // - filesystemProperties.Dirs
410 // - filesystemProperties.Symlinks
411 // - filesystemProperties.Fake_timestamp
412 // - filesystemProperties.Uuid
413 // - filesystemProperties.Mount_point
414 // - filesystemProperties.Include_make_built_files
415 // - filesystemProperties.Build_logtags
Jihoon Kang98047cf2024-10-02 17:13:54 +0000416 // - systemImageProperties.Linker_config_src
mrziwang4b0ca972024-10-17 14:56:19 -0700417
418 return fsProps, true
Cole Faust92ccbe22024-10-03 14:38:37 -0700419}
420
421func (f *filesystemCreator) createDiffTest(ctx android.ModuleContext, partitionType string) android.Path {
Jihoon Kang0d545b82024-10-11 00:21:57 +0000422 partitionModuleName := generatedModuleNameForPartition(ctx.Config(), partitionType)
Cole Faust92ccbe22024-10-03 14:38:37 -0700423 systemImage := ctx.GetDirectDepWithTag(partitionModuleName, generatedFilesystemDepTag)
424 filesystemInfo, ok := android.OtherModuleProvider(ctx, systemImage, filesystem.FilesystemProvider)
425 if !ok {
426 ctx.ModuleErrorf("Expected module %s to provide FileysystemInfo", partitionModuleName)
427 }
428 makeFileList := android.PathForArbitraryOutput(ctx, fmt.Sprintf("target/product/%s/obj/PACKAGING/%s_intermediates/file_list.txt", ctx.Config().DeviceName(), partitionType))
429 // For now, don't allowlist anything. The test will fail, but that's fine in the current
430 // early stages where we're just figuring out what we need
Jihoon Kang9e866c82024-10-07 22:39:18 +0000431 emptyAllowlistFile := android.PathForModuleOut(ctx, fmt.Sprintf("allowlist_%s.txt", partitionModuleName))
Cole Faust92ccbe22024-10-03 14:38:37 -0700432 android.WriteFileRule(ctx, emptyAllowlistFile, "")
Jihoon Kang9e866c82024-10-07 22:39:18 +0000433 diffTestResultFile := android.PathForModuleOut(ctx, fmt.Sprintf("diff_test_%s.txt", partitionModuleName))
Cole Faust92ccbe22024-10-03 14:38:37 -0700434
435 builder := android.NewRuleBuilder(pctx, ctx)
436 builder.Command().BuiltTool("file_list_diff").
437 Input(makeFileList).
438 Input(filesystemInfo.FileListFile).
Jihoon Kang9e866c82024-10-07 22:39:18 +0000439 Text(partitionModuleName).
440 FlagWithInput("--allowlists ", emptyAllowlistFile)
Cole Faust92ccbe22024-10-03 14:38:37 -0700441 builder.Command().Text("touch").Output(diffTestResultFile)
442 builder.Build(partitionModuleName+" diff test", partitionModuleName+" diff test")
443 return diffTestResultFile
444}
445
446func createFailingCommand(ctx android.ModuleContext, message string) android.Path {
447 hasher := sha256.New()
448 hasher.Write([]byte(message))
449 filename := fmt.Sprintf("failing_command_%x.txt", hasher.Sum(nil))
450 file := android.PathForModuleOut(ctx, filename)
451 builder := android.NewRuleBuilder(pctx, ctx)
452 builder.Command().Textf("echo %s", proptools.NinjaAndShellEscape(message))
453 builder.Command().Text("exit 1 #").Output(file)
454 builder.Build("failing command "+filename, "failing command "+filename)
455 return file
456}
457
458type systemImageDepTagType struct {
459 blueprint.BaseDependencyTag
460}
461
462var generatedFilesystemDepTag systemImageDepTagType
463
464func (f *filesystemCreator) DepsMutator(ctx android.BottomUpMutatorContext) {
465 for _, partitionType := range f.properties.Generated_partition_types {
Jihoon Kang0d545b82024-10-11 00:21:57 +0000466 ctx.AddDependency(ctx.Module(), generatedFilesystemDepTag, generatedModuleNameForPartition(ctx.Config(), partitionType))
Cole Faust92ccbe22024-10-03 14:38:37 -0700467 }
Jihoon Kang98047cf2024-10-02 17:13:54 +0000468}
469
470func (f *filesystemCreator) GenerateAndroidBuildActions(ctx android.ModuleContext) {
Cole Faust92ccbe22024-10-03 14:38:37 -0700471 if ctx.ModuleDir() != "build/soong/fsgen" {
472 ctx.ModuleErrorf("There can only be one soong_filesystem_creator in build/soong/fsgen")
473 }
474 f.HideFromMake()
Jihoon Kang98047cf2024-10-02 17:13:54 +0000475
Jihoon Kang4e5d8de2024-10-19 01:59:58 +0000476 var content strings.Builder
477 generatedBp := android.PathForModuleOut(ctx, "soong_generated_product_config.bp")
478 for _, partition := range ctx.Config().Get(fsGenStateOnceKey).(*FsGenState).soongGeneratedPartitions {
479 content.WriteString(generateBpContent(ctx, partition))
480 content.WriteString("\n")
481 }
482 android.WriteFileRule(ctx, generatedBp, content.String())
483
mrziwang8f86c882024-10-03 12:34:33 -0700484 ctx.Phony("product_config_to_bp", generatedBp)
485
Cole Faust92ccbe22024-10-03 14:38:37 -0700486 var diffTestFiles []android.Path
487 for _, partitionType := range f.properties.Generated_partition_types {
Jihoon Kang72f812f2024-10-17 18:46:24 +0000488 diffTestFile := f.createDiffTest(ctx, partitionType)
489 diffTestFiles = append(diffTestFiles, diffTestFile)
490 ctx.Phony(fmt.Sprintf("soong_generated_%s_filesystem_test", partitionType), diffTestFile)
Cole Faust92ccbe22024-10-03 14:38:37 -0700491 }
492 for _, partitionType := range f.properties.Unsupported_partition_types {
Jihoon Kang72f812f2024-10-17 18:46:24 +0000493 diffTestFile := createFailingCommand(ctx, fmt.Sprintf("Couldn't build %s partition", partitionType))
494 diffTestFiles = append(diffTestFiles, diffTestFile)
495 ctx.Phony(fmt.Sprintf("soong_generated_%s_filesystem_test", partitionType), diffTestFile)
Cole Faust92ccbe22024-10-03 14:38:37 -0700496 }
497 ctx.Phony("soong_generated_filesystem_tests", diffTestFiles...)
Jihoon Kang98047cf2024-10-02 17:13:54 +0000498}
mrziwang8f86c882024-10-03 12:34:33 -0700499
mrziwang8f86c882024-10-03 12:34:33 -0700500func generateBpContent(ctx android.EarlyModuleContext, partitionType string) string {
mrziwang4b0ca972024-10-17 14:56:19 -0700501 fsProps, fsTypeSupported := generateFsProps(ctx, partitionType)
502 if !fsTypeSupported {
503 return ""
mrziwang8f86c882024-10-03 12:34:33 -0700504 }
505
mrziwang4b0ca972024-10-17 14:56:19 -0700506 baseProps := generateBaseProps(proptools.StringPtr(generatedModuleNameForPartition(ctx.Config(), partitionType)))
mrziwang2a506cf2024-10-17 15:38:37 -0700507 deps := ctx.Config().Get(fsGenStateOnceKey).(*FsGenState).fsDeps[partitionType]
508 depProps := generateDepStruct(*deps)
mrziwang8f86c882024-10-03 12:34:33 -0700509
mrziwang4b0ca972024-10-17 14:56:19 -0700510 result, err := proptools.RepackProperties([]interface{}{baseProps, fsProps, depProps})
mrziwang8f86c882024-10-03 12:34:33 -0700511 if err != nil {
Cole Faustae3e1d32024-11-05 13:22:50 -0800512 ctx.ModuleErrorf("%s", err.Error())
513 return ""
mrziwang8f86c882024-10-03 12:34:33 -0700514 }
515
Jihoon Kang4e5d8de2024-10-19 01:59:58 +0000516 moduleType := "android_filesystem"
517 if partitionType == "system" {
518 moduleType = "android_system_image"
519 }
520
mrziwang8f86c882024-10-03 12:34:33 -0700521 file := &parser.File{
522 Defs: []parser.Definition{
523 &parser.Module{
Jihoon Kang4e5d8de2024-10-19 01:59:58 +0000524 Type: moduleType,
mrziwang8f86c882024-10-03 12:34:33 -0700525 Map: *result,
526 },
527 },
528 }
529 bytes, err := parser.Print(file)
530 if err != nil {
531 ctx.ModuleErrorf(err.Error())
532 }
533 return strings.TrimSpace(string(bytes))
534}