• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1// Copyright 2017 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 cc
16
17import (
18	"fmt"
19	"path/filepath"
20	"strings"
21
22	"github.com/google/blueprint/proptools"
23
24	"android/soong/android"
25)
26
27var (
28	// Add flags to ignore warnings that profiles are old or missing for
29	// some functions.
30	profileUseOtherFlags = []string{
31		"-Wno-backend-plugin",
32	}
33
34	globalPgoProfileProjects = []string{
35		"toolchain/pgo-profiles/pgo",
36		"vendor/google_data/pgo_profile/pgo",
37	}
38)
39
40var pgoProfileProjectsConfigKey = android.NewOnceKey("PgoProfileProjects")
41
42const profileInstrumentFlag = "-fprofile-generate=/data/local/tmp"
43const profileUseInstrumentFormat = "-fprofile-use=%s"
44const profileUseSamplingFormat = "-fprofile-sample-accurate -fprofile-sample-use=%s"
45
46func getPgoProfileProjects(config android.DeviceConfig) []string {
47	return config.OnceStringSlice(pgoProfileProjectsConfigKey, func() []string {
48		return append(globalPgoProfileProjects, config.PgoAdditionalProfileDirs()...)
49	})
50}
51
52func recordMissingProfileFile(ctx BaseModuleContext, missing string) {
53	getNamedMapForConfig(ctx.Config(), modulesMissingProfileFileKey).Store(missing, true)
54}
55
56type PgoProperties struct {
57	Pgo struct {
58		Instrumentation    *bool
59		Sampling           *bool   `android:"arch_variant"`
60		Profile_file       *string `android:"arch_variant"`
61		Benchmarks         []string
62		Enable_profile_use *bool `android:"arch_variant"`
63		// Additional compiler flags to use when building this module
64		// for profiling (either instrumentation or sampling).
65		Cflags []string `android:"arch_variant"`
66	} `android:"arch_variant"`
67
68	PgoPresent          bool `blueprint:"mutated"`
69	ShouldProfileModule bool `blueprint:"mutated"`
70	PgoCompile          bool `blueprint:"mutated"`
71	PgoInstrLink        bool `blueprint:"mutated"`
72}
73
74type pgo struct {
75	Properties PgoProperties
76}
77
78func (props *PgoProperties) isInstrumentation() bool {
79	return props.Pgo.Instrumentation != nil && *props.Pgo.Instrumentation == true
80}
81
82func (props *PgoProperties) isSampling() bool {
83	return props.Pgo.Sampling != nil && *props.Pgo.Sampling == true
84}
85
86func (pgo *pgo) props() []interface{} {
87	return []interface{}{&pgo.Properties}
88}
89
90func (props *PgoProperties) addInstrumentationProfileGatherFlags(ctx ModuleContext, flags Flags) Flags {
91	// Add to C flags iff PGO is explicitly enabled for this module.
92	if props.ShouldProfileModule {
93		flags.Local.CFlags = append(flags.Local.CFlags, props.Pgo.Cflags...)
94		flags.Local.CFlags = append(flags.Local.CFlags, profileInstrumentFlag)
95	}
96	flags.Local.LdFlags = append(flags.Local.LdFlags, profileInstrumentFlag)
97	return flags
98}
99
100func (props *PgoProperties) getPgoProfileFile(ctx BaseModuleContext) android.OptionalPath {
101	profileFile := *props.Pgo.Profile_file
102
103	// Test if the profile_file is present in any of the PGO profile projects
104	for _, profileProject := range getPgoProfileProjects(ctx.DeviceConfig()) {
105		// Bug: http://b/74395273 If the profile_file is unavailable,
106		// use a versioned file named
107		// <profile_file>.<arbitrary-version> when available.  This
108		// works around an issue where ccache serves stale cache
109		// entries when the profile file has changed.
110		globPattern := filepath.Join(profileProject, profileFile+".*")
111		versionedProfiles, err := ctx.GlobWithDeps(globPattern, nil)
112		if err != nil {
113			ctx.ModuleErrorf("glob: %s", err.Error())
114		}
115
116		path := android.ExistentPathForSource(ctx, profileProject, profileFile)
117		if path.Valid() {
118			if len(versionedProfiles) != 0 {
119				ctx.PropertyErrorf("pgo.profile_file", "Profile_file has multiple versions: "+filepath.Join(profileProject, profileFile)+", "+strings.Join(versionedProfiles, ", "))
120			}
121			return path
122		}
123
124		if len(versionedProfiles) > 1 {
125			ctx.PropertyErrorf("pgo.profile_file", "Profile_file has multiple versions: "+strings.Join(versionedProfiles, ", "))
126		} else if len(versionedProfiles) == 1 {
127			return android.OptionalPathForPath(android.PathForSource(ctx, versionedProfiles[0]))
128		}
129	}
130
131	// Record that this module's profile file is absent
132	missing := *props.Pgo.Profile_file + ":" + ctx.ModuleDir() + "/Android.bp:" + ctx.ModuleName()
133	recordMissingProfileFile(ctx, missing)
134
135	return android.OptionalPathForPath(nil)
136}
137
138func (props *PgoProperties) profileUseFlag(ctx ModuleContext, file string) string {
139	if props.isInstrumentation() {
140		return fmt.Sprintf(profileUseInstrumentFormat, file)
141	}
142	if props.isSampling() {
143		return fmt.Sprintf(profileUseSamplingFormat, file)
144	}
145	return ""
146}
147
148func (props *PgoProperties) profileUseFlags(ctx ModuleContext, file string) []string {
149	flags := []string{props.profileUseFlag(ctx, file)}
150	flags = append(flags, profileUseOtherFlags...)
151	return flags
152}
153
154func (props *PgoProperties) addProfileUseFlags(ctx ModuleContext, flags Flags) Flags {
155	// Return if 'pgo' property is not present in this module.
156	if !props.PgoPresent {
157		return flags
158	}
159
160	if props.PgoCompile {
161		profileFile := props.getPgoProfileFile(ctx)
162		profileFilePath := profileFile.Path()
163		profileUseFlags := props.profileUseFlags(ctx, profileFilePath.String())
164
165		flags.Local.CFlags = append(flags.Local.CFlags, profileUseFlags...)
166		flags.Local.LdFlags = append(flags.Local.LdFlags, profileUseFlags...)
167
168		// Update CFlagsDeps and LdFlagsDeps so the module is rebuilt
169		// if profileFile gets updated
170		flags.CFlagsDeps = append(flags.CFlagsDeps, profileFilePath)
171		flags.LdFlagsDeps = append(flags.LdFlagsDeps, profileFilePath)
172
173		if props.isSampling() {
174			flags.Local.LdFlags = append(flags.Local.LdFlags, "-Wl,-mllvm,-no-warn-sample-unused=true")
175		}
176	}
177	return flags
178}
179
180func (props *PgoProperties) isPGO(ctx BaseModuleContext) bool {
181	isInstrumentation := props.isInstrumentation()
182	isSampling := props.isSampling()
183
184	profileKindPresent := isInstrumentation || isSampling
185	filePresent := props.Pgo.Profile_file != nil
186	benchmarksPresent := len(props.Pgo.Benchmarks) > 0
187
188	// If all three properties are absent, PGO is OFF for this module
189	if !profileKindPresent && !filePresent && !benchmarksPresent {
190		return false
191	}
192
193	// profileKindPresent and filePresent are mandatory properties.
194	if !profileKindPresent || !filePresent {
195		var missing []string
196		if !profileKindPresent {
197			missing = append(missing, "profile kind (either \"instrumentation\" or \"sampling\" property)")
198		}
199		if !filePresent {
200			missing = append(missing, "profile_file property")
201		}
202		missingProps := strings.Join(missing, ", ")
203		ctx.ModuleErrorf("PGO specification is missing properties: " + missingProps)
204	}
205
206	// Benchmark property is mandatory for instrumentation PGO.
207	if isInstrumentation && !benchmarksPresent {
208		ctx.ModuleErrorf("Instrumentation PGO specification is missing benchmark property")
209	}
210
211	if isSampling {
212		ctx.ModuleErrorf("Sampling PGO is deprecated, use AFDO instead")
213	}
214
215	if isSampling && isInstrumentation {
216		ctx.PropertyErrorf("pgo", "Exactly one of \"instrumentation\" and \"sampling\" properties must be set")
217	}
218
219	return true
220}
221
222func (pgo *pgo) begin(ctx BaseModuleContext) {
223	// TODO Evaluate if we need to support PGO for host modules
224	if ctx.Host() {
225		return
226	}
227
228	// Check if PGO is needed for this module
229	pgo.Properties.PgoPresent = pgo.Properties.isPGO(ctx)
230
231	if !pgo.Properties.PgoPresent {
232		return
233	}
234
235	// This module should be instrumented if ANDROID_PGO_INSTRUMENT is set
236	// and includes 'all', 'ALL' or a benchmark listed for this module.
237	//
238	// TODO Validate that each benchmark instruments at least one module
239	pgo.Properties.ShouldProfileModule = false
240	pgoBenchmarks := ctx.Config().Getenv("ANDROID_PGO_INSTRUMENT")
241	pgoBenchmarksMap := make(map[string]bool)
242	for _, b := range strings.Split(pgoBenchmarks, ",") {
243		pgoBenchmarksMap[b] = true
244	}
245
246	if pgoBenchmarksMap["all"] == true || pgoBenchmarksMap["ALL"] == true {
247		pgo.Properties.ShouldProfileModule = true
248		pgo.Properties.PgoInstrLink = pgo.Properties.isInstrumentation()
249	} else {
250		for _, b := range pgo.Properties.Pgo.Benchmarks {
251			if pgoBenchmarksMap[b] == true {
252				pgo.Properties.ShouldProfileModule = true
253				pgo.Properties.PgoInstrLink = pgo.Properties.isInstrumentation()
254				break
255			}
256		}
257	}
258
259	// PGO profile use is not feasible for a Clang coverage build because
260	// -fprofile-use and -fprofile-instr-generate are incompatible.
261	if ctx.DeviceConfig().ClangCoverageEnabled() {
262		return
263	}
264
265	if !ctx.Config().IsEnvTrue("ANDROID_PGO_NO_PROFILE_USE") &&
266		proptools.BoolDefault(pgo.Properties.Pgo.Enable_profile_use, true) {
267		if profileFile := pgo.Properties.getPgoProfileFile(ctx); profileFile.Valid() {
268			pgo.Properties.PgoCompile = true
269		}
270	}
271}
272
273func (pgo *pgo) flags(ctx ModuleContext, flags Flags) Flags {
274	if ctx.Host() {
275		return flags
276	}
277
278	// Deduce PgoInstrLink property i.e. whether this module needs to be
279	// linked with profile-generation flags.  Here, we're setting it if any
280	// dependency needs PGO instrumentation.  It is initially set in
281	// begin() if PGO is directly enabled for this module.
282	if ctx.static() && !ctx.staticBinary() {
283		// For static libraries, check if any whole_static_libs are
284		// linked with profile generation
285		ctx.VisitDirectDeps(func(m android.Module) {
286			if depTag, ok := ctx.OtherModuleDependencyTag(m).(libraryDependencyTag); ok {
287				if depTag.static() && depTag.wholeStatic {
288					if cc, ok := m.(*Module); ok {
289						if cc.pgo.Properties.PgoInstrLink {
290							pgo.Properties.PgoInstrLink = true
291						}
292					}
293				}
294			}
295		})
296	} else {
297		// For executables and shared libraries, check all static dependencies.
298		ctx.VisitDirectDeps(func(m android.Module) {
299			if depTag, ok := ctx.OtherModuleDependencyTag(m).(libraryDependencyTag); ok {
300				if depTag.static() {
301					if cc, ok := m.(*Module); ok {
302						if cc.pgo.Properties.PgoInstrLink {
303							pgo.Properties.PgoInstrLink = true
304						}
305					}
306				}
307			}
308		})
309	}
310
311	props := pgo.Properties
312	// Add flags to profile this module based on its profile_kind
313	if (props.ShouldProfileModule && props.isInstrumentation()) || props.PgoInstrLink {
314		// Instrumentation PGO use and gather flags cannot coexist.
315		return props.addInstrumentationProfileGatherFlags(ctx, flags)
316	}
317
318	if !ctx.Config().IsEnvTrue("ANDROID_PGO_NO_PROFILE_USE") {
319		flags = props.addProfileUseFlags(ctx, flags)
320	}
321
322	return flags
323}
324