b115e819af
This patch allows one to provide the function name using an optional -base option to compare all other functions against. This is useful when pitching one implementation of a string function against alternatives. In the absence of this option, comparisons are done against the first ifunc in the list. * benchtests/scripts/compare_strings.py (main): Add an optional -base option. (process_results): New argument base_func.
138 lines
4.1 KiB
Python
Executable File
138 lines
4.1 KiB
Python
Executable File
#!/usr/bin/python
|
|
# Copyright (C) 2017 Free Software Foundation, Inc.
|
|
# This file is part of the GNU C Library.
|
|
#
|
|
# The GNU C Library is free software; you can redistribute it and/or
|
|
# modify it under the terms of the GNU Lesser General Public
|
|
# License as published by the Free Software Foundation; either
|
|
# version 2.1 of the License, or (at your option) any later version.
|
|
#
|
|
# The GNU C Library is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
# Lesser General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU Lesser General Public
|
|
# License along with the GNU C Library; if not, see
|
|
# <http://www.gnu.org/licenses/>.
|
|
"""Compare results of string functions
|
|
|
|
Given a string benchmark result file, print a table with comparisons with a
|
|
baseline. The baseline is the first function, which typically is the builtin
|
|
function.
|
|
"""
|
|
|
|
import sys
|
|
import os
|
|
import json
|
|
import pylab
|
|
|
|
try:
|
|
import jsonschema as validator
|
|
except ImportError:
|
|
print('Could not find jsonschema module.')
|
|
raise
|
|
|
|
|
|
def parse_file(filename, schema_filename):
|
|
with open(schema_filename, 'r') as schemafile:
|
|
schema = json.load(schemafile)
|
|
with open(filename, 'r') as benchfile:
|
|
bench = json.load(benchfile)
|
|
validator.validate(bench, schema)
|
|
return bench
|
|
|
|
|
|
def draw_graph(f, v, ifuncs, results):
|
|
"""Plot graphs for functions
|
|
|
|
Plot line graphs for each of the ifuncs
|
|
|
|
Args:
|
|
f: Function name
|
|
v: Benchmark variant for the function.
|
|
ifuncs: List of ifunc names
|
|
results: Dictionary of results for each test criterion
|
|
"""
|
|
xkeys = results.keys()
|
|
|
|
pylab.clf()
|
|
fig = pylab.figure(frameon=False)
|
|
fig.set_size_inches(32, 18)
|
|
pylab.ylabel('Performance improvement from base')
|
|
X = range(len(xkeys))
|
|
pylab.xticks(X, xkeys)
|
|
|
|
i = 0
|
|
|
|
while i < len(ifuncs):
|
|
Y = [results[k][i] for k in xkeys]
|
|
lines = pylab.plot(X, Y, label=':'+ifuncs[i])
|
|
i = i + 1
|
|
|
|
pylab.legend()
|
|
pylab.grid()
|
|
pylab.savefig('%s-%s.png' % (f, v), bbox_inches='tight')
|
|
|
|
|
|
def process_results(results, attrs, base_func):
|
|
""" Process results and print them
|
|
|
|
Args:
|
|
results: JSON dictionary of results
|
|
attrs: Attributes that form the test criteria
|
|
"""
|
|
|
|
for f in results['functions'].keys():
|
|
print('Function: %s' % f)
|
|
base_index = 0
|
|
if base_func:
|
|
base_index = results['functions'][f]['ifuncs'].index(base_func)
|
|
|
|
print('\t'.join(results['functions'][f]['ifuncs']))
|
|
v = results['functions'][f]['bench-variant']
|
|
print('Variant: %s' % v)
|
|
print("=" * 80)
|
|
graph_res = {}
|
|
for res in results['functions'][f]['results']:
|
|
attr_list = ['%s=%s' % (a, res[a]) for a in attrs]
|
|
i = 0
|
|
key = ','.join(attr_list)
|
|
sys.stdout.write('%s: \t' % key)
|
|
graph_res[key] = res['timings']
|
|
for t in res['timings']:
|
|
sys.stdout.write ('%.2f' % t)
|
|
if i != base_index:
|
|
diff = (res['timings'][base_index] - t) * 100 / res['timings'][base_index]
|
|
sys.stdout.write (' (%.2f%%)' % diff)
|
|
sys.stdout.write('\t')
|
|
i = i + 1
|
|
print('')
|
|
draw_graph(f, v, results['functions'][f]['ifuncs'], graph_res)
|
|
|
|
|
|
def main(args):
|
|
"""Program Entry Point
|
|
|
|
Take a string benchmark output file and compare timings.
|
|
"""
|
|
if len(args) < 3:
|
|
print('Usage: %s <input file> <schema file> [-base=ifunc_name] attr1 [attr2 ...]' % sys.argv[0])
|
|
sys.exit(os.EX_USAGE)
|
|
|
|
base_func = None
|
|
filename = args[0]
|
|
schema_filename = args[1]
|
|
if args[2].find('-base=') == 0:
|
|
base_func = args[2][6:]
|
|
attrs = args[3:]
|
|
else:
|
|
attrs = args[2:]
|
|
|
|
results = parse_file(filename, schema_filename)
|
|
process_results(results, attrs, base_func)
|
|
|
|
|
|
if __name__ == '__main__':
|
|
main(sys.argv[1:])
|