benchmark.py 2.9 KB
Newer Older
1
# -*- coding: utf-8 -*-
Wenzel Jakob's avatar
Wenzel Jakob committed
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
import random
import os
import time
import datetime as dt

nfns = 4  # Functions per class
nargs = 4  # Arguments per function


def generate_dummy_code_pybind11(nclasses=10):
    decl = ""
    bindings = ""

    for cl in range(nclasses):
        decl += "class cl%03i;\n" % cl
17
    decl += "\n"
Wenzel Jakob's avatar
Wenzel Jakob committed
18
19
20
21
22
23
24

    for cl in range(nclasses):
        decl += "class cl%03i {\n" % cl
        decl += "public:\n"
        bindings += '    py::class_<cl%03i>(m, "cl%03i")\n' % (cl, cl)
        for fn in range(nfns):
            ret = random.randint(0, nclasses - 1)
25
            params = [random.randint(0, nclasses - 1) for i in range(nargs)]
Wenzel Jakob's avatar
Wenzel Jakob committed
26
27
28
            decl += "    cl%03i *fn_%03i(" % (ret, fn)
            decl += ", ".join("cl%03i *" % p for p in params)
            decl += ");\n"
29
            bindings += '        .def("fn_%03i", &cl%03i::fn_%03i)\n' % (fn, cl, fn)
Wenzel Jakob's avatar
Wenzel Jakob committed
30
        decl += "};\n\n"
31
        bindings += "        ;\n"
Wenzel Jakob's avatar
Wenzel Jakob committed
32
33
34

    result = "#include <pybind11/pybind11.h>\n\n"
    result += "namespace py = pybind11;\n\n"
35
    result += decl + "\n"
36
    result += "PYBIND11_MODULE(example, m) {\n"
Wenzel Jakob's avatar
Wenzel Jakob committed
37
38
39
40
41
42
43
44
45
46
47
    result += bindings
    result += "}"
    return result


def generate_dummy_code_boost(nclasses=10):
    decl = ""
    bindings = ""

    for cl in range(nclasses):
        decl += "class cl%03i;\n" % cl
48
    decl += "\n"
Wenzel Jakob's avatar
Wenzel Jakob committed
49
50
51
52
53
54
55

    for cl in range(nclasses):
        decl += "class cl%03i {\n" % cl
        decl += "public:\n"
        bindings += '    py::class_<cl%03i>("cl%03i")\n' % (cl, cl)
        for fn in range(nfns):
            ret = random.randint(0, nclasses - 1)
56
            params = [random.randint(0, nclasses - 1) for i in range(nargs)]
Wenzel Jakob's avatar
Wenzel Jakob committed
57
58
59
            decl += "    cl%03i *fn_%03i(" % (ret, fn)
            decl += ", ".join("cl%03i *" % p for p in params)
            decl += ");\n"
60
61
62
63
            bindings += (
                '        .def("fn_%03i", &cl%03i::fn_%03i, py::return_value_policy<py::manage_new_object>())\n'
                % (fn, cl, fn)
            )
Wenzel Jakob's avatar
Wenzel Jakob committed
64
        decl += "};\n\n"
65
        bindings += "        ;\n"
Wenzel Jakob's avatar
Wenzel Jakob committed
66
67
68

    result = "#include <boost/python.hpp>\n\n"
    result += "namespace py = boost::python;\n\n"
69
    result += decl + "\n"
Wenzel Jakob's avatar
Wenzel Jakob committed
70
71
72
73
74
75
76
    result += "BOOST_PYTHON_MODULE(example) {\n"
    result += bindings
    result += "}"
    return result


for codegen in [generate_dummy_code_pybind11, generate_dummy_code_boost]:
77
    print("{")
Wenzel Jakob's avatar
Wenzel Jakob committed
78
79
80
81
82
    for i in range(0, 10):
        nclasses = 2 ** i
        with open("test.cpp", "w") as f:
            f.write(codegen(nclasses))
        n1 = dt.datetime.now()
83
84
        os.system(
            "g++ -Os -shared -rdynamic -undefined dynamic_lookup "
85
            "-fvisibility=hidden -std=c++14 test.cpp -I include "
86
87
            "-I /System/Library/Frameworks/Python.framework/Headers -o test.so"
        )
Wenzel Jakob's avatar
Wenzel Jakob committed
88
89
        n2 = dt.datetime.now()
        elapsed = (n2 - n1).total_seconds()
90
        size = os.stat("test.so").st_size
Wenzel Jakob's avatar
Wenzel Jakob committed
91
        print("   {%i, %f, %i}," % (nclasses * nfns, elapsed, size))
92
    print("}")