blob: 8acc74c270e2e9f2a47d6b22936aefd846204550 [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"
mrziwang8f86c882024-10-03 12:34:33 -070020 "slices"
Jihoon Kang98047cf2024-10-02 17:13:54 +000021 "strconv"
mrziwang8f86c882024-10-03 12:34:33 -070022 "strings"
23 "sync"
24
25 "android/soong/android"
26 "android/soong/filesystem"
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
44func RegisterCollectFileSystemDepsMutators(ctx android.RegisterMutatorsContext) {
45 ctx.BottomUp("fs_collect_deps", collectDepsMutator).MutatesGlobalState()
Jihoon Kangdc6492f2024-10-11 00:21:57 +000046 ctx.BottomUp("fs_set_deps", setDepsMutator)
mrziwang8f86c882024-10-03 12:34:33 -070047}
48
mrziwangc7e58c92024-10-11 09:49:48 -070049var fsDepsMutex = sync.Mutex{}
mrziwang8f86c882024-10-03 12:34:33 -070050var collectFsDepsOnceKey = android.NewOnceKey("CollectFsDeps")
51var depCandidatesOnceKey = android.NewOnceKey("DepCandidates")
52
Jihoon Kangdc6492f2024-10-11 00:21:57 +000053// List of partitions that the filesystem_creator module currently generates the partition
54var soongGeneratedPartitions = []string{"system"}
55
56// Map of partition module name to its partition that may be generated by Soong.
57// Note that it is not guaranteed that all modules returned by this function are successfully
58// created.
59func getAllSoongGeneratedPartitionNames(config android.Config) map[string]string {
60 ret := map[string]string{}
61 for _, partition := range soongGeneratedPartitions {
62 ret[generatedModuleNameForPartition(config, partition)] = partition
63 }
64 return ret
65}
66
mrziwang8f86c882024-10-03 12:34:33 -070067func collectDepsMutator(mctx android.BottomUpMutatorContext) {
68 // These additional deps are added according to the cuttlefish system image bp.
69 fsDeps := mctx.Config().Once(collectFsDepsOnceKey, func() interface{} {
Jihoon Kangdc6492f2024-10-11 00:21:57 +000070 deps := map[string][]string{
71 "system": {
72 "com.android.apex.cts.shim.v1_prebuilt",
73 "dex_bootjars",
74 "framework_compatibility_matrix.device.xml",
75 "idc_data",
76 "init.environ.rc-soong",
77 "keychars_data",
78 "keylayout_data",
79 "libclang_rt.asan",
80 "libcompiler_rt",
81 "libdmabufheap",
82 "libgsi",
83 "llndk.libraries.txt",
84 "logpersist.start",
85 "preloaded-classes",
86 "public.libraries.android.txt",
87 "update_engine_sideload",
88 },
mrziwang8f86c882024-10-03 12:34:33 -070089 }
90 return &deps
Jihoon Kangdc6492f2024-10-11 00:21:57 +000091 }).(*map[string][]string)
mrziwang8f86c882024-10-03 12:34:33 -070092
93 depCandidates := mctx.Config().Once(depCandidatesOnceKey, func() interface{} {
94 partitionVars := mctx.Config().ProductVariables().PartitionVarsForSoongMigrationOnlyDoNotUse
95 candidates := slices.Concat(partitionVars.ProductPackages, partitionVars.ProductPackagesDebug)
96 return &candidates
97 }).(*[]string)
98
99 m := mctx.Module()
mrziwang8f86c882024-10-03 12:34:33 -0700100 if slices.Contains(*depCandidates, m.Name()) {
Jihoon Kangdc6492f2024-10-11 00:21:57 +0000101 installPartition := getInstallPartition(m, mctx.DeviceConfig())
102 fsDepsMutex.Lock()
103 if _, ok := (*fsDeps)[installPartition]; !ok {
104 (*fsDeps)[installPartition] = make([]string, 0)
105 }
106 if m.Enabled(mctx) {
107 (*fsDeps)[installPartition] = append((*fsDeps)[installPartition], m.Name())
108 }
109 fsDepsMutex.Unlock()
110 }
111}
112
113type depsStruct struct {
114 Deps []string
115}
116
117func setDepsMutator(mctx android.BottomUpMutatorContext) {
118 fsDeps := mctx.Config().Get(collectFsDepsOnceKey).(*map[string][]string)
119 soongGeneratedPartitionMap := getAllSoongGeneratedPartitionNames(mctx.Config())
120 m := mctx.Module()
121 if partition, ok := soongGeneratedPartitionMap[m.Name()]; ok {
122 deps := (*fsDeps)[partition]
123 if err := proptools.AppendMatchingProperties(m.GetProperties(), &depsStruct{Deps: deps}, nil); err != nil {
124 mctx.ModuleErrorf(err.Error())
mrziwang8f86c882024-10-03 12:34:33 -0700125 }
126 }
Jihoon Kang98047cf2024-10-02 17:13:54 +0000127}
128
Cole Faust92ccbe22024-10-03 14:38:37 -0700129type filesystemCreatorProps struct {
130 Generated_partition_types []string `blueprint:"mutated"`
131 Unsupported_partition_types []string `blueprint:"mutated"`
132}
133
Jihoon Kang98047cf2024-10-02 17:13:54 +0000134type filesystemCreator struct {
135 android.ModuleBase
Cole Faust92ccbe22024-10-03 14:38:37 -0700136
137 properties filesystemCreatorProps
Jihoon Kang98047cf2024-10-02 17:13:54 +0000138}
139
140func filesystemCreatorFactory() android.Module {
141 module := &filesystemCreator{}
142
Cole Faust69788792024-10-10 11:00:36 -0700143 android.InitAndroidArchModule(module, android.DeviceSupported, android.MultilibCommon)
Cole Faust92ccbe22024-10-03 14:38:37 -0700144 module.AddProperties(&module.properties)
Jihoon Kang98047cf2024-10-02 17:13:54 +0000145 android.AddLoadHook(module, func(ctx android.LoadHookContext) {
146 module.createInternalModules(ctx)
147 })
148
149 return module
150}
151
152func (f *filesystemCreator) createInternalModules(ctx android.LoadHookContext) {
Jihoon Kangdc6492f2024-10-11 00:21:57 +0000153 for _, partitionType := range soongGeneratedPartitions {
Cole Faust92ccbe22024-10-03 14:38:37 -0700154 if f.createPartition(ctx, partitionType) {
155 f.properties.Generated_partition_types = append(f.properties.Generated_partition_types, partitionType)
156 } else {
157 f.properties.Unsupported_partition_types = append(f.properties.Unsupported_partition_types, partitionType)
158 }
159 }
Jihoon Kangf1c79ca2024-10-09 20:18:38 +0000160 f.createDeviceModule(ctx)
Jihoon Kang98047cf2024-10-02 17:13:54 +0000161}
162
Jihoon Kangdc6492f2024-10-11 00:21:57 +0000163func generatedModuleName(cfg android.Config, suffix string) string {
Cole Faust92ccbe22024-10-03 14:38:37 -0700164 prefix := "soong"
165 if cfg.HasDeviceProduct() {
166 prefix = cfg.DeviceProduct()
167 }
Jihoon Kangf1c79ca2024-10-09 20:18:38 +0000168 return fmt.Sprintf("%s_generated_%s", prefix, suffix)
169}
170
Jihoon Kangdc6492f2024-10-11 00:21:57 +0000171func generatedModuleNameForPartition(cfg android.Config, partitionType string) string {
172 return generatedModuleName(cfg, fmt.Sprintf("%s_image", partitionType))
Jihoon Kangf1c79ca2024-10-09 20:18:38 +0000173}
174
175func (f *filesystemCreator) createDeviceModule(ctx android.LoadHookContext) {
176 baseProps := &struct {
177 Name *string
178 }{
Jihoon Kangdc6492f2024-10-11 00:21:57 +0000179 Name: proptools.StringPtr(generatedModuleName(ctx.Config(), "device")),
Jihoon Kangf1c79ca2024-10-09 20:18:38 +0000180 }
181
Priyanka Advani (xWF)41e4c992024-10-11 16:53:20 +0000182 // Currently, only the system partition module is created.
Jihoon Kangf1c79ca2024-10-09 20:18:38 +0000183 partitionProps := &filesystem.PartitionNameProperties{}
184 if android.InList("system", f.properties.Generated_partition_types) {
Jihoon Kangdc6492f2024-10-11 00:21:57 +0000185 partitionProps.System_partition_name = proptools.StringPtr(generatedModuleNameForPartition(ctx.Config(), "system"))
Jihoon Kangf1c79ca2024-10-09 20:18:38 +0000186 }
187
188 ctx.CreateModule(filesystem.AndroidDeviceFactory, baseProps, partitionProps)
Cole Faust92ccbe22024-10-03 14:38:37 -0700189}
190
191// Creates a soong module to build the given partition. Returns false if we can't support building
192// it.
193func (f *filesystemCreator) createPartition(ctx android.LoadHookContext, partitionType string) bool {
Jihoon Kang98047cf2024-10-02 17:13:54 +0000194 baseProps := &struct {
195 Name *string
196 }{
Jihoon Kangdc6492f2024-10-11 00:21:57 +0000197 Name: proptools.StringPtr(generatedModuleNameForPartition(ctx.Config(), partitionType)),
Jihoon Kang98047cf2024-10-02 17:13:54 +0000198 }
199
Cole Faust92ccbe22024-10-03 14:38:37 -0700200 fsProps := &filesystem.FilesystemProperties{}
201
202 // Don't build this module on checkbuilds, the soong-built partitions are still in-progress
203 // and sometimes don't build.
204 fsProps.Unchecked_module = proptools.BoolPtr(true)
205
Jihoon Kang98047cf2024-10-02 17:13:54 +0000206 partitionVars := ctx.Config().ProductVariables().PartitionVarsForSoongMigrationOnlyDoNotUse
Cole Faust92ccbe22024-10-03 14:38:37 -0700207 specificPartitionVars := partitionVars.PartitionQualifiedVariables[partitionType]
Jihoon Kang98047cf2024-10-02 17:13:54 +0000208
209 // BOARD_AVB_ENABLE
210 fsProps.Use_avb = proptools.BoolPtr(partitionVars.BoardAvbEnable)
211 // BOARD_AVB_KEY_PATH
Cole Faust92ccbe22024-10-03 14:38:37 -0700212 fsProps.Avb_private_key = proptools.StringPtr(specificPartitionVars.BoardAvbKeyPath)
Jihoon Kang98047cf2024-10-02 17:13:54 +0000213 // BOARD_AVB_ALGORITHM
Cole Faust92ccbe22024-10-03 14:38:37 -0700214 fsProps.Avb_algorithm = proptools.StringPtr(specificPartitionVars.BoardAvbAlgorithm)
Jihoon Kang98047cf2024-10-02 17:13:54 +0000215 // BOARD_AVB_SYSTEM_ROLLBACK_INDEX
Cole Faust92ccbe22024-10-03 14:38:37 -0700216 if rollbackIndex, err := strconv.ParseInt(specificPartitionVars.BoardAvbRollbackIndex, 10, 64); err == nil {
Jihoon Kang98047cf2024-10-02 17:13:54 +0000217 fsProps.Rollback_index = proptools.Int64Ptr(rollbackIndex)
218 }
219
Cole Faust92ccbe22024-10-03 14:38:37 -0700220 fsProps.Partition_name = proptools.StringPtr(partitionType)
Jihoon Kang98047cf2024-10-02 17:13:54 +0000221 // BOARD_SYSTEMIMAGE_FILE_SYSTEM_TYPE
Priyanka Advani (xWF)41e4c992024-10-11 16:53:20 +0000222 fsProps.Type = proptools.StringPtr(specificPartitionVars.BoardFileSystemType)
223 if *fsProps.Type != "ext4" {
224 // TODO(b/372522486): Support other FS types.
225 // Currently the android_filesystem module type only supports ext4:
226 // https://cs.android.com/android/platform/superproject/main/+/main:build/soong/filesystem/filesystem.go;l=416;drc=98047cfd07944b297a12d173453bc984806760d2
Cole Faust92ccbe22024-10-03 14:38:37 -0700227 return false
228 }
Jihoon Kang98047cf2024-10-02 17:13:54 +0000229
Cole Faust92ccbe22024-10-03 14:38:37 -0700230 fsProps.Base_dir = proptools.StringPtr(partitionType)
Jihoon Kang98047cf2024-10-02 17:13:54 +0000231
232 fsProps.Gen_aconfig_flags_pb = proptools.BoolPtr(true)
233
Jihoon Kangdc6492f2024-10-11 00:21:57 +0000234 fsProps.Is_auto_generated = proptools.BoolPtr(true)
235
Jihoon Kang98047cf2024-10-02 17:13:54 +0000236 // Identical to that of the generic_system_image
237 fsProps.Fsverity.Inputs = []string{
238 "etc/boot-image.prof",
239 "etc/dirty-image-objects",
240 "etc/preloaded-classes",
241 "etc/classpaths/*.pb",
242 "framework/*",
243 "framework/*/*", // framework/{arch}
244 "framework/oat/*/*", // framework/oat/{arch}
245 }
246
247 // system_image properties that are not set:
248 // - filesystemProperties.Avb_hash_algorithm
249 // - filesystemProperties.File_contexts
250 // - filesystemProperties.Dirs
251 // - filesystemProperties.Symlinks
252 // - filesystemProperties.Fake_timestamp
253 // - filesystemProperties.Uuid
254 // - filesystemProperties.Mount_point
255 // - filesystemProperties.Include_make_built_files
256 // - filesystemProperties.Build_logtags
257 // - filesystemProperties.Fsverity.Libs
258 // - systemImageProperties.Linker_config_src
Cole Faust92ccbe22024-10-03 14:38:37 -0700259 var module android.Module
260 if partitionType == "system" {
261 module = ctx.CreateModule(filesystem.SystemImageFactory, baseProps, fsProps)
262 } else {
263 module = ctx.CreateModule(filesystem.FilesystemFactory, baseProps, fsProps)
264 }
265 module.HideFromMake()
266 return true
267}
268
269func (f *filesystemCreator) createDiffTest(ctx android.ModuleContext, partitionType string) android.Path {
Jihoon Kangdc6492f2024-10-11 00:21:57 +0000270 partitionModuleName := generatedModuleNameForPartition(ctx.Config(), partitionType)
Cole Faust92ccbe22024-10-03 14:38:37 -0700271 systemImage := ctx.GetDirectDepWithTag(partitionModuleName, generatedFilesystemDepTag)
272 filesystemInfo, ok := android.OtherModuleProvider(ctx, systemImage, filesystem.FilesystemProvider)
273 if !ok {
274 ctx.ModuleErrorf("Expected module %s to provide FileysystemInfo", partitionModuleName)
275 }
276 makeFileList := android.PathForArbitraryOutput(ctx, fmt.Sprintf("target/product/%s/obj/PACKAGING/%s_intermediates/file_list.txt", ctx.Config().DeviceName(), partitionType))
277 // For now, don't allowlist anything. The test will fail, but that's fine in the current
278 // early stages where we're just figuring out what we need
Jihoon Kang9e866c82024-10-07 22:39:18 +0000279 emptyAllowlistFile := android.PathForModuleOut(ctx, fmt.Sprintf("allowlist_%s.txt", partitionModuleName))
Cole Faust92ccbe22024-10-03 14:38:37 -0700280 android.WriteFileRule(ctx, emptyAllowlistFile, "")
Jihoon Kang9e866c82024-10-07 22:39:18 +0000281 diffTestResultFile := android.PathForModuleOut(ctx, fmt.Sprintf("diff_test_%s.txt", partitionModuleName))
Cole Faust92ccbe22024-10-03 14:38:37 -0700282
283 builder := android.NewRuleBuilder(pctx, ctx)
284 builder.Command().BuiltTool("file_list_diff").
285 Input(makeFileList).
286 Input(filesystemInfo.FileListFile).
Jihoon Kang9e866c82024-10-07 22:39:18 +0000287 Text(partitionModuleName).
288 FlagWithInput("--allowlists ", emptyAllowlistFile)
Cole Faust92ccbe22024-10-03 14:38:37 -0700289 builder.Command().Text("touch").Output(diffTestResultFile)
290 builder.Build(partitionModuleName+" diff test", partitionModuleName+" diff test")
291 return diffTestResultFile
292}
293
294func createFailingCommand(ctx android.ModuleContext, message string) android.Path {
295 hasher := sha256.New()
296 hasher.Write([]byte(message))
297 filename := fmt.Sprintf("failing_command_%x.txt", hasher.Sum(nil))
298 file := android.PathForModuleOut(ctx, filename)
299 builder := android.NewRuleBuilder(pctx, ctx)
300 builder.Command().Textf("echo %s", proptools.NinjaAndShellEscape(message))
301 builder.Command().Text("exit 1 #").Output(file)
302 builder.Build("failing command "+filename, "failing command "+filename)
303 return file
304}
305
306type systemImageDepTagType struct {
307 blueprint.BaseDependencyTag
308}
309
310var generatedFilesystemDepTag systemImageDepTagType
311
312func (f *filesystemCreator) DepsMutator(ctx android.BottomUpMutatorContext) {
313 for _, partitionType := range f.properties.Generated_partition_types {
Jihoon Kangdc6492f2024-10-11 00:21:57 +0000314 ctx.AddDependency(ctx.Module(), generatedFilesystemDepTag, generatedModuleNameForPartition(ctx.Config(), partitionType))
Cole Faust92ccbe22024-10-03 14:38:37 -0700315 }
Jihoon Kang98047cf2024-10-02 17:13:54 +0000316}
317
318func (f *filesystemCreator) GenerateAndroidBuildActions(ctx android.ModuleContext) {
Cole Faust92ccbe22024-10-03 14:38:37 -0700319 if ctx.ModuleDir() != "build/soong/fsgen" {
320 ctx.ModuleErrorf("There can only be one soong_filesystem_creator in build/soong/fsgen")
321 }
322 f.HideFromMake()
Jihoon Kang98047cf2024-10-02 17:13:54 +0000323
mrziwang8f86c882024-10-03 12:34:33 -0700324 content := generateBpContent(ctx, "system")
325 generatedBp := android.PathForOutput(ctx, "soong_generated_product_config.bp")
326 android.WriteFileRule(ctx, generatedBp, content)
327 ctx.Phony("product_config_to_bp", generatedBp)
328
Cole Faust92ccbe22024-10-03 14:38:37 -0700329 var diffTestFiles []android.Path
330 for _, partitionType := range f.properties.Generated_partition_types {
331 diffTestFiles = append(diffTestFiles, f.createDiffTest(ctx, partitionType))
332 }
333 for _, partitionType := range f.properties.Unsupported_partition_types {
334 diffTestFiles = append(diffTestFiles, createFailingCommand(ctx, fmt.Sprintf("Couldn't build %s partition", partitionType)))
335 }
336 ctx.Phony("soong_generated_filesystem_tests", diffTestFiles...)
Jihoon Kang98047cf2024-10-02 17:13:54 +0000337}
mrziwang8f86c882024-10-03 12:34:33 -0700338
Jihoon Kangdc6492f2024-10-11 00:21:57 +0000339func getInstallPartition(m android.Module, config android.DeviceConfig) string {
340 return m.PartitionTag(config)
mrziwang8f86c882024-10-03 12:34:33 -0700341}
342
343// TODO: assemble baseProps and fsProps here
344func generateBpContent(ctx android.EarlyModuleContext, partitionType string) string {
345 // Currently only system partition is supported
346 if partitionType != "system" {
347 return ""
348 }
349
Jihoon Kangdc6492f2024-10-11 00:21:57 +0000350 deps := ctx.Config().Get(collectFsDepsOnceKey).(*map[string][]string)
mrziwang8f86c882024-10-03 12:34:33 -0700351 depProps := &android.PackagingProperties{
Jihoon Kangdc6492f2024-10-11 00:21:57 +0000352 Deps: android.NewSimpleConfigurable(android.SortedUniqueStrings((*deps)[partitionType])),
mrziwang8f86c882024-10-03 12:34:33 -0700353 }
354
355 result, err := proptools.RepackProperties([]interface{}{depProps})
356 if err != nil {
357 ctx.ModuleErrorf(err.Error())
358 }
359
360 file := &parser.File{
361 Defs: []parser.Definition{
362 &parser.Module{
363 Type: "module",
364 Map: *result,
365 },
366 },
367 }
368 bytes, err := parser.Print(file)
369 if err != nil {
370 ctx.ModuleErrorf(err.Error())
371 }
372 return strings.TrimSpace(string(bytes))
373}