-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgofunction.go
86 lines (64 loc) · 1.93 KB
/
gofunction.go
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
package py
/*
#include <Python.h>
#include "gofunction.h"
static inline void decref(PyObject *obj) { Py_DECREF(obj); }
*/
import "C"
import "unsafe"
import "reflect"
type Closure [2]reflect.Value // closure = self.method
func (closure *Closure) NewFunction(name string, nin int, doc string) *Base {
d := new(C.PyMethodDef)
d.ml_name = C.CString(name)
defer C.free(unsafe.Pointer(d.ml_name))
if C.setMethod(d, C.int(nin)) != 0 {
panic("Invalid arguments: nin")
}
if doc != "" {
d.ml_doc = C.CString(doc)
defer C.free(unsafe.Pointer(d.ml_doc))
}
ctx := uintptr(unsafe.Pointer(closure))
self := C.PyLong_FromLongLong(C.longlong(ctx))
defer C.decref(self)
f := C.PyCFunction_NewEx(d, self, nil)
return (*Base)(unsafe.Pointer(f))
}
//export goClassCallMethodArgs
func goClassCallMethodArgs(obj, args unsafe.Pointer) unsafe.Pointer {
// Unpack context and self pointer from obj
t := (*C.PyObject)(obj)
closure := (*Closure)(unsafe.Pointer(uintptr(C.PyLong_AsLongLong(t))))
// Get args ready to use, by turning it into a pointer of the appropriate
// type
a := (*Tuple)(args)
in := []reflect.Value{closure[0], reflect.ValueOf(a)}
out := closure[1].Call(in)
err := out[1].Interface()
if err != nil {
Raise(err.(error))
return nil
}
ret := out[0].Interface().(*Base)
return unsafe.Pointer(ret)
}
//export goClassCallMethodKwds
func goClassCallMethodKwds(obj, args, kwds unsafe.Pointer) unsafe.Pointer {
// Unpack context and self pointer from obj
t := (*C.PyObject)(obj)
closure := (*Closure)(unsafe.Pointer(uintptr(C.PyLong_AsLongLong(t))))
// Get args and kwds ready to use, by turning them into pointers of the
// appropriate type
a := (*Tuple)(args)
k := (*Dict)(kwds)
in := []reflect.Value{closure[0], reflect.ValueOf(a), reflect.ValueOf(k)}
out := closure[1].Call(in)
err := out[1].Interface()
if err != nil {
Raise(err.(error))
return nil
}
ret := out[0].Interface().(*Base)
return unsafe.Pointer(ret)
}