forked from eklitzke/c.sh
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpython.sh
85 lines (64 loc) · 2.08 KB
/
python.sh
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
#!/bin/bash
# where ctypes.sh is installed; you will likely have to change this
LD_LIBRARY_PATH=$HOME/local/lib
. ~/code/ctypes.sh/ctypes.sh
# this is a 64-bit system
declare -i wordsize=8
s=$(cat<<EOF
def add(*args):
return sum(int(arg) for arg in args)
EOF
)
cat<<EOF >python.c
#include <assert.h>
#include <Python.h>
// Marshal the input words. This will return a Python tuple object containing
// all of the strings passed in.
PyObject *MarshalParams(const char **words, size_t count) {
PyObject *list = PyTuple_New(count);
for (size_t i = 0; i < count; i++) {
PyObject *s = PyString_FromString(words[i]);
PyTuple_SET_ITEM(list, (Py_ssize_t)i, s);
}
return list;
}
// XXX: leaks references
PyObject *CreateFunction(const char *funcname, const char *code) {
// compile a code object
PyObject *compiled = Py_CompileString(code, "", Py_file_input);
assert(compiled != NULL);
// build a module w/ the code object
PyObject *module = PyImport_ExecCodeModule((char *)funcname, compiled);
assert(module != NULL);
// get the function we created
PyObject *method = PyObject_GetAttrString(module, (char *)funcname);
assert(method != NULL);
return method;
}
EOF
function build_python {
sofile=$(mktemp /tmp/XXXXXX.so)
cc $(pkg-config --cflags --libs python) -fPIC -shared python.c -o $sofile 2>/dev/null
echo $sofile
}
sofile=$(build_python)
trap "rm -f $sofile python.c" EXIT
declare -a words
words=(string:1 string:2 string:3)
numwords=${#words[@]}
# allocate space for our packed words
dlcall -n buffer -r pointer malloc $((numwords * wordsize))
pack $buffer words
# load the code
dlopen $sofile
# initialize the python interpreter
dlcall Py_Initialize
# marshal the params
dlcall -n pytuple -r pointer MarshalParams $buffer long:$numwords
# create a python function object for "add"
dlcall -n pyfunc -r pointer CreateFunction string:add string:"$s"
# call the function
dlcall -n out -r pointer PyObject_CallObject $pyfunc $pytuple
# unmarshal the return value
dlcall -n res -r long PyInt_AsLong $out
printf "return value is %d\n" $(echo $res | egrep -o '[0-9]+')