benchmark.py 2.79 KB
Newer Older
1
import datetime as dt
Wenzel Jakob's avatar
Wenzel Jakob committed
2
import os
3
import random
Wenzel Jakob's avatar
Wenzel Jakob committed
4
5
6
7
8
9
10
11
12
13

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


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

    for cl in range(nclasses):
14
        decl += f"class cl{cl:03};\n"
15
    decl += "\n"
Wenzel Jakob's avatar
Wenzel Jakob committed
16
17

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

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


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

    for cl in range(nclasses):
45
        decl += f"class cl{cl:03};\n"
46
    decl += "\n"
Wenzel Jakob's avatar
Wenzel Jakob committed
47
48
49
50

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

    result = "#include <boost/python.hpp>\n\n"
    result += "namespace py = boost::python;\n\n"
64
    result += decl + "\n"
Wenzel Jakob's avatar
Wenzel Jakob committed
65
66
67
68
69
70
71
    result += "BOOST_PYTHON_MODULE(example) {\n"
    result += bindings
    result += "}"
    return result


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