blob: 38ed856ee09ce89bf452defcf533790bd6c749b1 [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 java
16
17import (
Jiakai Zhangca9bc982021-09-09 08:09:41 +000018 "path/filepath"
Jiakai Zhang51b2a8b2023-06-26 16:47:38 +010019 "sort"
Jiakai Zhangca9bc982021-09-09 08:09:41 +000020 "strings"
21
Colin Cross43f08db2018-11-12 10:13:39 -080022 "android/soong/android"
23 "android/soong/dexpreopt"
24)
25
Jiakai Zhangca9bc982021-09-09 08:09:41 +000026type DexpreopterInterface interface {
Jiakai Zhang81e46812023-02-08 21:56:07 +080027 // True if the java module is to be dexed and installed on devices.
28 // Structs that embed dexpreopter must implement this.
29 IsInstallable() bool
30
31 // True if dexpreopt is disabled for the java module.
Spandan Dase21a8d42024-01-23 23:56:29 +000032 dexpreoptDisabled(ctx android.BaseModuleContext, libraryName string) bool
Jiakai Zhang81e46812023-02-08 21:56:07 +080033
34 // If the java module is to be installed into an APEX, this list contains information about the
35 // dexpreopt outputs to be installed on devices. Note that these dexpreopt outputs are installed
36 // outside of the APEX.
Jiakai Zhangca9bc982021-09-09 08:09:41 +000037 DexpreoptBuiltInstalledForApex() []dexpreopterInstall
Jiakai Zhang81e46812023-02-08 21:56:07 +080038
39 // The Make entries to install the dexpreopt outputs. Derived from
40 // `DexpreoptBuiltInstalledForApex`.
Jiakai Zhangca9bc982021-09-09 08:09:41 +000041 AndroidMkEntriesForApex() []android.AndroidMkEntries
Jiakai Zhang81e46812023-02-08 21:56:07 +080042
43 // See `dexpreopter.outputProfilePathOnHost`.
44 OutputProfilePathOnHost() android.Path
Jiakai Zhangca9bc982021-09-09 08:09:41 +000045}
46
47type dexpreopterInstall struct {
48 // A unique name to distinguish an output from others for the same java library module. Usually in
49 // the form of `<arch>-<encoded-path>.odex/vdex/art`.
50 name string
51
52 // The name of the input java module.
53 moduleName string
54
55 // The path to the dexpreopt output on host.
56 outputPathOnHost android.Path
57
58 // The directory on the device for the output to install to.
59 installDirOnDevice android.InstallPath
60
61 // The basename (the last segment of the path) for the output to install as.
62 installFileOnDevice string
63}
64
65// The full module name of the output in the makefile.
66func (install *dexpreopterInstall) FullModuleName() string {
67 return install.moduleName + install.SubModuleName()
68}
69
70// The sub-module name of the output in the makefile (the name excluding the java module name).
71func (install *dexpreopterInstall) SubModuleName() string {
72 return "-dexpreopt-" + install.name
Martin Stjernholm6d415272020-01-31 17:10:36 +000073}
74
Jiakai Zhang6decef92022-01-12 17:56:19 +000075// Returns Make entries for installing the file.
76//
77// This function uses a value receiver rather than a pointer receiver to ensure that the object is
78// safe to use in `android.AndroidMkExtraEntriesFunc`.
79func (install dexpreopterInstall) ToMakeEntries() android.AndroidMkEntries {
80 return android.AndroidMkEntries{
81 Class: "ETC",
Jiakai Zhang6decef92022-01-12 17:56:19 +000082 OutputFile: android.OptionalPathForPath(install.outputPathOnHost),
83 ExtraEntries: []android.AndroidMkExtraEntriesFunc{
84 func(ctx android.AndroidMkExtraEntriesContext, entries *android.AndroidMkEntries) {
Spandan Das2069c3f2023-12-06 19:40:24 +000085 entries.SetString("LOCAL_MODULE", install.FullModuleName())
Jiakai Zhang6decef92022-01-12 17:56:19 +000086 entries.SetString("LOCAL_MODULE_PATH", install.installDirOnDevice.String())
87 entries.SetString("LOCAL_INSTALLED_MODULE_STEM", install.installFileOnDevice)
88 entries.SetString("LOCAL_NOT_AVAILABLE_FOR_PLATFORM", "false")
Spandan Das2069c3f2023-12-06 19:40:24 +000089 // Unset LOCAL_SOONG_INSTALLED_MODULE so that this does not default to the primary .apex file
90 // Without this, installation of the dexpreopt artifacts get skipped
91 entries.SetString("LOCAL_SOONG_INSTALLED_MODULE", "")
Jiakai Zhang6decef92022-01-12 17:56:19 +000092 },
93 },
94 }
95}
96
Spandan Das2069c3f2023-12-06 19:40:24 +000097type Dexpreopter struct {
98 dexpreopter
99}
100
Colin Cross43f08db2018-11-12 10:13:39 -0800101type dexpreopter struct {
Jiakai Zhang9c4dc192023-02-09 00:09:24 +0800102 dexpreoptProperties DexpreoptProperties
103 importDexpreoptProperties ImportDexpreoptProperties
Colin Cross43f08db2018-11-12 10:13:39 -0800104
Spandan Das0727ba72024-02-13 16:37:43 +0000105 // If true, the dexpreopt rules will not be generated
106 // Unlike Dex_preopt.Enabled which is user-facing,
107 // shouldDisableDexpreopt is a mutated propery.
108 shouldDisableDexpreopt bool
109
Colin Cross70dda7e2019-10-01 22:05:35 -0700110 installPath android.InstallPath
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700111 uncompressedDex bool
112 isSDKLibrary bool
Ulya Trafimovich76b08522021-01-14 17:52:43 +0000113 isApp bool
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700114 isTest bool
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700115 isPresignedPrebuilt bool
Colin Crossfa9bfcd2021-11-10 16:42:38 -0800116 preventInstall bool
Colin Cross43f08db2018-11-12 10:13:39 -0800117
Ulya Trafimovich8cbc5d22020-11-03 15:15:46 +0000118 manifestFile android.Path
Ulya Trafimovich8c35fcf2021-02-17 16:23:28 +0000119 statusFile android.WritablePath
Ulya Trafimovich8cbc5d22020-11-03 15:15:46 +0000120 enforceUsesLibs bool
121 classLoaderContexts dexpreopt.ClassLoaderContextMap
Colin Cross50ddcc42019-05-16 12:28:22 -0700122
Jiakai Zhangca9bc982021-09-09 08:09:41 +0000123 // See the `dexpreopt` function for details.
124 builtInstalled string
125 builtInstalledForApex []dexpreopterInstall
Ulya Trafimovich76b08522021-01-14 17:52:43 +0000126
Jeongik Chac6246672021-04-08 00:00:19 +0900127 // The config is used for two purposes:
128 // - Passing dexpreopt information about libraries from Soong to Make. This is needed when
129 // a <uses-library> is defined in Android.bp, but used in Android.mk (see dex_preopt_config_merger.py).
130 // Note that dexpreopt.config might be needed even if dexpreopt is disabled for the library itself.
131 // - Dexpreopt post-processing (using dexpreopt artifacts from a prebuilt system image to incrementally
132 // dexpreopt another partition).
Ulya Trafimovich76b08522021-01-14 17:52:43 +0000133 configPath android.WritablePath
Jiakai Zhang3317ce72023-02-08 01:19:19 +0800134
Jiakai Zhang81e46812023-02-08 21:56:07 +0800135 // The path to the profile on host that dexpreopter generates. This is used as the input for
136 // dex2oat.
137 outputProfilePathOnHost android.Path
138
139 // The path to the profile that dexpreopter accepts. It must be in the binary format. If this is
140 // set, it overrides the profile settings in `dexpreoptProperties`.
141 inputProfilePathOnHost android.Path
Colin Cross43f08db2018-11-12 10:13:39 -0800142}
143
144type DexpreoptProperties struct {
145 Dex_preopt struct {
Nicolas Geoffrayc1bf7242019-10-18 14:51:38 +0100146 // If false, prevent dexpreopting. Defaults to true.
Colin Cross43f08db2018-11-12 10:13:39 -0800147 Enabled *bool
148
149 // If true, generate an app image (.art file) for this module.
150 App_image *bool
151
152 // If true, use a checked-in profile to guide optimization. Defaults to false unless
153 // a matching profile is set or a profile is found in PRODUCT_DEX_PREOPT_PROFILE_DIR
154 // that matches the name of this module, in which case it is defaulted to true.
155 Profile_guided *bool
156
157 // If set, provides the path to profile relative to the Android.bp file. If not set,
158 // defaults to searching for a file that matches the name of this module in the default
159 // profile location set by PRODUCT_DEX_PREOPT_PROFILE_DIR, or empty if not found.
Colin Crossde4e4e62019-04-26 10:52:32 -0700160 Profile *string `android:"path"`
Colin Cross43f08db2018-11-12 10:13:39 -0800161 }
Jiakai Zhang9c4dc192023-02-09 00:09:24 +0800162
163 Dex_preopt_result struct {
164 // True if profile-guided optimization is actually enabled.
165 Profile_guided bool
166 } `blueprint:"mutated"`
167}
168
169type ImportDexpreoptProperties struct {
170 Dex_preopt struct {
171 // If true, use the profile in the prebuilt APEX to guide optimization. Defaults to false.
172 Profile_guided *bool
173 }
Colin Cross43f08db2018-11-12 10:13:39 -0800174}
175
Ulya Trafimovich6cf2c0c2020-04-24 12:15:20 +0100176func init() {
177 dexpreopt.DexpreoptRunningInSoong = true
178}
179
Jiakai Zhangca9bc982021-09-09 08:09:41 +0000180func isApexVariant(ctx android.BaseModuleContext) bool {
Colin Crossff694a82023-12-13 15:54:49 -0800181 apexInfo, _ := android.ModuleProvider(ctx, android.ApexInfoProvider)
Jiakai Zhangca9bc982021-09-09 08:09:41 +0000182 return !apexInfo.IsForPlatform()
183}
184
Jiakai Zhang28bc9a82021-12-20 15:08:57 +0000185func forPrebuiltApex(ctx android.BaseModuleContext) bool {
Colin Crossff694a82023-12-13 15:54:49 -0800186 apexInfo, _ := android.ModuleProvider(ctx, android.ApexInfoProvider)
Jiakai Zhang28bc9a82021-12-20 15:08:57 +0000187 return apexInfo.ForPrebuiltApex
188}
189
Spandan Dasa8afdcb2024-02-29 06:40:16 +0000190// For apex variant of modules, this returns true on the source variant if the prebuilt apex
191// has been selected using apex_contributions.
192// The prebuilt apex will be responsible for generating the dexpreopt rules of the deapexed java lib.
193func disableSourceApexVariant(ctx android.BaseModuleContext) bool {
194 if !isApexVariant(ctx) {
195 return false // platform variant
196 }
197 apexInfo, _ := android.ModuleProvider(ctx, android.ApexInfoProvider)
198 psi := android.PrebuiltSelectionInfoMap{}
199 ctx.VisitDirectDepsWithTag(android.PrebuiltDepTag, func(am android.Module) {
200 psi, _ = android.OtherModuleProvider(ctx, am, android.PrebuiltSelectionInfoProvider)
201 })
202 // Find the apex variant for this module
203 _, apexVariantsWithoutTestApexes, _ := android.ListSetDifference(apexInfo.InApexVariants, apexInfo.TestApexes)
204 disableSource := false
205 // find the selected apexes
206 for _, apexVariant := range apexVariantsWithoutTestApexes {
207 for _, selected := range psi.GetSelectedModulesForApiDomain(apexVariant) {
208 // If the apex_contribution for this api domain contains a prebuilt apex, disable the source variant
209 if strings.HasPrefix(selected, "prebuilt_com.google.android") {
210 disableSource = true
211 }
212 }
213 }
214 return disableSource
215}
216
Jiakai Zhangcf61e3c2023-05-08 16:28:38 +0000217// Returns whether dexpreopt is applicable to the module.
218// When it returns true, neither profile nor dexpreopt artifacts will be generated.
Spandan Dase21a8d42024-01-23 23:56:29 +0000219func (d *dexpreopter) dexpreoptDisabled(ctx android.BaseModuleContext, libName string) bool {
Colin Cross38310bb2021-12-01 10:34:14 -0800220 if !ctx.Device() {
Colin Cross43f08db2018-11-12 10:13:39 -0800221 return true
222 }
223
Colin Cross43f08db2018-11-12 10:13:39 -0800224 if d.isTest {
225 return true
226 }
227
228 if !BoolDefault(d.dexpreoptProperties.Dex_preopt.Enabled, true) {
229 return true
230 }
231
Spandan Das0727ba72024-02-13 16:37:43 +0000232 if d.shouldDisableDexpreopt {
233 return true
234 }
235
Jiakai Zhang28bc9a82021-12-20 15:08:57 +0000236 // If the module is from a prebuilt APEX, it shouldn't be installable, but it can still be
237 // dexpreopted.
238 if !ctx.Module().(DexpreopterInterface).IsInstallable() && !forPrebuiltApex(ctx) {
Martin Stjernholm6d415272020-01-31 17:10:36 +0000239 return true
240 }
241
Colin Cross38310bb2021-12-01 10:34:14 -0800242 if !android.IsModulePreferred(ctx.Module()) {
243 return true
244 }
245
Spandan Dasa8afdcb2024-02-29 06:40:16 +0000246 if disableSourceApexVariant(ctx) {
247 return true
248 }
249
Spandan Dase21a8d42024-01-23 23:56:29 +0000250 if _, isApex := android.ModuleProvider(ctx, android.ApexBundleInfoProvider); isApex {
251 // dexpreopt rules for system server jars can be generated in the ModuleCtx of prebuilt apexes
252 return false
253 }
254
Colin Cross38310bb2021-12-01 10:34:14 -0800255 global := dexpreopt.GetGlobalConfig(ctx)
256
Spandan Dase21a8d42024-01-23 23:56:29 +0000257 // Use the libName argument to determine if the library being dexpreopt'd is a system server jar
258 // ctx.ModuleName() is not safe. In case of prebuilt apexes, the dexpreopt rules of system server jars
259 // are created in the ctx object of the top-level prebuilt apex.
260 isApexSystemServerJar := global.AllApexSystemServerJars(ctx).ContainsJar(libName)
261
262 if _, isApex := android.ModuleProvider(ctx, android.ApexBundleInfoProvider); isApex || isApexVariant(ctx) {
263 // dexpreopt rules for system server jars can be generated in the ModuleCtx of prebuilt apexes
Jiakai Zhang3317ce72023-02-08 01:19:19 +0800264 if !isApexSystemServerJar {
Jiakai Zhangca9bc982021-09-09 08:09:41 +0000265 return true
266 }
267 } else {
268 // Don't preopt the platform variant of an APEX system server jar to avoid conflicts.
Jiakai Zhang389a6472021-12-14 18:54:06 +0000269 if isApexSystemServerJar {
Jiakai Zhangca9bc982021-09-09 08:09:41 +0000270 return true
271 }
Yo Chiangdbdf8f92020-01-09 19:00:27 +0800272 }
273
Colin Cross43f08db2018-11-12 10:13:39 -0800274 // TODO: contains no java code
275
276 return false
277}
278
Martin Stjernholm6d415272020-01-31 17:10:36 +0000279func dexpreoptToolDepsMutator(ctx android.BottomUpMutatorContext) {
Spandan Dase21a8d42024-01-23 23:56:29 +0000280 if _, isApex := android.ModuleProvider(ctx, android.ApexBundleInfoProvider); isApex && dexpreopt.IsDex2oatNeeded(ctx) {
281 // prebuilt apexes can genererate rules to dexpreopt deapexed jars
282 // Add a dex2oat dep aggressively on _every_ apex module
283 dexpreopt.RegisterToolDeps(ctx)
284 return
285 }
286 if d, ok := ctx.Module().(DexpreopterInterface); !ok || d.dexpreoptDisabled(ctx, android.RemoveOptionalPrebuiltPrefix(ctx.ModuleName())) || !dexpreopt.IsDex2oatNeeded(ctx) {
Martin Stjernholm6d415272020-01-31 17:10:36 +0000287 return
288 }
289 dexpreopt.RegisterToolDeps(ctx)
290}
291
Spandan Dase21a8d42024-01-23 23:56:29 +0000292func (d *dexpreopter) odexOnSystemOther(ctx android.ModuleContext, libName string, installPath android.InstallPath) bool {
293 return dexpreopt.OdexOnSystemOtherByName(libName, android.InstallPathToOnDevicePath(ctx, installPath), dexpreopt.GetGlobalConfig(ctx))
Jiakai Zhangca9bc982021-09-09 08:09:41 +0000294}
295
296// Returns the install path of the dex jar of a module.
297//
298// Do not rely on `ApexInfo.ApexVariationName` because it can be something like "apex1000", rather
299// than the `name` in the path `/apex/<name>` as suggested in its comment.
300//
301// This function is on a best-effort basis. It cannot handle the case where an APEX jar is not a
302// system server jar, which is fine because we currently only preopt system server jars for APEXes.
303func (d *dexpreopter) getInstallPath(
Spandan Dase21a8d42024-01-23 23:56:29 +0000304 ctx android.ModuleContext, libName string, defaultInstallPath android.InstallPath) android.InstallPath {
Jiakai Zhangca9bc982021-09-09 08:09:41 +0000305 global := dexpreopt.GetGlobalConfig(ctx)
Spandan Dase21a8d42024-01-23 23:56:29 +0000306 if global.AllApexSystemServerJars(ctx).ContainsJar(libName) {
307 dexLocation := dexpreopt.GetSystemServerDexLocation(ctx, global, libName)
Jiakai Zhangca9bc982021-09-09 08:09:41 +0000308 return android.PathForModuleInPartitionInstall(ctx, "", strings.TrimPrefix(dexLocation, "/"))
309 }
Spandan Dase21a8d42024-01-23 23:56:29 +0000310 if !d.dexpreoptDisabled(ctx, libName) && isApexVariant(ctx) &&
Jiakai Zhangca9bc982021-09-09 08:09:41 +0000311 filepath.Base(defaultInstallPath.PartitionDir()) != "apex" {
312 ctx.ModuleErrorf("unable to get the install path of the dex jar for dexpreopt")
313 }
314 return defaultInstallPath
Nicolas Geoffrayfa6e9ec2019-02-12 13:12:16 +0000315}
316
Spandan Das2069c3f2023-12-06 19:40:24 +0000317// DexpreoptPrebuiltApexSystemServerJars generates the dexpreopt artifacts from a jar file that has been deapexed from a prebuilt apex
318func (d *Dexpreopter) DexpreoptPrebuiltApexSystemServerJars(ctx android.ModuleContext, libraryName string, di *android.DeapexerInfo) {
319 // A single prebuilt apex can have multiple apex system jars
320 // initialize the output path for this dex jar
321 dc := dexpreopt.GetGlobalConfig(ctx)
322 d.installPath = android.PathForModuleInPartitionInstall(ctx, "", strings.TrimPrefix(dexpreopt.GetSystemServerDexLocation(ctx, dc, libraryName), "/"))
323 // generate the rules for creating the .odex and .vdex files for this system server jar
Spandan Das5be63332023-12-13 00:06:32 +0000324 dexJarFile := di.PrebuiltExportPath(ApexRootRelativePathToJavaLib(libraryName))
Spandan Das2ea84dd2024-01-25 22:12:50 +0000325
326 d.inputProfilePathOnHost = nil // reset: TODO(spandandas): Make dexpreopter stateless
327 if android.InList(libraryName, di.GetDexpreoptProfileGuidedExportedModuleNames()) {
328 // Set the profile path to guide optimization
329 prof := di.PrebuiltExportPath(ApexRootRelativePathToJavaLib(libraryName) + ".prof")
330 if prof == nil {
331 ctx.ModuleErrorf("Could not find a .prof file in this prebuilt apex")
332 }
333 d.inputProfilePathOnHost = prof
334 }
335
Spandan Dase21a8d42024-01-23 23:56:29 +0000336 d.dexpreopt(ctx, libraryName, dexJarFile)
Spandan Das2069c3f2023-12-06 19:40:24 +0000337}
338
Spandan Dase21a8d42024-01-23 23:56:29 +0000339func (d *dexpreopter) dexpreopt(ctx android.ModuleContext, libName string, dexJarFile android.WritablePath) {
Jiakai Zhangca9bc982021-09-09 08:09:41 +0000340 global := dexpreopt.GetGlobalConfig(ctx)
341
Martin Stjernholm6d415272020-01-31 17:10:36 +0000342 // TODO(b/148690468): The check on d.installPath is to bail out in cases where
343 // the dexpreopter struct hasn't been fully initialized before we're called,
344 // e.g. in aar.go. This keeps the behaviour that dexpreopting is effectively
345 // disabled, even if installable is true.
Ulya Trafimovich76b08522021-01-14 17:52:43 +0000346 if d.installPath.Base() == "." {
347 return
348 }
349
350 dexLocation := android.InstallPathToOnDevicePath(ctx, d.installPath)
351
Spandan Dase21a8d42024-01-23 23:56:29 +0000352 providesUsesLib := libName
Ulya Trafimovich76b08522021-01-14 17:52:43 +0000353 if ulib, ok := ctx.Module().(ProvidesUsesLib); ok {
354 name := ulib.ProvidesUsesLib()
355 if name != nil {
356 providesUsesLib = *name
357 }
358 }
359
Jeongik Cha4b073cd2021-06-08 11:35:00 +0900360 // If it is test, make config files regardless of its dexpreopt setting.
Jeongik Chac6246672021-04-08 00:00:19 +0900361 // The config files are required for apps defined in make which depend on the lib.
Spandan Dase21a8d42024-01-23 23:56:29 +0000362 if d.isTest && d.dexpreoptDisabled(ctx, libName) {
Jaewoong Jung4b97a562020-12-17 09:43:28 -0800363 return
Nicolas Geoffrayfa6e9ec2019-02-12 13:12:16 +0000364 }
365
Spandan Dase21a8d42024-01-23 23:56:29 +0000366 isSystemServerJar := global.AllSystemServerJars(ctx).ContainsJar(libName)
Ulya Trafimovich9023b022021-03-22 16:02:28 +0000367
Colin Cross44df5812019-02-15 23:06:46 -0800368 bootImage := defaultBootImageConfig(ctx)
Jiakai Zhangb8796202023-03-06 19:16:48 +0000369 // When `global.PreoptWithUpdatableBcp` is true, `bcpForDexpreopt` below includes the mainline
370 // boot jars into bootclasspath, so we should include the mainline boot image as well because it's
371 // generated from those jars.
372 if global.PreoptWithUpdatableBcp {
373 bootImage = mainlineBootImageConfig(ctx)
374 }
Jiakai Zhang02669e82021-09-11 03:44:06 +0000375 dexFiles, dexLocations := bcpForDexpreopt(ctx, global.PreoptWithUpdatableBcp)
Ulya Trafimovich9023b022021-03-22 16:02:28 +0000376
David Srbeckyc177ebe2020-02-18 20:43:06 +0000377 targets := ctx.MultiTargets()
378 if len(targets) == 0 {
Colin Cross43f08db2018-11-12 10:13:39 -0800379 // assume this is a java library, dexpreopt for all arches for now
380 for _, target := range ctx.Config().Targets[android.Android] {
dimitry1f33e402019-03-26 12:39:31 +0100381 if target.NativeBridge == android.NativeBridgeDisabled {
David Srbeckyc177ebe2020-02-18 20:43:06 +0000382 targets = append(targets, target)
dimitry1f33e402019-03-26 12:39:31 +0100383 }
Colin Cross43f08db2018-11-12 10:13:39 -0800384 }
Spandan Dase21a8d42024-01-23 23:56:29 +0000385 if isSystemServerJar && libName != "com.android.location.provider" {
Jiakai Zhang2fbc3552022-11-28 15:38:23 +0000386 // If the module is a system server jar, only preopt for the primary arch because the jar can
387 // only be loaded by system server. "com.android.location.provider" is a special case because
388 // it's also used by apps as a shared library.
David Srbeckyc177ebe2020-02-18 20:43:06 +0000389 targets = targets[:1]
Colin Cross43f08db2018-11-12 10:13:39 -0800390 }
391 }
Colin Cross43f08db2018-11-12 10:13:39 -0800392
David Srbeckyc177ebe2020-02-18 20:43:06 +0000393 var archs []android.ArchType
Colin Cross69f59a32019-02-15 10:39:37 -0800394 var images android.Paths
Ulya Trafimovich4d2eeed2019-11-08 10:54:21 +0000395 var imagesDeps []android.OutputPaths
David Srbeckyc177ebe2020-02-18 20:43:06 +0000396 for _, target := range targets {
397 archs = append(archs, target.Arch.ArchType)
398 variant := bootImage.getVariant(target)
Jeongik Chaa5969092021-05-07 18:53:21 +0900399 images = append(images, variant.imagePathOnHost)
David Srbeckyc177ebe2020-02-18 20:43:06 +0000400 imagesDeps = append(imagesDeps, variant.imagesDeps)
Colin Crossc7e40aa2019-02-08 21:37:00 -0800401 }
David Srbeckyab994982020-03-30 17:24:13 +0100402 // The image locations for all Android variants are identical.
Jeongik Cha4dda75e2021-04-27 23:56:44 +0900403 hostImageLocations, deviceImageLocations := bootImage.getAnyAndroidVariant().imageLocations()
Colin Crossc7e40aa2019-02-08 21:37:00 -0800404
Colin Cross43f08db2018-11-12 10:13:39 -0800405 var profileClassListing android.OptionalPath
Nicolas Geoffraye7102422019-07-24 13:19:29 +0100406 var profileBootListing android.OptionalPath
Colin Cross43f08db2018-11-12 10:13:39 -0800407 profileIsTextListing := false
Spandan Das2ea84dd2024-01-25 22:12:50 +0000408
Jiakai Zhang81e46812023-02-08 21:56:07 +0800409 if d.inputProfilePathOnHost != nil {
410 profileClassListing = android.OptionalPathForPath(d.inputProfilePathOnHost)
411 } else if BoolDefault(d.dexpreoptProperties.Dex_preopt.Profile_guided, true) && !forPrebuiltApex(ctx) {
Colin Cross43f08db2018-11-12 10:13:39 -0800412 // If dex_preopt.profile_guided is not set, default it based on the existence of the
413 // dexprepot.profile option or the profile class listing.
414 if String(d.dexpreoptProperties.Dex_preopt.Profile) != "" {
415 profileClassListing = android.OptionalPathForPath(
416 android.PathForModuleSrc(ctx, String(d.dexpreoptProperties.Dex_preopt.Profile)))
Nicolas Geoffraye7102422019-07-24 13:19:29 +0100417 profileBootListing = android.ExistentPathForSource(ctx,
418 ctx.ModuleDir(), String(d.dexpreoptProperties.Dex_preopt.Profile)+"-boot")
Colin Cross43f08db2018-11-12 10:13:39 -0800419 profileIsTextListing = true
Dan Willemsen78d51b02020-06-24 16:33:31 -0700420 } else if global.ProfileDir != "" {
Colin Cross43f08db2018-11-12 10:13:39 -0800421 profileClassListing = android.ExistentPathForSource(ctx,
Spandan Dase21a8d42024-01-23 23:56:29 +0000422 global.ProfileDir, libName+".prof")
Colin Cross43f08db2018-11-12 10:13:39 -0800423 }
424 }
425
Jiakai Zhang9c4dc192023-02-09 00:09:24 +0800426 d.dexpreoptProperties.Dex_preopt_result.Profile_guided = profileClassListing.Valid()
427
Spandan Das2069c3f2023-12-06 19:40:24 +0000428 // A single apex can have multiple system server jars
429 // Use the dexJar to create a unique scope for each
430 dexJarStem := strings.TrimSuffix(dexJarFile.Base(), dexJarFile.Ext())
431
Ulya Trafimovich76b08522021-01-14 17:52:43 +0000432 // Full dexpreopt config, used to create dexpreopt build rules.
Martin Stjernholm8d80cee2020-01-31 17:44:54 +0000433 dexpreoptConfig := &dexpreopt.ModuleConfig{
Spandan Dase21a8d42024-01-23 23:56:29 +0000434 Name: libName,
Victor Hsiehd181c8b2019-01-29 13:00:33 -0800435 DexLocation: dexLocation,
Spandan Dase21a8d42024-01-23 23:56:29 +0000436 BuildPath: android.PathForModuleOut(ctx, "dexpreopt", dexJarStem, libName+".jar").OutputPath,
Colin Cross69f59a32019-02-15 10:39:37 -0800437 DexPath: dexJarFile,
Jeongik Cha33a3a812021-04-15 09:12:49 +0900438 ManifestPath: android.OptionalPathForPath(d.manifestFile),
Victor Hsiehd181c8b2019-01-29 13:00:33 -0800439 UncompressedDex: d.uncompressedDex,
440 HasApkLibraries: false,
441 PreoptFlags: nil,
Colin Cross43f08db2018-11-12 10:13:39 -0800442
Colin Cross69f59a32019-02-15 10:39:37 -0800443 ProfileClassListing: profileClassListing,
Colin Cross43f08db2018-11-12 10:13:39 -0800444 ProfileIsTextListing: profileIsTextListing,
Nicolas Geoffraye7102422019-07-24 13:19:29 +0100445 ProfileBootListing: profileBootListing,
Colin Cross43f08db2018-11-12 10:13:39 -0800446
Ulya Trafimovich8c35fcf2021-02-17 16:23:28 +0000447 EnforceUsesLibrariesStatusFile: dexpreopt.UsesLibrariesStatusFile(ctx),
448 EnforceUsesLibraries: d.enforceUsesLibs,
449 ProvidesUsesLibrary: providesUsesLib,
450 ClassLoaderContexts: d.classLoaderContexts,
Colin Cross43f08db2018-11-12 10:13:39 -0800451
Jeongik Cha4dda75e2021-04-27 23:56:44 +0900452 Archs: archs,
453 DexPreoptImagesDeps: imagesDeps,
454 DexPreoptImageLocationsOnHost: hostImageLocations,
455 DexPreoptImageLocationsOnDevice: deviceImageLocations,
Colin Cross43f08db2018-11-12 10:13:39 -0800456
Ulya Trafimovich9023b022021-03-22 16:02:28 +0000457 PreoptBootClassPathDexFiles: dexFiles.Paths(),
Vladimir Marko40139d62020-02-06 15:14:29 +0000458 PreoptBootClassPathDexLocations: dexLocations,
Colin Cross800fe132019-02-11 14:21:24 -0800459
Colin Cross43f08db2018-11-12 10:13:39 -0800460 NoCreateAppImage: !BoolDefault(d.dexpreoptProperties.Dex_preopt.App_image, true),
461 ForceCreateAppImage: BoolDefault(d.dexpreoptProperties.Dex_preopt.App_image, false),
462
Jaewoong Jungccbb3932019-04-15 09:48:31 -0700463 PresignedPrebuilt: d.isPresignedPrebuilt,
Colin Cross43f08db2018-11-12 10:13:39 -0800464 }
465
Spandan Das2069c3f2023-12-06 19:40:24 +0000466 d.configPath = android.PathForModuleOut(ctx, "dexpreopt", dexJarStem, "dexpreopt.config")
Jeongik Chac6246672021-04-08 00:00:19 +0900467 dexpreopt.WriteModuleConfig(ctx, dexpreoptConfig, d.configPath)
468
Spandan Dase21a8d42024-01-23 23:56:29 +0000469 if d.dexpreoptDisabled(ctx, libName) {
Jeongik Chac6246672021-04-08 00:00:19 +0900470 return
471 }
472
473 globalSoong := dexpreopt.GetGlobalSoongConfig(ctx)
474
Jiakai Zhang51b2a8b2023-06-26 16:47:38 +0100475 // The root "product_packages.txt" is generated by `build/make/core/Makefile`. It contains a list
476 // of all packages that are installed on the device. We use `grep` to filter the list by the app's
477 // dependencies to create a per-app list, and use `rsync --checksum` to prevent the file's mtime
478 // from being changed if the contents don't change. This avoids unnecessary dexpreopt reruns.
Jiakai Zhanga4496782023-05-17 16:57:30 +0100479 productPackages := android.PathForModuleInPartitionInstall(ctx, "", "product_packages.txt")
Spandan Das2069c3f2023-12-06 19:40:24 +0000480 appProductPackages := android.PathForModuleOut(ctx, "dexpreopt", dexJarStem, "product_packages.txt")
Jiakai Zhang51b2a8b2023-06-26 16:47:38 +0100481 appProductPackagesStaging := appProductPackages.ReplaceExtension(ctx, "txt.tmp")
482 clcNames, _ := dexpreopt.ComputeClassLoaderContextDependencies(dexpreoptConfig.ClassLoaderContexts)
483 sort.Strings(clcNames) // The order needs to be deterministic.
484 productPackagesRule := android.NewRuleBuilder(pctx, ctx)
485 if len(clcNames) > 0 {
486 productPackagesRule.Command().
487 Text("grep -F -x").
488 FlagForEachArg("-e ", clcNames).
489 Input(productPackages).
490 FlagWithOutput("> ", appProductPackagesStaging).
491 Text("|| true")
492 } else {
493 productPackagesRule.Command().
494 Text("rm -f").Output(appProductPackagesStaging).
495 Text("&&").
496 Text("touch").Output(appProductPackagesStaging)
497 }
498 productPackagesRule.Command().
499 Text("rsync --checksum").
500 Input(appProductPackagesStaging).
501 Output(appProductPackages)
Spandan Das2069c3f2023-12-06 19:40:24 +0000502 productPackagesRule.Restat().Build("product_packages."+dexJarStem, "dexpreopt product_packages")
Jiakai Zhanga4496782023-05-17 16:57:30 +0100503
504 dexpreoptRule, err := dexpreopt.GenerateDexpreoptRule(
Jiakai Zhang51b2a8b2023-06-26 16:47:38 +0100505 ctx, globalSoong, global, dexpreoptConfig, appProductPackages)
Colin Cross43f08db2018-11-12 10:13:39 -0800506 if err != nil {
507 ctx.ModuleErrorf("error generating dexpreopt rule: %s", err.Error())
Jaewoong Jung4b97a562020-12-17 09:43:28 -0800508 return
Colin Cross43f08db2018-11-12 10:13:39 -0800509 }
510
Spandan Das2069c3f2023-12-06 19:40:24 +0000511 dexpreoptRule.Build("dexpreopt"+"."+dexJarStem, "dexpreopt")
Colin Cross43f08db2018-11-12 10:13:39 -0800512
Spandan Das2069c3f2023-12-06 19:40:24 +0000513 // The current ctx might be of a deapexer module created by a prebuilt apex
514 // Use the path of the dex file to determine the library name
515 isApexSystemServerJar := global.AllApexSystemServerJars(ctx).ContainsJar(dexJarStem)
Jiakai Zhang389a6472021-12-14 18:54:06 +0000516
Colin Cross1d0eb7a2021-11-03 14:08:20 -0700517 for _, install := range dexpreoptRule.Installs() {
518 // Remove the "/" prefix because the path should be relative to $ANDROID_PRODUCT_OUT.
519 installDir := strings.TrimPrefix(filepath.Dir(install.To), "/")
520 installBase := filepath.Base(install.To)
521 arch := filepath.Base(installDir)
522 installPath := android.PathForModuleInPartitionInstall(ctx, "", installDir)
Jiakai Zhang3317ce72023-02-08 01:19:19 +0800523 isProfile := strings.HasSuffix(installBase, ".prof")
524
525 if isProfile {
Jiakai Zhang81e46812023-02-08 21:56:07 +0800526 d.outputProfilePathOnHost = install.From
Jiakai Zhang3317ce72023-02-08 01:19:19 +0800527 }
Colin Cross1d0eb7a2021-11-03 14:08:20 -0700528
Jiakai Zhang389a6472021-12-14 18:54:06 +0000529 if isApexSystemServerJar {
Jiakai Zhang3317ce72023-02-08 01:19:19 +0800530 // Profiles are handled separately because they are installed into the APEX.
531 if !isProfile {
532 // APEX variants of java libraries are hidden from Make, so their dexpreopt
533 // outputs need special handling. Currently, for APEX variants of java
534 // libraries, only those in the system server classpath are handled here.
535 // Preopting of boot classpath jars in the ART APEX are handled in
536 // java/dexpreopt_bootjars.go, and other APEX jars are not preopted.
537 // The installs will be handled by Make as sub-modules of the java library.
538 d.builtInstalledForApex = append(d.builtInstalledForApex, dexpreopterInstall{
539 name: arch + "-" + installBase,
Spandan Dase21a8d42024-01-23 23:56:29 +0000540 moduleName: libName,
Jiakai Zhang3317ce72023-02-08 01:19:19 +0800541 outputPathOnHost: install.From,
542 installDirOnDevice: installPath,
543 installFileOnDevice: installBase,
544 })
545 }
Colin Crossfa9bfcd2021-11-10 16:42:38 -0800546 } else if !d.preventInstall {
Colin Cross1d0eb7a2021-11-03 14:08:20 -0700547 ctx.InstallFile(installPath, installBase, install.From)
Jiakai Zhangca9bc982021-09-09 08:09:41 +0000548 }
Colin Cross1d0eb7a2021-11-03 14:08:20 -0700549 }
550
Jiakai Zhang389a6472021-12-14 18:54:06 +0000551 if !isApexSystemServerJar {
Jiakai Zhangca9bc982021-09-09 08:09:41 +0000552 d.builtInstalled = dexpreoptRule.Installs().String()
553 }
554}
555
556func (d *dexpreopter) DexpreoptBuiltInstalledForApex() []dexpreopterInstall {
557 return d.builtInstalledForApex
558}
559
560func (d *dexpreopter) AndroidMkEntriesForApex() []android.AndroidMkEntries {
561 var entries []android.AndroidMkEntries
562 for _, install := range d.builtInstalledForApex {
Jiakai Zhang6decef92022-01-12 17:56:19 +0000563 entries = append(entries, install.ToMakeEntries())
Jiakai Zhangca9bc982021-09-09 08:09:41 +0000564 }
565 return entries
Colin Cross43f08db2018-11-12 10:13:39 -0800566}
Jiakai Zhang3317ce72023-02-08 01:19:19 +0800567
Jiakai Zhang81e46812023-02-08 21:56:07 +0800568func (d *dexpreopter) OutputProfilePathOnHost() android.Path {
569 return d.outputProfilePathOnHost
Jiakai Zhang3317ce72023-02-08 01:19:19 +0800570}
Spandan Das0727ba72024-02-13 16:37:43 +0000571
572func (d *dexpreopter) disableDexpreopt() {
573 d.shouldDisableDexpreopt = true
574}