blob: bc44b21015aba97827cba8a2ad74a9f1d311b052 [file] [log] [blame]
Colin Cross43f08db2018-11-12 10:13:39 -08001// Copyright 2018 Google Inc. All rights reserved.
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 dexpreopt
16
17import (
18 "encoding/json"
Martin Stjernholmd90676f2020-01-11 00:37:30 +000019 "fmt"
Colin Cross69f59a32019-02-15 10:39:37 -080020 "strings"
Colin Cross74ba9622019-02-11 15:11:14 -080021
Martin Stjernholmd90676f2020-01-11 00:37:30 +000022 "github.com/google/blueprint"
23
Colin Cross74ba9622019-02-11 15:11:14 -080024 "android/soong/android"
Colin Cross43f08db2018-11-12 10:13:39 -080025)
26
Martin Stjernholmc52aaf12020-01-06 23:11:37 +000027// GlobalConfig stores the configuration for dex preopting. The fields are set
Martin Stjernholm75a48d82020-01-10 20:32:59 +000028// from product variables via dex_preopt_config.mk.
Colin Cross43f08db2018-11-12 10:13:39 -080029type GlobalConfig struct {
Colin Cross69f59a32019-02-15 10:39:37 -080030 DisablePreopt bool // disable preopt for all modules
Colin Cross43f08db2018-11-12 10:13:39 -080031 DisablePreoptModules []string // modules with preopt disabled by product-specific config
32
33 OnlyPreoptBootImageAndSystemServer bool // only preopt jars in the boot image or system server
34
Vladimir Marko40139d62020-02-06 15:14:29 +000035 UseArtImage bool // use the art image (use other boot class path dex files without image)
36
Colin Cross43f08db2018-11-12 10:13:39 -080037 HasSystemOther bool // store odex files that match PatternsOnSystemOther on the system_other partition
38 PatternsOnSystemOther []string // patterns (using '%' to denote a prefix match) to put odex on the system_other partition
39
Colin Cross69f59a32019-02-15 10:39:37 -080040 DisableGenerateProfile bool // don't generate profiles
41 ProfileDir string // directory to find profiles in
Colin Cross43f08db2018-11-12 10:13:39 -080042
Roshan Piusccc26ef2019-11-27 09:37:46 -080043 BootJars []string // modules for jars that form the boot class path
44 UpdatableBootJars []string // jars within apex that form the boot class path
Vladimir Markod2ee5322018-12-19 17:57:57 +000045
Ulya Trafimovich4d2eeed2019-11-08 10:54:21 +000046 ArtApexJars []string // modules for jars that are in the ART APEX
Colin Cross800fe132019-02-11 14:21:24 -080047
Roshan Pius9b51a402019-11-21 12:36:53 -080048 SystemServerJars []string // jars that form the system server
49 SystemServerApps []string // apps that are loaded into system server
50 UpdatableSystemServerJars []string // jars within apex that are loaded into system server
51 SpeedApps []string // apps that should be speed optimized
Colin Cross43f08db2018-11-12 10:13:39 -080052
Ulya Trafimovichcd3203f2020-03-27 11:30:00 +000053 BrokenSuboptimalOrderOfSystemServerJars bool // if true, sub-optimal order does not cause a build error
54
Colin Cross43f08db2018-11-12 10:13:39 -080055 PreoptFlags []string // global dex2oat flags that should be used if no module-specific dex2oat flags are specified
56
57 DefaultCompilerFilter string // default compiler filter to pass to dex2oat, overridden by --compiler-filter= in module-specific dex2oat flags
58 SystemServerCompilerFilter string // default compiler filter to pass to dex2oat for system server jars
59
Nicolas Geoffrayc1bf7242019-10-18 14:51:38 +010060 GenerateDMFiles bool // generate Dex Metadata files
Colin Cross43f08db2018-11-12 10:13:39 -080061
62 NoDebugInfo bool // don't generate debug info by default
Mathieu Chartier3f7ddbb2019-04-29 09:33:50 -070063 DontResolveStartupStrings bool // don't resolve string literals loaded during application startup.
Colin Cross43f08db2018-11-12 10:13:39 -080064 AlwaysSystemServerDebugInfo bool // always generate mini debug info for system server modules (overrides NoDebugInfo=true)
65 NeverSystemServerDebugInfo bool // never generate mini debug info for system server modules (overrides NoDebugInfo=false)
66 AlwaysOtherDebugInfo bool // always generate mini debug info for non-system server modules (overrides NoDebugInfo=true)
67 NeverOtherDebugInfo bool // never generate mini debug info for non-system server modules (overrides NoDebugInfo=true)
68
Colin Cross43f08db2018-11-12 10:13:39 -080069 IsEng bool // build is a eng variant
70 SanitizeLite bool // build is the second phase of a SANITIZE_LITE build
71
72 DefaultAppImages bool // build app images (TODO: .art files?) by default
73
Colin Cross800fe132019-02-11 14:21:24 -080074 Dex2oatXmx string // max heap size for dex2oat
75 Dex2oatXms string // initial heap size for dex2oat
Colin Cross43f08db2018-11-12 10:13:39 -080076
77 EmptyDirectory string // path to an empty directory
78
Colin Cross74ba9622019-02-11 15:11:14 -080079 CpuVariant map[android.ArchType]string // cpu variant for each architecture
80 InstructionSetFeatures map[android.ArchType]string // instruction set for each architecture
Colin Cross43f08db2018-11-12 10:13:39 -080081
Colin Cross800fe132019-02-11 14:21:24 -080082 // Only used for boot image
Mathieu Chartier6adeee12019-06-26 10:01:36 -070083 DirtyImageObjects android.OptionalPath // path to a dirty-image-objects file
84 BootImageProfiles android.Paths // path to a boot-image-profile.txt file
85 BootFlags string // extra flags to pass to dex2oat for the boot image
86 Dex2oatImageXmx string // max heap size for dex2oat for the boot image
87 Dex2oatImageXms string // initial heap size for dex2oat for the boot image
Colin Cross43f08db2018-11-12 10:13:39 -080088}
89
Martin Stjernholmc52aaf12020-01-06 23:11:37 +000090// GlobalSoongConfig contains the global config that is generated from Soong,
91// stored in dexpreopt_soong.config.
92type GlobalSoongConfig struct {
93 // Paths to tools possibly used by the generated commands.
94 Profman android.Path
95 Dex2oat android.Path
96 Aapt android.Path
97 SoongZip android.Path
98 Zip2zip android.Path
99 ManifestCheck android.Path
Colin Cross38b96852019-05-22 10:21:09 -0700100 ConstructContext android.Path
Colin Cross43f08db2018-11-12 10:13:39 -0800101}
102
Ulya Trafimovichd4bcea42020-06-03 14:57:22 +0100103// LibraryPath contains paths to the library DEX jar on host and on device.
104type LibraryPath struct {
105 Host android.Path
106 Device string
107}
108
109// LibraryPaths is a map from library name to on-host and on-device paths to its DEX jar.
110type LibraryPaths map[string]*LibraryPath
111
Colin Cross43f08db2018-11-12 10:13:39 -0800112type ModuleConfig struct {
Victor Hsiehd181c8b2019-01-29 13:00:33 -0800113 Name string
114 DexLocation string // dex location on device
Colin Cross69f59a32019-02-15 10:39:37 -0800115 BuildPath android.OutputPath
116 DexPath android.Path
Colin Cross38b96852019-05-22 10:21:09 -0700117 ManifestPath android.Path
Victor Hsiehd181c8b2019-01-29 13:00:33 -0800118 UncompressedDex bool
119 HasApkLibraries bool
120 PreoptFlags []string
Colin Cross43f08db2018-11-12 10:13:39 -0800121
Colin Cross69f59a32019-02-15 10:39:37 -0800122 ProfileClassListing android.OptionalPath
Colin Cross43f08db2018-11-12 10:13:39 -0800123 ProfileIsTextListing bool
Nicolas Geoffraye7102422019-07-24 13:19:29 +0100124 ProfileBootListing android.OptionalPath
Colin Cross43f08db2018-11-12 10:13:39 -0800125
Ulya Trafimovich6e827482020-06-12 14:32:24 +0100126 EnforceUsesLibraries bool
127 OptionalUsesLibraries []string
128 UsesLibraries []string
129 LibraryPaths LibraryPaths
Colin Cross43f08db2018-11-12 10:13:39 -0800130
Ulya Trafimovich4d2eeed2019-11-08 10:54:21 +0000131 Archs []android.ArchType
132 DexPreoptImages []android.Path
133 DexPreoptImagesDeps []android.OutputPaths
134 DexPreoptImageLocations []string
Colin Cross43f08db2018-11-12 10:13:39 -0800135
Colin Cross69f59a32019-02-15 10:39:37 -0800136 PreoptBootClassPathDexFiles android.Paths // file paths of boot class path files
137 PreoptBootClassPathDexLocations []string // virtual locations of boot class path files
Colin Cross800fe132019-02-11 14:21:24 -0800138
Colin Cross43f08db2018-11-12 10:13:39 -0800139 PreoptExtractedApk bool // Overrides OnlyPreoptModules
140
141 NoCreateAppImage bool
142 ForceCreateAppImage bool
143
144 PresignedPrebuilt bool
Colin Cross43f08db2018-11-12 10:13:39 -0800145}
146
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000147type globalSoongConfigSingleton struct{}
148
149var pctx = android.NewPackageContext("android/soong/dexpreopt")
150
151func init() {
152 pctx.Import("android/soong/android")
153 android.RegisterSingletonType("dexpreopt-soong-config", func() android.Singleton {
154 return &globalSoongConfigSingleton{}
155 })
156}
157
Colin Cross69f59a32019-02-15 10:39:37 -0800158func constructPath(ctx android.PathContext, path string) android.Path {
159 buildDirPrefix := ctx.Config().BuildDir() + "/"
160 if path == "" {
161 return nil
162 } else if strings.HasPrefix(path, buildDirPrefix) {
163 return android.PathForOutput(ctx, strings.TrimPrefix(path, buildDirPrefix))
164 } else {
165 return android.PathForSource(ctx, path)
166 }
Colin Cross43f08db2018-11-12 10:13:39 -0800167}
168
Colin Cross69f59a32019-02-15 10:39:37 -0800169func constructPaths(ctx android.PathContext, paths []string) android.Paths {
170 var ret android.Paths
171 for _, path := range paths {
172 ret = append(ret, constructPath(ctx, path))
173 }
174 return ret
Colin Cross43f08db2018-11-12 10:13:39 -0800175}
176
Colin Cross69f59a32019-02-15 10:39:37 -0800177func constructWritablePath(ctx android.PathContext, path string) android.WritablePath {
178 if path == "" {
179 return nil
180 }
181 return constructPath(ctx, path).(android.WritablePath)
182}
183
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000184// ParseGlobalConfig parses the given data assumed to be read from the global
185// dexpreopt.config file into a GlobalConfig struct.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000186func ParseGlobalConfig(ctx android.PathContext, data []byte) (*GlobalConfig, error) {
Colin Cross69f59a32019-02-15 10:39:37 -0800187 type GlobalJSONConfig struct {
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000188 *GlobalConfig
Colin Cross69f59a32019-02-15 10:39:37 -0800189
190 // Copies of entries in GlobalConfig that are not constructable without extra parameters. They will be
191 // used to construct the real value manually below.
192 DirtyImageObjects string
Colin Cross69f59a32019-02-15 10:39:37 -0800193 BootImageProfiles []string
Colin Cross69f59a32019-02-15 10:39:37 -0800194 }
195
196 config := GlobalJSONConfig{}
Colin Cross988414c2020-01-11 01:11:46 +0000197 err := json.Unmarshal(data, &config)
Colin Cross69f59a32019-02-15 10:39:37 -0800198 if err != nil {
Colin Cross988414c2020-01-11 01:11:46 +0000199 return config.GlobalConfig, err
Colin Cross69f59a32019-02-15 10:39:37 -0800200 }
201
202 // Construct paths that require a PathContext.
203 config.GlobalConfig.DirtyImageObjects = android.OptionalPathForPath(constructPath(ctx, config.DirtyImageObjects))
Colin Cross69f59a32019-02-15 10:39:37 -0800204 config.GlobalConfig.BootImageProfiles = constructPaths(ctx, config.BootImageProfiles)
205
Colin Cross988414c2020-01-11 01:11:46 +0000206 return config.GlobalConfig, nil
Colin Cross69f59a32019-02-15 10:39:37 -0800207}
208
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000209type globalConfigAndRaw struct {
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000210 global *GlobalConfig
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000211 data []byte
212}
213
214// GetGlobalConfig returns the global dexpreopt.config that's created in the
215// make config phase. It is loaded once the first time it is called for any
216// ctx.Config(), and returns the same data for all future calls with the same
217// ctx.Config(). A value can be inserted for tests using
218// setDexpreoptTestGlobalConfig.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000219func GetGlobalConfig(ctx android.PathContext) *GlobalConfig {
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000220 return getGlobalConfigRaw(ctx).global
221}
222
223// GetGlobalConfigRawData is the same as GetGlobalConfig, except that it returns
224// the literal content of dexpreopt.config.
225func GetGlobalConfigRawData(ctx android.PathContext) []byte {
226 return getGlobalConfigRaw(ctx).data
227}
228
229var globalConfigOnceKey = android.NewOnceKey("DexpreoptGlobalConfig")
230var testGlobalConfigOnceKey = android.NewOnceKey("TestDexpreoptGlobalConfig")
231
232func getGlobalConfigRaw(ctx android.PathContext) globalConfigAndRaw {
233 return ctx.Config().Once(globalConfigOnceKey, func() interface{} {
234 if data, err := ctx.Config().DexpreoptGlobalConfig(ctx); err != nil {
235 panic(err)
236 } else if data != nil {
237 globalConfig, err := ParseGlobalConfig(ctx, data)
238 if err != nil {
239 panic(err)
240 }
241 return globalConfigAndRaw{globalConfig, data}
242 }
243
244 // No global config filename set, see if there is a test config set
245 return ctx.Config().Once(testGlobalConfigOnceKey, func() interface{} {
246 // Nope, return a config with preopting disabled
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000247 return globalConfigAndRaw{&GlobalConfig{
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000248 DisablePreopt: true,
249 DisableGenerateProfile: true,
250 }, nil}
251 })
252 }).(globalConfigAndRaw)
253}
254
255// SetTestGlobalConfig sets a GlobalConfig that future calls to GetGlobalConfig
256// will return. It must be called before the first call to GetGlobalConfig for
257// the config.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000258func SetTestGlobalConfig(config android.Config, globalConfig *GlobalConfig) {
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000259 config.Once(testGlobalConfigOnceKey, func() interface{} { return globalConfigAndRaw{globalConfig, nil} })
260}
261
262// ParseModuleConfig parses a per-module dexpreopt.config file into a
263// ModuleConfig struct. It is not used in Soong, which receives a ModuleConfig
264// struct directly from java/dexpreopt.go. It is used in dexpreopt_gen called
265// from Make to read the module dexpreopt.config written in the Make config
266// stage.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000267func ParseModuleConfig(ctx android.PathContext, data []byte) (*ModuleConfig, error) {
Ulya Trafimovichd4bcea42020-06-03 14:57:22 +0100268 type jsonLibraryPath struct {
269 Host string
270 Device string
271 }
272
273 type jsonLibraryPaths map[string]jsonLibraryPath
274
Colin Cross69f59a32019-02-15 10:39:37 -0800275 type ModuleJSONConfig struct {
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000276 *ModuleConfig
Colin Cross69f59a32019-02-15 10:39:37 -0800277
278 // Copies of entries in ModuleConfig that are not constructable without extra parameters. They will be
279 // used to construct the real value manually below.
280 BuildPath string
281 DexPath string
Colin Cross38b96852019-05-22 10:21:09 -0700282 ManifestPath string
Colin Cross69f59a32019-02-15 10:39:37 -0800283 ProfileClassListing string
Ulya Trafimovichd4bcea42020-06-03 14:57:22 +0100284 LibraryPaths jsonLibraryPaths
Colin Cross69f59a32019-02-15 10:39:37 -0800285 DexPreoptImages []string
Ulya Trafimovich4d2eeed2019-11-08 10:54:21 +0000286 DexPreoptImageLocations []string
Colin Cross69f59a32019-02-15 10:39:37 -0800287 PreoptBootClassPathDexFiles []string
Colin Cross69f59a32019-02-15 10:39:37 -0800288 }
289
Ulya Trafimovichd4bcea42020-06-03 14:57:22 +0100290 // convert JSON map of library paths to LibraryPaths
291 constructLibraryPaths := func(ctx android.PathContext, paths jsonLibraryPaths) LibraryPaths {
292 m := LibraryPaths{}
293 for lib, path := range paths {
294 m[lib] = &LibraryPath{
295 constructPath(ctx, path.Host),
296 path.Device,
297 }
298 }
299 return m
300 }
301
Colin Cross69f59a32019-02-15 10:39:37 -0800302 config := ModuleJSONConfig{}
303
Colin Cross988414c2020-01-11 01:11:46 +0000304 err := json.Unmarshal(data, &config)
Colin Cross69f59a32019-02-15 10:39:37 -0800305 if err != nil {
306 return config.ModuleConfig, err
307 }
308
309 // Construct paths that require a PathContext.
310 config.ModuleConfig.BuildPath = constructPath(ctx, config.BuildPath).(android.OutputPath)
311 config.ModuleConfig.DexPath = constructPath(ctx, config.DexPath)
Colin Cross38b96852019-05-22 10:21:09 -0700312 config.ModuleConfig.ManifestPath = constructPath(ctx, config.ManifestPath)
Colin Cross69f59a32019-02-15 10:39:37 -0800313 config.ModuleConfig.ProfileClassListing = android.OptionalPathForPath(constructPath(ctx, config.ProfileClassListing))
Ulya Trafimovichd4bcea42020-06-03 14:57:22 +0100314 config.ModuleConfig.LibraryPaths = constructLibraryPaths(ctx, config.LibraryPaths)
Colin Cross69f59a32019-02-15 10:39:37 -0800315 config.ModuleConfig.DexPreoptImages = constructPaths(ctx, config.DexPreoptImages)
Ulya Trafimovich4d2eeed2019-11-08 10:54:21 +0000316 config.ModuleConfig.DexPreoptImageLocations = config.DexPreoptImageLocations
Colin Cross69f59a32019-02-15 10:39:37 -0800317 config.ModuleConfig.PreoptBootClassPathDexFiles = constructPaths(ctx, config.PreoptBootClassPathDexFiles)
Colin Cross69f59a32019-02-15 10:39:37 -0800318
Dan Willemsen0f416782019-06-13 21:44:53 +0000319 // This needs to exist, but dependencies are already handled in Make, so we don't need to pass them through JSON.
Ulya Trafimovich4d2eeed2019-11-08 10:54:21 +0000320 config.ModuleConfig.DexPreoptImagesDeps = make([]android.OutputPaths, len(config.ModuleConfig.DexPreoptImages))
Dan Willemsen0f416782019-06-13 21:44:53 +0000321
Colin Cross69f59a32019-02-15 10:39:37 -0800322 return config.ModuleConfig, nil
323}
324
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000325// dex2oatModuleName returns the name of the module to use for the dex2oat host
326// tool. It should be a binary module with public visibility that is compiled
327// and installed for host.
328func dex2oatModuleName(config android.Config) string {
329 // Default to the debug variant of dex2oat to help find bugs.
330 // Set USE_DEX2OAT_DEBUG to false for only building non-debug versions.
331 if config.Getenv("USE_DEX2OAT_DEBUG") == "false" {
332 return "dex2oat"
333 } else {
334 return "dex2oatd"
335 }
336}
337
338var dex2oatDepTag = struct {
339 blueprint.BaseDependencyTag
340}{}
341
Martin Stjernholm6d415272020-01-31 17:10:36 +0000342// RegisterToolDeps adds the necessary dependencies to binary modules for tools
343// that are required later when Get(Cached)GlobalSoongConfig is called. It
344// should be called from a mutator that's registered with
345// android.RegistrationContext.FinalDepsMutators.
346func RegisterToolDeps(ctx android.BottomUpMutatorContext) {
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000347 dex2oatBin := dex2oatModuleName(ctx.Config())
348 v := ctx.Config().BuildOSTarget.Variations()
349 ctx.AddFarVariationDependencies(v, dex2oatDepTag, dex2oatBin)
350}
351
352func dex2oatPathFromDep(ctx android.ModuleContext) android.Path {
353 dex2oatBin := dex2oatModuleName(ctx.Config())
354
355 dex2oatModule := ctx.GetDirectDepWithTag(dex2oatBin, dex2oatDepTag)
356 if dex2oatModule == nil {
357 // If this happens there's probably a missing call to AddToolDeps in DepsMutator.
358 panic(fmt.Sprintf("Failed to lookup %s dependency", dex2oatBin))
359 }
360
361 dex2oatPath := dex2oatModule.(android.HostToolProvider).HostToolPath()
362 if !dex2oatPath.Valid() {
363 panic(fmt.Sprintf("Failed to find host tool path in %s", dex2oatModule))
364 }
365
366 return dex2oatPath.Path()
367}
368
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000369// createGlobalSoongConfig creates a GlobalSoongConfig from the current context.
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000370// Should not be used in dexpreopt_gen.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000371func createGlobalSoongConfig(ctx android.ModuleContext) *GlobalSoongConfig {
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000372 if ctx.Config().TestProductVariables != nil {
373 // If we're called in a test there'll be a confusing error from the path
374 // functions below that gets reported without a stack trace, so let's panic
375 // properly with a more helpful message.
376 panic("This should not be called from tests. Please call GlobalSoongConfigForTests somewhere in the test setup.")
377 }
378
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000379 return &GlobalSoongConfig{
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000380 Profman: ctx.Config().HostToolPath(ctx, "profman"),
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000381 Dex2oat: dex2oatPathFromDep(ctx),
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000382 Aapt: ctx.Config().HostToolPath(ctx, "aapt"),
383 SoongZip: ctx.Config().HostToolPath(ctx, "soong_zip"),
384 Zip2zip: ctx.Config().HostToolPath(ctx, "zip2zip"),
385 ManifestCheck: ctx.Config().HostToolPath(ctx, "manifest_check"),
386 ConstructContext: android.PathForSource(ctx, "build/make/core/construct_context.sh"),
387 }
388}
389
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000390// The main reason for this Once cache for GlobalSoongConfig is to make the
391// dex2oat path available to singletons. In ordinary modules we get it through a
392// dex2oatDepTag dependency, but in singletons there's no simple way to do the
393// same thing and ensure the right variant is selected, hence this cache to make
394// the resolved path available to singletons. This means we depend on there
395// being at least one ordinary module with a dex2oatDepTag dependency.
396//
397// TODO(b/147613152): Implement a way to deal with dependencies from singletons,
398// and then possibly remove this cache altogether (but the use in
399// GlobalSoongConfigForTests also needs to be rethought).
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000400var globalSoongConfigOnceKey = android.NewOnceKey("DexpreoptGlobalSoongConfig")
401
402// GetGlobalSoongConfig creates a GlobalSoongConfig the first time it's called,
403// and later returns the same cached instance.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000404func GetGlobalSoongConfig(ctx android.ModuleContext) *GlobalSoongConfig {
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000405 globalSoong := ctx.Config().Once(globalSoongConfigOnceKey, func() interface{} {
406 return createGlobalSoongConfig(ctx)
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000407 }).(*GlobalSoongConfig)
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000408
409 // Always resolve the tool path from the dependency, to ensure that every
410 // module has the dependency added properly.
411 myDex2oat := dex2oatPathFromDep(ctx)
412 if myDex2oat != globalSoong.Dex2oat {
413 panic(fmt.Sprintf("Inconsistent dex2oat path in cached config: expected %s, got %s", globalSoong.Dex2oat, myDex2oat))
414 }
415
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000416 return globalSoong
417}
418
419// GetCachedGlobalSoongConfig returns a cached GlobalSoongConfig created by an
420// earlier GetGlobalSoongConfig call. This function works with any context
421// compatible with a basic PathContext, since it doesn't try to create a
Martin Stjernholm6d415272020-01-31 17:10:36 +0000422// GlobalSoongConfig with the proper paths (which requires a full
423// ModuleContext). If there has been no prior call to GetGlobalSoongConfig, nil
424// is returned.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000425func GetCachedGlobalSoongConfig(ctx android.PathContext) *GlobalSoongConfig {
Martin Stjernholm6d415272020-01-31 17:10:36 +0000426 return ctx.Config().Once(globalSoongConfigOnceKey, func() interface{} {
427 return (*GlobalSoongConfig)(nil)
428 }).(*GlobalSoongConfig)
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000429}
430
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000431type globalJsonSoongConfig struct {
432 Profman string
433 Dex2oat string
434 Aapt string
435 SoongZip string
436 Zip2zip string
437 ManifestCheck string
438 ConstructContext string
439}
440
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000441// ParseGlobalSoongConfig parses the given data assumed to be read from the
442// global dexpreopt_soong.config file into a GlobalSoongConfig struct. It is
443// only used in dexpreopt_gen.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000444func ParseGlobalSoongConfig(ctx android.PathContext, data []byte) (*GlobalSoongConfig, error) {
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000445 var jc globalJsonSoongConfig
446
Colin Cross988414c2020-01-11 01:11:46 +0000447 err := json.Unmarshal(data, &jc)
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000448 if err != nil {
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000449 return &GlobalSoongConfig{}, err
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000450 }
451
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000452 config := &GlobalSoongConfig{
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000453 Profman: constructPath(ctx, jc.Profman),
454 Dex2oat: constructPath(ctx, jc.Dex2oat),
455 Aapt: constructPath(ctx, jc.Aapt),
456 SoongZip: constructPath(ctx, jc.SoongZip),
457 Zip2zip: constructPath(ctx, jc.Zip2zip),
458 ManifestCheck: constructPath(ctx, jc.ManifestCheck),
459 ConstructContext: constructPath(ctx, jc.ConstructContext),
460 }
461
462 return config, nil
463}
464
465func (s *globalSoongConfigSingleton) GenerateBuildActions(ctx android.SingletonContext) {
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000466 if GetGlobalConfig(ctx).DisablePreopt {
467 return
468 }
469
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000470 config := GetCachedGlobalSoongConfig(ctx)
Martin Stjernholm6d415272020-01-31 17:10:36 +0000471 if config == nil {
472 // No module has enabled dexpreopting, so we assume there will be no calls
473 // to dexpreopt_gen.
474 return
475 }
476
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000477 jc := globalJsonSoongConfig{
478 Profman: config.Profman.String(),
479 Dex2oat: config.Dex2oat.String(),
480 Aapt: config.Aapt.String(),
481 SoongZip: config.SoongZip.String(),
482 Zip2zip: config.Zip2zip.String(),
483 ManifestCheck: config.ManifestCheck.String(),
484 ConstructContext: config.ConstructContext.String(),
485 }
486
487 data, err := json.Marshal(jc)
488 if err != nil {
489 ctx.Errorf("failed to JSON marshal GlobalSoongConfig: %v", err)
490 return
491 }
492
493 ctx.Build(pctx, android.BuildParams{
494 Rule: android.WriteFile,
495 Output: android.PathForOutput(ctx, "dexpreopt_soong.config"),
496 Args: map[string]string{
497 "content": string(data),
498 },
499 })
500}
501
502func (s *globalSoongConfigSingleton) MakeVars(ctx android.MakeVarsContext) {
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000503 if GetGlobalConfig(ctx).DisablePreopt {
504 return
505 }
506
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000507 config := GetCachedGlobalSoongConfig(ctx)
Martin Stjernholm6d415272020-01-31 17:10:36 +0000508 if config == nil {
509 return
510 }
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000511
512 ctx.Strict("DEX2OAT", config.Dex2oat.String())
513 ctx.Strict("DEXPREOPT_GEN_DEPS", strings.Join([]string{
514 config.Profman.String(),
515 config.Dex2oat.String(),
516 config.Aapt.String(),
517 config.SoongZip.String(),
518 config.Zip2zip.String(),
519 config.ManifestCheck.String(),
520 config.ConstructContext.String(),
521 }, " "))
522}
523
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000524func GlobalConfigForTests(ctx android.PathContext) *GlobalConfig {
525 return &GlobalConfig{
Colin Cross69f59a32019-02-15 10:39:37 -0800526 DisablePreopt: false,
527 DisablePreoptModules: nil,
528 OnlyPreoptBootImageAndSystemServer: false,
529 HasSystemOther: false,
530 PatternsOnSystemOther: nil,
531 DisableGenerateProfile: false,
532 ProfileDir: "",
533 BootJars: nil,
Roshan Piusccc26ef2019-11-27 09:37:46 -0800534 UpdatableBootJars: nil,
Martin Stjernholmcc4b0ad2019-07-05 22:38:25 +0100535 ArtApexJars: nil,
Colin Cross69f59a32019-02-15 10:39:37 -0800536 SystemServerJars: nil,
537 SystemServerApps: nil,
Roshan Pius9b51a402019-11-21 12:36:53 -0800538 UpdatableSystemServerJars: nil,
Colin Cross69f59a32019-02-15 10:39:37 -0800539 SpeedApps: nil,
540 PreoptFlags: nil,
541 DefaultCompilerFilter: "",
542 SystemServerCompilerFilter: "",
543 GenerateDMFiles: false,
Colin Cross69f59a32019-02-15 10:39:37 -0800544 NoDebugInfo: false,
Mathieu Chartier3f7ddbb2019-04-29 09:33:50 -0700545 DontResolveStartupStrings: false,
Colin Cross69f59a32019-02-15 10:39:37 -0800546 AlwaysSystemServerDebugInfo: false,
547 NeverSystemServerDebugInfo: false,
548 AlwaysOtherDebugInfo: false,
549 NeverOtherDebugInfo: false,
Colin Cross69f59a32019-02-15 10:39:37 -0800550 IsEng: false,
551 SanitizeLite: false,
552 DefaultAppImages: false,
553 Dex2oatXmx: "",
554 Dex2oatXms: "",
555 EmptyDirectory: "empty_dir",
556 CpuVariant: nil,
557 InstructionSetFeatures: nil,
558 DirtyImageObjects: android.OptionalPath{},
Colin Cross69f59a32019-02-15 10:39:37 -0800559 BootImageProfiles: nil,
Colin Cross69f59a32019-02-15 10:39:37 -0800560 BootFlags: "",
561 Dex2oatImageXmx: "",
562 Dex2oatImageXms: "",
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000563 }
564}
565
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000566func GlobalSoongConfigForTests(config android.Config) *GlobalSoongConfig {
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000567 // Install the test GlobalSoongConfig in the Once cache so that later calls to
568 // Get(Cached)GlobalSoongConfig returns it without trying to create a real one.
569 return config.Once(globalSoongConfigOnceKey, func() interface{} {
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000570 return &GlobalSoongConfig{
Colin Cross38b96852019-05-22 10:21:09 -0700571 Profman: android.PathForTesting("profman"),
572 Dex2oat: android.PathForTesting("dex2oat"),
573 Aapt: android.PathForTesting("aapt"),
574 SoongZip: android.PathForTesting("soong_zip"),
575 Zip2zip: android.PathForTesting("zip2zip"),
576 ManifestCheck: android.PathForTesting("manifest_check"),
577 ConstructContext: android.PathForTesting("construct_context.sh"),
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000578 }
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000579 }).(*GlobalSoongConfig)
Colin Cross69f59a32019-02-15 10:39:37 -0800580}