benchmark.py 2.9 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192
  1. # -*- coding: utf-8 -*-
  2. import random
  3. import os
  4. import time
  5. import datetime as dt
  6. nfns = 4 # Functions per class
  7. nargs = 4 # Arguments per function
  8. def generate_dummy_code_pybind11(nclasses=10):
  9. decl = ""
  10. bindings = ""
  11. for cl in range(nclasses):
  12. decl += "class cl%03i;\n" % cl
  13. decl += "\n"
  14. for cl in range(nclasses):
  15. decl += "class cl%03i {\n" % cl
  16. decl += "public:\n"
  17. bindings += ' py::class_<cl%03i>(m, "cl%03i")\n' % (cl, cl)
  18. for fn in range(nfns):
  19. ret = random.randint(0, nclasses - 1)
  20. params = [random.randint(0, nclasses - 1) for i in range(nargs)]
  21. decl += " cl%03i *fn_%03i(" % (ret, fn)
  22. decl += ", ".join("cl%03i *" % p for p in params)
  23. decl += ");\n"
  24. bindings += ' .def("fn_%03i", &cl%03i::fn_%03i)\n' % (fn, cl, fn)
  25. decl += "};\n\n"
  26. bindings += " ;\n"
  27. result = "#include <pybind11/pybind11.h>\n\n"
  28. result += "namespace py = pybind11;\n\n"
  29. result += decl + "\n"
  30. result += "PYBIND11_MODULE(example, m) {\n"
  31. result += bindings
  32. result += "}"
  33. return result
  34. def generate_dummy_code_boost(nclasses=10):
  35. decl = ""
  36. bindings = ""
  37. for cl in range(nclasses):
  38. decl += "class cl%03i;\n" % cl
  39. decl += "\n"
  40. for cl in range(nclasses):
  41. decl += "class cl%03i {\n" % cl
  42. decl += "public:\n"
  43. bindings += ' py::class_<cl%03i>("cl%03i")\n' % (cl, cl)
  44. for fn in range(nfns):
  45. ret = random.randint(0, nclasses - 1)
  46. params = [random.randint(0, nclasses - 1) for i in range(nargs)]
  47. decl += " cl%03i *fn_%03i(" % (ret, fn)
  48. decl += ", ".join("cl%03i *" % p for p in params)
  49. decl += ");\n"
  50. bindings += (
  51. ' .def("fn_%03i", &cl%03i::fn_%03i, py::return_value_policy<py::manage_new_object>())\n'
  52. % (fn, cl, fn)
  53. )
  54. decl += "};\n\n"
  55. bindings += " ;\n"
  56. result = "#include <boost/python.hpp>\n\n"
  57. result += "namespace py = boost::python;\n\n"
  58. result += decl + "\n"
  59. result += "BOOST_PYTHON_MODULE(example) {\n"
  60. result += bindings
  61. result += "}"
  62. return result
  63. for codegen in [generate_dummy_code_pybind11, generate_dummy_code_boost]:
  64. print("{")
  65. for i in range(0, 10):
  66. nclasses = 2 ** i
  67. with open("test.cpp", "w") as f:
  68. f.write(codegen(nclasses))
  69. n1 = dt.datetime.now()
  70. os.system(
  71. "g++ -Os -shared -rdynamic -undefined dynamic_lookup "
  72. "-fvisibility=hidden -std=c++14 test.cpp -I include "
  73. "-I /System/Library/Frameworks/Python.framework/Headers -o test.so"
  74. )
  75. n2 = dt.datetime.now()
  76. elapsed = (n2 - n1).total_seconds()
  77. size = os.stat("test.so").st_size
  78. print(" {%i, %f, %i}," % (nclasses * nfns, elapsed, size))
  79. print("}")