๐
Here are some recent and important revisions. ๐ Complete list of results.
Key: ๐: table, ๐: time plot, ๐ง : memory plot
Most recent pystats on main (86c1a60)
date | fork/ref | hash/flags | vs. 3.10.4: | vs. 3.12.0: | vs. 3.13.0: | vs. base: |
---|---|---|---|---|---|---|
2025-01-23 | mdboom/aa_test_2025_2 | 8ffb2c1 (NOGIL) | 1.086x โ ๐๐ |
1.136x โ ๐๐ |
1.137x โ ๐๐ |
1.001x โ ๐๐๐ง |
2025-01-23 | mdboom/aa_test_2025_2 | 8ffb2c1 | 1.318x โ ๐๐ |
1.033x โ ๐๐ |
1.041x โ ๐๐ |
1.002x โ ๐๐๐ง |
2025-01-22 | nascheme/1b4e8c39e99ce39b39c7 | 1b4e8c3 (NOGIL) | 1.093x โ ๐๐ |
1.134x โ ๐๐ |
1.137x โ ๐๐ |
1.001x โ ๐๐๐ง |
2025-01-22 | diegorusso/fplt | 707b019 (JIT) | 1.204x โ ๐๐ |
1.047x โ ๐๐ |
1.042x โ ๐๐ |
1.004x โ ๐๐๐ง |
2025-01-22 | python/a16ded10ad3952406280 | a16ded1 (JIT) | 1.216x โ ๐๐ |
1.038x โ ๐๐ |
1.033x โ ๐๐ |
|
2025-01-21 | python/f18b2264929c56360c86 | f18b226 | 1.311x โ ๐๐ |
1.032x โ ๐๐ |
1.035x โ ๐๐ |
|
2025-01-21 | python/f18b2264929c56360c86 | f18b226 (NOGIL) | 1.083x โ ๐๐ |
1.138x โ ๐๐ |
1.139x โ ๐๐ |
1.167x โ ๐๐๐ง |
2025-01-21 | mdboom/aa_test_2025 | 4844db8 (NOGIL) | 1.084x โ ๐๐ |
1.138x โ ๐๐ |
1.140x โ ๐๐ |
1.000x โ ๐๐๐ง |
2025-01-21 | mdboom/aa_test_2025 | 4844db8 | 1.319x โ ๐๐ |
1.036x โ ๐๐ |
1.041x โ ๐๐ |
1.001x โ ๐๐๐ง |
2025-01-17 | python/3829104ab412a47bf3f3 | 3829104 (NOGIL) | 1.090x โ ๐๐ |
1.132x โ ๐๐ |
1.135x โ ๐๐ |
date | fork/ref | hash/flags | vs. 3.10.4: | vs. 3.12.0: | vs. 3.13.0: | vs. base: |
---|---|---|---|---|---|---|
2025-01-24 | python/7907203bc07387ff2d8e | 7907203 (JIT) | 1.433x โ ๐๐ |
1.104x โ ๐๐ |
1.034x โ ๐๐ |
|
2025-01-24 | faster-cpython/remove_most_conditio | 1e0f842 (NOGIL) | 1.230x โ ๐๐ |
1.045x โ ๐๐ |
1.105x โ ๐๐ |
1.010x โ ๐๐๐ง |
2025-01-24 | brandtbucher/justin_mcmodel_again | e88395f (JIT) | 1.441x โ ๐๐ |
1.111x โ ๐๐ |
1.039x โ ๐๐ |
1.005x โ ๐๐๐ง |
2025-01-23 | faster-cpython/remove_most_conditio | 584015a (NOGIL) | 1.226x โ ๐๐ |
1.048x โ ๐๐ |
1.107x โ ๐๐ |
1.013x โ ๐๐๐ง |
2025-01-23 | mdboom/aa_test_2025_2 | 8ffb2c1 (NOGIL) | 1.235x โ ๐๐ |
1.039x โ ๐๐ |
1.100x โ ๐๐ |
1.002x โ ๐๐๐ง |
2025-01-23 | mdboom/aa_test_2025_2 | 8ffb2c1 | 1.439x โ ๐๐ |
1.108x โ ๐๐ |
1.040x โ ๐๐ |
1.002x โ ๐๐๐ง |
2025-01-23 | python/a10f99375e7912df863c | a10f993 (NOGIL) | 1.244x โ ๐๐ |
1.034x โ ๐๐ |
1.095x โ ๐๐ |
|
2025-01-22 | nascheme/1b4e8c39e99ce39b39c7 | 1b4e8c3 (NOGIL) | 1.241x โ ๐๐ |
1.041x โ ๐๐ |
1.102x โ ๐๐ |
1.003x โ ๐๐๐ง |
2025-01-22 | python/86c1a60d5a28cfb51f88 | 86c1a60 (JIT) | 1.434x โ ๐๐ |
1.107x โ ๐๐ |
1.035x โ ๐๐ |
1.010x โ ๐๐๐ง |
2025-01-22 | python/86c1a60d5a28cfb51f88 | 86c1a60 | 1.447x โ ๐๐ |
1.114x โ ๐๐ |
1.046x โ ๐๐ |
|
2025-01-21 | python/f18b2264929c56360c86 | f18b226 | 1.442x โ ๐๐ |
1.110x โ ๐๐ |
1.042x โ ๐๐ |
|
2025-01-21 | python/f18b2264929c56360c86 | f18b226 (NOGIL) | 1.232x โ ๐๐ |
1.040x โ ๐๐ |
1.102x โ ๐๐ |
1.138x โ ๐๐๐ง |
2025-01-21 | brandtbucher/remove_optimizer_api | 085e172 | 1.445x โ ๐๐ |
1.112x โ ๐๐ |
1.044x โ ๐๐ |
1.002x โ ๐๐๐ง |
2025-01-21 | brandtbucher/remove_optimizer_api | 085e172 (JIT) | 1.429x โ ๐๐ |
1.104x โ ๐๐ |
1.033x โ ๐๐ |
1.002x โ ๐๐๐ง |
2025-01-21 | mdboom/aa_test_2025 | 4844db8 (NOGIL) | 1.232x โ ๐๐ |
1.042x โ ๐๐ |
1.103x โ ๐๐ |
1.000x โ ๐๐๐ง |
2025-01-21 | mdboom/aa_test_2025 | 4844db8 | 1.447x โ ๐๐ |
1.114x โ ๐๐ |
1.045x โ ๐๐ |
1.003x โ ๐๐๐ง |
2025-01-21 | iritkatriel/binaryops | 6476205 | 1.452x โ ๐๐ |
1.117x โ ๐๐ |
1.049x โ ๐๐ |
1.002x โ ๐๐๐ง |
2025-01-20 | faster-cpython/no_conditional_stack | d5e47ea (NOGIL) | 1.228x โ ๐๐ |
1.044x โ ๐๐ |
1.105x โ ๐๐ |
1.008x โ ๐๐๐ง |
2025-01-20 | python/f0f7b978be84c432139d | f0f7b97 (NOGIL) | 1.238x โ ๐๐ |
1.036x โ ๐๐ |
1.097x โ ๐๐ |
|
2025-01-17 | python/3829104ab412a47bf3f3 | 3829104 (NOGIL) | 1.241x โ ๐๐ |
1.035x โ ๐๐ |
1.096x โ ๐๐ |
date | fork/ref | hash/flags | vs. 3.10.4: | vs. 3.12.0: | vs. 3.13.0: | vs. base: |
---|---|---|---|---|---|---|
2025-01-24 | faster-cpython/remove_most_conditio | 1e0f842 | 1.363x โ ๐๐ |
1.055x โ ๐๐ |
1.067x โ ๐๐ |
1.000x โ ๐๐๐ง |
2025-01-23 | faster-cpython/remove_most_conditio | 584015a | 1.351x โ ๐๐ |
1.046x โ ๐๐ |
1.058x โ ๐๐ |
1.008x โ ๐๐๐ง |
2025-01-23 | mdboom/aa_test_2025_2 | 8ffb2c1 (NOGIL) | 1.182x โ ๐๐ |
1.073x โ ๐๐ |
1.064x โ ๐๐ |
1.000x โ ๐๐๐ง |
2025-01-23 | mdboom/aa_test_2025_2 | 8ffb2c1 | 1.353x โ ๐๐ |
1.050x โ ๐๐ |
1.062x โ ๐๐ |
1.000x โ ๐๐๐ง |
2025-01-23 | python/a10f99375e7912df863c | a10f993 | 1.361x โ ๐๐ |
1.053x โ ๐๐ |
1.067x โ ๐๐ |
|
2025-01-22 | nascheme/1b4e8c39e99ce39b39c7 | 1b4e8c3 (NOGIL) | 1.181x โ ๐๐ |
1.080x โ ๐๐ |
1.071x โ ๐๐ |
1.003x โ ๐๐๐ง |
2025-01-22 | faster-cpython/close_escapes | 08894e6 | 1.348x โ ๐๐ |
1.046x โ ๐๐ |
1.058x โ ๐๐ |
1.003x โ ๐๐๐ง |
2025-01-22 | python/470a0a68ebbbb4254f1a | 470a0a6 | 1.352x โ ๐๐ |
1.050x โ ๐๐ |
1.061x โ ๐๐ |
|
2025-01-21 | python/f18b2264929c56360c86 | f18b226 | 1.353x โ ๐๐ |
1.049x โ ๐๐ |
1.061x โ ๐๐ |
|
2025-01-21 | python/f18b2264929c56360c86 | f18b226 (NOGIL) | 1.182x โ ๐๐ |
1.073x โ ๐๐ |
1.064x โ ๐๐ |
1.118x โ ๐๐๐ง |
2025-01-21 | mdboom/aa_test_2025 | 4844db8 (NOGIL) | 1.188x โ ๐๐ |
1.069x โ ๐๐ |
1.061x โ ๐๐ |
1.004x โ ๐๐๐ง |
2025-01-21 | mdboom/aa_test_2025 | 4844db8 | 1.353x โ ๐๐ |
1.050x โ ๐๐ |
1.062x โ ๐๐ |
1.001x โ ๐๐๐ง |
2025-01-17 | python/3829104ab412a47bf3f3 | 3829104 (NOGIL) | 1.181x โ ๐๐ |
1.075x โ ๐๐ |
1.066x โ ๐๐ |
date | fork/ref | hash/flags | vs. 3.10.4: | vs. 3.12.0: | vs. 3.13.0: | vs. base: |
---|---|---|---|---|---|---|
2025-01-23 | mdboom/test_without_pgo_wor | 71a13ea | 1.164x โ ๐๐ |
1.013x โ ๐๐ |
1.039x โ ๐๐ |
1.004x โ ๐๐ |
2025-01-23 | mdboom/aa_test_2025_2 | 8ffb2c1 | 1.186x โ ๐๐ |
1.006x โ ๐๐ |
1.022x โ ๐๐ |
1.002x โ ๐๐ |
2025-01-21 | python/f18b2264929c56360c86 | f18b226 | 1.191x โ ๐๐ |
1.011x โ ๐๐ |
1.019x โ ๐๐ |
|
2025-01-21 | mdboom/aa_test_2025 | 4844db8 | 1.183x โ ๐๐ |
1.004x โ ๐๐ |
1.023x โ ๐๐ |
1.004x โ ๐๐ |
date | fork/ref | hash/flags | vs. 3.10.4: | vs. 3.12.0: | vs. 3.13.0: | vs. base: |
---|---|---|---|---|---|---|
2025-01-23 | mdboom/test_without_pgo_wor | 71a13ea | 1.113x โ ๐๐ |
1.123x โ ๐๐ |
1.001x โ ๐๐ |
1.003x โ ๐๐ |
2025-01-23 | mdboom/aa_test_2025_2 | 8ffb2c1 | 1.116x โ ๐๐ |
1.124x โ ๐๐ |
1.001x โ ๐๐ |
1.003x โ ๐๐ |
2025-01-21 | python/f18b2264929c56360c86 | f18b226 | 1.119x โ ๐๐ |
1.128x โ ๐๐ |
1.003x โ ๐๐ |
|
2025-01-21 | mdboom/aa_test_2025 | 4844db8 | 1.120x โ ๐๐ |
1.128x โ ๐๐ |
1.003x โ ๐๐ |
1.001x โ ๐๐ |
date | fork/ref | hash/flags | vs. 3.10.4: | vs. 3.12.0: | vs. 3.13.0: | vs. base: |
---|---|---|---|---|---|---|
2025-01-23 | mdboom/aa_test_2025_2 | 8ffb2c1 (NOGIL) | 1.257x โ ๐๐ |
1.001x โ ๐๐ |
1.004x โ ๐๐ |
1.000x โ ๐๐๐ง |
2025-01-23 | mdboom/aa_test_2025_2 | 8ffb2c1 | 1.359x โ ๐๐ |
1.068x โ ๐๐ |
1.072x โ ๐๐ |
1.002x โ ๐๐๐ง |
2025-01-22 | nascheme/1b4e8c39e99ce39b39c7 | 1b4e8c3 (NOGIL) | 1.295x โ ๐๐ |
1.032x โ ๐๐ |
1.035x โ ๐๐ |
1.007x โ ๐๐๐ง |
2025-01-21 | python/f18b2264929c56360c86 | f18b226 | 1.363x โ ๐๐ |
1.072x โ ๐๐ |
1.075x โ ๐๐ |
|
2025-01-21 | python/f18b2264929c56360c86 | f18b226 (NOGIL) | 1.257x โ ๐๐ |
1.000x โ ๐๐ |
1.004x โ ๐๐ |
1.067x โ ๐๐๐ง |
2025-01-21 | mdboom/aa_test_2025 | 4844db8 (NOGIL) | 1.258x โ ๐๐ |
1.002x โ ๐๐ |
1.005x โ ๐๐ |
1.001x โ ๐๐๐ง |
2025-01-21 | mdboom/aa_test_2025 | 4844db8 | 1.359x โ ๐๐ |
1.069x โ ๐๐ |
1.072x โ ๐๐ |
1.002x โ ๐๐๐ง |
2025-01-17 | python/3829104ab412a47bf3f3 | 3829104 (NOGIL) | 1.295x โ ๐๐ |
1.040x โ ๐๐ |
1.044x โ ๐๐ |
*
indicates that the exact same versions of pyperformance was not used.
For the results above, the "faster/slower" result is a geometric mean of each of the benchmarks.
Below are longitudinal timing results. There are also ๐ง longitudinal memory results.
The results have a resolution of 0.001 (0.1%).
- linux: Intelยฎ Xeonยฎ W-2255 CPU @ 3.70GHz, running Ubuntu 20.04 LTS, gcc 9.4.0
- linux2: 12th Gen Intelยฎ Coreโข i9-12900 @ 2.40 GHz, running Ubuntu 22.04 LTS, gcc 11.3.0
- linux-aarch64: ARM Neoverse N1, running Ubuntu 22.04 LTS, gcc 11.4.0
- macos: M1 arm64 Macยฎ Mini, running macOS 13.2.1, clang 1400.0.29.202
- windows: 12th Gen Intelยฎ Coreโข i9-12900 @ 2.40 GHz, running Windows 11 Pro (21H2, 22000.1696), MSVC v143
This is a CHANGELOG of how any derived data has changed:
- 2024-11-26: The longitudinal plots and index tables now use geometric mean rather than HPT.
- 2024-06-27: The HPT values (and the longitudinal plots that are based on them) now correctly exclude any benchmarks in
excluded_benchmarks.txt
.
Visit the ๐ benchmark action and click the "Run Workflow" button.
The available parameters are:
fork
: The fork of CPython to benchmark. If benchmarking a pull request, this would normally be your GitHub username.ref
: The branch, tag or commit SHA to benchmark. If a SHA, it must be the full SHA, since finding it by a prefix is not supported.machine
: The machine to run on. One oflinux-amd64
(default),windows-amd64
,darwin-arm64
orall
.benchmark_base
: If checked, the base of the selected branch will also be benchmarked. The base is determined by runninggit merge-base upstream/main $ref
.pystats
: If checked, collect the pystats from running the benchmarks.
To watch the progress of the benchmark, select it from the ๐ benchmark action page. It may be canceled from there as well. To show only your benchmark workflows, select your GitHub ID from the "Actor" dropdown.
When the benchmarking is complete, the results are published to this repository and will appear in the master table. Each set of benchmarks will have:
- The raw
.json
results from pyperformance. - Comparisons against important reference releases, as well as the merge base of the branch if
benchmark_base
was selected. These include- A markdown table produced by
pyperf compare_to
. - A set of "violin" plots showing the distribution of results for each benchmark.
- A set of plots showing the memory change for each benchmark (for immediate bases only, on non-Windows platforms).
- A markdown table produced by
The most convenient way to get results locally is to clone this repo and git pull
from it.
To automate benchmarking runs, it may be more convenient to use the GitHub CLI.
Once you have gh
installed and configured, you can run benchmarks by cloning this repository and then from inside it:
$ gh workflow run benchmark.yml -f fork=me -f ref=my_branch
Any of the parameters described above are available at the commandline using the -f key=value
syntax.
To collect Linux perf sampling profile data for a benchmarking run, run the _benchmark
action and check the perf
checkbox.
If the default comparisons generated by this tool aren't sufficient, you can check out the repo and use the same infrastructure to generate any arbitrary comparison.
Check out a local copy of this repo:
$ git clone https://github.com/faster-cpython/benchmarking-public
Create a new virtual environment, activate it and install the dependencies into it:
$ cd benchmarking-public
$ python -m venv venv
$ source venv/bin/activate
$ pip install -r requirements.txt
Run bench_runner
's compare
tool:
usage:
Generate a set of comparisons between arbitrary commits. The commits
must already exist in the dataset.
[-h] --output-dir OUTPUT_DIR [--type {1:n,n:n}] commit [commit ...]
positional arguments:
commit Commits to compare. Must be a git commit hash prefix. May optionally have a friendly name
after a comma, e.g. c0ffee,main. If ends with a "T", use the Tier 2 run for that commit. If
ends with a "J", use the JIT run for that commit. If ends with a "N", use the NOGIL run for
that commit.
options:
-h, --help show this help message and exit
--output-dir OUTPUT_DIR
Directory to output results to.
--type {1:n,n:n} Compare the first commit to all others, or do the full product of all commits
For example:
$ python -m bench_runner compare e418fc3,default e418fc3J,jit --output comparison --type 1:n
The infrastructure to make all of this work is the bench_runner project. Look there for more detailed developer docs.
The easiest way to reproduce what is here is to use the bench_runner project library directly, but if you want to run parts of it in a different context or better understand how the numbers are calculated, this section describes some of the things that the benchmarking infrastructure does.
These results combine benchmarks that live in the
pyperformance and
pyston/python-macrobenchmarks
projects, so running the default set from pyperformance
will definitely
produce different results. To combine these benchmarks in the same run, clone
both repos side-by-side in the same directory and use a manifest
file
to combine them. This file should be passed to pyperformance run
:
pyperformance run --manifest benchmarks.manifest
Benchmarks and stats collection can happen in three different configurations. Here "configuration" may be a combination of both build-time and run-time flags:
- Default: A PGO build of CPython (
./configure --enable-optimizations --with-lto=yes
). - Tier 2: The same build as above, but with the
PYTHON_UOPS
environment variable set at runtime to use the Tier 2 interpreter. - JIT: A JIT and PGO build of CPython (
./configure --enable-optimizations --with-lto=yes --enable-experimental-jit
).
Information about the configuration of the run is in the README.md
at the root
of each run directory. The directory name will also include PYTHON_UOPS
for
Tier 2 and JIT
for JIT.
To reduce the number of unknown variables when comparing results, runs are always compared against runs of the same configuration. Be aware that sometimes the base commit on main may predate the configuration becoming available, for example, before the JIT compiler was merged into main. (An exception to this rule are the weekly benchmarks of upstream main, there Tier 2 and JIT configurations are compared against default configurations of the same commit, but that isn't relevant for the common case of testing a pull request).
An additional sharp edge is that, by default, pyperformance
does not pass
environment variables to the child process that actually does the work.
Therefore for a Tier 2 configuration, the --inherit-environ=PYTHON_UOPS
flag
must be passed to pyperformance run
when running benchmarks.
For detailed information, see how configurations affect build time flags in the Github Actions configuration..
Timing benchmarks are notoriously noisy. There are a few techniques to reduce this:
- Where available (on Linux), we use
pyperf tune
to set CPU affinity and other things that make the benchmarks more reproducible. For this reason, we know that the benchmarks are more predictable on Linux than on the other platforms. pyperf
has the concept of "warmup" runs, while caches are warming up and other things about the system are still stabilizing. These runs are excluded from the timing results. This is generally effective at reducing variability, but also may exclude real work done during optimization, for example.- We also use the Hierarchical Performance Testing (HPT) method (see below) to
statistically reduce the effect of benchmarks that have more variability. This
is a different method than the simple geometric mean that
pyperf
uses by default. We provide both numbers in our results.
pystats
are a set of counters in CPython that measure things like the number
of times each bytecode instruction is executed. (Detailed documentation of all
of the counters should be added to CPython in the future).
Collecting pystats
requires a special build of CPython with pystats
enabled:
(./configure --enable-pystats
).
pystats
must also be enabled at runtime, either using the -Xpystats
command
line argument or sys._stats_on()
. pyperformance
/pyperf
handles this step
automatically when running on a pystats-enabled build. Stats collection is
enabled during actual benchmarking code, and disabled while running the
"benchmarking harness" code in pyperf
itself. pyperf
has the concept of
"warmup" runs, which allow things like cache lines to warmup before actually
timing benchmarks. While they aren't included in the timing benchmarks, these
warmup runs are included in pystats collection since often Tier 2/JIT traces are
created during warmup, and we don't want the stats to appear as if the traces
ran but were not created.
Any statistics collected are then dumped at exit to the /tmp/py_stats
directory with a random filename. Lastly, the Tools/scripts/summarize_stats.py
script (in the CPython repo) is used to read all of the files from
/tmp/py_stats
and produce a human-readable markdown summary and a JSON file
with aggregate data. Because of this design, it is imperative that:
- The
/tmp/py_stats
directory is cleared before data collection. - No other Python processes are run that could also produce pystats data. Especially, this means benchmarks can not run in parallel.
For more information, see the actual code to collect pystats.
Hierarchical performance testing (HPT) is a method introduced in this paper:
T. Chen, Y. Chen, Q. Guo, O. Temam, Y. Wu and W. Hu, "Statistical performance comparisons of computers," IEEE International Symposium on High-Performance Comp Architecture, New Orleans, LA, USA, 2012, pp. 1-12, doi: 10.1109/HPCA.2012.6169043.
From the abstract:
In traditional performance comparisons, the impact of performance variability is usually ignored (i.e., the means of performance measurements are compared regardless of the variability), or in the few cases where it is factored in using parametric confidence techniques, the confidence is either erroneously computed based on the distribution of performance measurements (with the implicit assumption that it obeys the normal law), instead of the distribution of sample mean of performance measurements, or too few measurements are considered for the distribution of sample mean to be normal. โฆ We propose a non-parametric Hierarchical Performance Testing (HPT) framework for performance comparison, which is significantly more practical than standard parametric techniques because it does not require to collect a large number of measurements in order to achieve a normal distribution of the sample mean.
For each result, we compute a reliability score, as well as the estimated speedup at the 90th, 95th and 99th percentile.
The inclusion of HPT scores is considered experimental as we learn about their usefulness for decision-making.