moc.py 1.7 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152
  1. mapping = {
  2. 'int': 'i64',
  3. 'float': 'f64',
  4. 'str': 'const char*',
  5. 'bool': 'bool',
  6. 'None': 'void'
  7. }
  8. def args(n, a: list, res: list, first=True):
  9. if n == 0:
  10. res.append(tuple(a))
  11. return
  12. for p_ret, c_ret in mapping.items():
  13. if not first and p_ret == 'None':
  14. continue
  15. a.append(p_ret)
  16. args(n-1, a, res, first=False)
  17. a.pop()
  18. data = []
  19. for n in [4,3,2,1]:
  20. res = []
  21. args(n, [], res)
  22. for p_ret,*p_args in res:
  23. c_args = ['VM*'] + [mapping[i] for i in p_args]
  24. c_ret = mapping[p_ret]
  25. name = f'__f_{p_ret}__{"_".join(p_args)}'
  26. s = f'typedef {c_ret} (*{name})({", ".join(c_args)});'
  27. s += '\n'
  28. impl = []
  29. for i, p_arg in enumerate(p_args):
  30. impl.append( f'{mapping[p_arg]} _{i} = vm->Py{p_arg.capitalize()}_AS_C(args[{i}]);' )
  31. call_impl = f'f({", ".join(["vm"] + [f"_{j}" for j in range(len(p_args))])})';
  32. if p_ret == 'None':
  33. impl.append( call_impl + ';' )
  34. impl.append( 'return vm->None;' )
  35. else:
  36. impl.append ( f'{mapping[p_ret]} ret = {call_impl};' )
  37. impl.append( f'return vm->Py{p_ret.capitalize()}(ret);' )
  38. impl = '\n'.join([' '*8 + i for i in impl])
  39. s += f'''__EXPORT\nvoid pkpy_vm_bind{name}(VM* vm, const char* mod, const char* name, {name} f) {{
  40. PyVar obj = vm->new_module_if_not_existed(mod);
  41. vm->bindFunc<{len(p_args)}>(obj, name, [f](VM* vm, const pkpy::ArgList& args) {{
  42. {impl}
  43. }});
  44. }}''' + '\n'
  45. data.append(s)
  46. with open('src/_bindings.h', 'w') as f:
  47. text = '\n'.join(data)
  48. f.write('extern "C" {\n' + text + '}')