f27a123653
This patch adds a basic framework to do performance testing for GDB. perftest.py is about the test case, testresult.py is about test results, and how are they saved. reporter.py is about how results are reported (in what format). measure.py is about measuring the execution of tests by a collection of measurements. In V5: - Simplify perftest.exp. In V4: - Rename MeasurementCPUTime to MeasurementCpuTime, - Add 'pass' in empty method, - Simplify string comparison in perftest.exp. - Rename GDB_PERFORMANCE to GDB_PERFTEST_MODE and rename GDB_PERFORMANCE_TIMEOUT to GDB_PERFTEST_TIMEOUT. In V3, there are some changes, - Add wall time measurement, cpu time measurement and vmsize measurement. - Rename SingleStatisticTestCase to TestCaseWithBasicMeasurements, which measures cpu time, wall time, and memory (vmsize). - GDB_PERFORMANCE=run|compile|both to control the mode of perf testing. - New GDB_PERFORMANCE_TIMEOUT to specify the timeout. - Split proc prepare to proc compile and startup. - Disable GC while doing measurements. In V2, there are several changes to address Doug and Sanimir's comments. - Add copyright header and docstring in perftest/__init__.py - Remove config.py. - Fix docstring format. - Rename classes "SingleVariable" to "SingleStatistic". - Don't extend gdb.Function in class TestCase. Add a new method run to run the test case so that we can pass parameters to test. - Allow to customize whether to warm up and to append test log. - Move time measurement into test harness. Add a new class Measurement for a specific measurement and a new class Measure to measure them for a given test case. - A new class ResultFactory to create instances of TestResult. - New file lib/perftest.exp, which is to do some preparations and cleanups to simplify each *.exp file. - Skip compilation step if GDB_PERFORMANCE_SKIP_COMPILE is set. gdb/testsuite/ 2013-11-06 Yao Qi <yao@codesourcery.com> * lib/perftest.exp: New. * gdb.perf/lib/perftest/__init__.py: New. * gdb.perf/lib/perftest/measure.py: New. * gdb.perf/lib/perftest/perftest.py: New. * gdb.perf/lib/perftest/reporter.py: New. * gdb.perf/lib/perftest/testresult.py: New.
146 lines
4.4 KiB
Python
146 lines
4.4 KiB
Python
# Copyright (C) 2013 Free Software Foundation, Inc.
|
|
|
|
# This program is free software; you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation; either version 3 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# This program 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 General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
import time
|
|
import os
|
|
import gc
|
|
|
|
class Measure(object):
|
|
"""A class that measure and collect the interesting data for a given testcase.
|
|
|
|
An instance of Measure has a collection of measurements, and each
|
|
of them is to measure a given aspect, such as time and memory.
|
|
"""
|
|
|
|
def __init__(self, measurements):
|
|
"""Constructor of measure.
|
|
|
|
measurements is a collection of Measurement objects.
|
|
"""
|
|
|
|
self.measurements = measurements
|
|
|
|
def measure(self, func, id):
|
|
"""Measure the operations done by func with a collection of measurements."""
|
|
# Enable GC, force GC and disable GC before running test in order to reduce
|
|
# the interference from GC.
|
|
gc.enable()
|
|
gc.collect()
|
|
gc.disable()
|
|
|
|
for m in self.measurements:
|
|
m.start(id)
|
|
|
|
func()
|
|
|
|
for m in self.measurements:
|
|
m.stop(id)
|
|
|
|
gc.enable()
|
|
|
|
def report(self, reporter, name):
|
|
"""Report the measured results."""
|
|
for m in self.measurements:
|
|
m.report(reporter, name)
|
|
|
|
class Measurement(object):
|
|
"""A measurement for a certain aspect."""
|
|
|
|
def __init__(self, name, result):
|
|
"""Constructor of Measurement.
|
|
|
|
Attribute result is the TestResult associated with measurement.
|
|
"""
|
|
self.name = name;
|
|
self.result = result
|
|
|
|
def start(self, id):
|
|
"""Abstract method to start the measurement."""
|
|
raise NotImplementedError("Abstract Method:start")
|
|
|
|
def stop(self, id):
|
|
"""Abstract method to stop the measurement.
|
|
|
|
When the measurement is stopped, we've got something, and
|
|
record them in result.
|
|
"""
|
|
raise NotImplementedError("Abstract Method:stop.")
|
|
|
|
def report(self, reporter, name):
|
|
"""Report the measured data by argument reporter."""
|
|
self.result.report(reporter, name + " " + self.name)
|
|
|
|
class MeasurementCpuTime(Measurement):
|
|
"""Measurement on CPU time."""
|
|
# On UNIX, time.clock() measures the amount of CPU time that has
|
|
# been used by the current process. On Windows it will measure
|
|
# wall-clock seconds elapsed since the first call to the function.
|
|
# Something other than time.clock() should be used to measure CPU
|
|
# time on Windows.
|
|
|
|
def __init__(self, result):
|
|
super(MeasurementCpuTime, self).__init__("cpu_time", result)
|
|
self.start_time = 0
|
|
|
|
def start(self, id):
|
|
self.start_time = time.clock()
|
|
|
|
def stop(self, id):
|
|
if os.name == 'nt':
|
|
cpu_time = 0
|
|
else:
|
|
cpu_time = time.clock() - self.start_time
|
|
self.result.record (id, cpu_time)
|
|
|
|
class MeasurementWallTime(Measurement):
|
|
"""Measurement on Wall time."""
|
|
|
|
def __init__(self, result):
|
|
super(MeasurementWallTime, self).__init__("wall_time", result)
|
|
self.start_time = 0
|
|
|
|
def start(self, id):
|
|
self.start_time = time.time()
|
|
|
|
def stop(self, id):
|
|
wall_time = time.time() - self.start_time
|
|
self.result.record (id, wall_time)
|
|
|
|
class MeasurementVmSize(Measurement):
|
|
"""Measurement on memory usage represented by VmSize."""
|
|
|
|
def __init__(self, result):
|
|
super(MeasurementVmSize, self).__init__("vmsize", result)
|
|
|
|
def _compute_process_memory_usage(self, key):
|
|
file_path = "/proc/%d/status" % os.getpid()
|
|
try:
|
|
t = open(file_path)
|
|
v = t.read()
|
|
t.close()
|
|
except:
|
|
return 0
|
|
i = v.index(key)
|
|
v = v[i:].split(None, 3)
|
|
if len(v) < 3:
|
|
return 0
|
|
return int(v[1])
|
|
|
|
def start(self, id):
|
|
pass
|
|
|
|
def stop(self, id):
|
|
memory_used = self._compute_process_memory_usage("VmSize:")
|
|
self.result.record (id, memory_used)
|