blob: 867ece62b536f703b3af9b04482e67639199f141 [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 {
Ulya Trafimovicha4a1c4e2021-01-15 18:40:04 +000030 DisablePreopt bool // disable preopt for all modules (excluding boot images)
31 DisablePreoptBootImages bool // disable prepot for boot images
32 DisablePreoptModules []string // modules with preopt disabled by product-specific config
Colin Cross43f08db2018-11-12 10:13:39 -080033
34 OnlyPreoptBootImageAndSystemServer bool // only preopt jars in the boot image or system server
35
Vladimir Marko40139d62020-02-06 15:14:29 +000036 UseArtImage bool // use the art image (use other boot class path dex files without image)
37
Colin Cross43f08db2018-11-12 10:13:39 -080038 HasSystemOther bool // store odex files that match PatternsOnSystemOther on the system_other partition
39 PatternsOnSystemOther []string // patterns (using '%' to denote a prefix match) to put odex on the system_other partition
40
Colin Cross69f59a32019-02-15 10:39:37 -080041 DisableGenerateProfile bool // don't generate profiles
42 ProfileDir string // directory to find profiles in
Colin Cross43f08db2018-11-12 10:13:39 -080043
Ulya Trafimovich249386a2020-07-01 14:31:13 +010044 BootJars android.ConfiguredJarList // modules for jars that form the boot class path
45 UpdatableBootJars android.ConfiguredJarList // jars within apex that form the boot class path
Vladimir Markod2ee5322018-12-19 17:57:57 +000046
Ulya Trafimovich249386a2020-07-01 14:31:13 +010047 ArtApexJars android.ConfiguredJarList // modules for jars that are in the ART APEX
Colin Cross800fe132019-02-11 14:21:24 -080048
Ulya Trafimovich249386a2020-07-01 14:31:13 +010049 SystemServerJars []string // jars that form the system server
50 SystemServerApps []string // apps that are loaded into system server
51 UpdatableSystemServerJars android.ConfiguredJarList // jars within apex that are loaded into system server
52 SpeedApps []string // apps that should be speed optimized
Colin Cross43f08db2018-11-12 10:13:39 -080053
Ulya Trafimovichcd3203f2020-03-27 11:30:00 +000054 BrokenSuboptimalOrderOfSystemServerJars bool // if true, sub-optimal order does not cause a build error
55
Colin Cross43f08db2018-11-12 10:13:39 -080056 PreoptFlags []string // global dex2oat flags that should be used if no module-specific dex2oat flags are specified
57
58 DefaultCompilerFilter string // default compiler filter to pass to dex2oat, overridden by --compiler-filter= in module-specific dex2oat flags
59 SystemServerCompilerFilter string // default compiler filter to pass to dex2oat for system server jars
60
Nicolas Geoffrayc1bf7242019-10-18 14:51:38 +010061 GenerateDMFiles bool // generate Dex Metadata files
Colin Cross43f08db2018-11-12 10:13:39 -080062
63 NoDebugInfo bool // don't generate debug info by default
Mathieu Chartier3f7ddbb2019-04-29 09:33:50 -070064 DontResolveStartupStrings bool // don't resolve string literals loaded during application startup.
Colin Cross43f08db2018-11-12 10:13:39 -080065 AlwaysSystemServerDebugInfo bool // always generate mini debug info for system server modules (overrides NoDebugInfo=true)
66 NeverSystemServerDebugInfo bool // never generate mini debug info for system server modules (overrides NoDebugInfo=false)
67 AlwaysOtherDebugInfo bool // always generate mini debug info for non-system server modules (overrides NoDebugInfo=true)
68 NeverOtherDebugInfo bool // never generate mini debug info for non-system server modules (overrides NoDebugInfo=true)
69
Colin Cross43f08db2018-11-12 10:13:39 -080070 IsEng bool // build is a eng variant
71 SanitizeLite bool // build is the second phase of a SANITIZE_LITE build
72
73 DefaultAppImages bool // build app images (TODO: .art files?) by default
74
Colin Cross800fe132019-02-11 14:21:24 -080075 Dex2oatXmx string // max heap size for dex2oat
76 Dex2oatXms string // initial heap size for dex2oat
Colin Cross43f08db2018-11-12 10:13:39 -080077
78 EmptyDirectory string // path to an empty directory
79
Colin Cross74ba9622019-02-11 15:11:14 -080080 CpuVariant map[android.ArchType]string // cpu variant for each architecture
81 InstructionSetFeatures map[android.ArchType]string // instruction set for each architecture
Colin Cross43f08db2018-11-12 10:13:39 -080082
Nicolas Geoffray1086e602021-01-20 14:30:40 +000083 BootImageProfiles android.Paths // path to a boot-image-profile.txt file
84 BootFlags string // extra flags to pass to dex2oat for the boot image
85 Dex2oatImageXmx string // max heap size for dex2oat for the boot image
86 Dex2oatImageXms string // initial heap size for dex2oat for the boot image
Colin Cross43f08db2018-11-12 10:13:39 -080087}
88
Martin Stjernholmc52aaf12020-01-06 23:11:37 +000089// GlobalSoongConfig contains the global config that is generated from Soong,
90// stored in dexpreopt_soong.config.
91type GlobalSoongConfig struct {
92 // Paths to tools possibly used by the generated commands.
93 Profman android.Path
94 Dex2oat android.Path
95 Aapt android.Path
96 SoongZip android.Path
97 Zip2zip android.Path
98 ManifestCheck android.Path
Colin Cross38b96852019-05-22 10:21:09 -070099 ConstructContext android.Path
Colin Cross43f08db2018-11-12 10:13:39 -0800100}
101
102type ModuleConfig struct {
Victor Hsiehd181c8b2019-01-29 13:00:33 -0800103 Name string
104 DexLocation string // dex location on device
Colin Cross69f59a32019-02-15 10:39:37 -0800105 BuildPath android.OutputPath
106 DexPath android.Path
Colin Cross38b96852019-05-22 10:21:09 -0700107 ManifestPath android.Path
Victor Hsiehd181c8b2019-01-29 13:00:33 -0800108 UncompressedDex bool
109 HasApkLibraries bool
110 PreoptFlags []string
Colin Cross43f08db2018-11-12 10:13:39 -0800111
Colin Cross69f59a32019-02-15 10:39:37 -0800112 ProfileClassListing android.OptionalPath
Colin Cross43f08db2018-11-12 10:13:39 -0800113 ProfileIsTextListing bool
Nicolas Geoffraye7102422019-07-24 13:19:29 +0100114 ProfileBootListing android.OptionalPath
Colin Cross43f08db2018-11-12 10:13:39 -0800115
Ulya Trafimovich8cbc5d22020-11-03 15:15:46 +0000116 EnforceUsesLibraries bool
117 ClassLoaderContexts ClassLoaderContextMap
Colin Cross43f08db2018-11-12 10:13:39 -0800118
Ulya Trafimovich4d2eeed2019-11-08 10:54:21 +0000119 Archs []android.ArchType
120 DexPreoptImages []android.Path
121 DexPreoptImagesDeps []android.OutputPaths
122 DexPreoptImageLocations []string
Colin Cross43f08db2018-11-12 10:13:39 -0800123
Colin Cross69f59a32019-02-15 10:39:37 -0800124 PreoptBootClassPathDexFiles android.Paths // file paths of boot class path files
125 PreoptBootClassPathDexLocations []string // virtual locations of boot class path files
Colin Cross800fe132019-02-11 14:21:24 -0800126
Colin Cross43f08db2018-11-12 10:13:39 -0800127 PreoptExtractedApk bool // Overrides OnlyPreoptModules
128
129 NoCreateAppImage bool
130 ForceCreateAppImage bool
131
132 PresignedPrebuilt bool
Colin Cross43f08db2018-11-12 10:13:39 -0800133}
134
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000135type globalSoongConfigSingleton struct{}
136
137var pctx = android.NewPackageContext("android/soong/dexpreopt")
138
139func init() {
140 pctx.Import("android/soong/android")
141 android.RegisterSingletonType("dexpreopt-soong-config", func() android.Singleton {
142 return &globalSoongConfigSingleton{}
143 })
144}
145
Colin Cross69f59a32019-02-15 10:39:37 -0800146func constructPath(ctx android.PathContext, path string) android.Path {
147 buildDirPrefix := ctx.Config().BuildDir() + "/"
148 if path == "" {
149 return nil
150 } else if strings.HasPrefix(path, buildDirPrefix) {
151 return android.PathForOutput(ctx, strings.TrimPrefix(path, buildDirPrefix))
152 } else {
153 return android.PathForSource(ctx, path)
154 }
Colin Cross43f08db2018-11-12 10:13:39 -0800155}
156
Colin Cross69f59a32019-02-15 10:39:37 -0800157func constructPaths(ctx android.PathContext, paths []string) android.Paths {
158 var ret android.Paths
159 for _, path := range paths {
160 ret = append(ret, constructPath(ctx, path))
161 }
162 return ret
Colin Cross43f08db2018-11-12 10:13:39 -0800163}
164
Colin Cross69f59a32019-02-15 10:39:37 -0800165func constructWritablePath(ctx android.PathContext, path string) android.WritablePath {
166 if path == "" {
167 return nil
168 }
169 return constructPath(ctx, path).(android.WritablePath)
170}
171
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000172// ParseGlobalConfig parses the given data assumed to be read from the global
173// dexpreopt.config file into a GlobalConfig struct.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000174func ParseGlobalConfig(ctx android.PathContext, data []byte) (*GlobalConfig, error) {
Colin Cross69f59a32019-02-15 10:39:37 -0800175 type GlobalJSONConfig struct {
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000176 *GlobalConfig
Colin Cross69f59a32019-02-15 10:39:37 -0800177
178 // Copies of entries in GlobalConfig that are not constructable without extra parameters. They will be
179 // used to construct the real value manually below.
Paul Duffin7ccacae2020-10-23 21:14:20 +0100180 BootImageProfiles []string
Colin Cross69f59a32019-02-15 10:39:37 -0800181 }
182
183 config := GlobalJSONConfig{}
Colin Cross988414c2020-01-11 01:11:46 +0000184 err := json.Unmarshal(data, &config)
Colin Cross69f59a32019-02-15 10:39:37 -0800185 if err != nil {
Colin Cross988414c2020-01-11 01:11:46 +0000186 return config.GlobalConfig, err
Colin Cross69f59a32019-02-15 10:39:37 -0800187 }
188
189 // Construct paths that require a PathContext.
Colin Cross69f59a32019-02-15 10:39:37 -0800190 config.GlobalConfig.BootImageProfiles = constructPaths(ctx, config.BootImageProfiles)
191
Colin Cross988414c2020-01-11 01:11:46 +0000192 return config.GlobalConfig, nil
Colin Cross69f59a32019-02-15 10:39:37 -0800193}
194
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000195type globalConfigAndRaw struct {
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000196 global *GlobalConfig
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000197 data []byte
198}
199
200// GetGlobalConfig returns the global dexpreopt.config that's created in the
201// make config phase. It is loaded once the first time it is called for any
202// ctx.Config(), and returns the same data for all future calls with the same
203// ctx.Config(). A value can be inserted for tests using
204// setDexpreoptTestGlobalConfig.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000205func GetGlobalConfig(ctx android.PathContext) *GlobalConfig {
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000206 return getGlobalConfigRaw(ctx).global
207}
208
209// GetGlobalConfigRawData is the same as GetGlobalConfig, except that it returns
210// the literal content of dexpreopt.config.
211func GetGlobalConfigRawData(ctx android.PathContext) []byte {
212 return getGlobalConfigRaw(ctx).data
213}
214
215var globalConfigOnceKey = android.NewOnceKey("DexpreoptGlobalConfig")
216var testGlobalConfigOnceKey = android.NewOnceKey("TestDexpreoptGlobalConfig")
217
218func getGlobalConfigRaw(ctx android.PathContext) globalConfigAndRaw {
219 return ctx.Config().Once(globalConfigOnceKey, func() interface{} {
220 if data, err := ctx.Config().DexpreoptGlobalConfig(ctx); err != nil {
221 panic(err)
222 } else if data != nil {
223 globalConfig, err := ParseGlobalConfig(ctx, data)
224 if err != nil {
225 panic(err)
226 }
227 return globalConfigAndRaw{globalConfig, data}
228 }
229
230 // No global config filename set, see if there is a test config set
231 return ctx.Config().Once(testGlobalConfigOnceKey, func() interface{} {
232 // Nope, return a config with preopting disabled
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000233 return globalConfigAndRaw{&GlobalConfig{
Ulya Trafimovicha4a1c4e2021-01-15 18:40:04 +0000234 DisablePreopt: true,
235 DisablePreoptBootImages: true,
236 DisableGenerateProfile: true,
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000237 }, nil}
238 })
239 }).(globalConfigAndRaw)
240}
241
242// SetTestGlobalConfig sets a GlobalConfig that future calls to GetGlobalConfig
243// will return. It must be called before the first call to GetGlobalConfig for
244// the config.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000245func SetTestGlobalConfig(config android.Config, globalConfig *GlobalConfig) {
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000246 config.Once(testGlobalConfigOnceKey, func() interface{} { return globalConfigAndRaw{globalConfig, nil} })
247}
248
249// ParseModuleConfig parses a per-module dexpreopt.config file into a
250// ModuleConfig struct. It is not used in Soong, which receives a ModuleConfig
251// struct directly from java/dexpreopt.go. It is used in dexpreopt_gen called
252// from Make to read the module dexpreopt.config written in the Make config
253// stage.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000254func ParseModuleConfig(ctx android.PathContext, data []byte) (*ModuleConfig, error) {
Colin Cross69f59a32019-02-15 10:39:37 -0800255 type ModuleJSONConfig struct {
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000256 *ModuleConfig
Colin Cross69f59a32019-02-15 10:39:37 -0800257
258 // Copies of entries in ModuleConfig that are not constructable without extra parameters. They will be
259 // used to construct the real value manually below.
260 BuildPath string
261 DexPath string
Colin Cross38b96852019-05-22 10:21:09 -0700262 ManifestPath string
Colin Cross69f59a32019-02-15 10:39:37 -0800263 ProfileClassListing string
Ulya Trafimovich8cbc5d22020-11-03 15:15:46 +0000264 ClassLoaderContexts jsonClassLoaderContextMap
Colin Cross69f59a32019-02-15 10:39:37 -0800265 DexPreoptImages []string
Ulya Trafimovich4d2eeed2019-11-08 10:54:21 +0000266 DexPreoptImageLocations []string
Colin Cross69f59a32019-02-15 10:39:37 -0800267 PreoptBootClassPathDexFiles []string
Colin Cross69f59a32019-02-15 10:39:37 -0800268 }
269
270 config := ModuleJSONConfig{}
271
Colin Cross988414c2020-01-11 01:11:46 +0000272 err := json.Unmarshal(data, &config)
Colin Cross69f59a32019-02-15 10:39:37 -0800273 if err != nil {
274 return config.ModuleConfig, err
275 }
276
277 // Construct paths that require a PathContext.
278 config.ModuleConfig.BuildPath = constructPath(ctx, config.BuildPath).(android.OutputPath)
279 config.ModuleConfig.DexPath = constructPath(ctx, config.DexPath)
Colin Cross38b96852019-05-22 10:21:09 -0700280 config.ModuleConfig.ManifestPath = constructPath(ctx, config.ManifestPath)
Colin Cross69f59a32019-02-15 10:39:37 -0800281 config.ModuleConfig.ProfileClassListing = android.OptionalPathForPath(constructPath(ctx, config.ProfileClassListing))
Ulya Trafimovich8cbc5d22020-11-03 15:15:46 +0000282 config.ModuleConfig.ClassLoaderContexts = fromJsonClassLoaderContext(ctx, config.ClassLoaderContexts)
Colin Cross69f59a32019-02-15 10:39:37 -0800283 config.ModuleConfig.DexPreoptImages = constructPaths(ctx, config.DexPreoptImages)
Ulya Trafimovich4d2eeed2019-11-08 10:54:21 +0000284 config.ModuleConfig.DexPreoptImageLocations = config.DexPreoptImageLocations
Colin Cross69f59a32019-02-15 10:39:37 -0800285 config.ModuleConfig.PreoptBootClassPathDexFiles = constructPaths(ctx, config.PreoptBootClassPathDexFiles)
Colin Cross69f59a32019-02-15 10:39:37 -0800286
Dan Willemsen0f416782019-06-13 21:44:53 +0000287 // 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 +0000288 config.ModuleConfig.DexPreoptImagesDeps = make([]android.OutputPaths, len(config.ModuleConfig.DexPreoptImages))
Dan Willemsen0f416782019-06-13 21:44:53 +0000289
Colin Cross69f59a32019-02-15 10:39:37 -0800290 return config.ModuleConfig, nil
291}
292
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000293// dex2oatModuleName returns the name of the module to use for the dex2oat host
294// tool. It should be a binary module with public visibility that is compiled
295// and installed for host.
296func dex2oatModuleName(config android.Config) string {
297 // Default to the debug variant of dex2oat to help find bugs.
298 // Set USE_DEX2OAT_DEBUG to false for only building non-debug versions.
299 if config.Getenv("USE_DEX2OAT_DEBUG") == "false" {
300 return "dex2oat"
301 } else {
302 return "dex2oatd"
303 }
304}
305
Ulya Trafimovicha4a1c4e2021-01-15 18:40:04 +0000306var Dex2oatDepTag = struct {
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000307 blueprint.BaseDependencyTag
308}{}
309
Martin Stjernholm6d415272020-01-31 17:10:36 +0000310// RegisterToolDeps adds the necessary dependencies to binary modules for tools
311// that are required later when Get(Cached)GlobalSoongConfig is called. It
312// should be called from a mutator that's registered with
313// android.RegistrationContext.FinalDepsMutators.
314func RegisterToolDeps(ctx android.BottomUpMutatorContext) {
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000315 dex2oatBin := dex2oatModuleName(ctx.Config())
316 v := ctx.Config().BuildOSTarget.Variations()
Ulya Trafimovicha4a1c4e2021-01-15 18:40:04 +0000317 ctx.AddFarVariationDependencies(v, Dex2oatDepTag, dex2oatBin)
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000318}
319
320func dex2oatPathFromDep(ctx android.ModuleContext) android.Path {
321 dex2oatBin := dex2oatModuleName(ctx.Config())
322
Martin Stjernholmc0048622020-08-18 17:37:41 +0100323 // Find the right dex2oat module, trying to follow PrebuiltDepTag from source
324 // to prebuilt if there is one. We wouldn't have to do this if the
325 // prebuilt_postdeps mutator that replaces source deps with prebuilt deps was
326 // run after RegisterToolDeps above, but changing that leads to ordering
327 // problems between mutators (RegisterToolDeps needs to run late to act on
328 // final variants, while prebuilt_postdeps needs to run before many of the
329 // PostDeps mutators, like the APEX mutators). Hence we need to dig out the
330 // prebuilt explicitly here instead.
331 var dex2oatModule android.Module
332 ctx.WalkDeps(func(child, parent android.Module) bool {
Ulya Trafimovicha4a1c4e2021-01-15 18:40:04 +0000333 if parent == ctx.Module() && ctx.OtherModuleDependencyTag(child) == Dex2oatDepTag {
Martin Stjernholmc0048622020-08-18 17:37:41 +0100334 // Found the source module, or prebuilt module that has replaced the source.
335 dex2oatModule = child
336 if p, ok := child.(android.PrebuiltInterface); ok && p.Prebuilt() != nil {
337 return false // If it's the prebuilt we're done.
338 } else {
339 return true // Recurse to check if the source has a prebuilt dependency.
340 }
341 }
342 if parent == dex2oatModule && ctx.OtherModuleDependencyTag(child) == android.PrebuiltDepTag {
343 if p, ok := child.(android.PrebuiltInterface); ok && p.Prebuilt() != nil && p.Prebuilt().UsePrebuilt() {
344 dex2oatModule = child // Found a prebuilt that should be used.
345 }
346 }
347 return false
348 })
349
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000350 if dex2oatModule == nil {
351 // If this happens there's probably a missing call to AddToolDeps in DepsMutator.
352 panic(fmt.Sprintf("Failed to lookup %s dependency", dex2oatBin))
353 }
354
355 dex2oatPath := dex2oatModule.(android.HostToolProvider).HostToolPath()
356 if !dex2oatPath.Valid() {
357 panic(fmt.Sprintf("Failed to find host tool path in %s", dex2oatModule))
358 }
359
360 return dex2oatPath.Path()
361}
362
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000363// createGlobalSoongConfig creates a GlobalSoongConfig from the current context.
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000364// Should not be used in dexpreopt_gen.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000365func createGlobalSoongConfig(ctx android.ModuleContext) *GlobalSoongConfig {
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000366 return &GlobalSoongConfig{
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000367 Profman: ctx.Config().HostToolPath(ctx, "profman"),
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000368 Dex2oat: dex2oatPathFromDep(ctx),
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000369 Aapt: ctx.Config().HostToolPath(ctx, "aapt"),
370 SoongZip: ctx.Config().HostToolPath(ctx, "soong_zip"),
371 Zip2zip: ctx.Config().HostToolPath(ctx, "zip2zip"),
372 ManifestCheck: ctx.Config().HostToolPath(ctx, "manifest_check"),
Ulya Trafimovich5f364b62020-06-30 12:39:01 +0100373 ConstructContext: ctx.Config().HostToolPath(ctx, "construct_context"),
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000374 }
375}
376
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000377// The main reason for this Once cache for GlobalSoongConfig is to make the
378// dex2oat path available to singletons. In ordinary modules we get it through a
Ulya Trafimovicha4a1c4e2021-01-15 18:40:04 +0000379// Dex2oatDepTag dependency, but in singletons there's no simple way to do the
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000380// same thing and ensure the right variant is selected, hence this cache to make
381// the resolved path available to singletons. This means we depend on there
Ulya Trafimovicha4a1c4e2021-01-15 18:40:04 +0000382// being at least one ordinary module with a Dex2oatDepTag dependency.
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000383//
384// TODO(b/147613152): Implement a way to deal with dependencies from singletons,
Paul Duffin9f045242021-01-21 15:05:11 +0000385// and then possibly remove this cache altogether.
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000386var globalSoongConfigOnceKey = android.NewOnceKey("DexpreoptGlobalSoongConfig")
387
388// GetGlobalSoongConfig creates a GlobalSoongConfig the first time it's called,
389// and later returns the same cached instance.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000390func GetGlobalSoongConfig(ctx android.ModuleContext) *GlobalSoongConfig {
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000391 globalSoong := ctx.Config().Once(globalSoongConfigOnceKey, func() interface{} {
392 return createGlobalSoongConfig(ctx)
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000393 }).(*GlobalSoongConfig)
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000394
395 // Always resolve the tool path from the dependency, to ensure that every
396 // module has the dependency added properly.
397 myDex2oat := dex2oatPathFromDep(ctx)
398 if myDex2oat != globalSoong.Dex2oat {
399 panic(fmt.Sprintf("Inconsistent dex2oat path in cached config: expected %s, got %s", globalSoong.Dex2oat, myDex2oat))
400 }
401
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000402 return globalSoong
403}
404
405// GetCachedGlobalSoongConfig returns a cached GlobalSoongConfig created by an
406// earlier GetGlobalSoongConfig call. This function works with any context
407// compatible with a basic PathContext, since it doesn't try to create a
Martin Stjernholm6d415272020-01-31 17:10:36 +0000408// GlobalSoongConfig with the proper paths (which requires a full
409// ModuleContext). If there has been no prior call to GetGlobalSoongConfig, nil
410// is returned.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000411func GetCachedGlobalSoongConfig(ctx android.PathContext) *GlobalSoongConfig {
Martin Stjernholm6d415272020-01-31 17:10:36 +0000412 return ctx.Config().Once(globalSoongConfigOnceKey, func() interface{} {
413 return (*GlobalSoongConfig)(nil)
414 }).(*GlobalSoongConfig)
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000415}
416
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000417type globalJsonSoongConfig struct {
418 Profman string
419 Dex2oat string
420 Aapt string
421 SoongZip string
422 Zip2zip string
423 ManifestCheck string
424 ConstructContext string
425}
426
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000427// ParseGlobalSoongConfig parses the given data assumed to be read from the
428// global dexpreopt_soong.config file into a GlobalSoongConfig struct. It is
429// only used in dexpreopt_gen.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000430func ParseGlobalSoongConfig(ctx android.PathContext, data []byte) (*GlobalSoongConfig, error) {
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000431 var jc globalJsonSoongConfig
432
Colin Cross988414c2020-01-11 01:11:46 +0000433 err := json.Unmarshal(data, &jc)
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000434 if err != nil {
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000435 return &GlobalSoongConfig{}, err
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000436 }
437
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000438 config := &GlobalSoongConfig{
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000439 Profman: constructPath(ctx, jc.Profman),
440 Dex2oat: constructPath(ctx, jc.Dex2oat),
441 Aapt: constructPath(ctx, jc.Aapt),
442 SoongZip: constructPath(ctx, jc.SoongZip),
443 Zip2zip: constructPath(ctx, jc.Zip2zip),
444 ManifestCheck: constructPath(ctx, jc.ManifestCheck),
445 ConstructContext: constructPath(ctx, jc.ConstructContext),
446 }
447
448 return config, nil
449}
450
451func (s *globalSoongConfigSingleton) GenerateBuildActions(ctx android.SingletonContext) {
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000452 if GetGlobalConfig(ctx).DisablePreopt {
453 return
454 }
455
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000456 config := GetCachedGlobalSoongConfig(ctx)
Martin Stjernholm6d415272020-01-31 17:10:36 +0000457 if config == nil {
458 // No module has enabled dexpreopting, so we assume there will be no calls
459 // to dexpreopt_gen.
460 return
461 }
462
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000463 jc := globalJsonSoongConfig{
464 Profman: config.Profman.String(),
465 Dex2oat: config.Dex2oat.String(),
466 Aapt: config.Aapt.String(),
467 SoongZip: config.SoongZip.String(),
468 Zip2zip: config.Zip2zip.String(),
469 ManifestCheck: config.ManifestCheck.String(),
470 ConstructContext: config.ConstructContext.String(),
471 }
472
473 data, err := json.Marshal(jc)
474 if err != nil {
475 ctx.Errorf("failed to JSON marshal GlobalSoongConfig: %v", err)
476 return
477 }
478
Colin Crosscf371cc2020-11-13 11:48:42 -0800479 android.WriteFileRule(ctx, android.PathForOutput(ctx, "dexpreopt_soong.config"), string(data))
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000480}
481
482func (s *globalSoongConfigSingleton) MakeVars(ctx android.MakeVarsContext) {
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000483 if GetGlobalConfig(ctx).DisablePreopt {
484 return
485 }
486
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000487 config := GetCachedGlobalSoongConfig(ctx)
Martin Stjernholm6d415272020-01-31 17:10:36 +0000488 if config == nil {
489 return
490 }
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000491
492 ctx.Strict("DEX2OAT", config.Dex2oat.String())
493 ctx.Strict("DEXPREOPT_GEN_DEPS", strings.Join([]string{
494 config.Profman.String(),
495 config.Dex2oat.String(),
496 config.Aapt.String(),
497 config.SoongZip.String(),
498 config.Zip2zip.String(),
499 config.ManifestCheck.String(),
500 config.ConstructContext.String(),
501 }, " "))
502}
503
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000504func GlobalConfigForTests(ctx android.PathContext) *GlobalConfig {
505 return &GlobalConfig{
Colin Cross69f59a32019-02-15 10:39:37 -0800506 DisablePreopt: false,
507 DisablePreoptModules: nil,
508 OnlyPreoptBootImageAndSystemServer: false,
509 HasSystemOther: false,
510 PatternsOnSystemOther: nil,
511 DisableGenerateProfile: false,
512 ProfileDir: "",
Ulya Trafimovich249386a2020-07-01 14:31:13 +0100513 BootJars: android.EmptyConfiguredJarList(),
514 UpdatableBootJars: android.EmptyConfiguredJarList(),
515 ArtApexJars: android.EmptyConfiguredJarList(),
Colin Cross69f59a32019-02-15 10:39:37 -0800516 SystemServerJars: nil,
517 SystemServerApps: nil,
Ulya Trafimovich249386a2020-07-01 14:31:13 +0100518 UpdatableSystemServerJars: android.EmptyConfiguredJarList(),
Colin Cross69f59a32019-02-15 10:39:37 -0800519 SpeedApps: nil,
520 PreoptFlags: nil,
521 DefaultCompilerFilter: "",
522 SystemServerCompilerFilter: "",
523 GenerateDMFiles: false,
Colin Cross69f59a32019-02-15 10:39:37 -0800524 NoDebugInfo: false,
Mathieu Chartier3f7ddbb2019-04-29 09:33:50 -0700525 DontResolveStartupStrings: false,
Colin Cross69f59a32019-02-15 10:39:37 -0800526 AlwaysSystemServerDebugInfo: false,
527 NeverSystemServerDebugInfo: false,
528 AlwaysOtherDebugInfo: false,
529 NeverOtherDebugInfo: false,
Colin Cross69f59a32019-02-15 10:39:37 -0800530 IsEng: false,
531 SanitizeLite: false,
532 DefaultAppImages: false,
533 Dex2oatXmx: "",
534 Dex2oatXms: "",
535 EmptyDirectory: "empty_dir",
536 CpuVariant: nil,
537 InstructionSetFeatures: nil,
Colin Cross69f59a32019-02-15 10:39:37 -0800538 BootImageProfiles: nil,
Colin Cross69f59a32019-02-15 10:39:37 -0800539 BootFlags: "",
540 Dex2oatImageXmx: "",
541 Dex2oatImageXms: "",
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000542 }
543}
544
Paul Duffin9f045242021-01-21 15:05:11 +0000545func globalSoongConfigForTests() *GlobalSoongConfig {
546 return &GlobalSoongConfig{
547 Profman: android.PathForTesting("profman"),
548 Dex2oat: android.PathForTesting("dex2oat"),
549 Aapt: android.PathForTesting("aapt"),
550 SoongZip: android.PathForTesting("soong_zip"),
551 Zip2zip: android.PathForTesting("zip2zip"),
552 ManifestCheck: android.PathForTesting("manifest_check"),
553 ConstructContext: android.PathForTesting("construct_context"),
554 }
Colin Cross69f59a32019-02-15 10:39:37 -0800555}