Skip to content

Commit 1fcdc2b

Browse files
authored
Add support for profiling benchmarks using perf-record (#214)
* Add support for profiling benchmarks using `perf-record` Add a hook that profiles benchmarks using `perf-record` when enabled * Rename hook to perf_record * Fix docstring * Fix formatting * Remove assignment to unused variable * Update docs * Inherit perf record env vars
1 parent f313174 commit 1fcdc2b

File tree

4 files changed

+79
-1
lines changed

4 files changed

+79
-1
lines changed

doc/run_benchmark.rst

+14
Original file line numberDiff line numberDiff line change
@@ -206,3 +206,17 @@ The `Tools/scripts/summarize_stats.py <https://github.com/python/cpython/blob/ma
206206

207207
Statistics are not cleared between runs.
208208
If you need to delete statistics from a previous run, remove the files in ``/tmp/py_stats`` (Unix) or ``C:\temp\py_stats`` (Windows).
209+
210+
Profiling benchmarks using ``perf record``
211+
==========================================
212+
``pyperf`` supports profiling benchmark execution using ``perf
213+
record``. ``perf`` is only enabled while the benchmark is running to avoid
214+
profiling unrelated parts of ``pyperf`` itself.
215+
216+
One profile data file is generated for each benchmark run. These files have
217+
the basename of ``perf.data.<uuid>`` and are written to the current directory
218+
by default. The directory can be overridden by setting the
219+
``PYPERF_PERF_RECORD_DATA_DIR`` environment variable.
220+
221+
The value of the ``PYPERF_PERF_RECORD_EXTRA_OPTS`` environment variable is
222+
appended to the command line of ``perf record`` if it is provided.

pyperf/_hooks.py

+60
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,12 @@
55

66
import abc
77
import importlib.metadata
8+
import os.path
9+
import shlex
10+
import subprocess
811
import sys
12+
import tempfile
13+
import uuid
914

1015

1116
def get_hooks():
@@ -108,3 +113,58 @@ def __enter__(self):
108113

109114
def __exit__(self, _exc_type, _exc_value, _traceback):
110115
sys._stats_off()
116+
117+
118+
class perf_record(HookBase):
119+
"""Profile the benchmark using perf-record.
120+
121+
Profile data is written to the current directory directory by default, or
122+
to the value of the `PYPERF_PERF_RECORD_DATA_DIR` environment variable, if
123+
it is provided.
124+
125+
Profile data files have a basename of the form `perf.data.<uuid>`
126+
127+
The value of the `PYPERF_PERF_RECORD_EXTRA_OPTS` environment variable is
128+
appended to the command line of perf-record, if provided.
129+
"""
130+
131+
def __init__(self):
132+
self.tempdir = tempfile.TemporaryDirectory()
133+
self.ctl_fifo = self.mkfifo(self.tempdir.name, "ctl_fifo")
134+
self.ack_fifo = self.mkfifo(self.tempdir.name, "ack_fifo")
135+
perf_data_dir = os.environ.get("PYPERF_PERF_RECORD_DATA_DIR", "")
136+
perf_data_basename = f"perf.data.{uuid.uuid4()}"
137+
cmd = ["perf", "record",
138+
"--pid", str(os.getpid()),
139+
"--output", os.path.join(perf_data_dir, perf_data_basename),
140+
"--control", f"fifo:{self.ctl_fifo},{self.ack_fifo}"]
141+
extra_opts = os.environ.get("PYPERF_PERF_RECORD_EXTRA_OPTS", "")
142+
cmd += shlex.split(extra_opts)
143+
self.perf = subprocess.Popen(
144+
cmd, stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)
145+
self.ctl_fd = open(self.ctl_fifo, "w")
146+
self.ack_fd = open(self.ack_fifo, "r")
147+
148+
def __enter__(self):
149+
self.exec_perf_cmd("enable")
150+
151+
def __exit__(self, _exc_type, _exc_value, _traceback):
152+
self.exec_perf_cmd("disable")
153+
154+
def teardown(self, metadata):
155+
try:
156+
self.exec_perf_cmd("stop")
157+
self.perf.wait(timeout=120)
158+
finally:
159+
self.ctl_fd.close()
160+
self.ack_fd.close()
161+
162+
def mkfifo(self, tmpdir, basename):
163+
path = os.path.join(tmpdir, basename)
164+
os.mkfifo(path)
165+
return path
166+
167+
def exec_perf_cmd(self, cmd):
168+
self.ctl_fd.write(f"{cmd}\n")
169+
self.ctl_fd.flush()
170+
self.ack_fd.readline()

pyperf/_utils.py

+4-1
Original file line numberDiff line numberDiff line change
@@ -270,7 +270,10 @@ def create_environ(inherit_environ, locale, copy_all):
270270
env = {}
271271
copy_env = ["PATH", "HOME", "TEMP", "COMSPEC", "SystemRoot", "SystemDrive",
272272
# Python specific variables
273-
"PYTHONPATH", "PYTHON_CPU_COUNT", "PYTHON_GIL"]
273+
"PYTHONPATH", "PYTHON_CPU_COUNT", "PYTHON_GIL",
274+
# Pyperf specific variables
275+
"PYPERF_PERF_RECORD_DATA_DIR", "PYPERF_PERF_RECORD_EXTRA_OPTS",
276+
]
274277
if locale:
275278
copy_env.extend(('LANG', 'LC_ADDRESS', 'LC_ALL', 'LC_COLLATE',
276279
'LC_CTYPE', 'LC_IDENTIFICATION', 'LC_MEASUREMENT',

pyproject.toml

+1
Original file line numberDiff line numberDiff line change
@@ -56,6 +56,7 @@ dev = [
5656
pyperf = "pyperf.__main__:main"
5757

5858
[project.entry-points."pyperf.hook"]
59+
perf_record = "pyperf._hooks:perf_record"
5960
pystats = "pyperf._hooks:pystats"
6061
_test_hook = "pyperf._hooks:_test_hook"
6162

0 commit comments

Comments
 (0)