blob: 5b3e19fa2e82e15b070354cdeb461a6ef04b9eaf [file] [log] [blame]
Liz Kammer2dd9ca42020-11-25 16:06:39 -08001package bp2build
2
3import (
Jingwen Chen0ee88a62022-01-07 14:55:29 +00004 "encoding/json"
Liz Kammer2dd9ca42020-11-25 16:06:39 -08005 "reflect"
Liz Kammer2dd9ca42020-11-25 16:06:39 -08006 "strings"
7
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux447f6c92021-08-31 20:30:36 +00008 "android/soong/android"
Trevor Radcliffe4f95ee92023-01-19 16:02:47 +00009 "android/soong/cc"
Sam Delmerico932c01c2022-03-25 16:33:26 +000010 cc_config "android/soong/cc/config"
11 java_config "android/soong/java/config"
Alex Márquez Pérez Muñíz Díaz Púras Thaureaux447f6c92021-08-31 20:30:36 +000012
Jingwen Chen7810e172022-07-29 02:25:34 +000013 "android/soong/apex"
14
Liz Kammer2dd9ca42020-11-25 16:06:39 -080015 "github.com/google/blueprint/proptools"
16)
17
18type BazelFile struct {
19 Dir string
20 Basename string
21 Contents string
22}
23
Spandan Das83e787e2023-01-11 02:50:00 +000024// PRIVATE: Use CreateSoongInjectionDirFiles instead
25func soongInjectionFiles(cfg android.Config, metrics CodegenMetrics) []BazelFile {
Jingwen Chenbf61afb2021-05-06 13:31:18 +000026 var files []BazelFile
27
Sam Delmerico46d08b42022-11-15 15:51:04 -050028 files = append(files, newFile("android", GeneratedBuildFileName, "")) // Creates a //cc_toolchain package.
29 files = append(files, newFile("android", "constants.bzl", android.BazelCcToolchainVars(cfg)))
30
Jingwen Chenc63677b2021-06-17 05:43:19 +000031 files = append(files, newFile("cc_toolchain", GeneratedBuildFileName, "")) // Creates a //cc_toolchain package.
Trevor Radcliffe4f95ee92023-01-19 16:02:47 +000032 files = append(files, newFile("cc_toolchain", "config_constants.bzl", cc_config.BazelCcToolchainVars(cfg)))
33 files = append(files, newFile("cc_toolchain", "sanitizer_constants.bzl", cc.BazelCcSanitizerToolchainVars(cfg)))
Sam Delmerico932c01c2022-03-25 16:33:26 +000034
35 files = append(files, newFile("java_toolchain", GeneratedBuildFileName, "")) // Creates a //java_toolchain package.
36 files = append(files, newFile("java_toolchain", "constants.bzl", java_config.BazelJavaToolchainVars(cfg)))
Jingwen Chenbf61afb2021-05-06 13:31:18 +000037
Jingwen Chen7810e172022-07-29 02:25:34 +000038 files = append(files, newFile("apex_toolchain", GeneratedBuildFileName, "")) // Creates a //apex_toolchain package.
39 files = append(files, newFile("apex_toolchain", "constants.bzl", apex.BazelApexToolchainVars()))
40
usta4f5d2c12022-10-28 23:32:01 -040041 files = append(files, newFile("metrics", "converted_modules.txt", strings.Join(metrics.Serialize().ConvertedModules, "\n")))
Jingwen Chenc63677b2021-06-17 05:43:19 +000042
Kevin Dagostino60f562a2022-09-20 03:54:47 +000043 convertedModulePathMap, err := json.MarshalIndent(metrics.convertedModulePathMap, "", "\t")
44 if err != nil {
45 panic(err)
46 }
47 files = append(files, newFile("metrics", "converted_modules_path_map.json", string(convertedModulePathMap)))
48
Jingwen Chen01812022021-11-19 14:29:43 +000049 files = append(files, newFile("product_config", "soong_config_variables.bzl", cfg.Bp2buildSoongConfigDefinitions.String()))
50
Liz Kammere8303bd2022-02-16 09:02:48 -050051 files = append(files, newFile("product_config", "arch_configuration.bzl", android.StarlarkArchConfigurations()))
52
Jingwen Chen0ee88a62022-01-07 14:55:29 +000053 apiLevelsContent, err := json.Marshal(android.GetApiLevelsMap(cfg))
54 if err != nil {
55 panic(err)
56 }
57 files = append(files, newFile("api_levels", GeneratedBuildFileName, `exports_files(["api_levels.json"])`))
58 files = append(files, newFile("api_levels", "api_levels.json", string(apiLevelsContent)))
Yu Liufc603162022-03-01 15:44:08 -080059 files = append(files, newFile("api_levels", "api_levels.bzl", android.StarlarkApiLevelConfigs(cfg)))
Jingwen Chen0ee88a62022-01-07 14:55:29 +000060
Sam Delmericocb3c52c2023-02-03 17:40:08 -050061 files = append(files, newFile("allowlists", GeneratedBuildFileName, ""))
62 files = append(files, newFile("allowlists", "env.bzl", android.EnvironmentVarsFile(cfg)))
Cole Faust705968d2022-12-14 11:32:05 -080063 // TODO(b/262781701): Create an alternate soong_build entrypoint for writing out these files only when requested
64 files = append(files, newFile("allowlists", "mixed_build_prod_allowlist.txt", strings.Join(android.GetBazelEnabledModules(android.BazelProdMode), "\n")+"\n"))
65 files = append(files, newFile("allowlists", "mixed_build_staging_allowlist.txt", strings.Join(android.GetBazelEnabledModules(android.BazelStagingMode), "\n")+"\n"))
66
Jingwen Chenbf61afb2021-05-06 13:31:18 +000067 return files
68}
69
Liz Kammer2dd9ca42020-11-25 16:06:39 -080070func CreateBazelFiles(
Sasha Smundak0fd93e02022-05-19 19:34:31 -070071 cfg android.Config,
Liz Kammer2dd9ca42020-11-25 16:06:39 -080072 ruleShims map[string]RuleShim,
Jingwen Chen40067de2021-01-26 21:58:43 -050073 buildToTargets map[string]BazelTargets,
Jingwen Chen33832f92021-01-24 22:55:54 -050074 mode CodegenMode) []BazelFile {
Liz Kammer2dd9ca42020-11-25 16:06:39 -080075
Jingwen Chen6c309cd2021-04-01 07:11:11 +000076 var files []BazelFile
Liz Kammer2dd9ca42020-11-25 16:06:39 -080077
Jingwen Chen33832f92021-01-24 22:55:54 -050078 if mode == QueryView {
Jingwen Chen6c309cd2021-04-01 07:11:11 +000079 // Write top level WORKSPACE.
80 files = append(files, newFile("", "WORKSPACE", ""))
81
Jingwen Chen12b4c272021-03-10 02:05:59 -050082 // Used to denote that the top level directory is a package.
83 files = append(files, newFile("", GeneratedBuildFileName, ""))
84
85 files = append(files, newFile(bazelRulesSubDir, GeneratedBuildFileName, ""))
86
Jingwen Chen73850672020-12-14 08:25:34 -050087 // These files are only used for queryview.
88 files = append(files, newFile(bazelRulesSubDir, "providers.bzl", providersBzl))
89
90 for bzlFileName, ruleShim := range ruleShims {
91 files = append(files, newFile(bazelRulesSubDir, bzlFileName+".bzl", ruleShim.content))
92 }
93 files = append(files, newFile(bazelRulesSubDir, "soong_module.bzl", generateSoongModuleBzl(ruleShims)))
Liz Kammer2dd9ca42020-11-25 16:06:39 -080094 }
Liz Kammer2dd9ca42020-11-25 16:06:39 -080095
Cole Faust324a92e2022-08-23 15:29:05 -070096 files = append(files, createBuildFiles(buildToTargets, mode)...)
Liz Kammer2dd9ca42020-11-25 16:06:39 -080097
98 return files
99}
100
Cole Faust324a92e2022-08-23 15:29:05 -0700101func createBuildFiles(buildToTargets map[string]BazelTargets, mode CodegenMode) []BazelFile {
Liz Kammer2dd9ca42020-11-25 16:06:39 -0800102 files := make([]BazelFile, 0, len(buildToTargets))
103 for _, dir := range android.SortedStringKeys(buildToTargets) {
Liz Kammer2dd9ca42020-11-25 16:06:39 -0800104 targets := buildToTargets[dir]
Jingwen Chen49109762021-05-25 05:16:48 +0000105 targets.sort()
106
107 var content string
Spandan Das5af0bd32022-09-28 20:43:08 +0000108 if mode == Bp2Build || mode == ApiBp2build {
Jingwen Chen49109762021-05-25 05:16:48 +0000109 content = `# READ THIS FIRST:
110# This file was automatically generated by bp2build for the Bazel migration project.
111# Feel free to edit or test it, but do *not* check it into your version control system.
112`
Jingwen Chen1c231732021-02-05 09:38:15 -0500113 content += targets.LoadStatements()
Sasha Smundak8bea2672022-08-04 13:31:14 -0700114 content += "\n\n"
115 // Get package rule from the handcrafted BUILD file, otherwise emit the default one.
116 prText := "package(default_visibility = [\"//visibility:public\"])\n"
117 if pr := targets.packageRule(); pr != nil {
118 prText = pr.content
119 }
120 content += prText
Jingwen Chen49109762021-05-25 05:16:48 +0000121 } else if mode == QueryView {
122 content = soongModuleLoad
Liz Kammer2dd9ca42020-11-25 16:06:39 -0800123 }
Jingwen Chen40067de2021-01-26 21:58:43 -0500124 if content != "" {
125 // If there are load statements, add a couple of newlines.
126 content += "\n\n"
127 }
128 content += targets.String()
Liz Kammerba3ea162021-02-17 13:22:03 -0500129 files = append(files, newFile(dir, GeneratedBuildFileName, content))
Liz Kammer2dd9ca42020-11-25 16:06:39 -0800130 }
131 return files
132}
133
134func newFile(dir, basename, content string) BazelFile {
135 return BazelFile{
136 Dir: dir,
137 Basename: basename,
138 Contents: content,
139 }
140}
141
142const (
143 bazelRulesSubDir = "build/bazel/queryview_rules"
144
145 // additional files:
146 // * workspace file
147 // * base BUILD file
148 // * rules BUILD file
149 // * rules providers.bzl file
150 // * rules soong_module.bzl file
151 numAdditionalFiles = 5
152)
153
154var (
155 // Certain module property names are blocklisted/ignored here, for the reasons commented.
156 ignoredPropNames = map[string]bool{
Sam Delmerico263efde2022-09-08 10:43:42 -0400157 "name": true, // redundant, since this is explicitly generated for every target
158 "from": true, // reserved keyword
159 "in": true, // reserved keyword
160 "size": true, // reserved for tests
161 "arch": true, // interface prop type is not supported yet.
162 "multilib": true, // interface prop type is not supported yet.
163 "target": true, // interface prop type is not supported yet.
164 "visibility": true, // Bazel has native visibility semantics. Handle later.
165 "features": true, // There is already a built-in attribute 'features' which cannot be overridden.
166 "for": true, // reserved keyword, b/233579439
167 "versions_with_info": true, // TODO(b/245730552) struct properties not fully supported
Liz Kammer2dd9ca42020-11-25 16:06:39 -0800168 }
169)
170
171func shouldGenerateAttribute(prop string) bool {
172 return !ignoredPropNames[prop]
173}
174
175func shouldSkipStructField(field reflect.StructField) bool {
Liz Kammer7a210ac2021-09-22 15:52:58 -0400176 if field.PkgPath != "" && !field.Anonymous {
Liz Kammer2dd9ca42020-11-25 16:06:39 -0800177 // Skip unexported fields. Some properties are
178 // internal to Soong only, and these fields do not have PkgPath.
179 return true
180 }
Sasha Smundak8bea2672022-08-04 13:31:14 -0700181 // fields with tag `blueprint:"mutated"` are exported to enable modification in mutators, etc.
Liz Kammer2dd9ca42020-11-25 16:06:39 -0800182 // but cannot be set in a .bp file
183 if proptools.HasTag(field, "blueprint", "mutated") {
184 return true
185 }
186 return false
187}
188
189// FIXME(b/168089390): In Bazel, rules ending with "_test" needs to be marked as
190// testonly = True, forcing other rules that depend on _test rules to also be
191// marked as testonly = True. This semantic constraint is not present in Soong.
192// To work around, rename "*_test" rules to "*_test_".
193func canonicalizeModuleType(moduleName string) string {
194 if strings.HasSuffix(moduleName, "_test") {
195 return moduleName + "_"
196 }
197
198 return moduleName
199}