comparison python/pydc/pydc.c @ 46:c21d1c2c84e1

- removed pydc.py wrapper overhead (which only called pydcext.so functions, directly, anyways) * implies renaming pydcext.* to pydc.* * while at it, iterate directly over args that are passed in (before we did extract fptr, sig and a tuple for the args and iterated over latter afterwards); we might have a tiny perf improvement now - added type stub as package_data
author Tassilo Philipp
date Fri, 13 Nov 2020 14:10:31 +0100
parents python/pydc/pydcext.c@0f86a5ecfe61
children d6670bd553dd
comparison
equal deleted inserted replaced
45:da553362fa7c 46:c21d1c2c84e1
1 /******************************************************************************
2 **
3 ** pydc - python dyncall package
4 **
5 ** python extension package in C
6 ** Copyright 2007-2016 Daniel Adler
7 ** 2018-2020 Tassilo Philipp
8 **
9 ** See README.txt for details (about changes, how to use, etc.).
10 **
11 *****************************************************************************/
12
13 #include <Python.h>
14 #include "dynload.h"
15 #include <limits.h>
16
17
18
19 #if ( (PY_VERSION_HEX < 0x02070000) \
20 || ((PY_VERSION_HEX >= 0x03000000) \
21 && (PY_VERSION_HEX < 0x03010000)) )
22 # define DcPyCObject_FromVoidPtr(ptr, dtor) PyCObject_FromVoidPtr((ptr), (dtor)) // !new ref!
23 # define DcPyCObject_AsVoidPtr(ppobj) PyCObject_AsVoidPtr((ppobj))
24 # define DcPyCObject_SetVoidPtr(ppobj, ptr) PyCObject_SetVoidPtr((ppobj), (ptr))
25 #else
26 # define USE_CAPSULE_API
27 # define DcPyCObject_FromVoidPtr(ptr, dtor) PyCapsule_New((ptr), NULL, (dtor)) // !new ref!
28 # define DcPyCObject_AsVoidPtr(ppobj) PyCapsule_GetPointer((ppobj), NULL)
29 # define DcPyCObject_SetVoidPtr(ppobj, ptr) //@@@ unsure what to do, cannot/shouldn't call this with a null pointer as this wants to call the dtor, so not doing anything: PyCapsule_SetPointer((ppobj), (ptr)) // this might need to call the dtor to behave like PyCObject_SetVoidPtr?
30 #endif
31
32 #if(PY_VERSION_HEX >= 0x03030000)
33 # define PYUNICODE_CACHES_UTF8
34 #endif
35
36 #if PY_MAJOR_VERSION >= 3
37 # define EXPECT_LONG_TYPE_STR "an int"
38 # define DcPyString_GET_SIZE PyBytes_GET_SIZE
39 # define DcPyString_Check PyBytes_Check
40 # define DcPyString_AsString PyBytes_AsString
41 # define DcPyInt_Check PyLong_Check
42 # define DcPyInt_AsLong PyLong_AsLong
43 # define DcPyInt_AS_LONG PyLong_AS_LONG
44 #else
45 # define EXPECT_LONG_TYPE_STR "an int or a long"
46 # define DcPyString_GET_SIZE PyString_GET_SIZE
47 # define DcPyString_Check PyString_Check
48 # define DcPyString_AsString PyString_AsString
49 # define DcPyInt_Check PyInt_Check
50 # define DcPyInt_AsLong PyInt_AsLong
51 # define DcPyInt_AS_LONG PyInt_AS_LONG
52 #endif
53
54 /* PyCObject destructor callback for libhandle */
55
56 #if defined(USE_CAPSULE_API)
57 void free_library(PyObject* capsule)
58 {
59 void* libhandle = PyCapsule_GetPointer(capsule, NULL);
60 #else
61 void free_library(void* libhandle)
62 {
63 #endif
64 if (libhandle != 0)
65 dlFreeLibrary(libhandle);
66 }
67
68
69 /* load function */
70
71 static PyObject*
72 pydc_load(PyObject* self, PyObject* args)
73 {
74 const char* libpath;
75 void* libhandle;
76
77 if (!PyArg_ParseTuple(args,"z", &libpath))
78 return PyErr_Format(PyExc_RuntimeError, "libpath argument (str) missing");
79
80 libhandle = dlLoadLibrary(libpath);
81
82 if (!libhandle)
83 return PyErr_Format(PyExc_RuntimeError, "dlLoadLibrary('%s') failed", libpath);
84
85 return DcPyCObject_FromVoidPtr(libhandle, &free_library); // !new ref!
86 }
87
88 /* find function */
89
90 static PyObject*
91 pydc_find(PyObject* self, PyObject* args)
92 {
93 PyObject* pcobj;
94 const char* symbol;
95 void* libhandle;
96 void* funcptr;
97
98 if (!PyArg_ParseTuple(args, "Os", &pcobj, &symbol))
99 return PyErr_Format(PyExc_RuntimeError, "argument mismatch");
100
101 libhandle = DcPyCObject_AsVoidPtr(pcobj);
102 if (!libhandle)
103 return PyErr_Format(PyExc_RuntimeError, "libhandle is null");
104
105 funcptr = dlFindSymbol(libhandle, symbol);
106 if (!funcptr)
107 return PyErr_Format(PyExc_RuntimeError, "symbol '%s' not found", symbol);
108
109 return DcPyCObject_FromVoidPtr(funcptr, NULL); // !new ref!
110 }
111
112 /* free function */
113
114 static PyObject*
115 pydc_free(PyObject* self, PyObject* args)
116 {
117 PyObject* pcobj;
118 void* libhandle;
119
120 if (!PyArg_ParseTuple(args, "O", &pcobj))
121 return PyErr_Format(PyExc_RuntimeError, "argument mismatch");
122
123 libhandle = DcPyCObject_AsVoidPtr(pcobj);
124 if (!libhandle)
125 return PyErr_Format(PyExc_RuntimeError, "libhandle is NULL");
126
127 dlFreeLibrary(libhandle);
128 DcPyCObject_SetVoidPtr(pcobj, NULL);
129
130 //don't think I need to release it, as the pyobj is not equivalent to the held handle
131 //Py_XDECREF(pcobj); // release ref from pydc_load()
132
133 Py_RETURN_NONE;
134 }
135
136 /* get_path function */
137
138 static PyObject*
139 pydc_get_path(PyObject* self, PyObject* args)
140 {
141 PyObject* pcobj;
142 PyObject* retobj;
143 void* libhandle;
144 char* path;
145 int path_bufSize;
146
147 if (!PyArg_ParseTuple(args, "O", &pcobj))
148 return PyErr_Format(PyExc_RuntimeError, "argument mismatch");
149
150 libhandle = (pcobj == Py_None)?NULL:DcPyCObject_AsVoidPtr(pcobj);
151 path_bufSize = dlGetLibraryPath(libhandle, NULL, 0);
152 if (!path_bufSize)
153 return PyErr_Format(PyExc_RuntimeError, "library path cannot be found");
154
155 path = malloc(path_bufSize);
156 if (path_bufSize != dlGetLibraryPath(libhandle, path, path_bufSize)) {
157 free(path);
158 return PyErr_Format(PyExc_RuntimeError, "library path cannot be queried");
159 }
160
161 retobj = Py_BuildValue("s", path); // !new ref! @@@ UTF-8 input...
162 free(path);
163 return retobj;
164 }
165
166
167 #include "dyncall.h"
168 #include "dyncall_signature.h"
169
170 DCCallVM* gpCall = NULL;
171
172 // helper to temporarily copy string arguments
173 #define NUM_AUX_STRS 64
174 static int n_str_aux;
175 static char* str_aux[NUM_AUX_STRS]; // hard limit, most likely enough and checked for below @@@ugly though
176
177
178 /* call function */
179
180 static PyObject*
181 pydc_call_impl(PyObject* self, PyObject* args) /* implementation, called by wrapper func pydc_call() */
182 {
183 const char *sig_ptr;
184 char ch;
185 int pos, ts;
186 void* pfunc;
187
188 pos = 0;
189 ts = PyTuple_Size(args);
190 if (ts < 2)
191 return PyErr_Format(PyExc_RuntimeError, "argument mismatch");
192
193 // get ptr to func to call
194 pfunc = DcPyCObject_AsVoidPtr(PyTuple_GetItem(args, pos++));
195 if (!pfunc)
196 return PyErr_Format( PyExc_RuntimeError, "function pointer is NULL" );
197
198 // get signature
199 #if !defined(PYUNICODE_CACHES_UTF8)
200 PyObject* sig_obj = NULL;
201 #endif
202 PyObject* so = PyTuple_GetItem(args, pos++);
203 if ( PyUnicode_Check(so) )
204 {
205 #if defined(PYUNICODE_CACHES_UTF8)
206 sig_ptr = PyUnicode_AsUTF8(so);
207 #else
208 // w/o PyUnicode_AsUTF8(), which caches the UTF-8 representation, itself, create new ref we'll dec below
209 if((sig_obj = PyUnicode_AsEncodedString(so, "utf-8", "strict"))) // !new ref!
210 sig_ptr = PyBytes_AS_STRING(sig_obj); // Borrowed pointer
211 #endif
212 } else if ( DcPyString_Check(so) )
213 sig_ptr = DcPyString_AsString(so); // @@@ test py 2
214
215
216
217 if (!sig_ptr)
218 return PyErr_Format( PyExc_RuntimeError, "signature is NULL" );
219
220
221 dcReset(gpCall);
222 dcMode(gpCall, DC_CALL_C_DEFAULT);
223
224 for (ch = *sig_ptr; ch != '\0' && ch != DC_SIGCHAR_ENDARG; ch = *++sig_ptr)
225 {
226 PyObject* po;
227
228 if (pos > ts)
229 return PyErr_Format( PyExc_RuntimeError, "expecting more arguments" );
230
231 po = PyTuple_GetItem(args, pos);
232
233 ++pos; // incr here, code below uses it as 1-based argument index for error strings
234
235 switch(ch)
236 {
237 case DC_SIGCHAR_CC_PREFIX:
238 {
239 if(*(sig_ptr+1) != '\0')
240 {
241 DCint mode = dcGetModeFromCCSigChar(*++sig_ptr);
242 if(mode != DC_ERROR_UNSUPPORTED_MODE)
243 dcMode(gpCall, mode);
244 }
245 --pos; // didn't count as arg
246 }
247 break;
248
249 case DC_SIGCHAR_BOOL:
250 if ( !PyBool_Check(po) )
251 return PyErr_Format( PyExc_RuntimeError, "arg %d - expecting a bool", pos );
252 dcArgBool(gpCall, (Py_True == po) ? DC_TRUE : DC_FALSE);
253 break;
254
255 case DC_SIGCHAR_CHAR:
256 case DC_SIGCHAR_UCHAR:
257 {
258 DCchar c;
259 if ( PyUnicode_Check(po) )
260 {
261 #if (PY_VERSION_HEX < 0x03030000)
262 Py_UNICODE cu;
263 if (PyUnicode_GET_SIZE(po) != 1)
264 #else
265 Py_UCS4 cu;
266 if (PyUnicode_GET_LENGTH(po) != 1)
267 #endif
268 return PyErr_Format( PyExc_RuntimeError, "arg %d - expecting a str with length of 1 (a char string)", pos );
269
270 #if (PY_VERSION_HEX < 0x03030000)
271 cu = PyUnicode_AS_UNICODE(po)[0];
272 #else
273 cu = PyUnicode_ReadChar(po, 0);
274 #endif
275 // check against UCHAR_MAX in every case b/c Py_UCS4 is unsigned
276 if ( (cu > UCHAR_MAX))
277 return PyErr_Format( PyExc_RuntimeError, "arg %d out of range - expecting a char code", pos );
278 c = (DCchar) cu;
279 }
280 else if ( DcPyString_Check(po) )
281 {
282 size_t l;
283 char* s;
284 l = DcPyString_GET_SIZE(po);
285 if (l != 1)
286 return PyErr_Format( PyExc_RuntimeError, "arg %d - expecting a str with length of 1 (a char string)", pos );
287 s = DcPyString_AsString(po);
288 c = (DCchar) s[0];
289 }
290 else if ( DcPyInt_Check(po) )
291 {
292 long l = DcPyInt_AsLong(po);
293 if (ch == DC_SIGCHAR_CHAR && (l < CHAR_MIN || l > CHAR_MAX))
294 return PyErr_Format( PyExc_RuntimeError, "arg %d out of range - expecting %d <= arg <= %d, got %ld", pos, CHAR_MIN, CHAR_MAX, l );
295 if (ch == DC_SIGCHAR_UCHAR && (l < 0 || l > UCHAR_MAX))
296 return PyErr_Format( PyExc_RuntimeError, "arg %d out of range - expecting 0 <= arg <= %d, got %ld", pos, UCHAR_MAX, l );
297 c = (DCchar) l;
298 }
299 else
300 return PyErr_Format( PyExc_RuntimeError, "arg %d - expecting a char", pos );
301 dcArgChar(gpCall, c);
302 }
303 break;
304
305 case DC_SIGCHAR_SHORT:
306 {
307 long l;
308 if ( !DcPyInt_Check(po) )
309 return PyErr_Format( PyExc_RuntimeError, "arg %d - expecting an int", pos );
310 l = DcPyInt_AS_LONG(po);
311 if (l < SHRT_MIN || l > SHRT_MAX)
312 return PyErr_Format( PyExc_RuntimeError, "arg %d out of range - expecting %d <= arg <= %d, got %ld", pos, SHRT_MIN, SHRT_MAX, l );
313 dcArgShort(gpCall, (DCshort)l);
314 }
315 break;
316
317 case DC_SIGCHAR_USHORT:
318 {
319 long l;
320 if ( !DcPyInt_Check(po) )
321 return PyErr_Format( PyExc_RuntimeError, "arg %d - expecting an int", pos );
322 l = DcPyInt_AS_LONG(po);
323 if (l < 0 || l > USHRT_MAX)
324 return PyErr_Format( PyExc_RuntimeError, "arg %d out of range - expecting 0 <= arg <= %d, got %ld", pos, USHRT_MAX, l );
325 dcArgShort(gpCall, (DCshort)l);
326 }
327 break;
328
329 case DC_SIGCHAR_INT:
330 case DC_SIGCHAR_UINT:
331 if ( !DcPyInt_Check(po) )
332 return PyErr_Format( PyExc_RuntimeError, "arg %d - expecting an int", pos );
333 dcArgInt(gpCall, (DCint) DcPyInt_AS_LONG(po));
334 break;
335
336 case DC_SIGCHAR_LONG:
337 case DC_SIGCHAR_ULONG:
338 if ( !DcPyInt_Check(po) )
339 return PyErr_Format( PyExc_RuntimeError, "arg %d - expecting an int", pos );
340 dcArgLong(gpCall, (DClong) PyLong_AsLong(po));
341 break;
342
343 case DC_SIGCHAR_LONGLONG:
344 case DC_SIGCHAR_ULONGLONG:
345 #if PY_MAJOR_VERSION < 3
346 if ( PyInt_Check(po) )
347 dcArgLongLong(gpCall, (DClonglong) PyInt_AS_LONG(po));
348 else
349 #endif
350 if ( !PyLong_Check(po) )
351 return PyErr_Format( PyExc_RuntimeError, "arg %d - expecting " EXPECT_LONG_TYPE_STR, pos );
352 dcArgLongLong(gpCall, (DClonglong)PyLong_AsLongLong(po));
353 break;
354
355 case DC_SIGCHAR_FLOAT:
356 if (!PyFloat_Check(po))
357 return PyErr_Format( PyExc_RuntimeError, "arg %d - expeecting a float", pos );
358 dcArgFloat(gpCall, (float)PyFloat_AsDouble(po));
359 break;
360
361 case DC_SIGCHAR_DOUBLE:
362 if (!PyFloat_Check(po))
363 return PyErr_Format( PyExc_RuntimeError, "arg %d - expeecting a float", pos );
364 dcArgDouble(gpCall, PyFloat_AsDouble(po));
365 break;
366
367 case DC_SIGCHAR_POINTER: // this will only accept integers or mutable array types (meaning only bytearray)
368 {
369 DCpointer p;
370 if ( PyByteArray_Check(po) )
371 p = (DCpointer) PyByteArray_AsString(po); // adds an extra '\0', but that's ok
372 #if PY_MAJOR_VERSION < 3
373 else if ( PyInt_Check(po) )
374 p = (DCpointer) PyInt_AS_LONG(po);
375 #endif
376 else if ( PyLong_Check(po) )
377 p = (DCpointer) PyLong_AsVoidPtr(po);
378 else if ( po == Py_None )
379 p = NULL;
380 else
381 return PyErr_Format( PyExc_RuntimeError, "arg %d - expecting a promoting pointer-type (int, bytearray)", pos );
382 dcArgPointer(gpCall, p);
383 }
384 break;
385
386 case DC_SIGCHAR_STRING: // strings are considered to be immutable objects
387 {
388 PyObject* bo = NULL;
389 const char* p;
390 size_t s;
391 if ( PyUnicode_Check(po) )
392 {
393 if(n_str_aux >= NUM_AUX_STRS)
394 return PyErr_Format( PyExc_RuntimeError, "too many arguments (implementation limit of %d new UTF-8 string references reached) - abort", n_str_aux );
395
396 #if defined(PYUNICODE_CACHES_UTF8)
397 p = PyUnicode_AsUTF8(po);
398 #else
399 // w/o PyUnicode_AsUTF8(), which caches the UTF-8 representation, itself, create new ref we'll dec below
400 if((bo = PyUnicode_AsEncodedString(po, "utf-8", "strict"))) // !new ref!
401 p = PyBytes_AS_STRING(bo); // Borrowed pointer
402 #endif
403 } else if ( DcPyString_Check(po) )
404 p = DcPyString_AsString(po);
405 else if ( PyByteArray_Check(po) )
406 p = (DCpointer) PyByteArray_AsString(po); // adds an extra '\0', but that's ok //@@@ not sure if allowed to modify
407 else
408 return PyErr_Format( PyExc_RuntimeError, "arg %d - expecting a str", pos );
409
410 // p points in any case to a buffer that shouldn't be modified, so pass a copy to dyncall (cleaned up after call)
411 s = strlen(p)+1;
412 str_aux[n_str_aux] = malloc(s);
413 strncpy(str_aux[n_str_aux], p, s);
414 Py_XDECREF(bo);
415 dcArgPointer(gpCall, (DCpointer)str_aux[n_str_aux++]);
416 }
417 break;
418
419 default:
420 return PyErr_Format( PyExc_RuntimeError, "unknown signature character '%c'", ch);
421 }
422 }
423
424 if (pos != ts)
425 return PyErr_Format( PyExc_RuntimeError, "too many arguments");
426
427 if (ch == '\0')
428 return PyErr_Format( PyExc_RuntimeError, "return value missing in signature");
429
430
431 ch = *++sig_ptr;
432 switch(ch)
433 {
434 // every line creates a new reference passed back to python
435 case DC_SIGCHAR_VOID: dcCallVoid (gpCall, pfunc); Py_RETURN_NONE; // !new ref!
436 case DC_SIGCHAR_BOOL: if(dcCallBool (gpCall, pfunc)){Py_RETURN_TRUE;}else{Py_RETURN_FALSE;} // !new ref!
437 case DC_SIGCHAR_CHAR: return Py_BuildValue("b", dcCallChar (gpCall, pfunc)); // !new ref!
438 case DC_SIGCHAR_UCHAR: return Py_BuildValue("B", dcCallChar (gpCall, pfunc)); // !new ref!
439 case DC_SIGCHAR_SHORT: return Py_BuildValue("h", dcCallShort (gpCall, pfunc)); // !new ref!
440 case DC_SIGCHAR_USHORT: return Py_BuildValue("H", dcCallShort (gpCall, pfunc)); // !new ref!
441 case DC_SIGCHAR_INT: return Py_BuildValue("i", dcCallInt (gpCall, pfunc)); // !new ref!
442 case DC_SIGCHAR_UINT: return Py_BuildValue("I", dcCallInt (gpCall, pfunc)); // !new ref!
443 case DC_SIGCHAR_LONG: return Py_BuildValue("l", dcCallLong (gpCall, pfunc)); // !new ref!
444 case DC_SIGCHAR_ULONG: return Py_BuildValue("k", dcCallLong (gpCall, pfunc)); // !new ref!
445 case DC_SIGCHAR_LONGLONG: return Py_BuildValue("L", dcCallLongLong(gpCall, pfunc)); // !new ref!
446 case DC_SIGCHAR_ULONGLONG: return Py_BuildValue("K", dcCallLongLong(gpCall, pfunc)); // !new ref!
447 case DC_SIGCHAR_FLOAT: return Py_BuildValue("f", dcCallFloat (gpCall, pfunc)); // !new ref!
448 case DC_SIGCHAR_DOUBLE: return Py_BuildValue("d", dcCallDouble (gpCall, pfunc)); // !new ref!
449 case DC_SIGCHAR_STRING: return Py_BuildValue("s", dcCallPointer (gpCall, pfunc)); // !new ref!
450 case DC_SIGCHAR_POINTER: return Py_BuildValue("n", dcCallPointer (gpCall, pfunc)); // !new ref!
451 default: return PyErr_Format(PyExc_RuntimeError, "invalid return type signature");
452 }
453
454 #if !defined(PYUNICODE_CACHES_UTF8)
455 Py_XDECREF(sig_obj);
456 #endif
457 }
458
459
460 static PyObject*
461 pydc_call(PyObject* self, PyObject* args)
462 {
463 int i;
464 n_str_aux = 0;
465 PyObject* o = pydc_call_impl(self, args);
466 for(i = 0; i<n_str_aux; ++i)
467 free(str_aux[i]);
468 return o;
469 }
470
471
472 // module deinit
473 static void deinit_pydc(void* x)
474 {
475 if(gpCall) {
476 dcFree(gpCall);
477 gpCall = NULL;
478 }
479 }
480
481
482 #define PYDC_TO_STR_(x) #x
483 #define PYDC_TO_STR(x) PYDC_TO_STR_(x)
484 #define PYDC_CONCAT_(x, y) x ## y
485 #define PYDC_CONCAT(x, y) PYDC_CONCAT_(x, y)
486
487 #define PYDC_MOD_NAME pydc
488 #define PYDC_MOD_NAME_STR PYDC_TO_STR(PYDC_MOD_NAME)
489 #define PYDC_MOD_DESC_STR "dyncall bindings for python"
490
491 #if PY_MAJOR_VERSION >= 3
492 # define PY_MOD_INIT_FUNC_NAME PYDC_CONCAT(PyInit_, PYDC_MOD_NAME)
493 #else
494 # define PY_MOD_INIT_FUNC_NAME PYDC_CONCAT(init, PYDC_MOD_NAME)
495 #endif
496
497
498 PyMODINIT_FUNC
499 PY_MOD_INIT_FUNC_NAME(void)
500 {
501 static PyMethodDef pydcMethods[] = {
502 {"load", pydc_load, METH_VARARGS, "load library" },
503 {"find", pydc_find, METH_VARARGS, "find symbols" },
504 {"free", pydc_free, METH_VARARGS, "free library" },
505 {"get_path", pydc_get_path, METH_VARARGS, "get library path"},
506 {"call", pydc_call, METH_VARARGS, "call function" },
507 {NULL,NULL,0,NULL}
508 };
509
510 PyObject* m;
511 #if PY_MAJOR_VERSION >= 3
512 static struct PyModuleDef moddef = { PyModuleDef_HEAD_INIT, PYDC_MOD_NAME_STR, PYDC_MOD_DESC_STR, -1, pydcMethods, NULL, NULL, NULL, deinit_pydc };
513 m = PyModule_Create(&moddef);
514 #else
515 m = Py_InitModule3(PYDC_MOD_NAME_STR, pydcMethods, PYDC_MOD_DESC_STR);
516 // NOTE: there is no way to pass a pointer to deinit_pydc - see PEP 3121 for details
517 #endif
518
519 if(m)
520 gpCall = dcNewCallVM(4096); //@@@ one shared callvm for the entire module, this is not reentrant
521
522 #if PY_MAJOR_VERSION >= 3
523 return m;
524 #endif
525 }
526