2 # Generate Intel taken branches Linux perf event script for autofdo profiling.
4 # Copyright (C) 2016 Free Software Foundation, Inc.
6 # GCC is free software; you can redistribute it and/or modify it under
7 # the terms of the GNU General Public License as published by the Free
8 # Software Foundation; either version 3, or (at your option) any later
11 # GCC is distributed in the hope that it will be useful, but WITHOUT ANY
12 # WARRANTY; without even the implied warranty of MERCHANTABILITY or
13 # FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
16 # You should have received a copy of the GNU General Public License
17 # along with GCC; see the file COPYING3. If not see
18 # <http://www.gnu.org/licenses/>. */
20 # Run it with perf record -b -e EVENT program ...
21 # The Linux Kernel needs to support the PMU of the current CPU, and
22 # It will likely not work in VMs.
23 # Add --all to print for all cpus, otherwise for current cpu.
24 # Add --script to generate shell script to run correct event.
26 # Requires internet (https) access. This may require setting up a proxy
27 # with export https_proxy=...
35 baseurl
= "https://download.01.org/perfmon"
37 target_events
= (u
'BR_INST_RETIRED.NEAR_TAKEN',
38 u
'BR_INST_EXEC.TAKEN',
39 u
'BR_INST_RETIRED.TAKEN_JCC',
40 u
'BR_INST_TYPE_RETIRED.COND_TAKEN')
42 ap
= argparse
.ArgumentParser()
43 ap
.add_argument('--all', '-a', help='Print for all CPUs', action
='store_true')
44 ap
.add_argument('--script', help='Generate shell script', action
='store_true')
45 args
= ap
.parse_args()
47 eventmap
= collections
.defaultdict(list)
50 cpuinfo
= os
.getenv("CPUINFO")
52 cpuinfo
= '/proc/cpuinfo'
53 f
= open(cpuinfo
, 'r')
54 cpu
= [None, None, None, None]
57 if n
[0] == 'vendor_id':
59 elif n
[0] == 'model' and n
[1] == ':':
61 elif n
[0] == 'cpu' and n
[1] == 'family':
63 elif n
[0] == 'stepping' and n
[1] == ':':
65 if all(v
is not None for v
in cpu
):
67 # stepping for SKX only
68 stepping
= cpu
[0] == "GenuineIntel" and cpu
[1] == 6 and cpu
[2] == 0x55
70 return "%s-%d-%X-%X" % tuple(cpu
)
71 return "%s-%d-%X" % tuple(cpu
)[:3]
73 def find_event(eventurl
, model
):
74 print >>sys
.stderr
, "Downloading", eventurl
75 u
= urllib2
.urlopen(eventurl
)
76 events
= json
.loads(u
.read())
81 if j
[u
'EventName'] in target_events
:
82 event
= "cpu/event=%s,umask=%s/" % (j
[u
'EventCode'], j
[u
'UMask'])
83 if u
'PEBS' in j
and j
[u
'PEBS'] > 0:
86 eventmap
[event
].append(model
)
88 print j
[u
'EventName'], "event for model", model
, "is", event
95 sys
.exit("Unknown CPU type")
97 url
= baseurl
+ "/mapfile.csv"
98 print >>sys
.stderr
, "Downloading", url
99 u
= urllib2
.urlopen(url
)
103 n
= j
.rstrip().split(',')
104 if len(n
) >= 4 and (args
.all
or n
[0] == cpu
) and n
[3] == "core":
106 components
= n
[0].split("-")
107 model
= components
[2]
108 model
= int(model
, 16)
110 found
+= find_event(baseurl
+ n
[2], model
)
115 # Profile workload for gcc profile feedback (autofdo) using Linux perf.
116 # Auto generated. To regenerate for new CPUs run
117 # contrib/gen_autofdo_event.py --script --all in gcc source
120 # gcc-auto-profile program (profile program and children)
121 # gcc-auto-profile -a sleep X (profile all for X secs, may need root)
122 # gcc-auto-profile -p PID sleep X (profile PID)
123 # gcc-auto-profile --kernel -a sleep X (profile kernel)
124 # gcc-auto-profile --all -a sleep X (profile kernel and user space)
126 # Identify branches taken event for CPU.
131 if [ "$1" = "--kernel" ] ; then
135 if [ "$1" = "--all" ] ; then
140 if ! grep -q Intel /proc/cpuinfo ; then
141 echo >&2 "Only Intel CPUs supported"
145 if grep -q hypervisor /proc/cpuinfo ; then
146 echo >&2 "Warning: branch profiling may not be functional in VMs"
149 case `egrep -q "^cpu family\s*: 6" /proc/cpuinfo &&
150 egrep "^model\s*:" /proc/cpuinfo | head -n1` in'''
151 for event
, mod
in eventmap
.iteritems():
153 print "model*:\ %s|\\" % m
154 print 'model*:\ %s) E="%s$FLAGS" ;;' % (mod
[-1], event
)
156 echo >&2 "Unknown CPU. Run contrib/gen_autofdo_event.py --all --script to update script."
160 print 'if ! perf record -e $E -b "$@" ; then'
161 print ' # PEBS may not actually be working even if the processor supports it'
162 print ' # (e.g., in a virtual machine). Trying to run without /p.'
164 print ' echo >&2 "Retrying without /p."'
165 print ' E="$(echo "${E}" | sed -e \'s/\/p/\//\')"'
167 print ' exec perf record -e $E -b "$@"'
171 if cpufound
== 0 and not args
.all
:
172 sys
.exit('CPU %s not found' % cpu
)
175 sys
.exit('Branch event not found')