• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1// Copyright 2019 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 (
18	"path/filepath"
19	"strings"
20
21	"android/soong/android"
22	"android/soong/dexpreopt"
23)
24
25// dexpreoptTargets returns the list of targets that are relevant to dexpreopting, which excludes architectures
26// supported through native bridge.
27func dexpreoptTargets(ctx android.PathContext) []android.Target {
28	var targets []android.Target
29	for _, target := range ctx.Config().Targets[android.Android] {
30		if target.NativeBridge == android.NativeBridgeDisabled {
31			targets = append(targets, target)
32		}
33	}
34	// We may also need the images on host in order to run host-based tests.
35	for _, target := range ctx.Config().Targets[ctx.Config().BuildOS] {
36		targets = append(targets, target)
37	}
38
39	return targets
40}
41
42var (
43	bootImageConfigKey       = android.NewOnceKey("bootImageConfig")
44	bootImageConfigRawKey    = android.NewOnceKey("bootImageConfigRaw")
45	frameworkBootImageName   = "boot"
46	mainlineBootImageName    = "mainline"
47	bootImageStem            = "boot"
48	ProfileInstallPathInApex = "etc/boot-image.prof"
49)
50
51// getImageNames returns an ordered list of image names. The order doesn't matter but needs to be
52// deterministic. The names listed here must match the map keys returned by genBootImageConfigs.
53func getImageNames() []string {
54	return []string{"art", "boot", "mainline"}
55}
56
57func genBootImageConfigRaw(ctx android.PathContext) map[string]*bootImageConfig {
58	return ctx.Config().Once(bootImageConfigRawKey, func() interface{} {
59		global := dexpreopt.GetGlobalConfig(ctx)
60
61		artBootImageName := "art"           // Keep this local to avoid accidental references.
62		frameworkModules := global.BootJars // This includes `global.ArtApexJars`.
63		mainlineBcpModules := global.ApexBootJars
64		frameworkSubdir := "system/framework"
65
66		profileImports := []string{"com.android.art"}
67
68		// ART boot image for testing only. Do not rely on it to make any build-time decision.
69		artCfg := bootImageConfig{
70			name:                  artBootImageName,
71			enabledIfExists:       "art-bootclasspath-fragment",
72			stem:                  bootImageStem,
73			installDir:            "apex/art_boot_images/javalib",
74			modules:               global.TestOnlyArtBootImageJars,
75			preloadedClassesFile:  "art/build/boot/preloaded-classes",
76			compilerFilter:        "speed-profile",
77			singleImage:           false,
78			profileImports:        profileImports,
79			profileProviderModule: "art-bootclasspath-fragment",
80		}
81
82		// Framework config for the boot image extension.
83		// It includes framework libraries and depends on the ART config.
84		frameworkCfg := bootImageConfig{
85			name:                 frameworkBootImageName,
86			enabledIfExists:      "platform-bootclasspath",
87			stem:                 bootImageStem,
88			installDir:           frameworkSubdir,
89			modules:              frameworkModules,
90			preloadedClassesFile: "frameworks/base/config/preloaded-classes",
91			compilerFilter:       "speed-profile",
92			singleImage:          false,
93			profileImports:       profileImports,
94		}
95
96		mainlineCfg := bootImageConfig{
97			extends:         &frameworkCfg,
98			name:            mainlineBootImageName,
99			enabledIfExists: "platform-bootclasspath",
100			stem:            bootImageStem,
101			installDir:      frameworkSubdir,
102			modules:         mainlineBcpModules,
103			compilerFilter:  "verify",
104			singleImage:     true,
105		}
106
107		return map[string]*bootImageConfig{
108			artBootImageName:       &artCfg,
109			frameworkBootImageName: &frameworkCfg,
110			mainlineBootImageName:  &mainlineCfg,
111		}
112	}).(map[string]*bootImageConfig)
113}
114
115// Construct the global boot image configs.
116func genBootImageConfigs(ctx android.PathContext) map[string]*bootImageConfig {
117	return ctx.Config().Once(bootImageConfigKey, func() interface{} {
118		targets := dexpreoptTargets(ctx)
119		deviceDir := android.PathForOutput(ctx, dexpreopt.GetDexpreoptDirName(ctx))
120
121		configs := genBootImageConfigRaw(ctx)
122
123		for _, c := range configs {
124			c.dir = deviceDir.Join(ctx, "dex_"+c.name+"jars")
125			c.symbolsDir = deviceDir.Join(ctx, "dex_"+c.name+"jars_unstripped")
126
127			// expands to <stem>.art for primary image and <stem>-<1st module>.art for extension
128			imageName := c.firstModuleNameOrStem(ctx) + ".art"
129
130			// The path to bootclasspath dex files needs to be known at module
131			// GenerateAndroidBuildAction time, before the bootclasspath modules have been compiled.
132			// Set up known paths for them, the singleton rules will copy them there.
133			// TODO(b/143682396): use module dependencies instead
134			inputDir := deviceDir.Join(ctx, "dex_"+c.name+"jars_input")
135			c.dexPaths = c.modules.BuildPaths(ctx, inputDir)
136			c.dexPathsByModule = c.modules.BuildPathsByModule(ctx, inputDir)
137			c.dexPathsDeps = c.dexPaths
138
139			// Create target-specific variants.
140			for _, target := range targets {
141				arch := target.Arch.ArchType
142				imageDir := c.dir.Join(ctx, target.Os.String(), c.installDir, arch.String())
143				variant := &bootImageVariant{
144					bootImageConfig:   c,
145					target:            target,
146					imagePathOnHost:   imageDir.Join(ctx, imageName),
147					imagePathOnDevice: filepath.Join("/", c.installDir, arch.String(), imageName),
148					imagesDeps:        c.moduleFiles(ctx, imageDir, ".art", ".oat", ".vdex"),
149					dexLocations:      c.modules.DevicePaths(ctx.Config(), target.Os),
150				}
151				variant.dexLocationsDeps = variant.dexLocations
152				c.variants = append(c.variants, variant)
153			}
154
155			c.zip = c.dir.Join(ctx, c.name+".zip")
156		}
157
158		visited := make(map[string]bool)
159		for _, c := range configs {
160			calculateDepsRecursive(c, targets, visited)
161		}
162
163		return configs
164	}).(map[string]*bootImageConfig)
165}
166
167// calculateDepsRecursive calculates the dependencies of the given boot image config and all its
168// ancestors, if they are not visited.
169// The boot images are supposed to form a tree, where the root is the primary boot image. We do not
170// expect loops (e.g., A extends B, B extends C, C extends A), and we let them crash soong with a
171// stack overflow.
172// Note that a boot image config only has a pointer to the parent, not to children. Therefore, we
173// first go up through the parent chain, and then go back down to visit every code along the path.
174// `visited` is a map where a key is a boot image name and the value indicates whether the boot
175// image config is visited. The boot image names are guaranteed to be unique because they come from
176// `genBootImageConfigRaw` above, which also returns a map and would fail in the first place if the
177// names were not unique.
178func calculateDepsRecursive(c *bootImageConfig, targets []android.Target, visited map[string]bool) {
179	if c.extends == nil || visited[c.name] {
180		return
181	}
182	if c.extends.extends != nil {
183		calculateDepsRecursive(c.extends, targets, visited)
184	}
185	visited[c.name] = true
186	c.dexPathsDeps = android.Concat(c.extends.dexPathsDeps, c.dexPathsDeps)
187	for i := range targets {
188		c.variants[i].baseImages = android.Concat(c.extends.variants[i].baseImages, android.OutputPaths{c.extends.variants[i].imagePathOnHost})
189		c.variants[i].baseImagesDeps = android.Concat(c.extends.variants[i].baseImagesDeps, c.extends.variants[i].imagesDeps.Paths())
190		c.variants[i].dexLocationsDeps = android.Concat(c.extends.variants[i].dexLocationsDeps, c.variants[i].dexLocationsDeps)
191	}
192}
193
194func defaultBootImageConfig(ctx android.PathContext) *bootImageConfig {
195	return genBootImageConfigs(ctx)[frameworkBootImageName]
196}
197
198func mainlineBootImageConfig(ctx android.PathContext) *bootImageConfig {
199	return genBootImageConfigs(ctx)[mainlineBootImageName]
200}
201
202// isProfileProviderApex returns true if this apex provides a boot image profile.
203func isProfileProviderApex(ctx android.PathContext, apexName string) bool {
204	for _, config := range genBootImageConfigs(ctx) {
205		for _, profileImport := range config.profileImports {
206			if profileImport == apexName {
207				return true
208			}
209		}
210	}
211	return false
212}
213
214// Returns a list of paths and a list of locations for the boot jars used in dexpreopt (to be
215// passed in -Xbootclasspath and -Xbootclasspath-locations arguments for dex2oat).
216func bcpForDexpreopt(ctx android.PathContext, withUpdatable bool) (android.WritablePaths, []string) {
217	bootImage := defaultBootImageConfig(ctx)
218	if withUpdatable {
219		bootImage = mainlineBootImageConfig(ctx)
220	}
221
222	dexPaths := bootImage.dexPathsDeps
223	// The dex locations for all Android variants are identical.
224	dexLocations := bootImage.getAnyAndroidVariant().dexLocationsDeps
225
226	return dexPaths, dexLocations
227}
228
229var defaultBootclasspathKey = android.NewOnceKey("defaultBootclasspath")
230
231func init() {
232	android.RegisterMakeVarsProvider(pctx, dexpreoptConfigMakevars)
233}
234
235func dexpreoptConfigMakevars(ctx android.MakeVarsContext) {
236	ctx.Strict("DEXPREOPT_BOOT_JARS_MODULES", strings.Join(defaultBootImageConfig(ctx).modules.CopyOfApexJarPairs(), ":"))
237}
238