xref: /aosp_15_r20/build/soong/android/build_prop.go (revision 333d2b3687b3a337dbcca9d65000bca186795e39)
1*333d2b36SAndroid Build Coastguard Worker// Copyright 2024 Google Inc. All rights reserved.
2*333d2b36SAndroid Build Coastguard Worker//
3*333d2b36SAndroid Build Coastguard Worker// Licensed under the Apache License, Version 2.0 (the "License");
4*333d2b36SAndroid Build Coastguard Worker// you may not use this file except in compliance with the License.
5*333d2b36SAndroid Build Coastguard Worker// You may obtain a copy of the License at
6*333d2b36SAndroid Build Coastguard Worker//
7*333d2b36SAndroid Build Coastguard Worker//     http://www.apache.org/licenses/LICENSE-2.0
8*333d2b36SAndroid Build Coastguard Worker//
9*333d2b36SAndroid Build Coastguard Worker// Unless required by applicable law or agreed to in writing, software
10*333d2b36SAndroid Build Coastguard Worker// distributed under the License is distributed on an "AS IS" BASIS,
11*333d2b36SAndroid Build Coastguard Worker// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12*333d2b36SAndroid Build Coastguard Worker// See the License for the specific language governing permissions and
13*333d2b36SAndroid Build Coastguard Worker// limitations under the License.
14*333d2b36SAndroid Build Coastguard Worker
15*333d2b36SAndroid Build Coastguard Workerpackage android
16*333d2b36SAndroid Build Coastguard Worker
17*333d2b36SAndroid Build Coastguard Workerimport (
18*333d2b36SAndroid Build Coastguard Worker	"fmt"
19*333d2b36SAndroid Build Coastguard Worker
20*333d2b36SAndroid Build Coastguard Worker	"github.com/google/blueprint/proptools"
21*333d2b36SAndroid Build Coastguard Worker)
22*333d2b36SAndroid Build Coastguard Worker
23*333d2b36SAndroid Build Coastguard Workerfunc init() {
24*333d2b36SAndroid Build Coastguard Worker	registerBuildPropComponents(InitRegistrationContext)
25*333d2b36SAndroid Build Coastguard Worker}
26*333d2b36SAndroid Build Coastguard Worker
27*333d2b36SAndroid Build Coastguard Workerfunc registerBuildPropComponents(ctx RegistrationContext) {
28*333d2b36SAndroid Build Coastguard Worker	ctx.RegisterModuleType("build_prop", BuildPropFactory)
29*333d2b36SAndroid Build Coastguard Worker	ctx.RegisterModuleType("android_info", AndroidInfoFactory)
30*333d2b36SAndroid Build Coastguard Worker}
31*333d2b36SAndroid Build Coastguard Worker
32*333d2b36SAndroid Build Coastguard Workertype buildPropProperties struct {
33*333d2b36SAndroid Build Coastguard Worker	// Output file name. Defaults to "build.prop"
34*333d2b36SAndroid Build Coastguard Worker	Stem *string
35*333d2b36SAndroid Build Coastguard Worker
36*333d2b36SAndroid Build Coastguard Worker	// List of prop names to exclude. This affects not only common build properties but also
37*333d2b36SAndroid Build Coastguard Worker	// properties in prop_files.
38*333d2b36SAndroid Build Coastguard Worker	Block_list []string
39*333d2b36SAndroid Build Coastguard Worker
40*333d2b36SAndroid Build Coastguard Worker	// Files to be appended at the end of build.prop. These files are appended after
41*333d2b36SAndroid Build Coastguard Worker	// post_process_props without any further checking.
42*333d2b36SAndroid Build Coastguard Worker	Footer_files []string `android:"path"`
43*333d2b36SAndroid Build Coastguard Worker
44*333d2b36SAndroid Build Coastguard Worker	// Path to a JSON file containing product configs.
45*333d2b36SAndroid Build Coastguard Worker	Product_config *string `android:"path"`
46*333d2b36SAndroid Build Coastguard Worker
47*333d2b36SAndroid Build Coastguard Worker	// Path to android-info.txt file containing board specific info.
48*333d2b36SAndroid Build Coastguard Worker	// This is empty for build.prop of all partitions except vendor.
49*333d2b36SAndroid Build Coastguard Worker	Android_info *string `android:"path"`
50*333d2b36SAndroid Build Coastguard Worker
51*333d2b36SAndroid Build Coastguard Worker	// Optional subdirectory under which this file is installed into
52*333d2b36SAndroid Build Coastguard Worker	Relative_install_path *string
53*333d2b36SAndroid Build Coastguard Worker}
54*333d2b36SAndroid Build Coastguard Worker
55*333d2b36SAndroid Build Coastguard Workertype buildPropModule struct {
56*333d2b36SAndroid Build Coastguard Worker	ModuleBase
57*333d2b36SAndroid Build Coastguard Worker
58*333d2b36SAndroid Build Coastguard Worker	properties buildPropProperties
59*333d2b36SAndroid Build Coastguard Worker
60*333d2b36SAndroid Build Coastguard Worker	outputFilePath Path
61*333d2b36SAndroid Build Coastguard Worker	installPath    InstallPath
62*333d2b36SAndroid Build Coastguard Worker}
63*333d2b36SAndroid Build Coastguard Worker
64*333d2b36SAndroid Build Coastguard Workerfunc (p *buildPropModule) stem() string {
65*333d2b36SAndroid Build Coastguard Worker	return proptools.StringDefault(p.properties.Stem, "build.prop")
66*333d2b36SAndroid Build Coastguard Worker}
67*333d2b36SAndroid Build Coastguard Worker
68*333d2b36SAndroid Build Coastguard Workerfunc (p *buildPropModule) propFiles(ctx ModuleContext) Paths {
69*333d2b36SAndroid Build Coastguard Worker	partition := p.partition(ctx.DeviceConfig())
70*333d2b36SAndroid Build Coastguard Worker	if partition == "system" {
71*333d2b36SAndroid Build Coastguard Worker		return ctx.Config().SystemPropFiles(ctx)
72*333d2b36SAndroid Build Coastguard Worker	} else if partition == "system_ext" {
73*333d2b36SAndroid Build Coastguard Worker		return ctx.Config().SystemExtPropFiles(ctx)
74*333d2b36SAndroid Build Coastguard Worker	} else if partition == "product" {
75*333d2b36SAndroid Build Coastguard Worker		return ctx.Config().ProductPropFiles(ctx)
76*333d2b36SAndroid Build Coastguard Worker	} else if partition == "odm" {
77*333d2b36SAndroid Build Coastguard Worker		return ctx.Config().OdmPropFiles(ctx)
78*333d2b36SAndroid Build Coastguard Worker	} else if partition == "vendor" {
79*333d2b36SAndroid Build Coastguard Worker		if p.properties.Android_info != nil {
80*333d2b36SAndroid Build Coastguard Worker			androidInfo := PathForModuleSrc(ctx, proptools.String(p.properties.Android_info))
81*333d2b36SAndroid Build Coastguard Worker			return append(ctx.Config().VendorPropFiles(ctx), androidInfo)
82*333d2b36SAndroid Build Coastguard Worker		}
83*333d2b36SAndroid Build Coastguard Worker		return ctx.Config().VendorPropFiles(ctx)
84*333d2b36SAndroid Build Coastguard Worker	}
85*333d2b36SAndroid Build Coastguard Worker	return nil
86*333d2b36SAndroid Build Coastguard Worker}
87*333d2b36SAndroid Build Coastguard Worker
88*333d2b36SAndroid Build Coastguard Workerfunc shouldAddBuildThumbprint(config Config) bool {
89*333d2b36SAndroid Build Coastguard Worker	knownOemProperties := []string{
90*333d2b36SAndroid Build Coastguard Worker		"ro.product.brand",
91*333d2b36SAndroid Build Coastguard Worker		"ro.product.name",
92*333d2b36SAndroid Build Coastguard Worker		"ro.product.device",
93*333d2b36SAndroid Build Coastguard Worker	}
94*333d2b36SAndroid Build Coastguard Worker
95*333d2b36SAndroid Build Coastguard Worker	for _, knownProp := range knownOemProperties {
96*333d2b36SAndroid Build Coastguard Worker		if InList(knownProp, config.OemProperties()) {
97*333d2b36SAndroid Build Coastguard Worker			return true
98*333d2b36SAndroid Build Coastguard Worker		}
99*333d2b36SAndroid Build Coastguard Worker	}
100*333d2b36SAndroid Build Coastguard Worker	return false
101*333d2b36SAndroid Build Coastguard Worker}
102*333d2b36SAndroid Build Coastguard Worker
103*333d2b36SAndroid Build Coastguard Worker// Can't use PartitionTag() because PartitionTag() returns the partition this module is actually
104*333d2b36SAndroid Build Coastguard Worker// installed (e.g. odm module's partition tag can be either "odm" or "vendor")
105*333d2b36SAndroid Build Coastguard Workerfunc (p *buildPropModule) partition(config DeviceConfig) string {
106*333d2b36SAndroid Build Coastguard Worker	if p.SocSpecific() {
107*333d2b36SAndroid Build Coastguard Worker		return "vendor"
108*333d2b36SAndroid Build Coastguard Worker	} else if p.DeviceSpecific() {
109*333d2b36SAndroid Build Coastguard Worker		return "odm"
110*333d2b36SAndroid Build Coastguard Worker	} else if p.ProductSpecific() {
111*333d2b36SAndroid Build Coastguard Worker		return "product"
112*333d2b36SAndroid Build Coastguard Worker	} else if p.SystemExtSpecific() {
113*333d2b36SAndroid Build Coastguard Worker		return "system_ext"
114*333d2b36SAndroid Build Coastguard Worker	} else if p.InstallInSystemDlkm() {
115*333d2b36SAndroid Build Coastguard Worker		return "system_dlkm"
116*333d2b36SAndroid Build Coastguard Worker	} else if p.InstallInVendorDlkm() {
117*333d2b36SAndroid Build Coastguard Worker		return "vendor_dlkm"
118*333d2b36SAndroid Build Coastguard Worker	} else if p.InstallInOdmDlkm() {
119*333d2b36SAndroid Build Coastguard Worker		return "odm_dlkm"
120*333d2b36SAndroid Build Coastguard Worker	} else if p.InstallInRamdisk() {
121*333d2b36SAndroid Build Coastguard Worker		// From this hardcoding in make:
122*333d2b36SAndroid Build Coastguard Worker		// https://cs.android.com/android/platform/superproject/main/+/main:build/make/core/sysprop.mk;l=311;drc=274435657e4682e5cee3fffd11fb301ab32a828d
123*333d2b36SAndroid Build Coastguard Worker		return "bootimage"
124*333d2b36SAndroid Build Coastguard Worker	}
125*333d2b36SAndroid Build Coastguard Worker	return "system"
126*333d2b36SAndroid Build Coastguard Worker}
127*333d2b36SAndroid Build Coastguard Worker
128*333d2b36SAndroid Build Coastguard Workerfunc (p *buildPropModule) GenerateAndroidBuildActions(ctx ModuleContext) {
129*333d2b36SAndroid Build Coastguard Worker	if !p.SocSpecific() && p.properties.Android_info != nil {
130*333d2b36SAndroid Build Coastguard Worker		ctx.ModuleErrorf("Android_info cannot be set if build.prop is not installed in vendor partition")
131*333d2b36SAndroid Build Coastguard Worker	}
132*333d2b36SAndroid Build Coastguard Worker
133*333d2b36SAndroid Build Coastguard Worker	outputFilePath := PathForModuleOut(ctx, "build.prop")
134*333d2b36SAndroid Build Coastguard Worker
135*333d2b36SAndroid Build Coastguard Worker	partition := p.partition(ctx.DeviceConfig())
136*333d2b36SAndroid Build Coastguard Worker
137*333d2b36SAndroid Build Coastguard Worker	rule := NewRuleBuilder(pctx, ctx)
138*333d2b36SAndroid Build Coastguard Worker
139*333d2b36SAndroid Build Coastguard Worker	config := ctx.Config()
140*333d2b36SAndroid Build Coastguard Worker
141*333d2b36SAndroid Build Coastguard Worker	cmd := rule.Command().BuiltTool("gen_build_prop")
142*333d2b36SAndroid Build Coastguard Worker
143*333d2b36SAndroid Build Coastguard Worker	cmd.FlagWithInput("--build-hostname-file=", config.BuildHostnameFile(ctx))
144*333d2b36SAndroid Build Coastguard Worker	cmd.FlagWithInput("--build-number-file=", config.BuildNumberFile(ctx))
145*333d2b36SAndroid Build Coastguard Worker	// shouldn't depend on BuildFingerprintFile and BuildThumbprintFile to prevent from rebuilding
146*333d2b36SAndroid Build Coastguard Worker	// on every incremental build.
147*333d2b36SAndroid Build Coastguard Worker	cmd.FlagWithArg("--build-fingerprint-file=", config.BuildFingerprintFile(ctx).String())
148*333d2b36SAndroid Build Coastguard Worker	// Export build thumbprint only if the product has specified at least one oem fingerprint property
149*333d2b36SAndroid Build Coastguard Worker	// b/17888863
150*333d2b36SAndroid Build Coastguard Worker	if shouldAddBuildThumbprint(config) {
151*333d2b36SAndroid Build Coastguard Worker		// In the previous make implementation, a dependency was not added on the thumbprint file
152*333d2b36SAndroid Build Coastguard Worker		cmd.FlagWithArg("--build-thumbprint-file=", config.BuildThumbprintFile(ctx).String())
153*333d2b36SAndroid Build Coastguard Worker	}
154*333d2b36SAndroid Build Coastguard Worker	cmd.FlagWithArg("--build-username=", config.Getenv("BUILD_USERNAME"))
155*333d2b36SAndroid Build Coastguard Worker	// shouldn't depend on BUILD_DATETIME_FILE to prevent from rebuilding on every incremental
156*333d2b36SAndroid Build Coastguard Worker	// build.
157*333d2b36SAndroid Build Coastguard Worker	cmd.FlagWithArg("--date-file=", ctx.Config().Getenv("BUILD_DATETIME_FILE"))
158*333d2b36SAndroid Build Coastguard Worker	cmd.FlagWithInput("--platform-preview-sdk-fingerprint-file=", ApiFingerprintPath(ctx))
159*333d2b36SAndroid Build Coastguard Worker	cmd.FlagWithInput("--product-config=", PathForModuleSrc(ctx, proptools.String(p.properties.Product_config)))
160*333d2b36SAndroid Build Coastguard Worker	cmd.FlagWithArg("--partition=", partition)
161*333d2b36SAndroid Build Coastguard Worker	cmd.FlagForEachInput("--prop-files=", p.propFiles(ctx))
162*333d2b36SAndroid Build Coastguard Worker	cmd.FlagWithOutput("--out=", outputFilePath)
163*333d2b36SAndroid Build Coastguard Worker
164*333d2b36SAndroid Build Coastguard Worker	postProcessCmd := rule.Command().BuiltTool("post_process_props")
165*333d2b36SAndroid Build Coastguard Worker	if ctx.DeviceConfig().BuildBrokenDupSysprop() {
166*333d2b36SAndroid Build Coastguard Worker		postProcessCmd.Flag("--allow-dup")
167*333d2b36SAndroid Build Coastguard Worker	}
168*333d2b36SAndroid Build Coastguard Worker	postProcessCmd.FlagWithArg("--sdk-version ", config.PlatformSdkVersion().String())
169*333d2b36SAndroid Build Coastguard Worker	if ctx.Config().EnableUffdGc() == "default" {
170*333d2b36SAndroid Build Coastguard Worker		postProcessCmd.FlagWithInput("--kernel-version-file-for-uffd-gc ", PathForOutput(ctx, "dexpreopt/kernel_version_for_uffd_gc.txt"))
171*333d2b36SAndroid Build Coastguard Worker	} else {
172*333d2b36SAndroid Build Coastguard Worker		// still need to pass an empty string to kernel-version-file-for-uffd-gc
173*333d2b36SAndroid Build Coastguard Worker		postProcessCmd.FlagWithArg("--kernel-version-file-for-uffd-gc ", `""`)
174*333d2b36SAndroid Build Coastguard Worker	}
175*333d2b36SAndroid Build Coastguard Worker	postProcessCmd.Text(outputFilePath.String())
176*333d2b36SAndroid Build Coastguard Worker	postProcessCmd.Flags(p.properties.Block_list)
177*333d2b36SAndroid Build Coastguard Worker
178*333d2b36SAndroid Build Coastguard Worker	for _, footer := range p.properties.Footer_files {
179*333d2b36SAndroid Build Coastguard Worker		path := PathForModuleSrc(ctx, footer)
180*333d2b36SAndroid Build Coastguard Worker		rule.appendText(outputFilePath, "####################################")
181*333d2b36SAndroid Build Coastguard Worker		rule.appendTextf(outputFilePath, "# Adding footer from %v", footer)
182*333d2b36SAndroid Build Coastguard Worker		rule.appendTextf(outputFilePath, "# with path %v", path)
183*333d2b36SAndroid Build Coastguard Worker		rule.appendText(outputFilePath, "####################################")
184*333d2b36SAndroid Build Coastguard Worker		rule.Command().Text("cat").FlagWithInput("", path).FlagWithArg(">> ", outputFilePath.String())
185*333d2b36SAndroid Build Coastguard Worker	}
186*333d2b36SAndroid Build Coastguard Worker
187*333d2b36SAndroid Build Coastguard Worker	rule.appendText(outputFilePath, "# end of file")
188*333d2b36SAndroid Build Coastguard Worker
189*333d2b36SAndroid Build Coastguard Worker	rule.Build(ctx.ModuleName(), "generating build.prop")
190*333d2b36SAndroid Build Coastguard Worker
191*333d2b36SAndroid Build Coastguard Worker	p.installPath = PathForModuleInstall(ctx, proptools.String(p.properties.Relative_install_path))
192*333d2b36SAndroid Build Coastguard Worker	ctx.InstallFile(p.installPath, p.stem(), outputFilePath)
193*333d2b36SAndroid Build Coastguard Worker
194*333d2b36SAndroid Build Coastguard Worker	ctx.SetOutputFiles(Paths{outputFilePath}, "")
195*333d2b36SAndroid Build Coastguard Worker	p.outputFilePath = outputFilePath
196*333d2b36SAndroid Build Coastguard Worker}
197*333d2b36SAndroid Build Coastguard Worker
198*333d2b36SAndroid Build Coastguard Workerfunc (r *RuleBuilder) appendText(path ModuleOutPath, text string) {
199*333d2b36SAndroid Build Coastguard Worker	r.Command().Text("echo").Text(proptools.NinjaAndShellEscape(text)).FlagWithArg(">> ", path.String())
200*333d2b36SAndroid Build Coastguard Worker}
201*333d2b36SAndroid Build Coastguard Worker
202*333d2b36SAndroid Build Coastguard Workerfunc (r *RuleBuilder) appendTextf(path ModuleOutPath, format string, a ...any) {
203*333d2b36SAndroid Build Coastguard Worker	r.appendText(path, fmt.Sprintf(format, a...))
204*333d2b36SAndroid Build Coastguard Worker}
205*333d2b36SAndroid Build Coastguard Worker
206*333d2b36SAndroid Build Coastguard Workerfunc (p *buildPropModule) AndroidMkEntries() []AndroidMkEntries {
207*333d2b36SAndroid Build Coastguard Worker	return []AndroidMkEntries{{
208*333d2b36SAndroid Build Coastguard Worker		Class:      "ETC",
209*333d2b36SAndroid Build Coastguard Worker		OutputFile: OptionalPathForPath(p.outputFilePath),
210*333d2b36SAndroid Build Coastguard Worker		ExtraEntries: []AndroidMkExtraEntriesFunc{
211*333d2b36SAndroid Build Coastguard Worker			func(ctx AndroidMkExtraEntriesContext, entries *AndroidMkEntries) {
212*333d2b36SAndroid Build Coastguard Worker				entries.SetString("LOCAL_MODULE_PATH", p.installPath.String())
213*333d2b36SAndroid Build Coastguard Worker				entries.SetString("LOCAL_INSTALLED_MODULE_STEM", p.outputFilePath.Base())
214*333d2b36SAndroid Build Coastguard Worker			},
215*333d2b36SAndroid Build Coastguard Worker		},
216*333d2b36SAndroid Build Coastguard Worker	}}
217*333d2b36SAndroid Build Coastguard Worker}
218*333d2b36SAndroid Build Coastguard Worker
219*333d2b36SAndroid Build Coastguard Worker// build_prop module generates {partition}/build.prop file. At first common build properties are
220*333d2b36SAndroid Build Coastguard Worker// printed based on Soong config variables. And then prop_files are printed as-is. Finally,
221*333d2b36SAndroid Build Coastguard Worker// post_process_props tool is run to check if the result build.prop is valid or not.
222*333d2b36SAndroid Build Coastguard Workerfunc BuildPropFactory() Module {
223*333d2b36SAndroid Build Coastguard Worker	module := &buildPropModule{}
224*333d2b36SAndroid Build Coastguard Worker	module.AddProperties(&module.properties)
225*333d2b36SAndroid Build Coastguard Worker	InitAndroidArchModule(module, DeviceSupported, MultilibCommon)
226*333d2b36SAndroid Build Coastguard Worker	return module
227*333d2b36SAndroid Build Coastguard Worker}
228