1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174
|
.. _benchmarks:
Benchmarks
==========
.. note::
**TL;DR**: nanobind bindings compile up to **~4× faster** and produce **~5×
smaller** binaries with **~10× lower** runtime overheads compared to
pybind11.
nanobind also outperforms Cython in important metrics (**3-12×** binary size
reduction, **1.6-4×** compilation time reduction, similar runtime performance).
The following experiments analyze the performance of a large function-heavy
(``func``) and class-heavy (``class``) binding microbenchmark compiled using
`Boost.Python <https://github.com/boostorg/python>`__, `Cython
<https://cython.org>`__, `pybind11 <https://github.com/pybind/pybind11>`__. The
``pybind11 + smart_holder`` results below refer to a `special branch
<https://github.com/pybind/pybind11/tree/smart_holder>`__ that addresses
long-standing issues related to holder types in pybind11.
Each experiment is shown twice: light gray ``[debug]`` columns provide data for
a debug build, and ``[opt]`` shows a size-optimized build that is representative
of a deployment scenario. The former is included to show that nanobind
performance is also good during a typical development workflow.
A comparison with `cppyy <https://cppyy.readthedocs.io/en/latest/>`_, which
uses dynamic compilation, is also shown later. Details on the experimental
setup can be found :ref:`below <benchmark_details>`.
Compilation time
----------------
The first plot contrasts the compilation time, where “*number* ×”
annotations denote the amount of time spent relative to nanobind. As
shown below, nanobind achieves a ~\ **2.7-4.4× improvement**
compared to pybind11 and a **1.6-4.4x improvement** compared to Cython.
.. image:: images/times.svg
:width: 800
:alt: Compilation time benchmark
Binary size
-----------
The extremely large size of generated binaries has been a persistent problem of
many prior binding libraries. nanobind significantly improves this metric in
size-optimized builds. There is a ~\ **11× improvement** compared to
Boost.Python, a **3-5× improvement** compared to pybind11, and a **3-12×
improvement** compared to Cython.
.. image:: images/sizes.svg
:width: 800
:alt: Compilation size benchmark
Performance
-----------
The last experiment compares the runtime performance overheads by calling a
bound function many times in a loop. Here, it is also interesting to
additionally compare against `cppyy
<https://cppyy.readthedocs.io/en/latest/>`__ (green bar) and a pure Python
implementation that runs bytecode without binding overheads (hatched gray bar).
The `smart_holder` branch of pybind11 is not explicitly listed since its
runtime performance matches the base version.
.. image:: images/perf.svg
:width: 850
:alt: Runtime performance benchmark
This data shows that the overhead of calling a nanobind function is
lower than that of an equivalent function call done within CPython. The
functions benchmarked here don’t perform CPU-intensive work, so this
this mainly measures the overheads of performing a function call,
boxing/unboxing arguments and return values, etc.
The difference to pybind11 is **significant**: a ~\ **3× improvement**
for simple functions, and an **~10× improvement** when classes are being
passed around. Complexities in pybind11 related to overload
resolution, multiple inheritance, and holders are the main reasons for
this difference. Those features were either simplified or completely
removed in nanobind.
The runtime performance of Cython and nanobind are similar (Cython leads in one
experiment and trails in another one). Cython generates specialized binding
code for every function and class, which is highly redundant (long compile
times, large binaries) but can also be beneficial for performance.
Finally, there is a **~1.6-2.1× improvement** in both experiments compared to
cppyy (please ignore the two ``[debug]`` columns—I did not feel comfortable
adjusting the JIT compilation flags; all cppyy bindings are therefore
optimized.)
Discussion
----------
Performance improvements compared to pybind11 are the result of optimizations
discussed in the :ref:`previous section <perf_improvements>`.
`cppyy <https://cppyy.readthedocs.io/en/latest/>`_ also achieves good
performance in the comparison above. It is based on dynamic parsing of C++ code
and *just-in-time* (JIT) compilation of bindings via the LLVM compiler
infrastructure. The authors of cppyy report that their tool produces bindings
with much lower overheads compared to pybind11, and the above plots show that
this is indeed true.
While nanobind retakes the performance lead, there are other qualitative
factors make these two tools appropriate to different audiences: cppyy has its
origin in CERN's ROOT mega-project and must be highly dynamic to work with that
codebase: it can parse header files to generate bindings as needed. cppyy works
particularly well together with PyPy and can avoid boxing/unboxing overheads
with this combination. The main downside of cppyy is that it depends on
Cling/Clang/LLVM that must be deployed on the user's side and then run there.
There isn't a way of pre-generating bindings and then shipping just the output
of this process.
nanobind is relatively static in comparison: you must tell it which functions
to expose via binding declarations. These declarations offer a high degree of
flexibility that users will typically use to create bindings that feel
*pythonic*. At compile-time, those declarations turn into a sequence of CPython
API calls, which produces self-contained bindings that are easy to redistribute
via `PyPI <https://pypi.org>`_ or elsewhere. Tools like `cibuildwheel
<https://cibuildwheel.readthedocs.io/en/stable/>`_ and `scikit-build
<https://scikit-build.readthedocs.io/en/latest/index.html>`_ can fully automate
the process of generating *Python wheels* for each target platform. A `minimal
example project <https://github.com/wjakob/nanobind_example>`_ shows how to do
this automatically via `GitHub Actions <https://github.com/features/actions>`_.
.. _benchmark_details:
Details
-------
The microbenchmark wraps a *large* number of trivial functions that only
perform a few additions. The objective of this is to quantify the overhead of
bindings on compilation time, binary size, and runtime performance. The
function-heavy benchmark (``func_*``) consists of 720 declarations of the form
(with permuted integer types)
.. code-block:: cpp
m.def("test_0050", [](uint16_t a, int64_t b, int32_t c, uint64_t d, uint32_t e, float f) {
return a+b+c+d+e+f;
});
while the latter (``class_*``) does exactly the same computation but packaged
up in ``struct``\ s with bindings.
.. code-block:: cpp
struct Struct50 {
uint16_t a; int64_t b; int32_t c; uint64_t d; uint32_t e; float f;
Struct50(uint16_t a, int64_t b, int32_t c, uint64_t d, uint32_t e, float f)
: a(a), b(b), c(c), d(d), e(e), f(f) { }
float sum() const { return a+b+c+d+e+f; }
};
py::class_<Struct50>(m, "Struct50")
.def(py::init<uint16_t, int64_t, int32_t, uint64_t, uint32_t, float>())
.def("sum", &Struct50::sum);
The code to generate the plots shown above is available `here
<https://github.com/wjakob/nanobind/blob/master/docs/microbenchmark.ipynb>`_.
Each test was compiled in debug mode (``debug``) and with optimizations
(``opt``) that minimize size (i.e., ``-Os``). Benchmarking was performed on a
AMD Ryzen 9 7950X workstation running Ubuntu 22.04.2 LTS. CPU boost was
disabled, and all core clock frequencies were pinned. Reported timings are the
median of five runs. Compilation used clang++ 15.0.7 with consistent compilation flags for
all experiments (see the referenced notebook file for detail). The used package
versions were Python 3.10.6, cppyy 1.12.13, Cython 0.29.28, and nanobind 1.2.0.
|