blob: 6e4bfee2936d4334e7cdac0b95d7ee336cf1a6cf [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
Ulya Trafimovich249386a2020-07-01 14:31:13 +010043 BootJars android.ConfiguredJarList // modules for jars that form the boot class path
44 UpdatableBootJars android.ConfiguredJarList // jars within apex that form the boot class path
Vladimir Markod2ee5322018-12-19 17:57:57 +000045
Ulya Trafimovich249386a2020-07-01 14:31:13 +010046 ArtApexJars android.ConfiguredJarList // modules for jars that are in the ART APEX
Colin Cross800fe132019-02-11 14:21:24 -080047
Ulya Trafimovich249386a2020-07-01 14:31:13 +010048 SystemServerJars []string // jars that form the system server
49 SystemServerApps []string // apps that are loaded into system server
50 UpdatableSystemServerJars android.ConfiguredJarList // 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 Trafimovich663dc532020-09-10 12:48:53 +0100103// These two libs are added as optional dependencies (<uses-library> with
104// android:required set to false). This is because they haven't existed in pre-P
105// devices, but classes in them were in bootclasspath jars, etc. So making them
106// hard dependencies (android:required=true) would prevent apps from being
107// installed to such legacy devices.
108var OptionalCompatUsesLibs = []string{
109 "android.test.base",
110 "android.test.mock",
111}
112
113var CompatUsesLibs = []string{
114 "org.apache.http.legacy",
115 "android.hidl.base-V1.0-java",
116 "android.hidl.manager-V1.0-java",
117}
118
Ulya Trafimovich31e444e2020-08-14 17:32:16 +0100119const UnknownInstallLibraryPath = "error"
120
Ulya Trafimovichd4bcea42020-06-03 14:57:22 +0100121// LibraryPath contains paths to the library DEX jar on host and on device.
122type LibraryPath struct {
123 Host android.Path
124 Device string
125}
126
127// LibraryPaths is a map from library name to on-host and on-device paths to its DEX jar.
128type LibraryPaths map[string]*LibraryPath
129
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +0100130// Add a new library path to the map, unless a path for this library already exists.
Ulya Trafimovich663dc532020-09-10 12:48:53 +0100131// If necessary, check that the build and install paths exist.
132func (libPaths LibraryPaths) addLibraryPath(ctx android.ModuleContext, lib string,
133 hostPath, installPath android.Path, strict bool) {
134
135 // If missing dependencies are allowed, the build shouldn't fail when a <uses-library> is
136 // not found. However, this is likely to result is disabling dexpreopt, as it won't be
137 // possible to construct class loader context without on-host and on-device library paths.
138 strict = strict && !ctx.Config().AllowMissingDependencies()
139
140 if hostPath == nil && strict {
141 android.ReportPathErrorf(ctx, "unknown build path to <uses-library> '%s'", lib)
142 }
143
144 if installPath == nil {
145 if android.InList(lib, CompatUsesLibs) || android.InList(lib, OptionalCompatUsesLibs) {
146 // Assume that compatibility libraries are installed in /system/framework.
147 installPath = android.PathForModuleInstall(ctx, "framework", lib+".jar")
148 } else if strict {
149 android.ReportPathErrorf(ctx, "unknown install path to <uses-library> '%s'", lib)
150 }
151 }
152
153 // Add a library only if the build and install path to it is known.
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +0100154 if _, present := libPaths[lib]; !present {
Ulya Trafimovich31e444e2020-08-14 17:32:16 +0100155 var devicePath string
156 if installPath != nil {
157 devicePath = android.InstallPathToOnDevicePath(ctx, installPath.(android.InstallPath))
158 } else {
159 // For some stub libraries the only known thing is the name of their implementation
160 // library, but the library itself is unavailable (missing or part of a prebuilt). In
161 // such cases we still need to add the library to <uses-library> tags in the manifest,
162 // but we cannot use if for dexpreopt.
163 devicePath = UnknownInstallLibraryPath
164 }
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +0100165 libPaths[lib] = &LibraryPath{hostPath, devicePath}
Ulya Trafimovich31e444e2020-08-14 17:32:16 +0100166 }
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +0100167}
168
Ulya Trafimovich663dc532020-09-10 12:48:53 +0100169// Add a new library path to the map. Enforce checks that the library paths exist.
170func (libPaths LibraryPaths) AddLibraryPath(ctx android.ModuleContext, lib string, hostPath, installPath android.Path) {
171 libPaths.addLibraryPath(ctx, lib, hostPath, installPath, true)
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +0100172}
173
174// Add a new library path to the map, if the library exists (name is not nil).
Ulya Trafimovich663dc532020-09-10 12:48:53 +0100175// Don't enforce checks that the library paths exist. Some libraries may be missing from the build,
176// but their names still need to be added to <uses-library> tags in the manifest.
177func (libPaths LibraryPaths) MaybeAddLibraryPath(ctx android.ModuleContext, lib *string, hostPath, installPath android.Path) {
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +0100178 if lib != nil {
Ulya Trafimovich663dc532020-09-10 12:48:53 +0100179 libPaths.addLibraryPath(ctx, *lib, hostPath, installPath, false)
Ulya Trafimovichfc24ad32020-08-19 16:32:54 +0100180 }
Ulya Trafimovich31e444e2020-08-14 17:32:16 +0100181}
182
183// Add library paths from the second map to the first map (do not override existing entries).
184func (libPaths LibraryPaths) AddLibraryPaths(otherPaths LibraryPaths) {
185 for lib, path := range otherPaths {
186 if _, present := libPaths[lib]; !present {
187 libPaths[lib] = path
188 }
189 }
190}
191
Colin Cross43f08db2018-11-12 10:13:39 -0800192type ModuleConfig struct {
Victor Hsiehd181c8b2019-01-29 13:00:33 -0800193 Name string
194 DexLocation string // dex location on device
Colin Cross69f59a32019-02-15 10:39:37 -0800195 BuildPath android.OutputPath
196 DexPath android.Path
Colin Cross38b96852019-05-22 10:21:09 -0700197 ManifestPath android.Path
Victor Hsiehd181c8b2019-01-29 13:00:33 -0800198 UncompressedDex bool
199 HasApkLibraries bool
200 PreoptFlags []string
Colin Cross43f08db2018-11-12 10:13:39 -0800201
Colin Cross69f59a32019-02-15 10:39:37 -0800202 ProfileClassListing android.OptionalPath
Colin Cross43f08db2018-11-12 10:13:39 -0800203 ProfileIsTextListing bool
Nicolas Geoffraye7102422019-07-24 13:19:29 +0100204 ProfileBootListing android.OptionalPath
Colin Cross43f08db2018-11-12 10:13:39 -0800205
Ulya Trafimovich6e827482020-06-12 14:32:24 +0100206 EnforceUsesLibraries bool
207 OptionalUsesLibraries []string
208 UsesLibraries []string
209 LibraryPaths LibraryPaths
Colin Cross43f08db2018-11-12 10:13:39 -0800210
Ulya Trafimovich4d2eeed2019-11-08 10:54:21 +0000211 Archs []android.ArchType
212 DexPreoptImages []android.Path
213 DexPreoptImagesDeps []android.OutputPaths
214 DexPreoptImageLocations []string
Colin Cross43f08db2018-11-12 10:13:39 -0800215
Colin Cross69f59a32019-02-15 10:39:37 -0800216 PreoptBootClassPathDexFiles android.Paths // file paths of boot class path files
217 PreoptBootClassPathDexLocations []string // virtual locations of boot class path files
Colin Cross800fe132019-02-11 14:21:24 -0800218
Colin Cross43f08db2018-11-12 10:13:39 -0800219 PreoptExtractedApk bool // Overrides OnlyPreoptModules
220
221 NoCreateAppImage bool
222 ForceCreateAppImage bool
223
224 PresignedPrebuilt bool
Colin Cross43f08db2018-11-12 10:13:39 -0800225}
226
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000227type globalSoongConfigSingleton struct{}
228
229var pctx = android.NewPackageContext("android/soong/dexpreopt")
230
231func init() {
232 pctx.Import("android/soong/android")
233 android.RegisterSingletonType("dexpreopt-soong-config", func() android.Singleton {
234 return &globalSoongConfigSingleton{}
235 })
236}
237
Colin Cross69f59a32019-02-15 10:39:37 -0800238func constructPath(ctx android.PathContext, path string) android.Path {
239 buildDirPrefix := ctx.Config().BuildDir() + "/"
240 if path == "" {
241 return nil
242 } else if strings.HasPrefix(path, buildDirPrefix) {
243 return android.PathForOutput(ctx, strings.TrimPrefix(path, buildDirPrefix))
244 } else {
245 return android.PathForSource(ctx, path)
246 }
Colin Cross43f08db2018-11-12 10:13:39 -0800247}
248
Colin Cross69f59a32019-02-15 10:39:37 -0800249func constructPaths(ctx android.PathContext, paths []string) android.Paths {
250 var ret android.Paths
251 for _, path := range paths {
252 ret = append(ret, constructPath(ctx, path))
253 }
254 return ret
Colin Cross43f08db2018-11-12 10:13:39 -0800255}
256
Colin Cross69f59a32019-02-15 10:39:37 -0800257func constructWritablePath(ctx android.PathContext, path string) android.WritablePath {
258 if path == "" {
259 return nil
260 }
261 return constructPath(ctx, path).(android.WritablePath)
262}
263
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000264// ParseGlobalConfig parses the given data assumed to be read from the global
265// dexpreopt.config file into a GlobalConfig struct.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000266func ParseGlobalConfig(ctx android.PathContext, data []byte) (*GlobalConfig, error) {
Colin Cross69f59a32019-02-15 10:39:37 -0800267 type GlobalJSONConfig struct {
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000268 *GlobalConfig
Colin Cross69f59a32019-02-15 10:39:37 -0800269
270 // Copies of entries in GlobalConfig that are not constructable without extra parameters. They will be
271 // used to construct the real value manually below.
Ulya Trafimovich249386a2020-07-01 14:31:13 +0100272 BootJars []string
273 UpdatableBootJars []string
274 ArtApexJars []string
275 UpdatableSystemServerJars []string
276 DirtyImageObjects string
277 BootImageProfiles []string
Colin Cross69f59a32019-02-15 10:39:37 -0800278 }
279
280 config := GlobalJSONConfig{}
Colin Cross988414c2020-01-11 01:11:46 +0000281 err := json.Unmarshal(data, &config)
Colin Cross69f59a32019-02-15 10:39:37 -0800282 if err != nil {
Colin Cross988414c2020-01-11 01:11:46 +0000283 return config.GlobalConfig, err
Colin Cross69f59a32019-02-15 10:39:37 -0800284 }
285
286 // Construct paths that require a PathContext.
Ulya Trafimovich249386a2020-07-01 14:31:13 +0100287 config.GlobalConfig.BootJars = android.CreateConfiguredJarList(ctx, config.BootJars)
288 config.GlobalConfig.UpdatableBootJars = android.CreateConfiguredJarList(ctx, config.UpdatableBootJars)
289 config.GlobalConfig.ArtApexJars = android.CreateConfiguredJarList(ctx, config.ArtApexJars)
290 config.GlobalConfig.UpdatableSystemServerJars = android.CreateConfiguredJarList(ctx, config.UpdatableSystemServerJars)
Colin Cross69f59a32019-02-15 10:39:37 -0800291 config.GlobalConfig.DirtyImageObjects = android.OptionalPathForPath(constructPath(ctx, config.DirtyImageObjects))
Colin Cross69f59a32019-02-15 10:39:37 -0800292 config.GlobalConfig.BootImageProfiles = constructPaths(ctx, config.BootImageProfiles)
293
Colin Cross988414c2020-01-11 01:11:46 +0000294 return config.GlobalConfig, nil
Colin Cross69f59a32019-02-15 10:39:37 -0800295}
296
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000297type globalConfigAndRaw struct {
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000298 global *GlobalConfig
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000299 data []byte
300}
301
302// GetGlobalConfig returns the global dexpreopt.config that's created in the
303// make config phase. It is loaded once the first time it is called for any
304// ctx.Config(), and returns the same data for all future calls with the same
305// ctx.Config(). A value can be inserted for tests using
306// setDexpreoptTestGlobalConfig.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000307func GetGlobalConfig(ctx android.PathContext) *GlobalConfig {
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000308 return getGlobalConfigRaw(ctx).global
309}
310
311// GetGlobalConfigRawData is the same as GetGlobalConfig, except that it returns
312// the literal content of dexpreopt.config.
313func GetGlobalConfigRawData(ctx android.PathContext) []byte {
314 return getGlobalConfigRaw(ctx).data
315}
316
317var globalConfigOnceKey = android.NewOnceKey("DexpreoptGlobalConfig")
318var testGlobalConfigOnceKey = android.NewOnceKey("TestDexpreoptGlobalConfig")
319
320func getGlobalConfigRaw(ctx android.PathContext) globalConfigAndRaw {
321 return ctx.Config().Once(globalConfigOnceKey, func() interface{} {
322 if data, err := ctx.Config().DexpreoptGlobalConfig(ctx); err != nil {
323 panic(err)
324 } else if data != nil {
325 globalConfig, err := ParseGlobalConfig(ctx, data)
326 if err != nil {
327 panic(err)
328 }
329 return globalConfigAndRaw{globalConfig, data}
330 }
331
332 // No global config filename set, see if there is a test config set
333 return ctx.Config().Once(testGlobalConfigOnceKey, func() interface{} {
334 // Nope, return a config with preopting disabled
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000335 return globalConfigAndRaw{&GlobalConfig{
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000336 DisablePreopt: true,
337 DisableGenerateProfile: true,
338 }, nil}
339 })
340 }).(globalConfigAndRaw)
341}
342
343// SetTestGlobalConfig sets a GlobalConfig that future calls to GetGlobalConfig
344// will return. It must be called before the first call to GetGlobalConfig for
345// the config.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000346func SetTestGlobalConfig(config android.Config, globalConfig *GlobalConfig) {
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000347 config.Once(testGlobalConfigOnceKey, func() interface{} { return globalConfigAndRaw{globalConfig, nil} })
348}
349
350// ParseModuleConfig parses a per-module dexpreopt.config file into a
351// ModuleConfig struct. It is not used in Soong, which receives a ModuleConfig
352// struct directly from java/dexpreopt.go. It is used in dexpreopt_gen called
353// from Make to read the module dexpreopt.config written in the Make config
354// stage.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000355func ParseModuleConfig(ctx android.PathContext, data []byte) (*ModuleConfig, error) {
Ulya Trafimovichd4bcea42020-06-03 14:57:22 +0100356 type jsonLibraryPath struct {
357 Host string
358 Device string
359 }
360
361 type jsonLibraryPaths map[string]jsonLibraryPath
362
Colin Cross69f59a32019-02-15 10:39:37 -0800363 type ModuleJSONConfig struct {
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000364 *ModuleConfig
Colin Cross69f59a32019-02-15 10:39:37 -0800365
366 // Copies of entries in ModuleConfig that are not constructable without extra parameters. They will be
367 // used to construct the real value manually below.
368 BuildPath string
369 DexPath string
Colin Cross38b96852019-05-22 10:21:09 -0700370 ManifestPath string
Colin Cross69f59a32019-02-15 10:39:37 -0800371 ProfileClassListing string
Ulya Trafimovichd4bcea42020-06-03 14:57:22 +0100372 LibraryPaths jsonLibraryPaths
Colin Cross69f59a32019-02-15 10:39:37 -0800373 DexPreoptImages []string
Ulya Trafimovich4d2eeed2019-11-08 10:54:21 +0000374 DexPreoptImageLocations []string
Colin Cross69f59a32019-02-15 10:39:37 -0800375 PreoptBootClassPathDexFiles []string
Colin Cross69f59a32019-02-15 10:39:37 -0800376 }
377
Ulya Trafimovichd4bcea42020-06-03 14:57:22 +0100378 // convert JSON map of library paths to LibraryPaths
379 constructLibraryPaths := func(ctx android.PathContext, paths jsonLibraryPaths) LibraryPaths {
380 m := LibraryPaths{}
381 for lib, path := range paths {
382 m[lib] = &LibraryPath{
383 constructPath(ctx, path.Host),
384 path.Device,
385 }
386 }
387 return m
388 }
389
Colin Cross69f59a32019-02-15 10:39:37 -0800390 config := ModuleJSONConfig{}
391
Colin Cross988414c2020-01-11 01:11:46 +0000392 err := json.Unmarshal(data, &config)
Colin Cross69f59a32019-02-15 10:39:37 -0800393 if err != nil {
394 return config.ModuleConfig, err
395 }
396
397 // Construct paths that require a PathContext.
398 config.ModuleConfig.BuildPath = constructPath(ctx, config.BuildPath).(android.OutputPath)
399 config.ModuleConfig.DexPath = constructPath(ctx, config.DexPath)
Colin Cross38b96852019-05-22 10:21:09 -0700400 config.ModuleConfig.ManifestPath = constructPath(ctx, config.ManifestPath)
Colin Cross69f59a32019-02-15 10:39:37 -0800401 config.ModuleConfig.ProfileClassListing = android.OptionalPathForPath(constructPath(ctx, config.ProfileClassListing))
Ulya Trafimovichd4bcea42020-06-03 14:57:22 +0100402 config.ModuleConfig.LibraryPaths = constructLibraryPaths(ctx, config.LibraryPaths)
Colin Cross69f59a32019-02-15 10:39:37 -0800403 config.ModuleConfig.DexPreoptImages = constructPaths(ctx, config.DexPreoptImages)
Ulya Trafimovich4d2eeed2019-11-08 10:54:21 +0000404 config.ModuleConfig.DexPreoptImageLocations = config.DexPreoptImageLocations
Colin Cross69f59a32019-02-15 10:39:37 -0800405 config.ModuleConfig.PreoptBootClassPathDexFiles = constructPaths(ctx, config.PreoptBootClassPathDexFiles)
Colin Cross69f59a32019-02-15 10:39:37 -0800406
Dan Willemsen0f416782019-06-13 21:44:53 +0000407 // 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 +0000408 config.ModuleConfig.DexPreoptImagesDeps = make([]android.OutputPaths, len(config.ModuleConfig.DexPreoptImages))
Dan Willemsen0f416782019-06-13 21:44:53 +0000409
Colin Cross69f59a32019-02-15 10:39:37 -0800410 return config.ModuleConfig, nil
411}
412
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000413// dex2oatModuleName returns the name of the module to use for the dex2oat host
414// tool. It should be a binary module with public visibility that is compiled
415// and installed for host.
416func dex2oatModuleName(config android.Config) string {
417 // Default to the debug variant of dex2oat to help find bugs.
418 // Set USE_DEX2OAT_DEBUG to false for only building non-debug versions.
419 if config.Getenv("USE_DEX2OAT_DEBUG") == "false" {
420 return "dex2oat"
421 } else {
422 return "dex2oatd"
423 }
424}
425
426var dex2oatDepTag = struct {
427 blueprint.BaseDependencyTag
428}{}
429
Martin Stjernholm6d415272020-01-31 17:10:36 +0000430// RegisterToolDeps adds the necessary dependencies to binary modules for tools
431// that are required later when Get(Cached)GlobalSoongConfig is called. It
432// should be called from a mutator that's registered with
433// android.RegistrationContext.FinalDepsMutators.
434func RegisterToolDeps(ctx android.BottomUpMutatorContext) {
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000435 dex2oatBin := dex2oatModuleName(ctx.Config())
436 v := ctx.Config().BuildOSTarget.Variations()
437 ctx.AddFarVariationDependencies(v, dex2oatDepTag, dex2oatBin)
438}
439
440func dex2oatPathFromDep(ctx android.ModuleContext) android.Path {
441 dex2oatBin := dex2oatModuleName(ctx.Config())
442
Martin Stjernholmc0048622020-08-18 17:37:41 +0100443 // Find the right dex2oat module, trying to follow PrebuiltDepTag from source
444 // to prebuilt if there is one. We wouldn't have to do this if the
445 // prebuilt_postdeps mutator that replaces source deps with prebuilt deps was
446 // run after RegisterToolDeps above, but changing that leads to ordering
447 // problems between mutators (RegisterToolDeps needs to run late to act on
448 // final variants, while prebuilt_postdeps needs to run before many of the
449 // PostDeps mutators, like the APEX mutators). Hence we need to dig out the
450 // prebuilt explicitly here instead.
451 var dex2oatModule android.Module
452 ctx.WalkDeps(func(child, parent android.Module) bool {
453 if parent == ctx.Module() && ctx.OtherModuleDependencyTag(child) == dex2oatDepTag {
454 // Found the source module, or prebuilt module that has replaced the source.
455 dex2oatModule = child
456 if p, ok := child.(android.PrebuiltInterface); ok && p.Prebuilt() != nil {
457 return false // If it's the prebuilt we're done.
458 } else {
459 return true // Recurse to check if the source has a prebuilt dependency.
460 }
461 }
462 if parent == dex2oatModule && ctx.OtherModuleDependencyTag(child) == android.PrebuiltDepTag {
463 if p, ok := child.(android.PrebuiltInterface); ok && p.Prebuilt() != nil && p.Prebuilt().UsePrebuilt() {
464 dex2oatModule = child // Found a prebuilt that should be used.
465 }
466 }
467 return false
468 })
469
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000470 if dex2oatModule == nil {
471 // If this happens there's probably a missing call to AddToolDeps in DepsMutator.
472 panic(fmt.Sprintf("Failed to lookup %s dependency", dex2oatBin))
473 }
474
475 dex2oatPath := dex2oatModule.(android.HostToolProvider).HostToolPath()
476 if !dex2oatPath.Valid() {
477 panic(fmt.Sprintf("Failed to find host tool path in %s", dex2oatModule))
478 }
479
480 return dex2oatPath.Path()
481}
482
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000483// createGlobalSoongConfig creates a GlobalSoongConfig from the current context.
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000484// Should not be used in dexpreopt_gen.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000485func createGlobalSoongConfig(ctx android.ModuleContext) *GlobalSoongConfig {
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000486 if ctx.Config().TestProductVariables != nil {
487 // If we're called in a test there'll be a confusing error from the path
488 // functions below that gets reported without a stack trace, so let's panic
489 // properly with a more helpful message.
490 panic("This should not be called from tests. Please call GlobalSoongConfigForTests somewhere in the test setup.")
491 }
492
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000493 return &GlobalSoongConfig{
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000494 Profman: ctx.Config().HostToolPath(ctx, "profman"),
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000495 Dex2oat: dex2oatPathFromDep(ctx),
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000496 Aapt: ctx.Config().HostToolPath(ctx, "aapt"),
497 SoongZip: ctx.Config().HostToolPath(ctx, "soong_zip"),
498 Zip2zip: ctx.Config().HostToolPath(ctx, "zip2zip"),
499 ManifestCheck: ctx.Config().HostToolPath(ctx, "manifest_check"),
Ulya Trafimovich5f364b62020-06-30 12:39:01 +0100500 ConstructContext: ctx.Config().HostToolPath(ctx, "construct_context"),
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000501 }
502}
503
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000504// The main reason for this Once cache for GlobalSoongConfig is to make the
505// dex2oat path available to singletons. In ordinary modules we get it through a
506// dex2oatDepTag dependency, but in singletons there's no simple way to do the
507// same thing and ensure the right variant is selected, hence this cache to make
508// the resolved path available to singletons. This means we depend on there
509// being at least one ordinary module with a dex2oatDepTag dependency.
510//
511// TODO(b/147613152): Implement a way to deal with dependencies from singletons,
512// and then possibly remove this cache altogether (but the use in
513// GlobalSoongConfigForTests also needs to be rethought).
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000514var globalSoongConfigOnceKey = android.NewOnceKey("DexpreoptGlobalSoongConfig")
515
516// GetGlobalSoongConfig creates a GlobalSoongConfig the first time it's called,
517// and later returns the same cached instance.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000518func GetGlobalSoongConfig(ctx android.ModuleContext) *GlobalSoongConfig {
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000519 globalSoong := ctx.Config().Once(globalSoongConfigOnceKey, func() interface{} {
520 return createGlobalSoongConfig(ctx)
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000521 }).(*GlobalSoongConfig)
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000522
523 // Always resolve the tool path from the dependency, to ensure that every
524 // module has the dependency added properly.
525 myDex2oat := dex2oatPathFromDep(ctx)
526 if myDex2oat != globalSoong.Dex2oat {
527 panic(fmt.Sprintf("Inconsistent dex2oat path in cached config: expected %s, got %s", globalSoong.Dex2oat, myDex2oat))
528 }
529
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000530 return globalSoong
531}
532
533// GetCachedGlobalSoongConfig returns a cached GlobalSoongConfig created by an
534// earlier GetGlobalSoongConfig call. This function works with any context
535// compatible with a basic PathContext, since it doesn't try to create a
Martin Stjernholm6d415272020-01-31 17:10:36 +0000536// GlobalSoongConfig with the proper paths (which requires a full
537// ModuleContext). If there has been no prior call to GetGlobalSoongConfig, nil
538// is returned.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000539func GetCachedGlobalSoongConfig(ctx android.PathContext) *GlobalSoongConfig {
Martin Stjernholm6d415272020-01-31 17:10:36 +0000540 return ctx.Config().Once(globalSoongConfigOnceKey, func() interface{} {
541 return (*GlobalSoongConfig)(nil)
542 }).(*GlobalSoongConfig)
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000543}
544
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000545type globalJsonSoongConfig struct {
546 Profman string
547 Dex2oat string
548 Aapt string
549 SoongZip string
550 Zip2zip string
551 ManifestCheck string
552 ConstructContext string
553}
554
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000555// ParseGlobalSoongConfig parses the given data assumed to be read from the
556// global dexpreopt_soong.config file into a GlobalSoongConfig struct. It is
557// only used in dexpreopt_gen.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000558func ParseGlobalSoongConfig(ctx android.PathContext, data []byte) (*GlobalSoongConfig, error) {
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000559 var jc globalJsonSoongConfig
560
Colin Cross988414c2020-01-11 01:11:46 +0000561 err := json.Unmarshal(data, &jc)
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000562 if err != nil {
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000563 return &GlobalSoongConfig{}, err
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000564 }
565
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000566 config := &GlobalSoongConfig{
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000567 Profman: constructPath(ctx, jc.Profman),
568 Dex2oat: constructPath(ctx, jc.Dex2oat),
569 Aapt: constructPath(ctx, jc.Aapt),
570 SoongZip: constructPath(ctx, jc.SoongZip),
571 Zip2zip: constructPath(ctx, jc.Zip2zip),
572 ManifestCheck: constructPath(ctx, jc.ManifestCheck),
573 ConstructContext: constructPath(ctx, jc.ConstructContext),
574 }
575
576 return config, nil
577}
578
579func (s *globalSoongConfigSingleton) GenerateBuildActions(ctx android.SingletonContext) {
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000580 if GetGlobalConfig(ctx).DisablePreopt {
581 return
582 }
583
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000584 config := GetCachedGlobalSoongConfig(ctx)
Martin Stjernholm6d415272020-01-31 17:10:36 +0000585 if config == nil {
586 // No module has enabled dexpreopting, so we assume there will be no calls
587 // to dexpreopt_gen.
588 return
589 }
590
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000591 jc := globalJsonSoongConfig{
592 Profman: config.Profman.String(),
593 Dex2oat: config.Dex2oat.String(),
594 Aapt: config.Aapt.String(),
595 SoongZip: config.SoongZip.String(),
596 Zip2zip: config.Zip2zip.String(),
597 ManifestCheck: config.ManifestCheck.String(),
598 ConstructContext: config.ConstructContext.String(),
599 }
600
601 data, err := json.Marshal(jc)
602 if err != nil {
603 ctx.Errorf("failed to JSON marshal GlobalSoongConfig: %v", err)
604 return
605 }
606
607 ctx.Build(pctx, android.BuildParams{
608 Rule: android.WriteFile,
609 Output: android.PathForOutput(ctx, "dexpreopt_soong.config"),
610 Args: map[string]string{
611 "content": string(data),
612 },
613 })
614}
615
616func (s *globalSoongConfigSingleton) MakeVars(ctx android.MakeVarsContext) {
Martin Stjernholmd90676f2020-01-11 00:37:30 +0000617 if GetGlobalConfig(ctx).DisablePreopt {
618 return
619 }
620
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000621 config := GetCachedGlobalSoongConfig(ctx)
Martin Stjernholm6d415272020-01-31 17:10:36 +0000622 if config == nil {
623 return
624 }
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000625
626 ctx.Strict("DEX2OAT", config.Dex2oat.String())
627 ctx.Strict("DEXPREOPT_GEN_DEPS", strings.Join([]string{
628 config.Profman.String(),
629 config.Dex2oat.String(),
630 config.Aapt.String(),
631 config.SoongZip.String(),
632 config.Zip2zip.String(),
633 config.ManifestCheck.String(),
634 config.ConstructContext.String(),
635 }, " "))
636}
637
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000638func GlobalConfigForTests(ctx android.PathContext) *GlobalConfig {
639 return &GlobalConfig{
Colin Cross69f59a32019-02-15 10:39:37 -0800640 DisablePreopt: false,
641 DisablePreoptModules: nil,
642 OnlyPreoptBootImageAndSystemServer: false,
643 HasSystemOther: false,
644 PatternsOnSystemOther: nil,
645 DisableGenerateProfile: false,
646 ProfileDir: "",
Ulya Trafimovich249386a2020-07-01 14:31:13 +0100647 BootJars: android.EmptyConfiguredJarList(),
648 UpdatableBootJars: android.EmptyConfiguredJarList(),
649 ArtApexJars: android.EmptyConfiguredJarList(),
Colin Cross69f59a32019-02-15 10:39:37 -0800650 SystemServerJars: nil,
651 SystemServerApps: nil,
Ulya Trafimovich249386a2020-07-01 14:31:13 +0100652 UpdatableSystemServerJars: android.EmptyConfiguredJarList(),
Colin Cross69f59a32019-02-15 10:39:37 -0800653 SpeedApps: nil,
654 PreoptFlags: nil,
655 DefaultCompilerFilter: "",
656 SystemServerCompilerFilter: "",
657 GenerateDMFiles: false,
Colin Cross69f59a32019-02-15 10:39:37 -0800658 NoDebugInfo: false,
Mathieu Chartier3f7ddbb2019-04-29 09:33:50 -0700659 DontResolveStartupStrings: false,
Colin Cross69f59a32019-02-15 10:39:37 -0800660 AlwaysSystemServerDebugInfo: false,
661 NeverSystemServerDebugInfo: false,
662 AlwaysOtherDebugInfo: false,
663 NeverOtherDebugInfo: false,
Colin Cross69f59a32019-02-15 10:39:37 -0800664 IsEng: false,
665 SanitizeLite: false,
666 DefaultAppImages: false,
667 Dex2oatXmx: "",
668 Dex2oatXms: "",
669 EmptyDirectory: "empty_dir",
670 CpuVariant: nil,
671 InstructionSetFeatures: nil,
672 DirtyImageObjects: android.OptionalPath{},
Colin Cross69f59a32019-02-15 10:39:37 -0800673 BootImageProfiles: nil,
Colin Cross69f59a32019-02-15 10:39:37 -0800674 BootFlags: "",
675 Dex2oatImageXmx: "",
676 Dex2oatImageXms: "",
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000677 }
678}
679
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000680func GlobalSoongConfigForTests(config android.Config) *GlobalSoongConfig {
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000681 // Install the test GlobalSoongConfig in the Once cache so that later calls to
682 // Get(Cached)GlobalSoongConfig returns it without trying to create a real one.
683 return config.Once(globalSoongConfigOnceKey, func() interface{} {
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000684 return &GlobalSoongConfig{
Colin Cross38b96852019-05-22 10:21:09 -0700685 Profman: android.PathForTesting("profman"),
686 Dex2oat: android.PathForTesting("dex2oat"),
687 Aapt: android.PathForTesting("aapt"),
688 SoongZip: android.PathForTesting("soong_zip"),
689 Zip2zip: android.PathForTesting("zip2zip"),
690 ManifestCheck: android.PathForTesting("manifest_check"),
Ulya Trafimovich5f364b62020-06-30 12:39:01 +0100691 ConstructContext: android.PathForTesting("construct_context"),
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000692 }
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000693 }).(*GlobalSoongConfig)
Colin Cross69f59a32019-02-15 10:39:37 -0800694}