2018-07-27 14:19:56 +01:00
|
|
|
# Copyright 2013-2015 ARM Limited
|
|
|
|
#
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
# you may not use this file except in compliance with the License.
|
|
|
|
# You may obtain a copy of the License at
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
# See the License for the specific language governing permissions and
|
|
|
|
# limitations under the License.
|
|
|
|
#
|
|
|
|
|
|
|
|
|
|
|
|
# pylint: disable=unused-argument
|
2019-08-15 08:46:17 +01:00
|
|
|
import csv
|
2018-07-27 14:19:56 +01:00
|
|
|
import os
|
|
|
|
import re
|
|
|
|
|
2019-12-03 16:46:26 +00:00
|
|
|
from devlib.collector.perf import PerfCollector
|
2018-07-27 14:19:56 +01:00
|
|
|
|
2021-04-27 16:36:15 +01:00
|
|
|
from wa import Instrument, Parameter, ConfigError
|
2019-08-15 08:46:17 +01:00
|
|
|
from wa.utils.types import list_or_string, list_of_strs, numeric
|
2018-07-27 14:19:56 +01:00
|
|
|
|
|
|
|
PERF_COUNT_REGEX = re.compile(r'^(CPU\d+)?\s*(\d+)\s*(.*?)\s*(\[\s*\d+\.\d+%\s*\])?\s*$')
|
|
|
|
|
|
|
|
|
|
|
|
class PerfInstrument(Instrument):
|
|
|
|
|
|
|
|
name = 'perf'
|
|
|
|
description = """
|
|
|
|
Perf is a Linux profiling with performance counters.
|
2019-08-15 08:46:17 +01:00
|
|
|
Simpleperf is an Android profiling tool with performance counters.
|
|
|
|
|
|
|
|
It is highly recomended to use perf_type = simpleperf when using this instrument
|
|
|
|
on android devices since it recognises android symbols in record mode and is much more stable
|
|
|
|
when reporting record .data files. For more information see simpleperf documentation at:
|
|
|
|
https://android.googlesource.com/platform/system/extras/+/master/simpleperf/doc/README.md
|
2018-07-27 14:19:56 +01:00
|
|
|
|
|
|
|
Performance counters are CPU hardware registers that count hardware events
|
|
|
|
such as instructions executed, cache-misses suffered, or branches
|
|
|
|
mispredicted. They form a basis for profiling applications to trace dynamic
|
|
|
|
control flow and identify hotspots.
|
|
|
|
|
2019-08-15 08:46:17 +01:00
|
|
|
perf accepts options and events. If no option is given the default '-a' is
|
|
|
|
used. For events, the default events for perf are migrations and cs. The default
|
|
|
|
events for simpleperf are raw-cpu-cycles, raw-l1-dcache, raw-l1-dcache-refill, raw-instructions-retired.
|
|
|
|
They both can be specified in the config file.
|
2018-07-27 14:19:56 +01:00
|
|
|
|
|
|
|
Events must be provided as a list that contains them and they will look like
|
|
|
|
this ::
|
|
|
|
|
2019-08-15 08:46:17 +01:00
|
|
|
(for perf_type = perf ) perf_events = ['migrations', 'cs']
|
|
|
|
(for perf_type = simpleperf) perf_events = ['raw-cpu-cycles', 'raw-l1-dcache']
|
|
|
|
|
2018-07-27 14:19:56 +01:00
|
|
|
|
|
|
|
Events can be obtained by typing the following in the command line on the
|
|
|
|
device ::
|
|
|
|
|
|
|
|
perf list
|
2019-08-15 08:46:17 +01:00
|
|
|
simpleperf list
|
2018-07-27 14:19:56 +01:00
|
|
|
|
|
|
|
Whereas options, they can be provided as a single string as following ::
|
|
|
|
|
|
|
|
perf_options = '-a -i'
|
2019-08-15 08:46:17 +01:00
|
|
|
perf_options = '--app com.adobe.reader'
|
2018-07-27 14:19:56 +01:00
|
|
|
|
|
|
|
Options can be obtained by running the following in the command line ::
|
|
|
|
|
|
|
|
man perf-stat
|
|
|
|
"""
|
|
|
|
|
|
|
|
parameters = [
|
2019-08-15 08:46:17 +01:00
|
|
|
Parameter('perf_type', kind=str, allowed_values=['perf', 'simpleperf'], default='perf',
|
|
|
|
global_alias='perf_type', description="""Specifies which type of perf binaries
|
|
|
|
to install. Use simpleperf for collecting perf data on android systems."""),
|
|
|
|
Parameter('command', kind=str, default='stat', allowed_values=['stat', 'record'],
|
|
|
|
global_alias='perf_command', description="""Specifies which perf command to use. If in record mode
|
|
|
|
report command will also be executed and results pulled from target along with raw data
|
|
|
|
file"""),
|
|
|
|
Parameter('events', kind=list_of_strs, global_alias='perf_events',
|
2018-07-27 14:19:56 +01:00
|
|
|
description="""Specifies the events to be counted."""),
|
|
|
|
Parameter('optionstring', kind=list_or_string, default='-a',
|
|
|
|
global_alias='perf_options',
|
|
|
|
description="""Specifies options to be used for the perf command. This
|
|
|
|
may be a list of option strings, in which case, multiple instances of perf
|
|
|
|
will be kicked off -- one for each option string. This may be used to e.g.
|
2019-08-15 08:46:17 +01:00
|
|
|
collected different events from different big.LITTLE clusters. In order to
|
|
|
|
profile a particular application process for android with simpleperf use
|
|
|
|
the --app option e.g. --app com.adobe.reader
|
2018-07-27 14:19:56 +01:00
|
|
|
"""),
|
2019-08-15 08:46:17 +01:00
|
|
|
Parameter('report_option_string', kind=str, global_alias='perf_report_options', default=None,
|
|
|
|
description="""Specifies options to be used to gather report when record command
|
|
|
|
is used. It's highly recommended to use perf_type simpleperf when running on
|
|
|
|
android devices as reporting options are unstable with perf"""),
|
2021-04-26 16:17:39 +01:00
|
|
|
Parameter('run_report_sample', kind=bool, default=False, description="""If true, run
|
|
|
|
'perf/simpleperf report-sample'. It only works with the record command."""),
|
|
|
|
Parameter('report_sample_options', kind=str, default=None,
|
|
|
|
description="""Specifies options to pass to report-samples when run_report_sample
|
|
|
|
is true."""),
|
2018-07-27 14:19:56 +01:00
|
|
|
Parameter('labels', kind=list_of_strs, default=None,
|
|
|
|
global_alias='perf_labels',
|
2019-08-15 08:46:17 +01:00
|
|
|
description="""Provides labels for perf/simpleperf output for each optionstring.
|
|
|
|
If specified, the number of labels must match the number of ``optionstring``\ s.
|
2018-07-27 14:19:56 +01:00
|
|
|
"""),
|
|
|
|
Parameter('force_install', kind=bool, default=False,
|
|
|
|
description="""
|
|
|
|
always install perf binary even if perf is already present on the device.
|
|
|
|
"""),
|
|
|
|
]
|
|
|
|
|
|
|
|
def __init__(self, target, **kwargs):
|
|
|
|
super(PerfInstrument, self).__init__(target, **kwargs)
|
|
|
|
self.collector = None
|
2019-12-03 16:46:26 +00:00
|
|
|
self.outdir = None
|
2018-07-27 14:19:56 +01:00
|
|
|
|
2021-04-27 16:36:15 +01:00
|
|
|
def validate(self):
|
|
|
|
if self.report_option_string and (self.command != "record"):
|
|
|
|
raise ConfigError("report_option_string only works with perf/simpleperf record. Set command to record or remove report_option_string")
|
|
|
|
if self.report_sample_options and (self.command != "record"):
|
|
|
|
raise ConfigError("report_sample_options only works with perf/simpleperf record. Set command to record or remove report_sample_options")
|
|
|
|
if self.run_report_sample and (self.command != "record"):
|
|
|
|
raise ConfigError("run_report_sample only works with perf/simpleperf record. Set command to record or remove run_report_sample")
|
|
|
|
|
2018-07-27 14:19:56 +01:00
|
|
|
def initialize(self, context):
|
2021-04-26 16:17:39 +01:00
|
|
|
if self.report_sample_options:
|
|
|
|
self.run_report_sample = True
|
|
|
|
|
2018-07-27 14:19:56 +01:00
|
|
|
self.collector = PerfCollector(self.target,
|
2019-08-15 08:46:17 +01:00
|
|
|
self.perf_type,
|
|
|
|
self.command,
|
2018-07-27 14:19:56 +01:00
|
|
|
self.events,
|
|
|
|
self.optionstring,
|
2019-08-15 08:46:17 +01:00
|
|
|
self.report_option_string,
|
2021-04-26 16:17:39 +01:00
|
|
|
self.run_report_sample,
|
|
|
|
self.report_sample_options,
|
2018-07-27 14:19:56 +01:00
|
|
|
self.labels,
|
|
|
|
self.force_install)
|
|
|
|
|
|
|
|
def setup(self, context):
|
2019-12-03 16:46:26 +00:00
|
|
|
self.outdir = os.path.join(context.output_directory, self.perf_type)
|
2019-12-20 15:21:18 +00:00
|
|
|
self.collector.set_output(self.outdir)
|
2018-07-27 14:19:56 +01:00
|
|
|
self.collector.reset()
|
|
|
|
|
|
|
|
def start(self, context):
|
|
|
|
self.collector.start()
|
|
|
|
|
|
|
|
def stop(self, context):
|
|
|
|
self.collector.stop()
|
|
|
|
|
|
|
|
def update_output(self, context):
|
|
|
|
self.logger.info('Extracting reports from target...')
|
2019-12-03 16:46:26 +00:00
|
|
|
self.collector.get_data()
|
2018-07-27 14:19:56 +01:00
|
|
|
|
2019-08-15 08:46:17 +01:00
|
|
|
if self.perf_type == 'perf':
|
2019-12-03 16:46:26 +00:00
|
|
|
self._process_perf_output(context)
|
2019-08-15 08:46:17 +01:00
|
|
|
else:
|
2019-12-03 16:46:26 +00:00
|
|
|
self._process_simpleperf_output(context)
|
2019-08-15 08:46:17 +01:00
|
|
|
|
|
|
|
def teardown(self, context):
|
|
|
|
self.collector.reset()
|
|
|
|
|
2019-12-03 16:46:26 +00:00
|
|
|
def _process_perf_output(self, context):
|
2019-08-15 08:46:17 +01:00
|
|
|
if self.command == 'stat':
|
2019-12-03 16:46:26 +00:00
|
|
|
self._process_perf_stat_output(context)
|
2019-08-15 08:46:17 +01:00
|
|
|
elif self.command == 'record':
|
2019-12-03 16:46:26 +00:00
|
|
|
self._process_perf_record_output(context)
|
2019-08-15 08:46:17 +01:00
|
|
|
|
2019-12-03 16:46:26 +00:00
|
|
|
def _process_simpleperf_output(self, context):
|
2019-08-15 08:46:17 +01:00
|
|
|
if self.command == 'stat':
|
2019-12-03 16:46:26 +00:00
|
|
|
self._process_simpleperf_stat_output(context)
|
2019-08-15 08:46:17 +01:00
|
|
|
elif self.command == 'record':
|
2019-12-03 16:46:26 +00:00
|
|
|
self._process_simpleperf_record_output(context)
|
2019-08-15 08:46:17 +01:00
|
|
|
|
2019-12-03 16:46:26 +00:00
|
|
|
def _process_perf_stat_output(self, context):
|
|
|
|
for host_file in os.listdir(self.outdir):
|
2018-07-27 14:19:56 +01:00
|
|
|
label = host_file.split('.out')[0]
|
2019-12-03 16:46:26 +00:00
|
|
|
host_file_path = os.path.join(self.outdir, host_file)
|
2018-07-27 14:19:56 +01:00
|
|
|
context.add_artifact(label, host_file_path, 'raw')
|
|
|
|
with open(host_file_path) as fh:
|
|
|
|
in_results_section = False
|
|
|
|
for line in fh:
|
|
|
|
if 'Performance counter stats' in line:
|
|
|
|
in_results_section = True
|
|
|
|
next(fh) # skip the following blank line
|
2019-08-15 08:46:17 +01:00
|
|
|
if not in_results_section:
|
|
|
|
continue
|
|
|
|
if not line.strip(): # blank line
|
|
|
|
in_results_section = False
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
self._add_perf_stat_metric(line, label, context)
|
2018-07-27 14:19:56 +01:00
|
|
|
|
2019-08-15 08:46:17 +01:00
|
|
|
@staticmethod
|
|
|
|
def _add_perf_stat_metric(line, label, context):
|
|
|
|
line = line.split('#')[0] # comment
|
|
|
|
match = PERF_COUNT_REGEX.search(line)
|
|
|
|
if not match:
|
|
|
|
return
|
|
|
|
classifiers = {}
|
|
|
|
cpu = match.group(1)
|
|
|
|
if cpu is not None:
|
|
|
|
classifiers['cpu'] = int(cpu.replace('CPU', ''))
|
|
|
|
count = int(match.group(2))
|
|
|
|
metric = '{}_{}'.format(label, match.group(3))
|
|
|
|
context.add_metric(metric, count, classifiers=classifiers)
|
|
|
|
|
2019-12-03 16:46:26 +00:00
|
|
|
def _process_perf_record_output(self, context):
|
|
|
|
for host_file in os.listdir(self.outdir):
|
2019-08-15 08:46:17 +01:00
|
|
|
label, ext = os.path.splitext(host_file)
|
2019-12-03 16:46:26 +00:00
|
|
|
context.add_artifact(label, os.path.join(self.outdir, host_file), 'raw')
|
2019-08-15 08:46:17 +01:00
|
|
|
column_headers = []
|
|
|
|
column_header_indeces = []
|
|
|
|
event_type = ''
|
|
|
|
if ext == '.rpt':
|
2019-12-03 16:46:26 +00:00
|
|
|
with open(os.path.join(self.outdir, host_file)) as fh:
|
2019-08-15 08:46:17 +01:00
|
|
|
for line in fh:
|
|
|
|
words = line.split()
|
|
|
|
if not words:
|
|
|
|
continue
|
|
|
|
event_type = self._get_report_event_type(words, event_type)
|
|
|
|
column_headers = self._get_report_column_headers(column_headers, words, 'perf')
|
|
|
|
for column_header in column_headers:
|
|
|
|
column_header_indeces.append(line.find(column_header))
|
|
|
|
self._add_report_metric(column_headers,
|
|
|
|
column_header_indeces,
|
|
|
|
line,
|
|
|
|
words,
|
|
|
|
context,
|
|
|
|
event_type,
|
|
|
|
label)
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def _get_report_event_type(words, event_type):
|
|
|
|
if words[0] != '#':
|
|
|
|
return event_type
|
|
|
|
if len(words) == 6 and words[4] == 'event':
|
|
|
|
event_type = words[5]
|
|
|
|
event_type = event_type.strip("'")
|
|
|
|
return event_type
|
|
|
|
|
2019-12-03 16:46:26 +00:00
|
|
|
def _process_simpleperf_stat_output(self, context):
|
2019-08-15 08:46:17 +01:00
|
|
|
labels = []
|
2019-12-03 16:46:26 +00:00
|
|
|
for host_file in os.listdir(self.outdir):
|
2019-08-15 08:46:17 +01:00
|
|
|
labels.append(host_file.split('.out')[0])
|
|
|
|
for opts, label in zip(self.optionstring, labels):
|
2019-12-03 16:46:26 +00:00
|
|
|
stat_file = os.path.join(self.outdir, '{}{}'.format(label, '.out'))
|
2019-08-15 08:46:17 +01:00
|
|
|
if '--csv' in opts:
|
|
|
|
self._process_simpleperf_stat_from_csv(stat_file, context, label)
|
|
|
|
else:
|
|
|
|
self._process_simpleperf_stat_from_raw(stat_file, context, label)
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def _process_simpleperf_stat_from_csv(stat_file, context, label):
|
|
|
|
with open(stat_file) as csv_file:
|
|
|
|
readCSV = csv.reader(csv_file, delimiter=',')
|
|
|
|
line_num = 0
|
|
|
|
for row in readCSV:
|
perf: correctly parse csv when using "--csv --interval-only-values"
With the perf instrument configured as:
perf:
perf_type: simpleperf
command: stat
optionstring: '-a --interval-only-values --csv'
WA fails to parse simpleperf's output:
INFO Extracting reports from target...
ERROR Error in instrument perf
ERROR File "/work/workload_automation/workload-automation/wa/framework/instrument.py", line 272, in __call__
ERROR self.callback(context)
ERROR File "/work/workload_automation/workload-automation/wa/instruments/perf.py", line 142, in update_output
ERROR self._process_simpleperf_output(context)
ERROR File "/work/workload_automation/workload-automation/wa/instruments/perf.py", line 155, in _process_simpleperf_output
ERROR self._process_simpleperf_stat_output(context)
ERROR File "/work/workload_automation/workload-automation/wa/instruments/perf.py", line 233, in _process_simpleperf_stat_output
ERROR self._process_simpleperf_stat_from_csv(stat_file, context, label)
ERROR File "/work/workload_automation/workload-automation/wa/instruments/perf.py", line 245, in _process_simpleperf_stat_from_csv
ERROR context.add_metric('{}_{}'.format(label, row[1]), row[0], 'count', classifiers=classifiers)
ERROR File "/work/workload_automation/workload-automation/wa/framework/execution.py", line 222, in add_metric
ERROR self.output.add_metric(name, value, units, lower_is_better, classifiers)
ERROR File "/work/workload_automation/workload-automation/wa/framework/output.py", line 142, in add_metric
ERROR self.result.add_metric(name, value, units, lower_is_better, classifiers)
ERROR File "/work/workload_automation/workload-automation/wa/framework/output.py", line 390, in add_metric
ERROR metric = Metric(name, value, units, lower_is_better, classifiers)
ERROR File "/work/workload_automation/workload-automation/wa/framework/output.py", line 653, in __init__
ERROR self.value = numeric(value)
ERROR File "/work/workload_automation/devlib/devlib/utils/types.py", line 88, in numeric
ERROR raise ValueError('Not numeric: {}'.format(value))
ERROR
ERROR ValueError(Not numeric: Performance counter statistics)
With the above options, the csv that simpleperf produces looks like
this:
Performance counter statistics,
123456789,raw-l1-dtlb,,(60%),
42424242,raw-l1-itlb,,(60%),
Total test time,1.001079,seconds,
Performance counter statistics,
123456789,raw-l1-dtlb,,(60%),
42424242,raw-l1-itlb,,(60%),
Total test time,2.001178,seconds,
Performance counter statistics,
[...]
That is, with "--interval-only-values", the "Performance counter
statistics," header is repeated every interval. WA current expects
it only in the first line. Modify the condition so that it is ignored
every time we find it in the file and not just the first time.
2021-01-11 16:17:34 +00:00
|
|
|
if 'Performance counter statistics' not in row and 'Total test time' not in row:
|
2019-08-15 08:46:17 +01:00
|
|
|
classifiers = {'scaled from(%)': row[len(row) - 2].replace('(', '').replace(')', '').replace('%', '')}
|
|
|
|
context.add_metric('{}_{}'.format(label, row[1]), row[0], 'count', classifiers=classifiers)
|
|
|
|
line_num += 1
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def _process_simpleperf_stat_from_raw(stat_file, context, label):
|
|
|
|
with open(stat_file) as fh:
|
|
|
|
for line in fh:
|
2022-08-06 17:53:52 +01:00
|
|
|
if '#' in line and not line.startswith('#'):
|
|
|
|
units = 'count'
|
|
|
|
if "(ms)" in line:
|
|
|
|
line = line.replace("(ms)", "")
|
|
|
|
units = 'ms'
|
2019-08-15 08:46:17 +01:00
|
|
|
tmp_line = line.split('#')[0]
|
|
|
|
tmp_line = line.strip()
|
|
|
|
count, metric = tmp_line.split(' ')[0], tmp_line.split(' ')[2]
|
2022-08-06 17:53:52 +01:00
|
|
|
count = float(count) if "." in count else int(count.replace(',', ''))
|
2019-08-15 08:46:17 +01:00
|
|
|
scaled_percentage = line.split('(')[1].strip().replace(')', '').replace('%', '')
|
|
|
|
scaled_percentage = int(scaled_percentage)
|
|
|
|
metric = '{}_{}'.format(label, metric)
|
2022-08-06 17:53:52 +01:00
|
|
|
context.add_metric(metric, count, units, classifiers={'scaled from(%)': scaled_percentage})
|
2019-08-15 08:46:17 +01:00
|
|
|
|
2019-12-03 16:46:26 +00:00
|
|
|
def _process_simpleperf_record_output(self, context):
|
|
|
|
for host_file in os.listdir(self.outdir):
|
2019-08-15 08:46:17 +01:00
|
|
|
label, ext = os.path.splitext(host_file)
|
2019-12-03 16:46:26 +00:00
|
|
|
context.add_artifact(label, os.path.join(self.outdir, host_file), 'raw')
|
2019-08-15 08:46:17 +01:00
|
|
|
if ext != '.rpt':
|
|
|
|
continue
|
|
|
|
column_headers = []
|
|
|
|
column_header_indeces = []
|
|
|
|
event_type = ''
|
2019-12-03 16:46:26 +00:00
|
|
|
with open(os.path.join(self.outdir, host_file)) as fh:
|
2019-08-15 08:46:17 +01:00
|
|
|
for line in fh:
|
|
|
|
words = line.split()
|
|
|
|
if not words:
|
|
|
|
continue
|
|
|
|
if words[0] == 'Event:':
|
|
|
|
event_type = words[1]
|
|
|
|
column_headers = self._get_report_column_headers(column_headers,
|
|
|
|
words,
|
|
|
|
'simpleperf')
|
|
|
|
for column_header in column_headers:
|
|
|
|
column_header_indeces.append(line.find(column_header))
|
|
|
|
self._add_report_metric(column_headers,
|
|
|
|
column_header_indeces,
|
|
|
|
line,
|
|
|
|
words,
|
|
|
|
context,
|
|
|
|
event_type,
|
|
|
|
label)
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def _get_report_column_headers(column_headers, words, perf_type):
|
|
|
|
if 'Overhead' not in words:
|
|
|
|
return column_headers
|
|
|
|
if perf_type == 'perf':
|
|
|
|
words.remove('#')
|
|
|
|
column_headers = words
|
|
|
|
# Concatonate Shared Objects header
|
|
|
|
if 'Shared' in column_headers:
|
|
|
|
shared_index = column_headers.index('Shared')
|
|
|
|
column_headers[shared_index:shared_index + 2] = ['{} {}'.format(column_headers[shared_index],
|
|
|
|
column_headers[shared_index + 1])]
|
|
|
|
return column_headers
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def _add_report_metric(column_headers, column_header_indeces, line, words, context, event_type, label):
|
|
|
|
if '%' not in words[0]:
|
|
|
|
return
|
|
|
|
classifiers = {}
|
|
|
|
for i in range(1, len(column_headers)):
|
|
|
|
classifiers[column_headers[i]] = line[column_header_indeces[i]:column_header_indeces[i + 1]].strip()
|
|
|
|
|
|
|
|
context.add_metric('{}_{}_Overhead'.format(label, event_type),
|
|
|
|
numeric(words[0].strip('%')),
|
|
|
|
'percent',
|
|
|
|
classifiers=classifiers)
|