| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 1 | // 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 |  | 
|  | 15 | package fsgen | 
|  | 16 |  | 
|  | 17 | import ( | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 18 | "crypto/sha256" | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 19 | "fmt" | 
| mrziwang | 8f86c88 | 2024-10-03 12:34:33 -0700 | [diff] [blame] | 20 | "slices" | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 21 | "strconv" | 
| mrziwang | 8f86c88 | 2024-10-03 12:34:33 -0700 | [diff] [blame] | 22 | "strings" | 
|  | 23 | "sync" | 
|  | 24 |  | 
|  | 25 | "android/soong/android" | 
|  | 26 | "android/soong/filesystem" | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 27 |  | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 28 | "github.com/google/blueprint" | 
| mrziwang | 8f86c88 | 2024-10-03 12:34:33 -0700 | [diff] [blame] | 29 | "github.com/google/blueprint/parser" | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 30 | "github.com/google/blueprint/proptools" | 
|  | 31 | ) | 
|  | 32 |  | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 33 | var pctx = android.NewPackageContext("android/soong/fsgen") | 
|  | 34 |  | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 35 | func init() { | 
|  | 36 | registerBuildComponents(android.InitRegistrationContext) | 
|  | 37 | } | 
|  | 38 |  | 
|  | 39 | func registerBuildComponents(ctx android.RegistrationContext) { | 
|  | 40 | ctx.RegisterModuleType("soong_filesystem_creator", filesystemCreatorFactory) | 
| mrziwang | 8f86c88 | 2024-10-03 12:34:33 -0700 | [diff] [blame] | 41 | ctx.PreDepsMutators(RegisterCollectFileSystemDepsMutators) | 
|  | 42 | } | 
|  | 43 |  | 
|  | 44 | func RegisterCollectFileSystemDepsMutators(ctx android.RegisterMutatorsContext) { | 
|  | 45 | ctx.BottomUp("fs_collect_deps", collectDepsMutator).MutatesGlobalState() | 
| Jihoon Kang | 0d545b8 | 2024-10-11 00:21:57 +0000 | [diff] [blame] | 46 | ctx.BottomUp("fs_set_deps", setDepsMutator) | 
| mrziwang | 8f86c88 | 2024-10-03 12:34:33 -0700 | [diff] [blame] | 47 | } | 
|  | 48 |  | 
| Jihoon Kang | 0d545b8 | 2024-10-11 00:21:57 +0000 | [diff] [blame] | 49 | var fsGenStateOnceKey = android.NewOnceKey("FsGenState") | 
|  | 50 |  | 
|  | 51 | // Map of partition module name to its partition that may be generated by Soong. | 
|  | 52 | // Note that it is not guaranteed that all modules returned by this function are successfully | 
|  | 53 | // created. | 
|  | 54 | func getAllSoongGeneratedPartitionNames(config android.Config, partitions []string) map[string]string { | 
|  | 55 | ret := map[string]string{} | 
|  | 56 | for _, partition := range partitions { | 
|  | 57 | ret[generatedModuleNameForPartition(config, partition)] = partition | 
|  | 58 | } | 
|  | 59 | return ret | 
|  | 60 | } | 
|  | 61 |  | 
|  | 62 | type depCandidateProps struct { | 
|  | 63 | Namespace string | 
|  | 64 | Multilib  string | 
|  | 65 | Arch      []android.ArchType | 
|  | 66 | } | 
|  | 67 |  | 
|  | 68 | // Map of module name to depCandidateProps | 
|  | 69 | type multilibDeps *map[string]*depCandidateProps | 
|  | 70 |  | 
|  | 71 | // Information necessary to generate the filesystem modules, including details about their | 
|  | 72 | // dependencies | 
|  | 73 | type FsGenState struct { | 
|  | 74 | // List of modules in `PRODUCT_PACKAGES` and `PRODUCT_PACKAGES_DEBUG` | 
|  | 75 | depCandidates []string | 
|  | 76 | // Map of names of partition to the information of modules to be added as deps | 
|  | 77 | fsDeps map[string]multilibDeps | 
|  | 78 | // List of name of partitions to be generated by the filesystem_creator module | 
|  | 79 | soongGeneratedPartitions []string | 
|  | 80 | // Mutex to protect the fsDeps | 
|  | 81 | fsDepsMutex sync.Mutex | 
|  | 82 | } | 
|  | 83 |  | 
|  | 84 | func newMultilibDeps() multilibDeps { | 
|  | 85 | return &map[string]*depCandidateProps{} | 
|  | 86 | } | 
|  | 87 |  | 
|  | 88 | func defaultDepCandidateProps(config android.Config) *depCandidateProps { | 
|  | 89 | return &depCandidateProps{ | 
|  | 90 | Namespace: ".", | 
|  | 91 | Arch:      []android.ArchType{config.BuildArch}, | 
|  | 92 | } | 
|  | 93 | } | 
|  | 94 |  | 
|  | 95 | func createFsGenState(ctx android.LoadHookContext) *FsGenState { | 
|  | 96 | return ctx.Config().Once(fsGenStateOnceKey, func() interface{} { | 
|  | 97 | partitionVars := ctx.Config().ProductVariables().PartitionVarsForSoongMigrationOnlyDoNotUse | 
|  | 98 | candidates := android.FirstUniqueStrings(android.Concat(partitionVars.ProductPackages, partitionVars.ProductPackagesDebug)) | 
|  | 99 |  | 
|  | 100 | generatedPartitions := []string{"system"} | 
|  | 101 | if ctx.DeviceConfig().SystemExtPath() == "system_ext" { | 
|  | 102 | generatedPartitions = append(generatedPartitions, "system_ext") | 
|  | 103 | } | 
|  | 104 |  | 
|  | 105 | return &FsGenState{ | 
|  | 106 | depCandidates: candidates, | 
|  | 107 | fsDeps: map[string]multilibDeps{ | 
|  | 108 | // These additional deps are added according to the cuttlefish system image bp. | 
|  | 109 | "system": &map[string]*depCandidateProps{ | 
|  | 110 | "com.android.apex.cts.shim.v1_prebuilt":     defaultDepCandidateProps(ctx.Config()), | 
|  | 111 | "dex_bootjars":                              defaultDepCandidateProps(ctx.Config()), | 
|  | 112 | "framework_compatibility_matrix.device.xml": defaultDepCandidateProps(ctx.Config()), | 
|  | 113 | "idc_data":                     defaultDepCandidateProps(ctx.Config()), | 
|  | 114 | "init.environ.rc-soong":        defaultDepCandidateProps(ctx.Config()), | 
|  | 115 | "keychars_data":                defaultDepCandidateProps(ctx.Config()), | 
|  | 116 | "keylayout_data":               defaultDepCandidateProps(ctx.Config()), | 
|  | 117 | "libclang_rt.asan":             defaultDepCandidateProps(ctx.Config()), | 
|  | 118 | "libcompiler_rt":               defaultDepCandidateProps(ctx.Config()), | 
|  | 119 | "libdmabufheap":                defaultDepCandidateProps(ctx.Config()), | 
|  | 120 | "libgsi":                       defaultDepCandidateProps(ctx.Config()), | 
|  | 121 | "llndk.libraries.txt":          defaultDepCandidateProps(ctx.Config()), | 
|  | 122 | "logpersist.start":             defaultDepCandidateProps(ctx.Config()), | 
|  | 123 | "preloaded-classes":            defaultDepCandidateProps(ctx.Config()), | 
|  | 124 | "public.libraries.android.txt": defaultDepCandidateProps(ctx.Config()), | 
|  | 125 | "update_engine_sideload":       defaultDepCandidateProps(ctx.Config()), | 
|  | 126 | }, | 
| Spandan Das | d9875bc | 2024-10-17 21:36:17 +0000 | [diff] [blame] | 127 | "vendor":  newMultilibDeps(), | 
|  | 128 | "odm":     newMultilibDeps(), | 
|  | 129 | "product": newMultilibDeps(), | 
|  | 130 | "system_ext": &map[string]*depCandidateProps{ | 
|  | 131 | // VNDK apexes are automatically included. | 
|  | 132 | // This hardcoded list will need to be updated if `PRODUCT_EXTRA_VNDK_VERSIONS` is updated. | 
|  | 133 | // https://cs.android.com/android/_/android/platform/build/+/adba533072b00c53ac0f198c550a3cbd7a00e4cd:core/main.mk;l=984;bpv=1;bpt=0;drc=174db7b179592cf07cbfd2adb0119486fda911e7 | 
|  | 134 | "com.android.vndk.v30": defaultDepCandidateProps(ctx.Config()), | 
|  | 135 | "com.android.vndk.v31": defaultDepCandidateProps(ctx.Config()), | 
|  | 136 | "com.android.vndk.v32": defaultDepCandidateProps(ctx.Config()), | 
|  | 137 | "com.android.vndk.v33": defaultDepCandidateProps(ctx.Config()), | 
|  | 138 | "com.android.vndk.v34": defaultDepCandidateProps(ctx.Config()), | 
|  | 139 | }, | 
| Jihoon Kang | 0d545b8 | 2024-10-11 00:21:57 +0000 | [diff] [blame] | 140 | }, | 
|  | 141 | soongGeneratedPartitions: generatedPartitions, | 
|  | 142 | fsDepsMutex:              sync.Mutex{}, | 
|  | 143 | } | 
|  | 144 | }).(*FsGenState) | 
|  | 145 | } | 
|  | 146 |  | 
|  | 147 | func checkDepModuleInMultipleNamespaces(mctx android.BottomUpMutatorContext, foundDeps map[string]*depCandidateProps, module string, partitionName string) { | 
|  | 148 | otherNamespace := mctx.Namespace().Path | 
|  | 149 | if val, found := foundDeps[module]; found && otherNamespace != "." && !android.InList(val.Namespace, []string{".", otherNamespace}) { | 
|  | 150 | mctx.ModuleErrorf("found in multiple namespaces(%s and %s) when including in %s partition", val.Namespace, otherNamespace, partitionName) | 
|  | 151 | } | 
|  | 152 | } | 
|  | 153 |  | 
|  | 154 | func appendDepIfAppropriate(mctx android.BottomUpMutatorContext, deps *map[string]*depCandidateProps, installPartition string) { | 
|  | 155 | checkDepModuleInMultipleNamespaces(mctx, *deps, mctx.Module().Name(), installPartition) | 
|  | 156 | if _, ok := (*deps)[mctx.Module().Name()]; ok { | 
|  | 157 | // Prefer the namespace-specific module over the platform module | 
|  | 158 | if mctx.Namespace().Path != "." { | 
|  | 159 | (*deps)[mctx.Module().Name()].Namespace = mctx.Namespace().Path | 
|  | 160 | } | 
|  | 161 | (*deps)[mctx.Module().Name()].Arch = append((*deps)[mctx.Module().Name()].Arch, mctx.Module().Target().Arch.ArchType) | 
|  | 162 | } else { | 
|  | 163 | multilib, _ := mctx.Module().DecodeMultilib(mctx) | 
|  | 164 | (*deps)[mctx.Module().Name()] = &depCandidateProps{ | 
|  | 165 | Namespace: mctx.Namespace().Path, | 
|  | 166 | Multilib:  multilib, | 
|  | 167 | Arch:      []android.ArchType{mctx.Module().Target().Arch.ArchType}, | 
|  | 168 | } | 
|  | 169 | } | 
|  | 170 | } | 
| mrziwang | 8f86c88 | 2024-10-03 12:34:33 -0700 | [diff] [blame] | 171 |  | 
|  | 172 | func collectDepsMutator(mctx android.BottomUpMutatorContext) { | 
| Jihoon Kang | 0d545b8 | 2024-10-11 00:21:57 +0000 | [diff] [blame] | 173 | fsGenState := mctx.Config().Get(fsGenStateOnceKey).(*FsGenState) | 
| mrziwang | 8f86c88 | 2024-10-03 12:34:33 -0700 | [diff] [blame] | 174 |  | 
|  | 175 | m := mctx.Module() | 
| Jihoon Kang | 0d545b8 | 2024-10-11 00:21:57 +0000 | [diff] [blame] | 176 | if slices.Contains(fsGenState.depCandidates, m.Name()) { | 
|  | 177 | installPartition := m.PartitionTag(mctx.DeviceConfig()) | 
|  | 178 | fsGenState.fsDepsMutex.Lock() | 
|  | 179 | // Only add the module as dependency when: | 
|  | 180 | // - its enabled | 
|  | 181 | // - its namespace is included in PRODUCT_SOONG_NAMESPACES | 
|  | 182 | if m.Enabled(mctx) && m.ExportedToMake() { | 
|  | 183 | appendDepIfAppropriate(mctx, fsGenState.fsDeps[installPartition], installPartition) | 
|  | 184 | } | 
|  | 185 | fsGenState.fsDepsMutex.Unlock() | 
|  | 186 | } | 
|  | 187 | } | 
|  | 188 |  | 
|  | 189 | type depsStruct struct { | 
|  | 190 | Deps []string | 
|  | 191 | } | 
|  | 192 |  | 
|  | 193 | type multilibDepsStruct struct { | 
|  | 194 | Common   depsStruct | 
|  | 195 | Lib32    depsStruct | 
|  | 196 | Lib64    depsStruct | 
|  | 197 | Both     depsStruct | 
|  | 198 | Prefer32 depsStruct | 
|  | 199 | } | 
|  | 200 |  | 
|  | 201 | type packagingPropsStruct struct { | 
|  | 202 | Deps     []string | 
|  | 203 | Multilib multilibDepsStruct | 
|  | 204 | } | 
|  | 205 |  | 
|  | 206 | func fullyQualifiedModuleName(moduleName, namespace string) string { | 
|  | 207 | if namespace == "." { | 
|  | 208 | return moduleName | 
|  | 209 | } | 
|  | 210 | return fmt.Sprintf("//%s:%s", namespace, moduleName) | 
|  | 211 | } | 
|  | 212 |  | 
|  | 213 | // Returns the sorted unique list of module names with namespace, if the module specifies one. | 
|  | 214 | func fullyQualifiedModuleNames(modules multilibDeps) (ret []string) { | 
|  | 215 | for moduleName, moduleProp := range *modules { | 
|  | 216 | ret = append(ret, fullyQualifiedModuleName(moduleName, moduleProp.Namespace)) | 
|  | 217 | } | 
|  | 218 | return android.SortedUniqueStrings(ret) | 
|  | 219 | } | 
|  | 220 |  | 
|  | 221 | func getBitness(archTypes []android.ArchType) (ret []string) { | 
|  | 222 | for _, archType := range archTypes { | 
|  | 223 | if archType.Multilib == "" { | 
|  | 224 | ret = append(ret, android.COMMON_VARIANT) | 
|  | 225 | } else { | 
|  | 226 | ret = append(ret, archType.Bitness()) | 
|  | 227 | } | 
|  | 228 | } | 
|  | 229 | return ret | 
|  | 230 | } | 
|  | 231 |  | 
|  | 232 | func setDepsMutator(mctx android.BottomUpMutatorContext) { | 
|  | 233 | fsGenState := mctx.Config().Get(fsGenStateOnceKey).(*FsGenState) | 
|  | 234 | fsDeps := fsGenState.fsDeps | 
|  | 235 | soongGeneratedPartitionMap := getAllSoongGeneratedPartitionNames(mctx.Config(), fsGenState.soongGeneratedPartitions) | 
|  | 236 | m := mctx.Module() | 
|  | 237 | if partition, ok := soongGeneratedPartitionMap[m.Name()]; ok { | 
| mrziwang | 2a506cf | 2024-10-17 15:38:37 -0700 | [diff] [blame] | 238 | depsStruct := generateDepStruct(*fsDeps[partition]) | 
|  | 239 | if err := proptools.AppendMatchingProperties(m.GetProperties(), depsStruct, nil); err != nil { | 
| Jihoon Kang | 0d545b8 | 2024-10-11 00:21:57 +0000 | [diff] [blame] | 240 | mctx.ModuleErrorf(err.Error()) | 
| mrziwang | 8f86c88 | 2024-10-03 12:34:33 -0700 | [diff] [blame] | 241 | } | 
|  | 242 | } | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 243 | } | 
|  | 244 |  | 
| mrziwang | 2a506cf | 2024-10-17 15:38:37 -0700 | [diff] [blame] | 245 | func generateDepStruct(deps map[string]*depCandidateProps) *packagingPropsStruct { | 
|  | 246 | depsStruct := packagingPropsStruct{} | 
|  | 247 | for depName, depProps := range deps { | 
|  | 248 | bitness := getBitness(depProps.Arch) | 
|  | 249 | fullyQualifiedDepName := fullyQualifiedModuleName(depName, depProps.Namespace) | 
|  | 250 | if android.InList("32", bitness) && android.InList("64", bitness) { | 
|  | 251 | // If both 32 and 64 bit variants are enabled for this module | 
|  | 252 | switch depProps.Multilib { | 
|  | 253 | case string(android.MultilibBoth): | 
|  | 254 | depsStruct.Multilib.Both.Deps = append(depsStruct.Multilib.Both.Deps, fullyQualifiedDepName) | 
|  | 255 | case string(android.MultilibCommon), string(android.MultilibFirst): | 
|  | 256 | depsStruct.Deps = append(depsStruct.Deps, fullyQualifiedDepName) | 
|  | 257 | case "32": | 
|  | 258 | depsStruct.Multilib.Lib32.Deps = append(depsStruct.Multilib.Lib32.Deps, fullyQualifiedDepName) | 
|  | 259 | case "64", "darwin_universal": | 
|  | 260 | depsStruct.Multilib.Lib64.Deps = append(depsStruct.Multilib.Lib64.Deps, fullyQualifiedDepName) | 
|  | 261 | case "prefer32", "first_prefer32": | 
|  | 262 | depsStruct.Multilib.Prefer32.Deps = append(depsStruct.Multilib.Prefer32.Deps, fullyQualifiedDepName) | 
|  | 263 | default: | 
|  | 264 | depsStruct.Multilib.Both.Deps = append(depsStruct.Multilib.Both.Deps, fullyQualifiedDepName) | 
|  | 265 | } | 
|  | 266 | } else if android.InList("64", bitness) { | 
|  | 267 | // If only 64 bit variant is enabled | 
|  | 268 | depsStruct.Multilib.Lib64.Deps = append(depsStruct.Multilib.Lib64.Deps, fullyQualifiedDepName) | 
|  | 269 | } else if android.InList("32", bitness) { | 
|  | 270 | // If only 32 bit variant is enabled | 
|  | 271 | depsStruct.Multilib.Lib32.Deps = append(depsStruct.Multilib.Lib32.Deps, fullyQualifiedDepName) | 
|  | 272 | } else { | 
|  | 273 | // If only common variant is enabled | 
|  | 274 | depsStruct.Multilib.Common.Deps = append(depsStruct.Multilib.Common.Deps, fullyQualifiedDepName) | 
|  | 275 | } | 
|  | 276 | } | 
|  | 277 | return &depsStruct | 
|  | 278 | } | 
|  | 279 |  | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 280 | type filesystemCreatorProps struct { | 
|  | 281 | Generated_partition_types   []string `blueprint:"mutated"` | 
|  | 282 | Unsupported_partition_types []string `blueprint:"mutated"` | 
|  | 283 | } | 
|  | 284 |  | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 285 | type filesystemCreator struct { | 
|  | 286 | android.ModuleBase | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 287 |  | 
|  | 288 | properties filesystemCreatorProps | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 289 | } | 
|  | 290 |  | 
|  | 291 | func filesystemCreatorFactory() android.Module { | 
|  | 292 | module := &filesystemCreator{} | 
|  | 293 |  | 
| Cole Faust | 6978879 | 2024-10-10 11:00:36 -0700 | [diff] [blame] | 294 | android.InitAndroidArchModule(module, android.DeviceSupported, android.MultilibCommon) | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 295 | module.AddProperties(&module.properties) | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 296 | android.AddLoadHook(module, func(ctx android.LoadHookContext) { | 
| Jihoon Kang | 0d545b8 | 2024-10-11 00:21:57 +0000 | [diff] [blame] | 297 | createFsGenState(ctx) | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 298 | module.createInternalModules(ctx) | 
|  | 299 | }) | 
|  | 300 |  | 
|  | 301 | return module | 
|  | 302 | } | 
|  | 303 |  | 
|  | 304 | func (f *filesystemCreator) createInternalModules(ctx android.LoadHookContext) { | 
| Jihoon Kang | 0d545b8 | 2024-10-11 00:21:57 +0000 | [diff] [blame] | 305 | soongGeneratedPartitions := &ctx.Config().Get(fsGenStateOnceKey).(*FsGenState).soongGeneratedPartitions | 
|  | 306 | for _, partitionType := range *soongGeneratedPartitions { | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 307 | if f.createPartition(ctx, partitionType) { | 
|  | 308 | f.properties.Generated_partition_types = append(f.properties.Generated_partition_types, partitionType) | 
|  | 309 | } else { | 
|  | 310 | f.properties.Unsupported_partition_types = append(f.properties.Unsupported_partition_types, partitionType) | 
| Jihoon Kang | 0d545b8 | 2024-10-11 00:21:57 +0000 | [diff] [blame] | 311 | _, *soongGeneratedPartitions = android.RemoveFromList(partitionType, *soongGeneratedPartitions) | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 312 | } | 
|  | 313 | } | 
| Jihoon Kang | f1c79ca | 2024-10-09 20:18:38 +0000 | [diff] [blame] | 314 | f.createDeviceModule(ctx) | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 315 | } | 
|  | 316 |  | 
| Jihoon Kang | 0d545b8 | 2024-10-11 00:21:57 +0000 | [diff] [blame] | 317 | func generatedModuleName(cfg android.Config, suffix string) string { | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 318 | prefix := "soong" | 
|  | 319 | if cfg.HasDeviceProduct() { | 
|  | 320 | prefix = cfg.DeviceProduct() | 
|  | 321 | } | 
| Jihoon Kang | f1c79ca | 2024-10-09 20:18:38 +0000 | [diff] [blame] | 322 | return fmt.Sprintf("%s_generated_%s", prefix, suffix) | 
|  | 323 | } | 
|  | 324 |  | 
| Jihoon Kang | 0d545b8 | 2024-10-11 00:21:57 +0000 | [diff] [blame] | 325 | func generatedModuleNameForPartition(cfg android.Config, partitionType string) string { | 
|  | 326 | return generatedModuleName(cfg, fmt.Sprintf("%s_image", partitionType)) | 
| Jihoon Kang | f1c79ca | 2024-10-09 20:18:38 +0000 | [diff] [blame] | 327 | } | 
|  | 328 |  | 
|  | 329 | func (f *filesystemCreator) createDeviceModule(ctx android.LoadHookContext) { | 
|  | 330 | baseProps := &struct { | 
|  | 331 | Name *string | 
|  | 332 | }{ | 
| Jihoon Kang | 0d545b8 | 2024-10-11 00:21:57 +0000 | [diff] [blame] | 333 | Name: proptools.StringPtr(generatedModuleName(ctx.Config(), "device")), | 
| Jihoon Kang | f1c79ca | 2024-10-09 20:18:38 +0000 | [diff] [blame] | 334 | } | 
|  | 335 |  | 
| Spandan Das | 7a46f6c | 2024-10-14 18:41:18 +0000 | [diff] [blame] | 336 | // Currently, only the system and system_ext partition module is created. | 
| Jihoon Kang | f1c79ca | 2024-10-09 20:18:38 +0000 | [diff] [blame] | 337 | partitionProps := &filesystem.PartitionNameProperties{} | 
|  | 338 | if android.InList("system", f.properties.Generated_partition_types) { | 
| Jihoon Kang | 0d545b8 | 2024-10-11 00:21:57 +0000 | [diff] [blame] | 339 | partitionProps.System_partition_name = proptools.StringPtr(generatedModuleNameForPartition(ctx.Config(), "system")) | 
| Jihoon Kang | f1c79ca | 2024-10-09 20:18:38 +0000 | [diff] [blame] | 340 | } | 
| Spandan Das | 7a46f6c | 2024-10-14 18:41:18 +0000 | [diff] [blame] | 341 | if android.InList("system_ext", f.properties.Generated_partition_types) { | 
| Jihoon Kang | 0d545b8 | 2024-10-11 00:21:57 +0000 | [diff] [blame] | 342 | partitionProps.System_ext_partition_name = proptools.StringPtr(generatedModuleNameForPartition(ctx.Config(), "system_ext")) | 
| Spandan Das | 7a46f6c | 2024-10-14 18:41:18 +0000 | [diff] [blame] | 343 | } | 
| Jihoon Kang | f1c79ca | 2024-10-09 20:18:38 +0000 | [diff] [blame] | 344 |  | 
|  | 345 | ctx.CreateModule(filesystem.AndroidDeviceFactory, baseProps, partitionProps) | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 346 | } | 
|  | 347 |  | 
| Jihoon Kang | 6850d8f | 2024-10-17 20:45:58 +0000 | [diff] [blame] | 348 | func partitionSpecificFsProps(fsProps *filesystem.FilesystemProperties, partitionType string) { | 
|  | 349 | switch partitionType { | 
|  | 350 | case "system": | 
|  | 351 | fsProps.Build_logtags = proptools.BoolPtr(true) | 
|  | 352 | // https://source.corp.google.com/h/googleplex-android/platform/build//639d79f5012a6542ab1f733b0697db45761ab0f3:core/packaging/flags.mk;l=21;drc=5ba8a8b77507f93aa48cc61c5ba3f31a4d0cbf37;bpv=1;bpt=0 | 
|  | 353 | fsProps.Gen_aconfig_flags_pb = proptools.BoolPtr(true) | 
|  | 354 | case "product": | 
|  | 355 | fsProps.Gen_aconfig_flags_pb = proptools.BoolPtr(true) | 
|  | 356 | case "vendor": | 
|  | 357 | fsProps.Gen_aconfig_flags_pb = proptools.BoolPtr(true) | 
|  | 358 | } | 
|  | 359 | } | 
| Spandan Das | cbe641a | 2024-10-14 21:07:34 +0000 | [diff] [blame] | 360 |  | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 361 | // Creates a soong module to build the given partition. Returns false if we can't support building | 
|  | 362 | // it. | 
|  | 363 | func (f *filesystemCreator) createPartition(ctx android.LoadHookContext, partitionType string) bool { | 
| mrziwang | 4b0ca97 | 2024-10-17 14:56:19 -0700 | [diff] [blame] | 364 | baseProps := generateBaseProps(proptools.StringPtr(generatedModuleNameForPartition(ctx.Config(), partitionType))) | 
|  | 365 |  | 
|  | 366 | fsProps, supported := generateFsProps(ctx, partitionType) | 
|  | 367 | if !supported { | 
|  | 368 | return false | 
| mrziwang | a077b94 | 2024-10-16 16:00:06 -0700 | [diff] [blame] | 369 | } | 
| mrziwang | a077b94 | 2024-10-16 16:00:06 -0700 | [diff] [blame] | 370 |  | 
| mrziwang | 4b0ca97 | 2024-10-17 14:56:19 -0700 | [diff] [blame] | 371 | var module android.Module | 
|  | 372 | if partitionType == "system" { | 
|  | 373 | module = ctx.CreateModule(filesystem.SystemImageFactory, baseProps, fsProps) | 
|  | 374 | } else { | 
|  | 375 | // Explicitly set the partition. | 
|  | 376 | fsProps.Partition_type = proptools.StringPtr(partitionType) | 
|  | 377 | module = ctx.CreateModule(filesystem.FilesystemFactory, baseProps, fsProps) | 
|  | 378 | } | 
|  | 379 | module.HideFromMake() | 
|  | 380 | return true | 
|  | 381 | } | 
|  | 382 |  | 
|  | 383 | type filesystemBaseProperty struct { | 
|  | 384 | Name             *string | 
|  | 385 | Compile_multilib *string | 
|  | 386 | } | 
|  | 387 |  | 
|  | 388 | func generateBaseProps(namePtr *string) *filesystemBaseProperty { | 
|  | 389 | return &filesystemBaseProperty{ | 
|  | 390 | Name:             namePtr, | 
|  | 391 | Compile_multilib: proptools.StringPtr("both"), | 
|  | 392 | } | 
|  | 393 | } | 
|  | 394 |  | 
|  | 395 | func generateFsProps(ctx android.EarlyModuleContext, partitionType string) (*filesystem.FilesystemProperties, bool) { | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 396 | fsProps := &filesystem.FilesystemProperties{} | 
|  | 397 |  | 
| mrziwang | 4b0ca97 | 2024-10-17 14:56:19 -0700 | [diff] [blame] | 398 | partitionVars := ctx.Config().ProductVariables().PartitionVarsForSoongMigrationOnlyDoNotUse | 
|  | 399 | specificPartitionVars := partitionVars.PartitionQualifiedVariables[partitionType] | 
|  | 400 |  | 
|  | 401 | // BOARD_SYSTEMIMAGE_FILE_SYSTEM_TYPE | 
|  | 402 | fsType := specificPartitionVars.BoardFileSystemType | 
|  | 403 | if fsType == "" { | 
|  | 404 | fsType = "ext4" //default | 
|  | 405 | } | 
|  | 406 | fsProps.Type = proptools.StringPtr(fsType) | 
|  | 407 | if filesystem.GetFsTypeFromString(ctx, *fsProps.Type).IsUnknown() { | 
|  | 408 | // Currently the android_filesystem module type only supports a handful of FS types like ext4, erofs | 
|  | 409 | return nil, false | 
|  | 410 | } | 
|  | 411 |  | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 412 | // Don't build this module on checkbuilds, the soong-built partitions are still in-progress | 
|  | 413 | // and sometimes don't build. | 
|  | 414 | fsProps.Unchecked_module = proptools.BoolPtr(true) | 
|  | 415 |  | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 416 | // BOARD_AVB_ENABLE | 
|  | 417 | fsProps.Use_avb = proptools.BoolPtr(partitionVars.BoardAvbEnable) | 
|  | 418 | // BOARD_AVB_KEY_PATH | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 419 | fsProps.Avb_private_key = proptools.StringPtr(specificPartitionVars.BoardAvbKeyPath) | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 420 | // BOARD_AVB_ALGORITHM | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 421 | fsProps.Avb_algorithm = proptools.StringPtr(specificPartitionVars.BoardAvbAlgorithm) | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 422 | // BOARD_AVB_SYSTEM_ROLLBACK_INDEX | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 423 | if rollbackIndex, err := strconv.ParseInt(specificPartitionVars.BoardAvbRollbackIndex, 10, 64); err == nil { | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 424 | fsProps.Rollback_index = proptools.Int64Ptr(rollbackIndex) | 
|  | 425 | } | 
|  | 426 |  | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 427 | fsProps.Partition_name = proptools.StringPtr(partitionType) | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 428 |  | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 429 | fsProps.Base_dir = proptools.StringPtr(partitionType) | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 430 |  | 
| Jihoon Kang | 0d545b8 | 2024-10-11 00:21:57 +0000 | [diff] [blame] | 431 | fsProps.Is_auto_generated = proptools.BoolPtr(true) | 
|  | 432 |  | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 433 | // Identical to that of the generic_system_image | 
|  | 434 | fsProps.Fsverity.Inputs = []string{ | 
|  | 435 | "etc/boot-image.prof", | 
|  | 436 | "etc/dirty-image-objects", | 
|  | 437 | "etc/preloaded-classes", | 
|  | 438 | "etc/classpaths/*.pb", | 
|  | 439 | "framework/*", | 
|  | 440 | "framework/*/*",     // framework/{arch} | 
|  | 441 | "framework/oat/*/*", // framework/oat/{arch} | 
|  | 442 | } | 
| Jihoon Kang | 3c7be41 | 2024-10-10 22:14:22 +0000 | [diff] [blame] | 443 | fsProps.Fsverity.Libs = []string{":framework-res{.export-package.apk}"} | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 444 |  | 
| Jihoon Kang | 6850d8f | 2024-10-17 20:45:58 +0000 | [diff] [blame] | 445 | partitionSpecificFsProps(fsProps, partitionType) | 
|  | 446 |  | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 447 | // system_image properties that are not set: | 
|  | 448 | // - filesystemProperties.Avb_hash_algorithm | 
|  | 449 | // - filesystemProperties.File_contexts | 
|  | 450 | // - filesystemProperties.Dirs | 
|  | 451 | // - filesystemProperties.Symlinks | 
|  | 452 | // - filesystemProperties.Fake_timestamp | 
|  | 453 | // - filesystemProperties.Uuid | 
|  | 454 | // - filesystemProperties.Mount_point | 
|  | 455 | // - filesystemProperties.Include_make_built_files | 
|  | 456 | // - filesystemProperties.Build_logtags | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 457 | // - systemImageProperties.Linker_config_src | 
| mrziwang | 4b0ca97 | 2024-10-17 14:56:19 -0700 | [diff] [blame] | 458 |  | 
|  | 459 | return fsProps, true | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 460 | } | 
|  | 461 |  | 
|  | 462 | func (f *filesystemCreator) createDiffTest(ctx android.ModuleContext, partitionType string) android.Path { | 
| Jihoon Kang | 0d545b8 | 2024-10-11 00:21:57 +0000 | [diff] [blame] | 463 | partitionModuleName := generatedModuleNameForPartition(ctx.Config(), partitionType) | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 464 | systemImage := ctx.GetDirectDepWithTag(partitionModuleName, generatedFilesystemDepTag) | 
|  | 465 | filesystemInfo, ok := android.OtherModuleProvider(ctx, systemImage, filesystem.FilesystemProvider) | 
|  | 466 | if !ok { | 
|  | 467 | ctx.ModuleErrorf("Expected module %s to provide FileysystemInfo", partitionModuleName) | 
|  | 468 | } | 
|  | 469 | makeFileList := android.PathForArbitraryOutput(ctx, fmt.Sprintf("target/product/%s/obj/PACKAGING/%s_intermediates/file_list.txt", ctx.Config().DeviceName(), partitionType)) | 
|  | 470 | // For now, don't allowlist anything. The test will fail, but that's fine in the current | 
|  | 471 | // early stages where we're just figuring out what we need | 
| Jihoon Kang | 9e866c8 | 2024-10-07 22:39:18 +0000 | [diff] [blame] | 472 | emptyAllowlistFile := android.PathForModuleOut(ctx, fmt.Sprintf("allowlist_%s.txt", partitionModuleName)) | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 473 | android.WriteFileRule(ctx, emptyAllowlistFile, "") | 
| Jihoon Kang | 9e866c8 | 2024-10-07 22:39:18 +0000 | [diff] [blame] | 474 | diffTestResultFile := android.PathForModuleOut(ctx, fmt.Sprintf("diff_test_%s.txt", partitionModuleName)) | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 475 |  | 
|  | 476 | builder := android.NewRuleBuilder(pctx, ctx) | 
|  | 477 | builder.Command().BuiltTool("file_list_diff"). | 
|  | 478 | Input(makeFileList). | 
|  | 479 | Input(filesystemInfo.FileListFile). | 
| Jihoon Kang | 9e866c8 | 2024-10-07 22:39:18 +0000 | [diff] [blame] | 480 | Text(partitionModuleName). | 
|  | 481 | FlagWithInput("--allowlists ", emptyAllowlistFile) | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 482 | builder.Command().Text("touch").Output(diffTestResultFile) | 
|  | 483 | builder.Build(partitionModuleName+" diff test", partitionModuleName+" diff test") | 
|  | 484 | return diffTestResultFile | 
|  | 485 | } | 
|  | 486 |  | 
|  | 487 | func createFailingCommand(ctx android.ModuleContext, message string) android.Path { | 
|  | 488 | hasher := sha256.New() | 
|  | 489 | hasher.Write([]byte(message)) | 
|  | 490 | filename := fmt.Sprintf("failing_command_%x.txt", hasher.Sum(nil)) | 
|  | 491 | file := android.PathForModuleOut(ctx, filename) | 
|  | 492 | builder := android.NewRuleBuilder(pctx, ctx) | 
|  | 493 | builder.Command().Textf("echo %s", proptools.NinjaAndShellEscape(message)) | 
|  | 494 | builder.Command().Text("exit 1 #").Output(file) | 
|  | 495 | builder.Build("failing command "+filename, "failing command "+filename) | 
|  | 496 | return file | 
|  | 497 | } | 
|  | 498 |  | 
|  | 499 | type systemImageDepTagType struct { | 
|  | 500 | blueprint.BaseDependencyTag | 
|  | 501 | } | 
|  | 502 |  | 
|  | 503 | var generatedFilesystemDepTag systemImageDepTagType | 
|  | 504 |  | 
|  | 505 | func (f *filesystemCreator) DepsMutator(ctx android.BottomUpMutatorContext) { | 
|  | 506 | for _, partitionType := range f.properties.Generated_partition_types { | 
| Jihoon Kang | 0d545b8 | 2024-10-11 00:21:57 +0000 | [diff] [blame] | 507 | ctx.AddDependency(ctx.Module(), generatedFilesystemDepTag, generatedModuleNameForPartition(ctx.Config(), partitionType)) | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 508 | } | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 509 | } | 
|  | 510 |  | 
|  | 511 | func (f *filesystemCreator) GenerateAndroidBuildActions(ctx android.ModuleContext) { | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 512 | if ctx.ModuleDir() != "build/soong/fsgen" { | 
|  | 513 | ctx.ModuleErrorf("There can only be one soong_filesystem_creator in build/soong/fsgen") | 
|  | 514 | } | 
|  | 515 | f.HideFromMake() | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 516 |  | 
| mrziwang | 8f86c88 | 2024-10-03 12:34:33 -0700 | [diff] [blame] | 517 | content := generateBpContent(ctx, "system") | 
|  | 518 | generatedBp := android.PathForOutput(ctx, "soong_generated_product_config.bp") | 
|  | 519 | android.WriteFileRule(ctx, generatedBp, content) | 
|  | 520 | ctx.Phony("product_config_to_bp", generatedBp) | 
|  | 521 |  | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 522 | var diffTestFiles []android.Path | 
|  | 523 | for _, partitionType := range f.properties.Generated_partition_types { | 
| Jihoon Kang | 72f812f | 2024-10-17 18:46:24 +0000 | [diff] [blame] | 524 | diffTestFile := f.createDiffTest(ctx, partitionType) | 
|  | 525 | diffTestFiles = append(diffTestFiles, diffTestFile) | 
|  | 526 | ctx.Phony(fmt.Sprintf("soong_generated_%s_filesystem_test", partitionType), diffTestFile) | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 527 | } | 
|  | 528 | for _, partitionType := range f.properties.Unsupported_partition_types { | 
| Jihoon Kang | 72f812f | 2024-10-17 18:46:24 +0000 | [diff] [blame] | 529 | diffTestFile := createFailingCommand(ctx, fmt.Sprintf("Couldn't build %s partition", partitionType)) | 
|  | 530 | diffTestFiles = append(diffTestFiles, diffTestFile) | 
|  | 531 | ctx.Phony(fmt.Sprintf("soong_generated_%s_filesystem_test", partitionType), diffTestFile) | 
| Cole Faust | 92ccbe2 | 2024-10-03 14:38:37 -0700 | [diff] [blame] | 532 | } | 
|  | 533 | ctx.Phony("soong_generated_filesystem_tests", diffTestFiles...) | 
| Jihoon Kang | 98047cf | 2024-10-02 17:13:54 +0000 | [diff] [blame] | 534 | } | 
| mrziwang | 8f86c88 | 2024-10-03 12:34:33 -0700 | [diff] [blame] | 535 |  | 
| mrziwang | 8f86c88 | 2024-10-03 12:34:33 -0700 | [diff] [blame] | 536 | func generateBpContent(ctx android.EarlyModuleContext, partitionType string) string { | 
|  | 537 | // Currently only system partition is supported | 
|  | 538 | if partitionType != "system" { | 
|  | 539 | return "" | 
|  | 540 | } | 
| mrziwang | 4b0ca97 | 2024-10-17 14:56:19 -0700 | [diff] [blame] | 541 | fsProps, fsTypeSupported := generateFsProps(ctx, partitionType) | 
|  | 542 | if !fsTypeSupported { | 
|  | 543 | return "" | 
| mrziwang | 8f86c88 | 2024-10-03 12:34:33 -0700 | [diff] [blame] | 544 | } | 
|  | 545 |  | 
| mrziwang | 4b0ca97 | 2024-10-17 14:56:19 -0700 | [diff] [blame] | 546 | baseProps := generateBaseProps(proptools.StringPtr(generatedModuleNameForPartition(ctx.Config(), partitionType))) | 
| mrziwang | 2a506cf | 2024-10-17 15:38:37 -0700 | [diff] [blame] | 547 | deps := ctx.Config().Get(fsGenStateOnceKey).(*FsGenState).fsDeps[partitionType] | 
|  | 548 | depProps := generateDepStruct(*deps) | 
| mrziwang | 8f86c88 | 2024-10-03 12:34:33 -0700 | [diff] [blame] | 549 |  | 
| mrziwang | 4b0ca97 | 2024-10-17 14:56:19 -0700 | [diff] [blame] | 550 | result, err := proptools.RepackProperties([]interface{}{baseProps, fsProps, depProps}) | 
| mrziwang | 8f86c88 | 2024-10-03 12:34:33 -0700 | [diff] [blame] | 551 | if err != nil { | 
|  | 552 | ctx.ModuleErrorf(err.Error()) | 
|  | 553 | } | 
|  | 554 |  | 
|  | 555 | file := &parser.File{ | 
|  | 556 | Defs: []parser.Definition{ | 
|  | 557 | &parser.Module{ | 
|  | 558 | Type: "module", | 
|  | 559 | Map:  *result, | 
|  | 560 | }, | 
|  | 561 | }, | 
|  | 562 | } | 
|  | 563 | bytes, err := parser.Print(file) | 
|  | 564 | if err != nil { | 
|  | 565 | ctx.ModuleErrorf(err.Error()) | 
|  | 566 | } | 
|  | 567 | return strings.TrimSpace(string(bytes)) | 
|  | 568 | } |