| #define PY_SSIZE_T_CLEAN |
| #include <Python.h> |
| #include "structmember.h" |
| |
| #define CFFI_VERSION "1.15.0" |
| |
| #ifdef MS_WIN32 |
| #include <windows.h> |
| #include "misc_win32.h" |
| #else |
| #include <stddef.h> |
| #include <stdint.h> |
| #include <dlfcn.h> |
| #include <errno.h> |
| #include <ffi.h> |
| #include <sys/mman.h> |
| #endif |
| |
| /* this block of #ifs should be kept exactly identical between |
| c/_cffi_backend.c, cffi/vengine_cpy.py, cffi/vengine_gen.py */ |
| #if defined(_MSC_VER) |
| # include <malloc.h> /* for alloca() */ |
| # if _MSC_VER < 1600 /* MSVC < 2010 */ |
| typedef __int8 int8_t; |
| typedef __int16 int16_t; |
| typedef __int32 int32_t; |
| typedef __int64 int64_t; |
| typedef unsigned __int8 uint8_t; |
| typedef unsigned __int16 uint16_t; |
| typedef unsigned __int32 uint32_t; |
| typedef unsigned __int64 uint64_t; |
| typedef __int8 int_least8_t; |
| typedef __int16 int_least16_t; |
| typedef __int32 int_least32_t; |
| typedef __int64 int_least64_t; |
| typedef unsigned __int8 uint_least8_t; |
| typedef unsigned __int16 uint_least16_t; |
| typedef unsigned __int32 uint_least32_t; |
| typedef unsigned __int64 uint_least64_t; |
| typedef __int8 int_fast8_t; |
| typedef __int16 int_fast16_t; |
| typedef __int32 int_fast32_t; |
| typedef __int64 int_fast64_t; |
| typedef unsigned __int8 uint_fast8_t; |
| typedef unsigned __int16 uint_fast16_t; |
| typedef unsigned __int32 uint_fast32_t; |
| typedef unsigned __int64 uint_fast64_t; |
| typedef __int64 intmax_t; |
| typedef unsigned __int64 uintmax_t; |
| # else |
| # include <stdint.h> |
| # endif |
| # if _MSC_VER < 1800 /* MSVC < 2013 */ |
| typedef unsigned char _Bool; |
| # endif |
| #else |
| # include <stdint.h> |
| # if (defined (__SVR4) && defined (__sun)) || defined(_AIX) || defined(__hpux) |
| # include <alloca.h> |
| # endif |
| #endif |
| |
| |
| /* Define the following macro ONLY if you trust libffi's version of |
| * ffi_closure_alloc() more than the code in malloc_closure.h. |
| * IMPORTANT: DO NOT ENABLE THIS ON LINUX, unless you understand exactly |
| * why I recommend against it and decide that you trust it more than my |
| * analysis below. |
| * |
| * There are two versions of this code: one inside libffi itself, and |
| * one inside malloc_closure.h here. Both should be fine as long as the |
| * Linux distribution does _not_ enable extra security features. If it |
| * does, then the code in malloc_closure.h will cleanly crash because |
| * there is no reasonable way to obtain a read-write-execute memory |
| * page. On the other hand, the code in libffi will appear to |
| * work---but will actually randomly crash after a fork() if the child |
| * does not immediately call exec(). This second crash is of the kind |
| * that can be turned into an attack vector by a motivated attacker. |
| * So, _enabling_ extra security features _opens_ an attack vector. |
| * That sounds like a horribly bad idea to me, and is the reason for why |
| * I prefer CFFI crashing cleanly. |
| * |
| * Currently, we use libffi's ffi_closure_alloc() on NetBSD. It is |
| * known that on the NetBSD kernel, a different strategy is used which |
| * should not be open to the fork() bug. |
| * |
| * This is also used on macOS, provided we are executing on macOS 10.15 or |
| * above. It's a mess because it needs runtime checks in that case. |
| */ |
| #ifdef __NetBSD__ |
| |
| # define CFFI_CHECK_FFI_CLOSURE_ALLOC 1 |
| # define CFFI_CHECK_FFI_CLOSURE_ALLOC_MAYBE 1 |
| # define CFFI_CHECK_FFI_PREP_CLOSURE_LOC 1 |
| # define CFFI_CHECK_FFI_PREP_CLOSURE_LOC_MAYBE 1 |
| # define CFFI_CHECK_FFI_PREP_CIF_VAR 0 |
| # define CFFI_CHECK_FFI_PREP_CIF_VAR_MAYBE 0 |
| |
| #elif defined(__APPLE__) && defined(FFI_AVAILABLE_APPLE) |
| |
| # define CFFI_CHECK_FFI_CLOSURE_ALLOC __builtin_available(macos 10.15, ios 13, watchos 6, tvos 13, *) |
| # define CFFI_CHECK_FFI_CLOSURE_ALLOC_MAYBE 1 |
| # define CFFI_CHECK_FFI_PREP_CLOSURE_LOC __builtin_available(macos 10.15, ios 13, watchos 6, tvos 13, *) |
| # define CFFI_CHECK_FFI_PREP_CLOSURE_LOC_MAYBE 1 |
| # define CFFI_CHECK_FFI_PREP_CIF_VAR __builtin_available(macos 10.15, ios 13, watchos 6, tvos 13, *) |
| # define CFFI_CHECK_FFI_PREP_CIF_VAR_MAYBE 1 |
| |
| #else |
| |
| # define CFFI_CHECK_FFI_CLOSURE_ALLOC 0 |
| # define CFFI_CHECK_FFI_CLOSURE_ALLOC_MAYBE 0 |
| # define CFFI_CHECK_FFI_PREP_CLOSURE_LOC 0 |
| # define CFFI_CHECK_FFI_PREP_CLOSURE_LOC_MAYBE 0 |
| # define CFFI_CHECK_FFI_PREP_CIF_VAR 0 |
| # define CFFI_CHECK_FFI_PREP_CIF_VAR_MAYBE 0 |
| |
| #endif |
| |
| /* always includes this, even if it turns out not to be used on NetBSD |
| because calls are behind "if (0)" */ |
| #include "malloc_closure.h" |
| |
| |
| #if PY_MAJOR_VERSION >= 3 |
| # define STR_OR_BYTES "bytes" |
| # define PyText_Type PyUnicode_Type |
| # define PyText_Check PyUnicode_Check |
| # define PyTextAny_Check PyUnicode_Check |
| # define PyText_FromFormat PyUnicode_FromFormat |
| # define PyText_AsUTF8 _PyUnicode_AsString /* PyUnicode_AsUTF8 in Py3.3 */ |
| # define PyText_AS_UTF8 _PyUnicode_AsString |
| # if PY_VERSION_HEX >= 0x03030000 |
| # define PyText_GetSize PyUnicode_GetLength |
| # else |
| # define PyText_GetSize PyUnicode_GetSize |
| # endif |
| # define PyText_FromString PyUnicode_FromString |
| # define PyText_FromStringAndSize PyUnicode_FromStringAndSize |
| # define PyText_InternInPlace PyUnicode_InternInPlace |
| # define PyText_InternFromString PyUnicode_InternFromString |
| # define PyIntOrLong_Check PyLong_Check |
| #else |
| # define STR_OR_BYTES "str" |
| # define PyText_Type PyString_Type |
| # define PyText_Check PyString_Check |
| # define PyTextAny_Check(op) (PyString_Check(op) || PyUnicode_Check(op)) |
| # define PyText_FromFormat PyString_FromFormat |
| # define PyText_AsUTF8 PyString_AsString |
| # define PyText_AS_UTF8 PyString_AS_STRING |
| # define PyText_GetSize PyString_Size |
| # define PyText_FromString PyString_FromString |
| # define PyText_FromStringAndSize PyString_FromStringAndSize |
| # define PyText_InternInPlace PyString_InternInPlace |
| # define PyText_InternFromString PyString_InternFromString |
| # define PyIntOrLong_Check(op) (PyInt_Check(op) || PyLong_Check(op)) |
| #endif |
| |
| #if PY_MAJOR_VERSION >= 3 |
| # define PyInt_FromLong PyLong_FromLong |
| # define PyInt_FromSsize_t PyLong_FromSsize_t |
| # define PyInt_AsSsize_t PyLong_AsSsize_t |
| # define PyInt_AsLong PyLong_AsLong |
| #endif |
| |
| #if PY_MAJOR_VERSION >= 3 |
| /* This is the default on Python3 and constant has been removed. */ |
| # define Py_TPFLAGS_CHECKTYPES 0 |
| #endif |
| |
| #if PY_MAJOR_VERSION < 3 |
| # undef PyCapsule_GetPointer |
| # undef PyCapsule_New |
| # define PyCapsule_GetPointer(capsule, name) \ |
| (PyCObject_AsVoidPtr(capsule)) |
| # define PyCapsule_New(pointer, name, destructor) \ |
| (PyCObject_FromVoidPtr(pointer, destructor)) |
| #endif |
| |
| #if PY_VERSION_HEX < 0x030900a4 |
| # define Py_SET_REFCNT(obj, val) (Py_REFCNT(obj) = (val)) |
| #endif |
| |
| #if PY_VERSION_HEX >= 0x03080000 |
| # define USE_WRITEUNRAISABLEMSG |
| #endif |
| |
| /************************************************************/ |
| |
| /* base type flag: exactly one of the following: */ |
| #define CT_PRIMITIVE_SIGNED 0x001 /* signed integer */ |
| #define CT_PRIMITIVE_UNSIGNED 0x002 /* unsigned integer */ |
| #define CT_PRIMITIVE_CHAR 0x004 /* char, wchar_t, charN_t */ |
| #define CT_PRIMITIVE_FLOAT 0x008 /* float, double, long double */ |
| #define CT_POINTER 0x010 /* pointer, excluding ptr-to-func */ |
| #define CT_ARRAY 0x020 /* array */ |
| #define CT_STRUCT 0x040 /* struct */ |
| #define CT_UNION 0x080 /* union */ |
| #define CT_FUNCTIONPTR 0x100 /* pointer to function */ |
| #define CT_VOID 0x200 /* void */ |
| #define CT_PRIMITIVE_COMPLEX 0x400 /* float _Complex, double _Complex */ |
| |
| /* other flags that may also be set in addition to the base flag: */ |
| #define CT_IS_VOIDCHAR_PTR 0x00001000 |
| #define CT_PRIMITIVE_FITS_LONG 0x00002000 |
| #define CT_IS_OPAQUE 0x00004000 |
| #define CT_IS_ENUM 0x00008000 |
| #define CT_IS_PTR_TO_OWNED 0x00010000 /* only owned if CDataOwning_Type */ |
| #define CT_CUSTOM_FIELD_POS 0x00020000 |
| #define CT_IS_LONGDOUBLE 0x00040000 |
| #define CT_IS_BOOL 0x00080000 |
| #define CT_IS_FILE 0x00100000 |
| #define CT_IS_VOID_PTR 0x00200000 |
| #define CT_WITH_VAR_ARRAY 0x00400000 /* with open-ended array, anywhere */ |
| /* unused 0x00800000 */ |
| #define CT_LAZY_FIELD_LIST 0x01000000 |
| #define CT_WITH_PACKED_CHANGE 0x02000000 |
| #define CT_IS_SIGNED_WCHAR 0x04000000 |
| #define CT_PRIMITIVE_ANY (CT_PRIMITIVE_SIGNED | \ |
| CT_PRIMITIVE_UNSIGNED | \ |
| CT_PRIMITIVE_CHAR | \ |
| CT_PRIMITIVE_FLOAT | \ |
| CT_PRIMITIVE_COMPLEX) |
| |
| typedef struct _ctypedescr { |
| PyObject_VAR_HEAD |
| |
| struct _ctypedescr *ct_itemdescr; /* ptrs and arrays: the item type */ |
| PyObject *ct_stuff; /* structs: dict of the fields |
| arrays: ctypedescr of the ptr type |
| function: tuple(abi, ctres, ctargs..) |
| enum: pair {"name":x},{x:"name"} |
| ptrs: lazily, ctypedescr of array */ |
| void *ct_extra; /* structs: first field (not a ref!) |
| function types: cif_description |
| primitives: prebuilt "cif" object */ |
| |
| PyObject *ct_weakreflist; /* weakref support */ |
| |
| PyObject *ct_unique_key; /* key in unique_cache (a string, but not |
| human-readable) */ |
| |
| Py_ssize_t ct_size; /* size of instances, or -1 if unknown */ |
| Py_ssize_t ct_length; /* length of arrays, or -1 if unknown; |
| or alignment of primitive and struct types; |
| always -1 for pointers */ |
| int ct_flags; /* CT_xxx flags */ |
| |
| int ct_name_position; /* index in ct_name of where to put a var name */ |
| char ct_name[1]; /* string, e.g. "int *" for pointers to ints */ |
| } CTypeDescrObject; |
| |
| typedef struct { |
| PyObject_HEAD |
| CTypeDescrObject *c_type; |
| char *c_data; |
| PyObject *c_weakreflist; |
| } CDataObject; |
| |
| typedef struct cfieldobject_s { |
| PyObject_HEAD |
| CTypeDescrObject *cf_type; |
| Py_ssize_t cf_offset; |
| short cf_bitshift; /* >= 0: bitshift; or BS_REGULAR or BS_EMPTY_ARRAY */ |
| short cf_bitsize; |
| unsigned char cf_flags; /* BF_... */ |
| struct cfieldobject_s *cf_next; |
| } CFieldObject; |
| #define BS_REGULAR (-1) /* a regular field, not with bitshift */ |
| #define BS_EMPTY_ARRAY (-2) /* a field declared 'type[0]' or 'type[]' */ |
| #define BF_IGNORE_IN_CTOR 0x01 /* union field not in the first place */ |
| |
| static PyTypeObject CTypeDescr_Type; |
| static PyTypeObject CField_Type; |
| static PyTypeObject CData_Type; |
| static PyTypeObject CDataOwning_Type; |
| static PyTypeObject CDataOwningGC_Type; |
| static PyTypeObject CDataFromBuf_Type; |
| static PyTypeObject CDataGCP_Type; |
| |
| #define CTypeDescr_Check(ob) (Py_TYPE(ob) == &CTypeDescr_Type) |
| #define CData_Check(ob) (Py_TYPE(ob) == &CData_Type || \ |
| Py_TYPE(ob) == &CDataOwning_Type || \ |
| Py_TYPE(ob) == &CDataOwningGC_Type || \ |
| Py_TYPE(ob) == &CDataFromBuf_Type || \ |
| Py_TYPE(ob) == &CDataGCP_Type) |
| #define CDataOwn_Check(ob) (Py_TYPE(ob) == &CDataOwning_Type || \ |
| Py_TYPE(ob) == &CDataOwningGC_Type) |
| |
| typedef union { |
| unsigned char m_char; |
| unsigned short m_short; |
| unsigned int m_int; |
| unsigned long m_long; |
| unsigned long long m_longlong; |
| float m_float; |
| double m_double; |
| long double m_longdouble; |
| } union_alignment; |
| |
| typedef struct { |
| CDataObject head; |
| union_alignment alignment; |
| } CDataObject_casted_primitive; |
| |
| typedef struct { |
| CDataObject head; |
| union_alignment alignment; |
| } CDataObject_own_nolength; |
| |
| typedef struct { |
| CDataObject head; |
| Py_ssize_t length; |
| union_alignment alignment; |
| } CDataObject_own_length; |
| |
| typedef struct { |
| CDataObject head; |
| PyObject *structobj; /* for ffi.new_handle() or ffi.new("struct *") */ |
| } CDataObject_own_structptr; |
| |
| typedef struct { |
| CDataObject head; |
| Py_ssize_t length; /* same as CDataObject_own_length up to here */ |
| Py_buffer *bufferview; |
| } CDataObject_frombuf; |
| |
| typedef struct { |
| CDataObject head; |
| Py_ssize_t length; /* same as CDataObject_own_length up to here */ |
| PyObject *origobj; |
| PyObject *destructor; |
| } CDataObject_gcp; |
| |
| typedef struct { |
| CDataObject head; |
| ffi_closure *closure; |
| } CDataObject_closure; |
| |
| typedef struct { |
| ffi_cif cif; |
| /* the following information is used when doing the call: |
| - a buffer of size 'exchange_size' is malloced |
| - the arguments are converted from Python objects to raw data |
| - the i'th raw data is stored at 'buffer + exchange_offset_arg[1+i]' |
| - the call is done |
| - the result is read back from 'buffer + exchange_offset_arg[0]' */ |
| Py_ssize_t exchange_size; |
| Py_ssize_t exchange_offset_arg[1]; |
| } cif_description_t; |
| |
| #define ADD_WRAPAROUND(x, y) ((Py_ssize_t)(((size_t)(x)) + ((size_t)(y)))) |
| #define MUL_WRAPAROUND(x, y) ((Py_ssize_t)(((size_t)(x)) * ((size_t)(y)))) |
| |
| |
| /* whenever running Python code, the errno is saved in this thread-local |
| variable */ |
| #ifndef MS_WIN32 |
| # include "misc_thread_posix.h" |
| #endif |
| |
| #include "minibuffer.h" |
| |
| #if PY_MAJOR_VERSION >= 3 |
| # include "file_emulator.h" |
| #endif |
| |
| #ifdef PyUnicode_KIND /* Python >= 3.3 */ |
| # include "wchar_helper_3.h" |
| #else |
| # include "wchar_helper.h" |
| #endif |
| |
| #include "../cffi/_cffi_errors.h" |
| |
| typedef struct _cffi_allocator_s { |
| PyObject *ca_alloc, *ca_free; |
| int ca_dont_clear; |
| } cffi_allocator_t; |
| static const cffi_allocator_t default_allocator = { NULL, NULL, 0 }; |
| static PyObject *FFIError; |
| static PyObject *unique_cache; |
| |
| /************************************************************/ |
| |
| static CTypeDescrObject * |
| ctypedescr_new(int name_size) |
| { |
| CTypeDescrObject *ct = PyObject_GC_NewVar(CTypeDescrObject, |
| &CTypeDescr_Type, |
| name_size); |
| if (ct == NULL) |
| return NULL; |
| |
| ct->ct_itemdescr = NULL; |
| ct->ct_stuff = NULL; |
| ct->ct_weakreflist = NULL; |
| ct->ct_unique_key = NULL; |
| PyObject_GC_Track(ct); |
| return ct; |
| } |
| |
| static CTypeDescrObject * |
| ctypedescr_new_on_top(CTypeDescrObject *ct_base, const char *extra_text, |
| int extra_position) |
| { |
| int base_name_len = strlen(ct_base->ct_name); |
| int extra_name_len = strlen(extra_text); |
| CTypeDescrObject *ct = ctypedescr_new(base_name_len + extra_name_len + 1); |
| char *p; |
| if (ct == NULL) |
| return NULL; |
| |
| Py_INCREF(ct_base); |
| ct->ct_itemdescr = ct_base; |
| ct->ct_name_position = ct_base->ct_name_position + extra_position; |
| |
| p = ct->ct_name; |
| memcpy(p, ct_base->ct_name, ct_base->ct_name_position); |
| p += ct_base->ct_name_position; |
| memcpy(p, extra_text, extra_name_len); |
| p += extra_name_len; |
| memcpy(p, ct_base->ct_name + ct_base->ct_name_position, |
| base_name_len - ct_base->ct_name_position + 1); |
| |
| return ct; |
| } |
| |
| static PyObject * |
| ctypedescr_repr(CTypeDescrObject *ct) |
| { |
| return PyText_FromFormat("<ctype '%s'>", ct->ct_name); |
| } |
| |
| static void |
| ctypedescr_dealloc(CTypeDescrObject *ct) |
| { |
| PyObject_GC_UnTrack(ct); |
| if (ct->ct_weakreflist != NULL) |
| PyObject_ClearWeakRefs((PyObject *) ct); |
| |
| if (ct->ct_unique_key != NULL) { |
| /* revive dead object temporarily for DelItem */ |
| Py_SET_REFCNT(ct, 43); |
| PyDict_DelItem(unique_cache, ct->ct_unique_key); |
| assert(Py_REFCNT(ct) == 42); |
| Py_SET_REFCNT(ct, 0); |
| Py_DECREF(ct->ct_unique_key); |
| } |
| Py_XDECREF(ct->ct_itemdescr); |
| Py_XDECREF(ct->ct_stuff); |
| if (ct->ct_flags & CT_FUNCTIONPTR) |
| PyObject_Free(ct->ct_extra); |
| Py_TYPE(ct)->tp_free((PyObject *)ct); |
| } |
| |
| static int |
| ctypedescr_traverse(CTypeDescrObject *ct, visitproc visit, void *arg) |
| { |
| Py_VISIT(ct->ct_itemdescr); |
| Py_VISIT(ct->ct_stuff); |
| return 0; |
| } |
| |
| static int |
| ctypedescr_clear(CTypeDescrObject *ct) |
| { |
| Py_CLEAR(ct->ct_itemdescr); |
| Py_CLEAR(ct->ct_stuff); |
| return 0; |
| } |
| |
| |
| static PyObject *nosuchattr(const char *attr) |
| { |
| PyErr_SetString(PyExc_AttributeError, attr); |
| return NULL; |
| } |
| |
| static PyObject *ctypeget_kind(CTypeDescrObject *ct, void *context) |
| { |
| char *result; |
| if (ct->ct_flags & CT_PRIMITIVE_ANY) { |
| if (ct->ct_flags & CT_IS_ENUM) |
| result = "enum"; |
| else |
| result = "primitive"; |
| } |
| else if (ct->ct_flags & CT_POINTER) { |
| result = "pointer"; |
| } |
| else if (ct->ct_flags & CT_ARRAY) { |
| result = "array"; |
| } |
| else if (ct->ct_flags & CT_VOID) { |
| result = "void"; |
| } |
| else if (ct->ct_flags & CT_STRUCT) { |
| result = "struct"; |
| } |
| else if (ct->ct_flags & CT_UNION) { |
| result = "union"; |
| } |
| else if (ct->ct_flags & CT_FUNCTIONPTR) { |
| result = "function"; |
| } |
| else |
| result = "?"; |
| |
| return PyText_FromString(result); |
| } |
| |
| static PyObject *ctypeget_cname(CTypeDescrObject *ct, void *context) |
| { |
| return PyText_FromString(ct->ct_name); |
| } |
| |
| static PyObject *ctypeget_item(CTypeDescrObject *ct, void *context) |
| { |
| if (ct->ct_flags & (CT_POINTER | CT_ARRAY)) { |
| Py_INCREF(ct->ct_itemdescr); |
| return (PyObject *)ct->ct_itemdescr; |
| } |
| return nosuchattr("item"); |
| } |
| |
| static PyObject *ctypeget_length(CTypeDescrObject *ct, void *context) |
| { |
| if (ct->ct_flags & CT_ARRAY) { |
| if (ct->ct_length >= 0) { |
| return PyInt_FromSsize_t(ct->ct_length); |
| } |
| else { |
| Py_INCREF(Py_None); |
| return Py_None; |
| } |
| } |
| return nosuchattr("length"); |
| } |
| |
| static PyObject * |
| get_field_name(CTypeDescrObject *ct, CFieldObject *cf); /* forward */ |
| |
| /* returns 0 if the struct ctype is opaque, 1 if it is not, or -1 if |
| an exception occurs */ |
| #define force_lazy_struct(ct) \ |
| ((ct)->ct_stuff != NULL ? 1 : do_realize_lazy_struct(ct)) |
| |
| static int do_realize_lazy_struct(CTypeDescrObject *ct); |
| /* forward, implemented in realize_c_type.c */ |
| |
| static PyObject *ctypeget_fields(CTypeDescrObject *ct, void *context) |
| { |
| if (ct->ct_flags & (CT_STRUCT | CT_UNION)) { |
| if (!(ct->ct_flags & CT_IS_OPAQUE)) { |
| CFieldObject *cf; |
| PyObject *res; |
| if (force_lazy_struct(ct) < 0) |
| return NULL; |
| res = PyList_New(0); |
| if (res == NULL) |
| return NULL; |
| for (cf = (CFieldObject *)ct->ct_extra; |
| cf != NULL; cf = cf->cf_next) { |
| PyObject *o = PyTuple_Pack(2, get_field_name(ct, cf), |
| (PyObject *)cf); |
| int err = (o != NULL) ? PyList_Append(res, o) : -1; |
| Py_XDECREF(o); |
| if (err < 0) { |
| Py_DECREF(res); |
| return NULL; |
| } |
| } |
| return res; |
| } |
| else { |
| Py_INCREF(Py_None); |
| return Py_None; |
| } |
| } |
| return nosuchattr("fields"); |
| } |
| |
| static PyObject *ctypeget_args(CTypeDescrObject *ct, void *context) |
| { |
| if (ct->ct_flags & CT_FUNCTIONPTR) { |
| PyObject *t = ct->ct_stuff; |
| return PyTuple_GetSlice(t, 2, PyTuple_GET_SIZE(t)); |
| } |
| return nosuchattr("args"); |
| } |
| |
| static PyObject *ctypeget_result(CTypeDescrObject *ct, void *context) |
| { |
| if (ct->ct_flags & CT_FUNCTIONPTR) { |
| PyObject *res = PyTuple_GetItem(ct->ct_stuff, 1); |
| Py_XINCREF(res); |
| return res; |
| } |
| return nosuchattr("result"); |
| } |
| |
| static PyObject *ctypeget_ellipsis(CTypeDescrObject *ct, void *context) |
| { |
| if (ct->ct_flags & CT_FUNCTIONPTR) { |
| PyObject *res = ct->ct_extra ? Py_False : Py_True; |
| Py_INCREF(res); |
| return res; |
| } |
| return nosuchattr("ellipsis"); |
| } |
| |
| static PyObject *ctypeget_abi(CTypeDescrObject *ct, void *context) |
| { |
| if (ct->ct_flags & CT_FUNCTIONPTR) { |
| PyObject *res = PyTuple_GetItem(ct->ct_stuff, 0); |
| Py_XINCREF(res); |
| return res; |
| } |
| return nosuchattr("abi"); |
| } |
| |
| static PyObject *ctypeget_elements(CTypeDescrObject *ct, void *context) |
| { |
| if (ct->ct_flags & CT_IS_ENUM) { |
| PyObject *res = PyTuple_GetItem(ct->ct_stuff, 1); |
| if (res) res = PyDict_Copy(res); |
| return res; |
| } |
| return nosuchattr("elements"); |
| } |
| |
| static PyObject *ctypeget_relements(CTypeDescrObject *ct, void *context) |
| { |
| if (ct->ct_flags & CT_IS_ENUM) { |
| PyObject *res = PyTuple_GetItem(ct->ct_stuff, 0); |
| if (res) res = PyDict_Copy(res); |
| return res; |
| } |
| return nosuchattr("relements"); |
| } |
| |
| static PyGetSetDef ctypedescr_getsets[] = { |
| {"kind", (getter)ctypeget_kind, NULL, "kind"}, |
| {"cname", (getter)ctypeget_cname, NULL, "C name"}, |
| {"item", (getter)ctypeget_item, NULL, "pointer to, or array of"}, |
| {"length", (getter)ctypeget_length, NULL, "array length or None"}, |
| {"fields", (getter)ctypeget_fields, NULL, "struct or union fields"}, |
| {"args", (getter)ctypeget_args, NULL, "function argument types"}, |
| {"result", (getter)ctypeget_result, NULL, "function result type"}, |
| {"ellipsis", (getter)ctypeget_ellipsis, NULL, "function has '...'"}, |
| {"abi", (getter)ctypeget_abi, NULL, "function ABI"}, |
| {"elements", (getter)ctypeget_elements, NULL, "enum elements"}, |
| {"relements", (getter)ctypeget_relements, NULL, "enum elements, reverse"}, |
| {NULL} /* sentinel */ |
| }; |
| |
| static PyObject * |
| ctypedescr_dir(PyObject *ct, PyObject *noarg) |
| { |
| int err; |
| struct PyGetSetDef *gsdef; |
| PyObject *res = PyList_New(0); |
| if (res == NULL) |
| return NULL; |
| |
| for (gsdef = ctypedescr_getsets; gsdef->name; gsdef++) { |
| PyObject *x = PyObject_GetAttrString(ct, gsdef->name); |
| if (x == NULL) { |
| PyErr_Clear(); |
| } |
| else { |
| Py_DECREF(x); |
| x = PyText_FromString(gsdef->name); |
| err = (x != NULL) ? PyList_Append(res, x) : -1; |
| Py_XDECREF(x); |
| if (err < 0) { |
| Py_DECREF(res); |
| return NULL; |
| } |
| } |
| } |
| return res; |
| } |
| |
| static PyMethodDef ctypedescr_methods[] = { |
| {"__dir__", ctypedescr_dir, METH_NOARGS}, |
| {NULL, NULL} /* sentinel */ |
| }; |
| |
| static PyTypeObject CTypeDescr_Type = { |
| PyVarObject_HEAD_INIT(NULL, 0) |
| "_cffi_backend.CType", |
| offsetof(CTypeDescrObject, ct_name), |
| sizeof(char), |
| (destructor)ctypedescr_dealloc, /* tp_dealloc */ |
| 0, /* tp_print */ |
| 0, /* tp_getattr */ |
| 0, /* tp_setattr */ |
| 0, /* tp_compare */ |
| (reprfunc)ctypedescr_repr, /* tp_repr */ |
| 0, /* tp_as_number */ |
| 0, /* tp_as_sequence */ |
| 0, /* tp_as_mapping */ |
| 0, /* tp_hash */ |
| 0, /* tp_call */ |
| 0, /* tp_str */ |
| PyObject_GenericGetAttr, /* tp_getattro */ |
| 0, /* tp_setattro */ |
| 0, /* tp_as_buffer */ |
| Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC, /* tp_flags */ |
| 0, /* tp_doc */ |
| (traverseproc)ctypedescr_traverse, /* tp_traverse */ |
| (inquiry)ctypedescr_clear, /* tp_clear */ |
| 0, /* tp_richcompare */ |
| offsetof(CTypeDescrObject, ct_weakreflist), /* tp_weaklistoffset */ |
| 0, /* tp_iter */ |
| 0, /* tp_iternext */ |
| ctypedescr_methods, /* tp_methods */ |
| 0, /* tp_members */ |
| ctypedescr_getsets, /* tp_getset */ |
| }; |
| |
| /************************************************************/ |
| |
| static PyObject * |
| get_field_name(CTypeDescrObject *ct, CFieldObject *cf) |
| { |
| Py_ssize_t i = 0; |
| PyObject *d_key, *d_value; |
| while (PyDict_Next(ct->ct_stuff, &i, &d_key, &d_value)) { |
| if (d_value == (PyObject *)cf) |
| return d_key; |
| } |
| Py_FatalError("_cffi_backend: get_field_name()"); |
| return NULL; |
| } |
| |
| static void |
| cfield_dealloc(CFieldObject *cf) |
| { |
| Py_DECREF(cf->cf_type); |
| PyObject_Del(cf); |
| } |
| |
| #undef OFF |
| #define OFF(x) offsetof(CFieldObject, x) |
| |
| static PyMemberDef cfield_members[] = { |
| {"type", T_OBJECT, OFF(cf_type), READONLY}, |
| {"offset", T_PYSSIZET, OFF(cf_offset), READONLY}, |
| {"bitshift", T_SHORT, OFF(cf_bitshift), READONLY}, |
| {"bitsize", T_SHORT, OFF(cf_bitsize), READONLY}, |
| {"flags", T_UBYTE, OFF(cf_flags), READONLY}, |
| {NULL} /* Sentinel */ |
| }; |
| #undef OFF |
| |
| static PyTypeObject CField_Type = { |
| PyVarObject_HEAD_INIT(NULL, 0) |
| "_cffi_backend.CField", |
| sizeof(CFieldObject), |
| 0, |
| (destructor)cfield_dealloc, /* tp_dealloc */ |
| 0, /* tp_print */ |
| 0, /* tp_getattr */ |
| 0, /* tp_setattr */ |
| 0, /* tp_compare */ |
| 0, /* tp_repr */ |
| 0, /* tp_as_number */ |
| 0, /* tp_as_sequence */ |
| 0, /* tp_as_mapping */ |
| 0, /* tp_hash */ |
| 0, /* tp_call */ |
| 0, /* tp_str */ |
| PyObject_GenericGetAttr, /* tp_getattro */ |
| 0, /* tp_setattro */ |
| 0, /* tp_as_buffer */ |
| Py_TPFLAGS_DEFAULT, /* tp_flags */ |
| 0, /* tp_doc */ |
| 0, /* tp_traverse */ |
| 0, /* tp_clear */ |
| 0, /* tp_richcompare */ |
| 0, /* tp_weaklistoffset */ |
| 0, /* tp_iter */ |
| 0, /* tp_iternext */ |
| 0, /* tp_methods */ |
| cfield_members, /* tp_members */ |
| }; |
| |
| /************************************************************/ |
| |
| static int |
| CDataObject_Or_PyFloat_Check(PyObject *ob) |
| { |
| return (PyFloat_Check(ob) || |
| (CData_Check(ob) && |
| (((CDataObject *)ob)->c_type->ct_flags & CT_PRIMITIVE_FLOAT))); |
| } |
| |
| static PY_LONG_LONG |
| _my_PyLong_AsLongLong(PyObject *ob) |
| { |
| /* (possibly) convert and cast a Python object to a long long. |
| Like PyLong_AsLongLong(), this version accepts a Python int too, and |
| does convertions from other types of objects. The difference is that |
| this version refuses floats. */ |
| #if PY_MAJOR_VERSION < 3 |
| if (PyInt_Check(ob)) { |
| return PyInt_AS_LONG(ob); |
| } |
| else |
| #endif |
| if (PyLong_Check(ob)) { |
| return PyLong_AsLongLong(ob); |
| } |
| else { |
| PyObject *io; |
| PY_LONG_LONG res; |
| PyNumberMethods *nb = ob->ob_type->tp_as_number; |
| |
| if (CDataObject_Or_PyFloat_Check(ob) || |
| nb == NULL || nb->nb_int == NULL) { |
| PyErr_SetString(PyExc_TypeError, "an integer is required"); |
| return -1; |
| } |
| io = (*nb->nb_int) (ob); |
| if (io == NULL) |
| return -1; |
| |
| if (PyIntOrLong_Check(io)) { |
| res = _my_PyLong_AsLongLong(io); |
| } |
| else { |
| PyErr_SetString(PyExc_TypeError, "integer conversion failed"); |
| res = -1; |
| } |
| Py_DECREF(io); |
| return res; |
| } |
| } |
| |
| static unsigned PY_LONG_LONG |
| _my_PyLong_AsUnsignedLongLong(PyObject *ob, int strict) |
| { |
| /* (possibly) convert and cast a Python object to an unsigned long long. |
| Like PyLong_AsLongLong(), this version accepts a Python int too, and |
| does convertions from other types of objects. If 'strict', complains |
| with OverflowError and refuses floats. If '!strict', rounds floats |
| and masks the result. */ |
| #if PY_MAJOR_VERSION < 3 |
| if (PyInt_Check(ob)) { |
| long value1 = PyInt_AS_LONG(ob); |
| if (strict && value1 < 0) |
| goto negative; |
| return (unsigned PY_LONG_LONG)(PY_LONG_LONG)value1; |
| } |
| else |
| #endif |
| if (PyLong_Check(ob)) { |
| if (strict) { |
| if (_PyLong_Sign(ob) < 0) |
| goto negative; |
| return PyLong_AsUnsignedLongLong(ob); |
| } |
| else { |
| return PyLong_AsUnsignedLongLongMask(ob); |
| } |
| } |
| else { |
| PyObject *io; |
| unsigned PY_LONG_LONG res; |
| PyNumberMethods *nb = ob->ob_type->tp_as_number; |
| |
| if ((strict && CDataObject_Or_PyFloat_Check(ob)) || |
| nb == NULL || nb->nb_int == NULL) { |
| PyErr_SetString(PyExc_TypeError, "an integer is required"); |
| return (unsigned PY_LONG_LONG)-1; |
| } |
| io = (*nb->nb_int) (ob); |
| if (io == NULL) |
| return (unsigned PY_LONG_LONG)-1; |
| |
| if (PyIntOrLong_Check(io)) { |
| res = _my_PyLong_AsUnsignedLongLong(io, strict); |
| } |
| else { |
| PyErr_SetString(PyExc_TypeError, "integer conversion failed"); |
| res = (unsigned PY_LONG_LONG)-1; |
| } |
| Py_DECREF(io); |
| return res; |
| } |
| |
| negative: |
| PyErr_SetString(PyExc_OverflowError, |
| "can't convert negative number to unsigned"); |
| return (unsigned PY_LONG_LONG)-1; |
| } |
| |
| #define _read_raw_data(type) \ |
| do { \ |
| if (size == sizeof(type)) { \ |
| type r; \ |
| memcpy(&r, target, sizeof(type)); \ |
| return r; \ |
| } \ |
| } while(0) |
| |
| static PY_LONG_LONG |
| read_raw_signed_data(char *target, int size) |
| { |
| _read_raw_data(signed char); |
| _read_raw_data(short); |
| _read_raw_data(int); |
| _read_raw_data(long); |
| _read_raw_data(PY_LONG_LONG); |
| Py_FatalError("read_raw_signed_data: bad integer size"); |
| return 0; |
| } |
| |
| static unsigned PY_LONG_LONG |
| read_raw_unsigned_data(char *target, int size) |
| { |
| _read_raw_data(unsigned char); |
| _read_raw_data(unsigned short); |
| _read_raw_data(unsigned int); |
| _read_raw_data(unsigned long); |
| _read_raw_data(unsigned PY_LONG_LONG); |
| Py_FatalError("read_raw_unsigned_data: bad integer size"); |
| return 0; |
| } |
| |
| #ifdef __GNUC__ |
| /* This is a workaround for what I think is a GCC bug on several |
| platforms. See issue #378. */ |
| __attribute__((noinline)) |
| #endif |
| void _cffi_memcpy(char *target, const void *src, size_t size) |
| { |
| memcpy(target, src, size); |
| } |
| |
| #define _write_raw_data(type) \ |
| do { \ |
| if (size == sizeof(type)) { \ |
| type r = (type)source; \ |
| _cffi_memcpy(target, &r, sizeof(type)); \ |
| return; \ |
| } \ |
| } while(0) |
| |
| static void |
| write_raw_integer_data(char *target, unsigned PY_LONG_LONG source, int size) |
| { |
| _write_raw_data(unsigned char); |
| _write_raw_data(unsigned short); |
| _write_raw_data(unsigned int); |
| _write_raw_data(unsigned long); |
| _write_raw_data(unsigned PY_LONG_LONG); |
| Py_FatalError("write_raw_integer_data: bad integer size"); |
| } |
| |
| static double |
| read_raw_float_data(char *target, int size) |
| { |
| _read_raw_data(float); |
| _read_raw_data(double); |
| Py_FatalError("read_raw_float_data: bad float size"); |
| return 0; |
| } |
| |
| static long double |
| read_raw_longdouble_data(char *target) |
| { |
| int size = sizeof(long double); |
| _read_raw_data(long double); |
| Py_FatalError("read_raw_longdouble_data: bad long double size"); |
| return 0; |
| } |
| |
| static Py_complex |
| read_raw_complex_data(char *target, int size) |
| { |
| Py_complex r = {0.0, 0.0}; |
| if (size == 2*sizeof(float)) { |
| float real_part, imag_part; |
| memcpy(&real_part, target + 0, sizeof(float)); |
| memcpy(&imag_part, target + sizeof(float), sizeof(float)); |
| r.real = real_part; |
| r.imag = imag_part; |
| return r; |
| } |
| if (size == 2*sizeof(double)) { |
| memcpy(&r, target, 2*sizeof(double)); |
| return r; |
| } |
| Py_FatalError("read_raw_complex_data: bad complex size"); |
| return r; |
| } |
| |
| static void |
| write_raw_float_data(char *target, double source, int size) |
| { |
| _write_raw_data(float); |
| _write_raw_data(double); |
| Py_FatalError("write_raw_float_data: bad float size"); |
| } |
| |
| static void |
| write_raw_longdouble_data(char *target, long double source) |
| { |
| int size = sizeof(long double); |
| _write_raw_data(long double); |
| } |
| |
| #define _write_raw_complex_data(type) \ |
| do { \ |
| if (size == 2*sizeof(type)) { \ |
| type r = (type)source.real; \ |
| type i = (type)source.imag; \ |
| _cffi_memcpy(target, &r, sizeof(type)); \ |
| _cffi_memcpy(target+sizeof(type), &i, sizeof(type)); \ |
| return; \ |
| } \ |
| } while(0) |
| |
| static void |
| write_raw_complex_data(char *target, Py_complex source, int size) |
| { |
| _write_raw_complex_data(float); |
| _write_raw_complex_data(double); |
| Py_FatalError("write_raw_complex_data: bad complex size"); |
| } |
| |
| static PyObject * |
| new_simple_cdata(char *data, CTypeDescrObject *ct) |
| { |
| CDataObject *cd = PyObject_New(CDataObject, &CData_Type); |
| if (cd == NULL) |
| return NULL; |
| Py_INCREF(ct); |
| cd->c_data = data; |
| cd->c_type = ct; |
| cd->c_weakreflist = NULL; |
| return (PyObject *)cd; |
| } |
| |
| static PyObject * |
| new_sized_cdata(char *data, CTypeDescrObject *ct, Py_ssize_t length) |
| { |
| CDataObject_own_length *scd; |
| |
| scd = (CDataObject_own_length *)PyObject_Malloc( |
| offsetof(CDataObject_own_length, alignment)); |
| if (PyObject_Init((PyObject *)scd, &CData_Type) == NULL) |
| return NULL; |
| Py_INCREF(ct); |
| scd->head.c_type = ct; |
| scd->head.c_data = data; |
| scd->head.c_weakreflist = NULL; |
| scd->length = length; |
| return (PyObject *)scd; |
| } |
| |
| static CDataObject *_new_casted_primitive(CTypeDescrObject *ct); /*forward*/ |
| |
| static PyObject * |
| convert_to_object(char *data, CTypeDescrObject *ct) |
| { |
| if (!(ct->ct_flags & CT_PRIMITIVE_ANY)) { |
| /* non-primitive types (check done just for performance) */ |
| if (ct->ct_flags & (CT_POINTER|CT_FUNCTIONPTR)) { |
| char *ptrdata = *(char **)data; |
| /*READ(data, sizeof(char *))*/ |
| return new_simple_cdata(ptrdata, ct); |
| } |
| else if (ct->ct_flags & CT_IS_OPAQUE) { |
| PyErr_Format(PyExc_TypeError, "cdata '%s' is opaque", |
| ct->ct_name); |
| return NULL; |
| } |
| else if (ct->ct_flags & (CT_STRUCT|CT_UNION)) { |
| return new_simple_cdata(data, ct); |
| } |
| else if (ct->ct_flags & CT_ARRAY) { |
| if (ct->ct_length < 0) { |
| /* we can't return a <cdata 'int[]'> here, because we don't |
| know the length to give it. As a compromize, returns |
| <cdata 'int *'> in this case. */ |
| ct = (CTypeDescrObject *)ct->ct_stuff; |
| } |
| return new_simple_cdata(data, ct); |
| } |
| } |
| else if (ct->ct_flags & CT_PRIMITIVE_SIGNED) { |
| PY_LONG_LONG value; |
| /*READ(data, ct->ct_size)*/ |
| value = read_raw_signed_data(data, ct->ct_size); |
| if (ct->ct_flags & CT_PRIMITIVE_FITS_LONG) |
| return PyInt_FromLong((long)value); |
| else |
| return PyLong_FromLongLong(value); |
| } |
| else if (ct->ct_flags & CT_PRIMITIVE_UNSIGNED) { |
| unsigned PY_LONG_LONG value; |
| /*READ(data, ct->ct_size)*/ |
| value = read_raw_unsigned_data(data, ct->ct_size); |
| |
| if (ct->ct_flags & CT_PRIMITIVE_FITS_LONG) { |
| if (ct->ct_flags & CT_IS_BOOL) { |
| PyObject *x; |
| switch ((int)value) { |
| case 0: x = Py_False; break; |
| case 1: x = Py_True; break; |
| default: |
| PyErr_Format(PyExc_ValueError, |
| "got a _Bool of value %d, expected 0 or 1", |
| (int)value); |
| return NULL; |
| } |
| Py_INCREF(x); |
| return x; |
| } |
| return PyInt_FromLong((long)value); |
| } |
| else |
| return PyLong_FromUnsignedLongLong(value); |
| } |
| else if (ct->ct_flags & CT_PRIMITIVE_FLOAT) { |
| /*READ(data, ct->ct_size)*/ |
| if (!(ct->ct_flags & CT_IS_LONGDOUBLE)) { |
| double value = read_raw_float_data(data, ct->ct_size); |
| return PyFloat_FromDouble(value); |
| } |
| else { |
| long double value = read_raw_longdouble_data(data); |
| CDataObject *cd = _new_casted_primitive(ct); |
| if (cd != NULL) |
| write_raw_longdouble_data(cd->c_data, value); |
| return (PyObject *)cd; |
| } |
| } |
| else if (ct->ct_flags & CT_PRIMITIVE_CHAR) { |
| /*READ(data, ct->ct_size)*/ |
| switch (ct->ct_size) { |
| case sizeof(char): |
| return PyBytes_FromStringAndSize(data, 1); |
| case 2: |
| return _my_PyUnicode_FromChar16((cffi_char16_t *)data, 1); |
| case 4: |
| return _my_PyUnicode_FromChar32((cffi_char32_t *)data, 1); |
| } |
| } |
| else if (ct->ct_flags & CT_PRIMITIVE_COMPLEX) { |
| Py_complex value = read_raw_complex_data(data, ct->ct_size); |
| return PyComplex_FromCComplex(value); |
| } |
| |
| PyErr_Format(PyExc_SystemError, |
| "convert_to_object: '%s'", ct->ct_name); |
| return NULL; |
| } |
| |
| static PyObject * |
| convert_to_object_bitfield(char *data, CFieldObject *cf) |
| { |
| CTypeDescrObject *ct = cf->cf_type; |
| /*READ(data, ct->ct_size)*/ |
| |
| if (ct->ct_flags & CT_PRIMITIVE_SIGNED) { |
| unsigned PY_LONG_LONG value, valuemask, shiftforsign; |
| PY_LONG_LONG result; |
| |
| value = (unsigned PY_LONG_LONG)read_raw_signed_data(data, ct->ct_size); |
| valuemask = (1ULL << cf->cf_bitsize) - 1ULL; |
| shiftforsign = 1ULL << (cf->cf_bitsize - 1); |
| value = ((value >> cf->cf_bitshift) + shiftforsign) & valuemask; |
| result = ((PY_LONG_LONG)value) - (PY_LONG_LONG)shiftforsign; |
| |
| if (ct->ct_flags & CT_PRIMITIVE_FITS_LONG) |
| return PyInt_FromLong((long)result); |
| else |
| return PyLong_FromLongLong(result); |
| } |
| else { |
| unsigned PY_LONG_LONG value, valuemask; |
| |
| value = read_raw_unsigned_data(data, ct->ct_size); |
| valuemask = (1ULL << cf->cf_bitsize) - 1ULL; |
| value = (value >> cf->cf_bitshift) & valuemask; |
| |
| if (ct->ct_flags & CT_PRIMITIVE_FITS_LONG) |
| return PyInt_FromLong((long)value); |
| else |
| return PyLong_FromUnsignedLongLong(value); |
| } |
| } |
| |
| static int _convert_overflow(PyObject *init, const char *ct_name) |
| { |
| PyObject *s; |
| if (PyErr_Occurred()) /* already an exception pending */ |
| return -1; |
| s = PyObject_Str(init); |
| if (s == NULL) |
| return -1; |
| PyErr_Format(PyExc_OverflowError, "integer %s does not fit '%s'", |
| PyText_AS_UTF8(s), ct_name); |
| Py_DECREF(s); |
| return -1; |
| } |
| |
| static int _convert_to_char(PyObject *init) |
| { |
| if (PyBytes_Check(init) && PyBytes_GET_SIZE(init) == 1) { |
| return (unsigned char)(PyBytes_AS_STRING(init)[0]); |
| } |
| if (CData_Check(init) && |
| (((CDataObject *)init)->c_type->ct_flags & CT_PRIMITIVE_CHAR) && |
| (((CDataObject *)init)->c_type->ct_size == sizeof(char))) { |
| char *data = ((CDataObject *)init)->c_data; |
| /*READ(data, 1)*/ |
| return *(unsigned char *)data; |
| } |
| PyErr_Format(PyExc_TypeError, |
| "initializer for ctype 'char' must be a "STR_OR_BYTES |
| " of length 1, not %.200s", Py_TYPE(init)->tp_name); |
| return -1; |
| } |
| |
| static cffi_char16_t _convert_to_char16_t(PyObject *init) |
| { |
| char err_got[80]; |
| err_got[0] = 0; |
| |
| if (PyUnicode_Check(init)) { |
| cffi_char16_t ordinal; |
| if (_my_PyUnicode_AsSingleChar16(init, &ordinal, err_got) == 0) |
| return ordinal; |
| } |
| if (CData_Check(init) && |
| (((CDataObject *)init)->c_type->ct_flags & CT_PRIMITIVE_CHAR) && |
| (((CDataObject *)init)->c_type->ct_size == 2)) { |
| char *data = ((CDataObject *)init)->c_data; |
| /*READ(data, 2)*/ |
| return *(cffi_char16_t *)data; |
| } |
| PyErr_Format(PyExc_TypeError, |
| "initializer for ctype 'char16_t' must be a unicode string " |
| "of length 1, not %.200s", |
| err_got[0] == 0 ? Py_TYPE(init)->tp_name : err_got); |
| return (cffi_char16_t)-1; |
| } |
| |
| static cffi_char32_t _convert_to_char32_t(PyObject *init) |
| { |
| char err_got[80]; |
| err_got[0] = 0; |
| |
| if (PyUnicode_Check(init)) { |
| cffi_char32_t ordinal; |
| if (_my_PyUnicode_AsSingleChar32(init, &ordinal, err_got) == 0) |
| return ordinal; |
| } |
| if (CData_Check(init) && |
| (((CDataObject *)init)->c_type->ct_flags & CT_PRIMITIVE_CHAR) && |
| (((CDataObject *)init)->c_type->ct_size == 4)) { |
| char *data = ((CDataObject *)init)->c_data; |
| /*READ(data, 4)*/ |
| return *(cffi_char32_t *)data; |
| } |
| PyErr_Format(PyExc_TypeError, |
| "initializer for ctype 'char32_t' must be a unicode string " |
| "of length 1, not %.200s", |
| err_got[0] == 0 ? Py_TYPE(init)->tp_name : err_got); |
| return (cffi_char32_t)-1; |
| } |
| |
| static int _convert_error(PyObject *init, CTypeDescrObject *ct, |
| const char *expected) |
| { |
| if (CData_Check(init)) { |
| CTypeDescrObject *ct2 = ((CDataObject *)init)->c_type; |
| if (strcmp(ct->ct_name, ct2->ct_name) != 0) |
| PyErr_Format(PyExc_TypeError, |
| "initializer for ctype '%s' must be a %s, " |
| "not cdata '%s'", |
| ct->ct_name, expected, ct2->ct_name); |
| else if (ct != ct2) { |
| /* in case we'd give the error message "initializer for |
| ctype 'A' must be a pointer to same type, not cdata |
| 'B'", but with A=B, then give instead a different error |
| message to try to clear up the confusion */ |
| PyErr_Format(PyExc_TypeError, |
| "initializer for ctype '%s' appears indeed to be '%s'," |
| " but the types are different (check that you are not" |
| " e.g. mixing up different ffi instances)", |
| ct->ct_name, ct2->ct_name); |
| } |
| else |
| { |
| PyErr_Format(PyExc_SystemError, |
| "initializer for ctype '%s' is correct, but we get " |
| "an internal mismatch--please report a bug", |
| ct->ct_name); |
| } |
| } |
| else |
| PyErr_Format(PyExc_TypeError, |
| "initializer for ctype '%s' must be a %s, " |
| "not %.200s", |
| ct->ct_name, expected, Py_TYPE(init)->tp_name); |
| return -1; |
| } |
| |
| static int /* forward */ |
| convert_from_object(char *data, CTypeDescrObject *ct, PyObject *init); |
| static int /* forward */ |
| convert_from_object_bitfield(char *data, CFieldObject *cf, PyObject *init); |
| |
| static Py_ssize_t |
| get_new_array_length(CTypeDescrObject *ctitem, PyObject **pvalue) |
| { |
| PyObject *value = *pvalue; |
| |
| if (PyList_Check(value) || PyTuple_Check(value)) { |
| return PySequence_Fast_GET_SIZE(value); |
| } |
| else if (PyBytes_Check(value)) { |
| /* from a string, we add the null terminator */ |
| return PyBytes_GET_SIZE(value) + 1; |
| } |
| else if (PyUnicode_Check(value)) { |
| /* from a unicode, we add the null terminator */ |
| int length; |
| if (ctitem->ct_size == 2) |
| length = _my_PyUnicode_SizeAsChar16(value); |
| else |
| length = _my_PyUnicode_SizeAsChar32(value); |
| return length + 1; |
| } |
| else { |
| Py_ssize_t explicitlength; |
| explicitlength = PyNumber_AsSsize_t(value, PyExc_OverflowError); |
| if (explicitlength < 0) { |
| if (PyErr_Occurred()) { |
| if (PyErr_ExceptionMatches(PyExc_TypeError)) |
| PyErr_Format(PyExc_TypeError, |
| "expected new array length or list/tuple/str, " |
| "not %.200s", Py_TYPE(value)->tp_name); |
| } |
| else |
| PyErr_SetString(PyExc_ValueError, "negative array length"); |
| return -1; |
| } |
| *pvalue = Py_None; |
| return explicitlength; |
| } |
| } |
| |
| static int |
| convert_field_from_object(char *data, CFieldObject *cf, PyObject *value) |
| { |
| data += cf->cf_offset; |
| if (cf->cf_bitshift >= 0) |
| return convert_from_object_bitfield(data, cf, value); |
| else |
| return convert_from_object(data, cf->cf_type, value); |
| } |
| |
| static int |
| add_varsize_length(Py_ssize_t offset, Py_ssize_t itemsize, |
| Py_ssize_t varsizelength, Py_ssize_t *optvarsize) |
| { |
| /* update '*optvarsize' to account for an array of 'varsizelength' |
| elements, each of size 'itemsize', that starts at 'offset'. */ |
| Py_ssize_t size = ADD_WRAPAROUND(offset, |
| MUL_WRAPAROUND(itemsize, varsizelength)); |
| if (size < 0 || |
| ((size - offset) / itemsize) != varsizelength) { |
| PyErr_SetString(PyExc_OverflowError, |
| "array size would overflow a Py_ssize_t"); |
| return -1; |
| } |
| if (size > *optvarsize) |
| *optvarsize = size; |
| return 0; |
| } |
| |
| static int |
| convert_struct_from_object(char *data, CTypeDescrObject *ct, PyObject *init, |
| Py_ssize_t *optvarsize); /* forward */ |
| |
| static int |
| convert_vfield_from_object(char *data, CFieldObject *cf, PyObject *value, |
| Py_ssize_t *optvarsize) |
| { |
| /* a special case for var-sized C99 arrays */ |
| if ((cf->cf_type->ct_flags & CT_ARRAY) && cf->cf_type->ct_size < 0) { |
| Py_ssize_t varsizelength = get_new_array_length( |
| cf->cf_type->ct_itemdescr, &value); |
| if (varsizelength < 0) |
| return -1; |
| if (optvarsize != NULL) { |
| /* in this mode, the only purpose of this function is to compute |
| the real size of the structure from a var-sized C99 array */ |
| assert(data == NULL); |
| return add_varsize_length(cf->cf_offset, |
| cf->cf_type->ct_itemdescr->ct_size, |
| varsizelength, |
| optvarsize); |
| } |
| /* if 'value' was only an integer, get_new_array_length() returns |
| it and convert 'value' to be None. Detect if this was the case, |
| and if so, stop here, leaving the content uninitialized |
| (it should be zero-initialized from somewhere else). */ |
| if (value == Py_None) |
| return 0; |
| } |
| if (optvarsize == NULL) { |
| return convert_field_from_object(data, cf, value); |
| } |
| else if ((cf->cf_type->ct_flags & CT_WITH_VAR_ARRAY) != 0 && |
| !CData_Check(value)) { |
| Py_ssize_t subsize = cf->cf_type->ct_size; |
| if (convert_struct_from_object(NULL, cf->cf_type, value, &subsize) < 0) |
| return -1; |
| return add_varsize_length(cf->cf_offset, 1, subsize, optvarsize); |
| } |
| else |
| return 0; |
| } |
| |
| static int |
| must_be_array_of_zero_or_one(const char *data, Py_ssize_t n) |
| { |
| Py_ssize_t i; |
| for (i = 0; i < n; i++) { |
| if (((unsigned char)data[i]) > 1) { |
| PyErr_SetString(PyExc_ValueError, |
| "an array of _Bool can only contain \\x00 or \\x01"); |
| return -1; |
| } |
| } |
| return 0; |
| } |
| |
| static Py_ssize_t |
| get_array_length(CDataObject *cd) |
| { |
| if (cd->c_type->ct_length < 0) |
| return ((CDataObject_own_length *)cd)->length; |
| else |
| return cd->c_type->ct_length; |
| } |
| |
| static int |
| convert_array_from_object(char *data, CTypeDescrObject *ct, PyObject *init) |
| { |
| /* used by convert_from_object(), and also to decode lists/tuples/unicodes |
| passed as function arguments. 'ct' is an CT_ARRAY in the first case |
| and a CT_POINTER in the second case. */ |
| const char *expected; |
| CTypeDescrObject *ctitem = ct->ct_itemdescr; |
| |
| if (PyList_Check(init) || PyTuple_Check(init)) { |
| PyObject **items; |
| Py_ssize_t i, n; |
| n = PySequence_Fast_GET_SIZE(init); |
| if (ct->ct_length >= 0 && n > ct->ct_length) { |
| PyErr_Format(PyExc_IndexError, |
| "too many initializers for '%s' (got %zd)", |
| ct->ct_name, n); |
| return -1; |
| } |
| items = PySequence_Fast_ITEMS(init); |
| for (i=0; i<n; i++) { |
| if (convert_from_object(data, ctitem, items[i]) < 0) |
| return -1; |
| data += ctitem->ct_size; |
| } |
| return 0; |
| } |
| else if ((ctitem->ct_flags & CT_PRIMITIVE_CHAR) || |
| ((ctitem->ct_flags & (CT_PRIMITIVE_SIGNED|CT_PRIMITIVE_UNSIGNED)) |
| && (ctitem->ct_size == sizeof(char)))) { |
| if (ctitem->ct_size == sizeof(char)) { |
| char *srcdata; |
| Py_ssize_t n; |
| if (!PyBytes_Check(init)) { |
| expected = STR_OR_BYTES" or list or tuple"; |
| goto cannot_convert; |
| } |
| n = PyBytes_GET_SIZE(init); |
| if (ct->ct_length >= 0 && n > ct->ct_length) { |
| PyErr_Format(PyExc_IndexError, |
| "initializer "STR_OR_BYTES" is too long for '%s' " |
| "(got %zd characters)", ct->ct_name, n); |
| return -1; |
| } |
| if (n != ct->ct_length) |
| n++; |
| srcdata = PyBytes_AS_STRING(init); |
| if (ctitem->ct_flags & CT_IS_BOOL) |
| if (must_be_array_of_zero_or_one(srcdata, n) < 0) |
| return -1; |
| memcpy(data, srcdata, n); |
| return 0; |
| } |
| else { |
| Py_ssize_t n; |
| if (!PyUnicode_Check(init)) { |
| expected = "unicode or list or tuple"; |
| goto cannot_convert; |
| } |
| |
| if (ctitem->ct_size == 4) |
| n = _my_PyUnicode_SizeAsChar32(init); |
| else |
| n = _my_PyUnicode_SizeAsChar16(init); |
| |
| if (ct->ct_length >= 0 && n > ct->ct_length) { |
| PyErr_Format(PyExc_IndexError, |
| "initializer unicode is too long for '%s' " |
| "(got %zd characters)", ct->ct_name, n); |
| return -1; |
| } |
| if (n != ct->ct_length) |
| n++; |
| if (ctitem->ct_size == 4) |
| return _my_PyUnicode_AsChar32(init, (cffi_char32_t *)data, n); |
| else |
| return _my_PyUnicode_AsChar16(init, (cffi_char16_t *)data, n); |
| } |
| } |
| else { |
| expected = "list or tuple"; |
| goto cannot_convert; |
| } |
| |
| cannot_convert: |
| if ((ct->ct_flags & CT_ARRAY) && CData_Check(init)) |
| { |
| CDataObject *cd = (CDataObject *)init; |
| if (cd->c_type == ct) |
| { |
| Py_ssize_t n = get_array_length(cd); |
| memcpy(data, cd->c_data, n * ctitem->ct_size); |
| return 0; |
| } |
| } |
| return _convert_error(init, ct, expected); |
| } |
| |
| static int |
| convert_struct_from_object(char *data, CTypeDescrObject *ct, PyObject *init, |
| Py_ssize_t *optvarsize) |
| { |
| /* does not accept 'init' being already a CData */ |
| const char *expected; |
| |
| if (force_lazy_struct(ct) <= 0) { |
| if (!PyErr_Occurred()) |
| PyErr_Format(PyExc_TypeError, "'%s' is opaque", ct->ct_name); |
| return -1; |
| } |
| |
| if (PyList_Check(init) || PyTuple_Check(init)) { |
| PyObject **items = PySequence_Fast_ITEMS(init); |
| Py_ssize_t i, n = PySequence_Fast_GET_SIZE(init); |
| CFieldObject *cf = (CFieldObject *)ct->ct_extra; |
| |
| for (i=0; i<n; i++) { |
| while (cf != NULL && (cf->cf_flags & BF_IGNORE_IN_CTOR)) |
| cf = cf->cf_next; |
| if (cf == NULL) { |
| PyErr_Format(PyExc_ValueError, |
| "too many initializers for '%s' (got %zd)", |
| ct->ct_name, n); |
| return -1; |
| } |
| if (convert_vfield_from_object(data, cf, items[i], optvarsize) < 0) |
| return -1; |
| cf = cf->cf_next; |
| } |
| return 0; |
| } |
| if (PyDict_Check(init)) { |
| PyObject *d_key, *d_value; |
| Py_ssize_t i = 0; |
| CFieldObject *cf; |
| |
| while (PyDict_Next(init, &i, &d_key, &d_value)) { |
| cf = (CFieldObject *)PyDict_GetItem(ct->ct_stuff, d_key); |
| if (cf == NULL) { |
| PyErr_SetObject(PyExc_KeyError, d_key); |
| return -1; |
| } |
| if (convert_vfield_from_object(data, cf, d_value, optvarsize) < 0) |
| return -1; |
| } |
| return 0; |
| } |
| expected = optvarsize == NULL ? "list or tuple or dict or struct-cdata" |
| : "list or tuple or dict"; |
| return _convert_error(init, ct, expected); |
| } |
| |
| #ifdef __GNUC__ |
| # if __GNUC__ >= 4 |
| /* Don't go inlining this huge function. Needed because occasionally |
| it gets inlined in places where is causes a warning: call to |
| __builtin___memcpy_chk will always overflow destination buffer |
| (which is places where the 'ct' should never represent such a large |
| primitive type anyway). */ |
| __attribute__((noinline)) |
| # endif |
| #endif |
| static int |
| convert_from_object(char *data, CTypeDescrObject *ct, PyObject *init) |
| { |
| const char *expected; |
| char buf[sizeof(PY_LONG_LONG)]; |
| |
| /*if (ct->ct_size > 0)*/ |
| /*WRITE(data, ct->ct_size)*/ |
| |
| if (ct->ct_flags & CT_ARRAY) { |
| return convert_array_from_object(data, ct, init); |
| } |
| if (ct->ct_flags & (CT_POINTER|CT_FUNCTIONPTR)) { |
| char *ptrdata; |
| CTypeDescrObject *ctinit; |
| |
| if (!CData_Check(init)) { |
| expected = "cdata pointer"; |
| goto cannot_convert; |
| } |
| ctinit = ((CDataObject *)init)->c_type; |
| if (!(ctinit->ct_flags & (CT_POINTER|CT_FUNCTIONPTR))) { |
| if (ctinit->ct_flags & CT_ARRAY) |
| ctinit = (CTypeDescrObject *)ctinit->ct_stuff; |
| else { |
| expected = "pointer or array"; |
| goto cannot_convert; |
| } |
| } |
| if (ctinit != ct) { |
| int combined_flags = ct->ct_flags | ctinit->ct_flags; |
| if (combined_flags & CT_IS_VOID_PTR) |
| ; /* accept "void *" as either source or target */ |
| else if (combined_flags & CT_IS_VOIDCHAR_PTR) { |
| /* for backward compatibility, accept "char *" as either |
| source of target. This is not what C does, though, |
| so emit a warning that will eventually turn into an |
| error. The warning is turned off if both types are |
| pointers to single bytes. */ |
| char *msg = (ct->ct_flags & CT_IS_VOIDCHAR_PTR ? |
| "implicit cast to 'char *' from a different pointer type: " |
| "will be forbidden in the future (check that the types " |
| "are as you expect; use an explicit ffi.cast() if they " |
| "are correct)" : |
| "implicit cast from 'char *' to a different pointer type: " |
| "will be forbidden in the future (check that the types " |
| "are as you expect; use an explicit ffi.cast() if they " |
| "are correct)"); |
| if ((ct->ct_flags & ctinit->ct_flags & CT_POINTER) && |
| ct->ct_itemdescr->ct_size == 1 && |
| ctinit->ct_itemdescr->ct_size == 1) { |
| /* no warning */ |
| } |
| else if (PyErr_WarnEx(PyExc_UserWarning, msg, 1)) |
| return -1; |
| } |
| else { |
| expected = "pointer to same type"; |
| goto cannot_convert; |
| } |
| } |
| ptrdata = ((CDataObject *)init)->c_data; |
| |
| *(char **)data = ptrdata; |
| return 0; |
| } |
| if (ct->ct_flags & CT_PRIMITIVE_SIGNED) { |
| PY_LONG_LONG value = _my_PyLong_AsLongLong(init); |
| if (value == -1 && PyErr_Occurred()) |
| return -1; |
| write_raw_integer_data(buf, value, ct->ct_size); |
| if (value != read_raw_signed_data(buf, ct->ct_size)) |
| goto overflow; |
| write_raw_integer_data(data, value, ct->ct_size); |
| return 0; |
| } |
| if (ct->ct_flags & CT_PRIMITIVE_UNSIGNED) { |
| unsigned PY_LONG_LONG value = _my_PyLong_AsUnsignedLongLong(init, 1); |
| if (value == (unsigned PY_LONG_LONG)-1 && PyErr_Occurred()) |
| return -1; |
| if (ct->ct_flags & CT_IS_BOOL) { |
| if (value > 1ULL) /* value != 0 && value != 1 */ |
| goto overflow; |
| } |
| else { |
| write_raw_integer_data(buf, value, ct->ct_size); |
| if (value != read_raw_unsigned_data(buf, ct->ct_size)) |
| goto overflow; |
| } |
| write_raw_integer_data(data, value, ct->ct_size); |
| return 0; |
| } |
| if (ct->ct_flags & CT_PRIMITIVE_FLOAT) { |
| double value; |
| if ((ct->ct_flags & CT_IS_LONGDOUBLE) && |
| CData_Check(init) && |
| (((CDataObject *)init)->c_type->ct_flags & CT_IS_LONGDOUBLE)) { |
| long double lvalue; |
| char *initdata = ((CDataObject *)init)->c_data; |
| /*READ(initdata, sizeof(long double))*/ |
| lvalue = read_raw_longdouble_data(initdata); |
| write_raw_longdouble_data(data, lvalue); |
| return 0; |
| } |
| value = PyFloat_AsDouble(init); |
| if (value == -1.0 && PyErr_Occurred()) |
| return -1; |
| if (!(ct->ct_flags & CT_IS_LONGDOUBLE)) |
| write_raw_float_data(data, value, ct->ct_size); |
| else |
| write_raw_longdouble_data(data, (long double)value); |
| return 0; |
| } |
| if (ct->ct_flags & CT_PRIMITIVE_CHAR) { |
| switch (ct->ct_size) { |
| case sizeof(char): { |
| int res = _convert_to_char(init); |
| if (res < 0) |
| return -1; |
| data[0] = res; |
| return 0; |
| } |
| case 2: { |
| cffi_char16_t res = _convert_to_char16_t(init); |
| if (res == (cffi_char16_t)-1 && PyErr_Occurred()) |
| return -1; |
| *(cffi_char16_t *)data = res; |
| return 0; |
| } |
| case 4: { |
| cffi_char32_t res = _convert_to_char32_t(init); |
| if (res == (cffi_char32_t)-1 && PyErr_Occurred()) |
| return -1; |
| *(cffi_char32_t *)data = res; |
| return 0; |
| } |
| } |
| } |
| if (ct->ct_flags & (CT_STRUCT|CT_UNION)) { |
| |
| if (CData_Check(init)) { |
| if (((CDataObject *)init)->c_type == ct && ct->ct_size >= 0) { |
| memcpy(data, ((CDataObject *)init)->c_data, ct->ct_size); |
| return 0; |
| } |
| } |
| return convert_struct_from_object(data, ct, init, NULL); |
| } |
| if (ct->ct_flags & CT_PRIMITIVE_COMPLEX) { |
| Py_complex value = PyComplex_AsCComplex(init); |
| if (PyErr_Occurred()) |
| return -1; |
| write_raw_complex_data(data, value, ct->ct_size); |
| return 0; |
| } |
| PyErr_Format(PyExc_SystemError, |
| "convert_from_object: '%s'", ct->ct_name); |
| return -1; |
| |
| overflow: |
| return _convert_overflow(init, ct->ct_name); |
| |
| cannot_convert: |
| return _convert_error(init, ct, expected); |
| } |
| |
| static int |
| convert_from_object_bitfield(char *data, CFieldObject *cf, PyObject *init) |
| { |
| CTypeDescrObject *ct = cf->cf_type; |
| PY_LONG_LONG fmin, fmax, value = PyLong_AsLongLong(init); |
| unsigned PY_LONG_LONG rawfielddata, rawvalue, rawmask; |
| if (value == -1 && PyErr_Occurred()) |
| return -1; |
| |
| if (ct->ct_flags & CT_PRIMITIVE_SIGNED) { |
| fmin = -(1LL << (cf->cf_bitsize-1)); |
| fmax = (1LL << (cf->cf_bitsize-1)) - 1LL; |
| if (fmax == 0) |
| fmax = 1; /* special case to let "int x:1" receive "1" */ |
| } |
| else { |
| fmin = 0LL; |
| fmax = (PY_LONG_LONG)((1ULL << cf->cf_bitsize) - 1ULL); |
| } |
| if (value < fmin || value > fmax) { |
| /* phew, PyErr_Format does not support "%lld" in Python 2.6 */ |
| PyObject *svalue = NULL, *sfmin = NULL, *sfmax = NULL; |
| PyObject *lfmin = NULL, *lfmax = NULL; |
| svalue = PyObject_Str(init); |
| if (svalue == NULL) goto skip; |
| lfmin = PyLong_FromLongLong(fmin); |
| if (lfmin == NULL) goto skip; |
| sfmin = PyObject_Str(lfmin); |
| if (sfmin == NULL) goto skip; |
| lfmax = PyLong_FromLongLong(fmax); |
| if (lfmax == NULL) goto skip; |
| sfmax = PyObject_Str(lfmax); |
| if (sfmax == NULL) goto skip; |
| PyErr_Format(PyExc_OverflowError, |
| "value %s outside the range allowed by the " |
| "bit field width: %s <= x <= %s", |
| PyText_AS_UTF8(svalue), |
| PyText_AS_UTF8(sfmin), |
| PyText_AS_UTF8(sfmax)); |
| skip: |
| Py_XDECREF(svalue); |
| Py_XDECREF(sfmin); |
| Py_XDECREF(sfmax); |
| Py_XDECREF(lfmin); |
| Py_XDECREF(lfmax); |
| return -1; |
| } |
| |
| rawmask = ((1ULL << cf->cf_bitsize) - 1ULL) << cf->cf_bitshift; |
| rawvalue = ((unsigned PY_LONG_LONG)value) << cf->cf_bitshift; |
| /*WRITE(data, ct->ct_size)*/ |
| rawfielddata = read_raw_unsigned_data(data, ct->ct_size); |
| rawfielddata = (rawfielddata & ~rawmask) | (rawvalue & rawmask); |
| write_raw_integer_data(data, rawfielddata, ct->ct_size); |
| return 0; |
| } |
| |
| static int |
| get_alignment(CTypeDescrObject *ct) |
| { |
| int align; |
| retry: |
| if ((ct->ct_flags & (CT_PRIMITIVE_ANY|CT_STRUCT|CT_UNION)) && |
| !(ct->ct_flags & CT_IS_OPAQUE)) { |
| align = ct->ct_length; |
| if (align == -1 && (ct->ct_flags & CT_LAZY_FIELD_LIST)) { |
| force_lazy_struct(ct); |
| align = ct->ct_length; |
| } |
| } |
| else if (ct->ct_flags & (CT_POINTER|CT_FUNCTIONPTR)) { |
| struct aligncheck_ptr { char x; char *y; }; |
| align = offsetof(struct aligncheck_ptr, y); |
| } |
| else if (ct->ct_flags & CT_ARRAY) { |
| ct = ct->ct_itemdescr; |
| goto retry; |
| } |
| else { |
| PyErr_Format(PyExc_ValueError, "ctype '%s' is of unknown alignment", |
| ct->ct_name); |
| return -1; |
| } |
| |
| if ((align < 1) || (align & (align-1))) { |
| PyErr_Format(PyExc_SystemError, |
| "found for ctype '%s' bogus alignment '%d'", |
| ct->ct_name, align); |
| return -1; |
| } |
| return align; |
| } |
| |
| static void cdata_dealloc(CDataObject *cd) |
| { |
| if (cd->c_weakreflist != NULL) |
| PyObject_ClearWeakRefs((PyObject *) cd); |
| |
| Py_DECREF(cd->c_type); |
| #ifndef CFFI_MEM_LEAK /* never release anything, tests only */ |
| Py_TYPE(cd)->tp_free((PyObject *)cd); |
| #endif |
| } |
| |
| static void cdataowning_dealloc(CDataObject *cd) |
| { |
| assert(!(cd->c_type->ct_flags & (CT_IS_VOID_PTR | CT_FUNCTIONPTR))); |
| |
| if (cd->c_type->ct_flags & CT_IS_PTR_TO_OWNED) { |
| /* for ffi.new("struct *") */ |
| Py_DECREF(((CDataObject_own_structptr *)cd)->structobj); |
| } |
| #if defined(CFFI_MEM_DEBUG) || defined(CFFI_MEM_LEAK) |
| if (cd->c_type->ct_flags & (CT_PRIMITIVE_ANY | CT_STRUCT | CT_UNION)) { |
| assert(cd->c_type->ct_size >= 0); |
| memset(cd->c_data, 0xDD, cd->c_type->ct_size); |
| } |
| else if (cd->c_type->ct_flags & CT_ARRAY) { |
| Py_ssize_t x = get_array_length(cd); |
| assert(x >= 0); |
| x *= cd->c_type->ct_itemdescr->ct_size; |
| assert(x >= 0); |
| memset(cd->c_data, 0xDD, x); |
| } |
| #endif |
| cdata_dealloc(cd); |
| } |
| |
| static void cdataowninggc_dealloc(CDataObject *cd) |
| { |
| PyObject_GC_UnTrack(cd); |
| |
| if (cd->c_type->ct_flags & CT_IS_VOID_PTR) { /* a handle */ |
| PyObject *x = ((CDataObject_own_structptr *)cd)->structobj; |
| Py_DECREF(x); |
| } |
| else if (cd->c_type->ct_flags & CT_FUNCTIONPTR) { /* a callback */ |
| ffi_closure *closure = ((CDataObject_closure *)cd)->closure; |
| PyObject *args = (PyObject *)(closure->user_data); |
| Py_XDECREF(args); |
| #if CFFI_CHECK_FFI_CLOSURE_ALLOC_MAYBE |
| if (CFFI_CHECK_FFI_CLOSURE_ALLOC) { |
| ffi_closure_free(closure); |
| } else |
| #endif |
| cffi_closure_free(closure); |
| } |
| else { |
| Py_FatalError("cdata CDataOwningGC_Type with unexpected type flags"); |
| } |
| cdata_dealloc(cd); |
| } |
| |
| static void cdatafrombuf_dealloc(CDataObject *cd) |
| { |
| Py_buffer *view = ((CDataObject_frombuf *)cd)->bufferview; |
| cdata_dealloc(cd); |
| |
| PyBuffer_Release(view); |
| PyObject_Free(view); |
| } |
| |
| static int cdataowninggc_traverse(CDataObject *cd, visitproc visit, void *arg) |
| { |
| if (cd->c_type->ct_flags & CT_IS_VOID_PTR) { /* a handle */ |
| PyObject *x = ((CDataObject_own_structptr *)cd)->structobj; |
| Py_VISIT(x); |
| } |
| else if (cd->c_type->ct_flags & CT_FUNCTIONPTR) { /* a callback */ |
| ffi_closure *closure = ((CDataObject_closure *)cd)->closure; |
| PyObject *args = (PyObject *)(closure->user_data); |
| Py_VISIT(args); |
| } |
| return 0; |
| } |
| |
| static int cdatafrombuf_traverse(CDataObject *cd, visitproc visit, void *arg) |
| { |
| Py_buffer *view = ((CDataObject_frombuf *)cd)->bufferview; |
| Py_VISIT(view->obj); |
| return 0; |
| } |
| |
| static int cdataowninggc_clear(CDataObject *cd) |
| { |
| if (cd->c_type->ct_flags & CT_IS_VOID_PTR) { /* a handle */ |
| CDataObject_own_structptr *cd1 = (CDataObject_own_structptr *)cd; |
| PyObject *x = cd1->structobj; |
| Py_INCREF(Py_None); |
| cd1->structobj = Py_None; |
| Py_DECREF(x); |
| } |
| else if (cd->c_type->ct_flags & CT_FUNCTIONPTR) { /* a callback */ |
| ffi_closure *closure = ((CDataObject_closure *)cd)->closure; |
| PyObject *args = (PyObject *)(closure->user_data); |
| closure->user_data = NULL; |
| Py_XDECREF(args); |
| } |
| return 0; |
| } |
| |
| static int cdatafrombuf_clear(CDataObject *cd) |
| { |
| Py_buffer *view = ((CDataObject_frombuf *)cd)->bufferview; |
| PyBuffer_Release(view); |
| return 0; |
| } |
| |
| /* forward */ |
| static void _my_PyErr_WriteUnraisable(PyObject *t, PyObject *v, PyObject *tb, |
| char *objdescr, PyObject *obj, |
| char *extra_error_line); |
| |
| |
| static void gcp_finalize(PyObject *destructor, PyObject *origobj) |
| { |
| /* NOTE: this decrements the reference count of the two arguments */ |
| |
| if (destructor != NULL) { |
| PyObject *result; |
| PyObject *error_type, *error_value, *error_traceback; |
| |
| /* Save the current exception */ |
| PyErr_Fetch(&error_type, &error_value, &error_traceback); |
| |
| result = PyObject_CallFunctionObjArgs(destructor, origobj, NULL); |
| if (result != NULL) { |
| Py_DECREF(result); |
| } |
| else { |
| PyObject *t, *v, *tb; |
| PyErr_Fetch(&t, &v, &tb); |
| /* Don't use error capture here, because it is very much |
| * like errors at __del__(), and these ones are not captured |
| * either */ |
| /* ecap = _cffi_start_error_capture(); */ |
| _my_PyErr_WriteUnraisable(t, v, tb, "From callback for ffi.gc ", |
| origobj, NULL); |
| /* _cffi_stop_error_capture(ecap); */ |
| } |
| Py_DECREF(destructor); |
| |
| /* Restore the saved exception */ |
| PyErr_Restore(error_type, error_value, error_traceback); |
| } |
| Py_XDECREF(origobj); |
| } |
| |
| static void cdatagcp_finalize(CDataObject_gcp *cd) |
| { |
| PyObject *destructor = cd->destructor; |
| PyObject *origobj = cd->origobj; |
| cd->destructor = NULL; |
| cd->origobj = NULL; |
| gcp_finalize(destructor, origobj); |
| } |
| |
| static void cdatagcp_dealloc(CDataObject_gcp *cd) |
| { |
| PyObject *destructor = cd->destructor; |
| PyObject *origobj = cd->origobj; |
| cdata_dealloc((CDataObject *)cd); |
| |
| gcp_finalize(destructor, origobj); |
| } |
| |
| static int cdatagcp_traverse(CDataObject_gcp *cd, visitproc visit, void *arg) |
| { |
| Py_VISIT(cd->destructor); |
| Py_VISIT(cd->origobj); |
| return 0; |
| } |
| |
| static PyObject *cdata_float(CDataObject *cd); /*forward*/ |
| |
| static PyObject *convert_cdata_to_enum_string(CDataObject *cd, int both) |
| { |
| PyObject *d_key, *d_value; |
| CTypeDescrObject *ct = cd->c_type; |
| |
| assert(ct->ct_flags & CT_IS_ENUM); |
| d_key = convert_to_object(cd->c_data, ct); |
| if (d_key == NULL) |
| return NULL; |
| |
| d_value = PyDict_GetItem(PyTuple_GET_ITEM(ct->ct_stuff, 1), d_key); |
| if (d_value != NULL) { |
| if (both) { |
| PyObject *o = PyObject_Str(d_key); |
| if (o == NULL) |
| d_value = NULL; |
| else { |
| d_value = PyText_FromFormat("%s: %s", |
| PyText_AS_UTF8(o), |
| PyText_AS_UTF8(d_value)); |
| Py_DECREF(o); |
| } |
| } |
| else |
| Py_INCREF(d_value); |
| } |
| else |
| d_value = PyObject_Str(d_key); |
| Py_DECREF(d_key); |
| return d_value; |
| } |
| |
| static PyObject *cdata_repr(CDataObject *cd) |
| { |
| char *extra; |
| PyObject *result, *s; |
| |
| if (cd->c_type->ct_flags & CT_PRIMITIVE_ANY) { |
| if (cd->c_type->ct_flags & CT_IS_ENUM) { |
| s = convert_cdata_to_enum_string(cd, 1); |
| } |
| else if (cd->c_type->ct_flags & CT_IS_LONGDOUBLE) { |
| long double lvalue; |
| char buffer[128]; /* big enough */ |
| /*READ(cd->c_data, sizeof(long double)*/ |
| lvalue = read_raw_longdouble_data(cd->c_data); |
| sprintf(buffer, "%LE", lvalue); |
| s = PyText_FromString(buffer); |
| } |
| else { |
| PyObject *o = convert_to_object(cd->c_data, cd->c_type); |
| if (o == NULL) |
| return NULL; |
| s = PyObject_Repr(o); |
| Py_DECREF(o); |
| } |
| } |
| else if ((cd->c_type->ct_flags & CT_ARRAY) && cd->c_type->ct_length < 0) { |
| s = PyText_FromFormat("sliced length %zd", get_array_length(cd)); |
| } |
| else { |
| if (cd->c_data != NULL) { |
| s = PyText_FromFormat("%p", cd->c_data); |
| } |
| else |
| s = PyText_FromString("NULL"); |
| } |
| if (s == NULL) |
| return NULL; |
| /* it's slightly confusing to get "<cdata 'struct foo' 0x...>" because the |
| struct foo is not owned. Trying to make it clearer, write in this |
| case "<cdata 'struct foo &' 0x...>". */ |
| if (cd->c_type->ct_flags & (CT_STRUCT|CT_UNION)) |
| extra = " &"; |
| else |
| extra = ""; |
| result = PyText_FromFormat("<cdata '%s%s' %s>", |
| cd->c_type->ct_name, extra, |
| PyText_AsUTF8(s)); |
| Py_DECREF(s); |
| return result; |
| } |
| |
| static PyObject *_cdata_repr2(CDataObject *cd, char *text, PyObject *x) |
| { |
| PyObject *res, *s = PyObject_Repr(x); |
| if (s == NULL) |
| return NULL; |
| res = PyText_FromFormat("<cdata '%s' %s %s>", |
| cd->c_type->ct_name, text, PyText_AsUTF8(s)); |
| Py_DECREF(s); |
| return res; |
| } |
| |
| static Py_ssize_t _cdata_var_byte_size(CDataObject *cd) |
| { |
| /* If 'cd' is a 'struct foo' or 'struct foo *' allocated with |
| ffi.new(), and if the struct foo contains a varsize array, |
| then return the real allocated size. Otherwise, return -1. */ |
| if (!CDataOwn_Check(cd)) |
| return -1; |
| |
| if (cd->c_type->ct_flags & CT_IS_PTR_TO_OWNED) { |
| cd = (CDataObject *)((CDataObject_own_structptr *)cd)->structobj; |
| } |
| if (cd->c_type->ct_flags & CT_WITH_VAR_ARRAY) { |
| return ((CDataObject_own_length *)cd)->length; |
| } |
| return -1; |
| } |
| |
| static PyObject *_frombuf_repr(CDataObject *cd, const char *cd_type_name) |
| { |
| Py_buffer *view = ((CDataObject_frombuf *)cd)->bufferview; |
| const char *obj_tp_name; |
| if (view->obj == NULL) { |
| return PyText_FromFormat( |
| "<cdata '%s' buffer RELEASED>", |
| cd_type_name); |
| } |
| |
| obj_tp_name = Py_TYPE(view->obj)->tp_name; |
| if (cd->c_type->ct_flags & CT_ARRAY) |
| { |
| Py_ssize_t buflen = get_array_length(cd); |
| return PyText_FromFormat( |
| "<cdata '%s' buffer len %zd from '%.200s' object>", |
| cd_type_name, |
| buflen, |
| obj_tp_name); |
| } |
| else |
| { |
| return PyText_FromFormat( |
| "<cdata '%s' buffer from '%.200s' object>", |
| cd_type_name, |
| obj_tp_name); |
| } |
| } |
| |
| static PyObject *cdataowning_repr(CDataObject *cd) |
| { |
| Py_ssize_t size = _cdata_var_byte_size(cd); |
| if (size < 0) { |
| if (cd->c_type->ct_flags & CT_POINTER) |
| size = cd->c_type->ct_itemdescr->ct_size; |
| else if (cd->c_type->ct_flags & CT_ARRAY) |
| size = get_array_length(cd) * cd->c_type->ct_itemdescr->ct_size; |
| else |
| size = cd->c_type->ct_size; |
| } |
| return PyText_FromFormat("<cdata '%s' owning %zd bytes>", |
| cd->c_type->ct_name, size); |
| } |
| |
| static PyObject *cdataowninggc_repr(CDataObject *cd) |
| { |
| if (cd->c_type->ct_flags & CT_IS_VOID_PTR) { /* a handle */ |
| PyObject *x = ((CDataObject_own_structptr *)cd)->structobj; |
| return _cdata_repr2(cd, "handle to", x); |
| } |
| else if (cd->c_type->ct_flags & CT_FUNCTIONPTR) { /* a callback */ |
| ffi_closure *closure = ((CDataObject_closure *)cd)->closure; |
| PyObject *args = (PyObject *)closure->user_data; |
| if (args == NULL) |
| return cdata_repr(cd); |
| else |
| return _cdata_repr2(cd, "calling", PyTuple_GET_ITEM(args, 1)); |
| } |
| return cdataowning_repr(cd); /* but should be unreachable */ |
| } |
| |
| static PyObject *cdatafrombuf_repr(CDataObject *cd) |
| { |
| return _frombuf_repr(cd, cd->c_type->ct_name); |
| } |
| |
| static int cdata_nonzero(CDataObject *cd) |
| { |
| if (cd->c_type->ct_flags & CT_PRIMITIVE_ANY) { |
| if (cd->c_type->ct_flags & (CT_PRIMITIVE_SIGNED | |
| CT_PRIMITIVE_UNSIGNED | |
| CT_PRIMITIVE_CHAR)) |
| return read_raw_unsigned_data(cd->c_data, cd->c_type->ct_size) != 0; |
| |
| if (cd->c_type->ct_flags & CT_PRIMITIVE_FLOAT) { |
| if (cd->c_type->ct_flags & CT_IS_LONGDOUBLE) |
| return read_raw_longdouble_data(cd->c_data) != 0.0; |
| return read_raw_float_data(cd->c_data, cd->c_type->ct_size) != 0.0; |
| } |
| if (cd->c_type->ct_flags & CT_PRIMITIVE_COMPLEX) { |
| Py_complex value = read_raw_complex_data(cd->c_data, |
| cd->c_type->ct_size); |
| return value.real != 0.0 || value.imag != 0.0; |
| } |
| } |
| return cd->c_data != NULL; |
| } |
| |
| static PyObject *cdata_int(CDataObject *cd) |
| { |
| if ((cd->c_type->ct_flags & (CT_PRIMITIVE_SIGNED|CT_PRIMITIVE_FITS_LONG)) |
| == (CT_PRIMITIVE_SIGNED|CT_PRIMITIVE_FITS_LONG)) { |
| /* this case is to handle enums, but also serves as a slight |
| performance improvement for some other primitive types */ |
| long value; |
| /*READ(cd->c_data, cd->c_type->ct_size)*/ |
| value = (long)read_raw_signed_data(cd->c_data, cd->c_type->ct_size); |
| return PyInt_FromLong(value); |
| } |
| if (cd->c_type->ct_flags & (CT_PRIMITIVE_SIGNED|CT_PRIMITIVE_UNSIGNED)) { |
| PyObject *result = convert_to_object(cd->c_data, cd->c_type); |
| if (result != NULL && PyBool_Check(result)) |
| result = PyInt_FromLong(PyInt_AsLong(result)); |
| return result; |
| } |
| else if (cd->c_type->ct_flags & CT_PRIMITIVE_CHAR) { |
| /*READ(cd->c_data, cd->c_type->ct_size)*/ |
| switch (cd->c_type->ct_size) { |
| case sizeof(char): |
| return PyInt_FromLong((unsigned char)cd->c_data[0]); |
| case 2: |
| return PyInt_FromLong((long)*(cffi_char16_t *)cd->c_data); |
| case 4: |
| if (cd->c_type->ct_flags & CT_IS_SIGNED_WCHAR) |
| return PyInt_FromLong((long)*(int32_t *)cd->c_data); |
| else if (sizeof(long) > 4) |
| return PyInt_FromLong(*(uint32_t *)cd->c_data); |
| else |
| return PyLong_FromUnsignedLong(*(uint32_t *)cd->c_data); |
| } |
| } |
| else if (cd->c_type->ct_flags & CT_PRIMITIVE_FLOAT) { |
| PyObject *o = cdata_float(cd); |
| #if PY_MAJOR_VERSION < 3 |
| PyObject *r = o ? PyNumber_Int(o) : NULL; |
| #else |
| PyObject *r = o ? PyNumber_Long(o) : NULL; |
| #endif |
| Py_XDECREF(o); |
| return r; |
| } |
| PyErr_Format(PyExc_TypeError, "int() not supported on cdata '%s'", |
| cd->c_type->ct_name); |
| return NULL; |
| } |
| |
| #if PY_MAJOR_VERSION < 3 |
| static PyObject *cdata_long(CDataObject *cd) |
| { |
| PyObject *res = cdata_int(cd); |
| if (res != NULL && PyInt_CheckExact(res)) { |
| PyObject *o = PyLong_FromLong(PyInt_AS_LONG(res)); |
| Py_DECREF(res); |
| res = o; |
| } |
| return res; |
| } |
| #endif |
| |
| static PyObject *cdata_float(CDataObject *cd) |
| { |
| if (cd->c_type->ct_flags & CT_PRIMITIVE_FLOAT) { |
| double value; |
| /*READ(cd->c_data, cd->c_type->ct_size)*/ |
| if (!(cd->c_type->ct_flags & CT_IS_LONGDOUBLE)) { |
| value = read_raw_float_data(cd->c_data, cd->c_type->ct_size); |
| } |
| else { |
| value = (double)read_raw_longdouble_data(cd->c_data); |
| } |
| return PyFloat_FromDouble(value); |
| } |
| PyErr_Format(PyExc_TypeError, "float() not supported on cdata '%s'", |
| cd->c_type->ct_name); |
| return NULL; |
| } |
| |
| static PyObject *cdata_richcompare(PyObject *v, PyObject *w, int op) |
| { |
| int v_is_ptr, w_is_ptr; |
| PyObject *pyres; |
| |
| assert(CData_Check(v)); |
| |
| /* Comparisons involving a primitive cdata work differently than |
| * comparisons involving a struct/array/pointer. |
| * |
| * If v or w is a struct/array/pointer, then the other must be too |
| * (otherwise we return NotImplemented and leave the case to |
| * Python). If both are, then we compare the addresses. |
| * |
| * If v and/or w is a primitive cdata, then we convert the cdata(s) |
| * to regular Python objects and redo the comparison there. |
| */ |
| |
| v_is_ptr = !(((CDataObject *)v)->c_type->ct_flags & CT_PRIMITIVE_ANY); |
| w_is_ptr = CData_Check(w) && |
| !(((CDataObject *)w)->c_type->ct_flags & CT_PRIMITIVE_ANY); |
| |
| if (v_is_ptr && w_is_ptr) { |
| int res; |
| char *v_cdata = ((CDataObject *)v)->c_data; |
| char *w_cdata = ((CDataObject *)w)->c_data; |
| |
| switch (op) { |
| case Py_EQ: res = (v_cdata == w_cdata); break; |
| case Py_NE: res = (v_cdata != w_cdata); break; |
| case Py_LT: res = (v_cdata < w_cdata); break; |
| case Py_LE: res = (v_cdata <= w_cdata); break; |
| case Py_GT: res = (v_cdata > w_cdata); break; |
| case Py_GE: res = (v_cdata >= w_cdata); break; |
| default: res = -1; |
| } |
| pyres = res ? Py_True : Py_False; |
| } |
| else if (v_is_ptr || w_is_ptr) { |
| pyres = Py_NotImplemented; |
| } |
| else { |
| PyObject *aa[2]; |
| int i; |
| |
| aa[0] = v; Py_INCREF(v); |
| aa[1] = w; Py_INCREF(w); |
| pyres = NULL; |
| |
| for (i = 0; i < 2; i++) { |
| v = aa[i]; |
| if (!CData_Check(v)) |
| continue; |
| w = convert_to_object(((CDataObject *)v)->c_data, |
| ((CDataObject *)v)->c_type); |
| if (w == NULL) |
| goto error; |
| if (CData_Check(w)) { |
| Py_DECREF(w); |
| PyErr_Format(PyExc_NotImplementedError, |
| "cannot use <cdata '%s'> in a comparison", |
| ((CDataObject *)v)->c_type->ct_name); |
| goto error; |
| } |
| aa[i] = w; |
| Py_DECREF(v); |
| } |
| pyres = PyObject_RichCompare(aa[0], aa[1], op); |
| error: |
| Py_DECREF(aa[1]); |
| Py_DECREF(aa[0]); |
| return pyres; |
| } |
| |
| Py_INCREF(pyres); |
| return pyres; |
| } |
| |
| #if PY_MAJOR_VERSION < 3 |
| typedef long Py_hash_t; |
| #endif |
| |
| static Py_hash_t cdata_hash(PyObject *v) |
| { |
| if (((CDataObject *)v)->c_type->ct_flags & CT_PRIMITIVE_ANY) { |
| PyObject *vv = convert_to_object(((CDataObject *)v)->c_data, |
| ((CDataObject *)v)->c_type); |
| if (vv == NULL) |
| return -1; |
| if (!CData_Check(vv)) { |
| Py_hash_t hash = PyObject_Hash(vv); |
| Py_DECREF(vv); |
| return hash; |
| } |
| Py_DECREF(vv); |
| } |
| return _Py_HashPointer(((CDataObject *)v)->c_data); |
| } |
| |
| static Py_ssize_t |
| cdata_length(CDataObject *cd) |
| { |
| if (cd->c_type->ct_flags & CT_ARRAY) { |
| return get_array_length(cd); |
| } |
| PyErr_Format(PyExc_TypeError, "cdata of type '%s' has no len()", |
| cd->c_type->ct_name); |
| return -1; |
| } |
| |
| static char * |
| _cdata_get_indexed_ptr(CDataObject *cd, PyObject *key) |
| { |
| Py_ssize_t i = PyNumber_AsSsize_t(key, PyExc_IndexError); |
| if (i == -1 && PyErr_Occurred()) |
| return NULL; |
| |
| if (cd->c_type->ct_flags & CT_POINTER) { |
| if (CDataOwn_Check(cd)) { |
| if (i != 0) { |
| PyErr_Format(PyExc_IndexError, |
| "cdata '%s' can only be indexed by 0", |
| cd->c_type->ct_name); |
| return NULL; |
| } |
| } |
| else { |
| if (cd->c_data == NULL) { |
| PyErr_Format(PyExc_RuntimeError, |
| "cannot dereference null pointer from cdata '%s'", |
| cd->c_type->ct_name); |
| return NULL; |
| } |
| } |
| } |
| else if (cd->c_type->ct_flags & CT_ARRAY) { |
| if (i < 0) { |
| PyErr_SetString(PyExc_IndexError, |
| "negative index"); |
| return NULL; |
| } |
| if (i >= get_array_length(cd)) { |
| PyErr_Format(PyExc_IndexError, |
| "index too large for cdata '%s' (expected %zd < %zd)", |
| cd->c_type->ct_name, |
| i, get_array_length(cd)); |
| return NULL; |
| } |
| } |
| else { |
| PyErr_Format(PyExc_TypeError, "cdata of type '%s' cannot be indexed", |
| cd->c_type->ct_name); |
| return NULL; |
| } |
| return cd->c_data + i * cd->c_type->ct_itemdescr->ct_size; |
| } |
| |
| static PyObject * |
| new_array_type(CTypeDescrObject *ctptr, Py_ssize_t length); /* forward */ |
| |
| static CTypeDescrObject * |
| _cdata_getslicearg(CDataObject *cd, PySliceObject *slice, Py_ssize_t bounds[]) |
| { |
| Py_ssize_t start, stop; |
| CTypeDescrObject *ct; |
| |
| start = PyInt_AsSsize_t(slice->start); |
| if (start == -1 && PyErr_Occurred()) { |
| if (slice->start == Py_None) |
| PyErr_SetString(PyExc_IndexError, "slice start must be specified"); |
| return NULL; |
| } |
| stop = PyInt_AsSsize_t(slice->stop); |
| if (stop == -1 && PyErr_Occurred()) { |
| if (slice->stop == Py_None) |
| PyErr_SetString(PyExc_IndexError, "slice stop must be specified"); |
| return NULL; |
| } |
| if (slice->step != Py_None) { |
| PyErr_SetString(PyExc_IndexError, "slice with step not supported"); |
| return NULL; |
| } |
| if (start > stop) { |
| PyErr_SetString(PyExc_IndexError, "slice start > stop"); |
| return NULL; |
| } |
| |
| ct = cd->c_type; |
| if (ct->ct_flags & CT_ARRAY) { |
| if (start < 0) { |
| PyErr_SetString(PyExc_IndexError, |
| "negative index"); |
| return NULL; |
| } |
| if (stop > get_array_length(cd)) { |
| PyErr_Format(PyExc_IndexError, |
| "index too large (expected %zd <= %zd)", |
| stop, get_array_length(cd)); |
| return NULL; |
| } |
| ct = (CTypeDescrObject *)ct->ct_stuff; |
| } |
| else if (!(ct->ct_flags & CT_POINTER)) { |
| PyErr_Format(PyExc_TypeError, "cdata of type '%s' cannot be indexed", |
| ct->ct_name); |
| return NULL; |
| } |
| |
| bounds[0] = start; |
| bounds[1] = stop - start; |
| return ct; |
| } |
| |
| static PyObject * |
| cdata_slice(CDataObject *cd, PySliceObject *slice) |
| { |
| char *cdata; |
| Py_ssize_t bounds[2]; |
| CTypeDescrObject *ct = _cdata_getslicearg(cd, slice, bounds); |
| if (ct == NULL) |
| return NULL; |
| |
| if (ct->ct_stuff == NULL) { |
| ct->ct_stuff = new_array_type(ct, -1); |
| if (ct->ct_stuff == NULL) |
| return NULL; |
| } |
| ct = (CTypeDescrObject *)ct->ct_stuff; |
| |
| cdata = cd->c_data + ct->ct_itemdescr->ct_size * bounds[0]; |
| return new_sized_cdata(cdata, ct, bounds[1]); |
| } |
| |
| static int |
| cdata_ass_slice(CDataObject *cd, PySliceObject *slice, PyObject *v) |
| { |
| Py_ssize_t bounds[2], i, length, itemsize; |
| PyObject *it, *item; |
| PyObject *(*iternext)(PyObject *); |
| char *cdata; |
| int err; |
| CTypeDescrObject *ct = _cdata_getslicearg(cd, slice, bounds); |
| if (ct == NULL) |
| return -1; |
| ct = ct->ct_itemdescr; |
| itemsize = ct->ct_size; |
| cdata = cd->c_data + itemsize * bounds[0]; |
| length = bounds[1]; |
| |
| if (CData_Check(v)) { |
| CTypeDescrObject *ctv = ((CDataObject *)v)->c_type; |
| if ((ctv->ct_flags & CT_ARRAY) && (ctv->ct_itemdescr == ct) && |
| (get_array_length((CDataObject *)v) == length)) { |
| /* fast path: copying from exactly the correct type */ |
| memmove(cdata, ((CDataObject *)v)->c_data, itemsize * length); |
| return 0; |
| } |
| } |
| |
| /* A fast path for <char[]>[0:N] = b"somestring" or bytearray, which |
| also adds support for Python 3: otherwise, you get integers while |
| enumerating the string, and you can't set them to characters :-/ |
| */ |
| if ((ct->ct_flags & CT_PRIMITIVE_CHAR) && itemsize == sizeof(char)) { |
| char *src; |
| Py_ssize_t srclen; |
| if (PyBytes_Check(v)) { |
| srclen = PyBytes_GET_SIZE(v); |
| src = PyBytes_AS_STRING(v); |
| } |
| else if (PyByteArray_Check(v)) { |
| srclen = PyByteArray_GET_SIZE(v); |
| src = PyByteArray_AS_STRING(v); |
| } |
| else |
| goto other_types; |
| |
| if (srclen != length) { |
| PyErr_Format(PyExc_ValueError, |
| "need a string of length %zd, got %zd", |
| length, srclen); |
| return -1; |
| } |
| memcpy(cdata, src, length); |
| return 0; |
| } |
| other_types: |
| |
| it = PyObject_GetIter(v); |
| if (it == NULL) |
| return -1; |
| iternext = *it->ob_type->tp_iternext; |
| |
| for (i = 0; i < length; i++) { |
| item = iternext(it); |
| if (item == NULL) { |
| if (!PyErr_Occurred()) |
| PyErr_Format(PyExc_ValueError, |
| "need %zd values to unpack, got %zd", |
| length, i); |
| goto error; |
| } |
| err = convert_from_object(cdata, ct, item); |
| Py_DECREF(item); |
| if (err < 0) |
| goto error; |
| |
| cdata += itemsize; |
| } |
| item = iternext(it); |
| if (item != NULL) { |
| Py_DECREF(item); |
| PyErr_Format(PyExc_ValueError, |
| "got more than %zd values to unpack", length); |
| } |
| error: |
| Py_DECREF(it); |
| return PyErr_Occurred() ? -1 : 0; |
| } |
| |
| static PyObject * |
| cdataowning_subscript(CDataObject *cd, PyObject *key) |
| { |
| char *c; |
| if (PySlice_Check(key)) |
| return cdata_slice(cd, (PySliceObject *)key); |
| |
| c = _cdata_get_indexed_ptr(cd, key); |
| /* use 'mp_subscript' instead of 'sq_item' because we don't want |
| negative indexes to be corrected automatically */ |
| if (c == NULL && PyErr_Occurred()) |
| return NULL; |
| |
| if (cd->c_type->ct_flags & CT_IS_PTR_TO_OWNED) { |
| PyObject *res = ((CDataObject_own_structptr *)cd)->structobj; |
| Py_INCREF(res); |
| return res; |
| } |
| else { |
| return convert_to_object(c, cd->c_type->ct_itemdescr); |
| } |
| } |
| |
| static PyObject * |
| cdata_subscript(CDataObject *cd, PyObject *key) |
| { |
| char *c; |
| if (PySlice_Check(key)) |
| return cdata_slice(cd, (PySliceObject *)key); |
| |
| c = _cdata_get_indexed_ptr(cd, key); |
| /* use 'mp_subscript' instead of 'sq_item' because we don't want |
| negative indexes to be corrected automatically */ |
| if (c == NULL && PyErr_Occurred()) |
| return NULL; |
| return convert_to_object(c, cd->c_type->ct_itemdescr); |
| } |
| |
| static int |
| cdata_ass_sub(CDataObject *cd, PyObject *key, PyObject *v) |
| { |
| char *c; |
| CTypeDescrObject *ctitem; |
| if (PySlice_Check(key)) |
| return cdata_ass_slice(cd, (PySliceObject *)key, v); |
| |
| c = _cdata_get_indexed_ptr(cd, key); |
| ctitem = cd->c_type->ct_itemdescr; |
| /* use 'mp_ass_subscript' instead of 'sq_ass_item' because we don't want |
| negative indexes to be corrected automatically */ |
| if (c == NULL && PyErr_Occurred()) |
| return -1; |
| if (v == NULL) { |
| PyErr_SetString(PyExc_TypeError, |
| "'del x[n]' not supported for cdata objects"); |
| return -1; |
| } |
| return convert_from_object(c, ctitem, v); |
| } |
| |
| static PyObject * |
| _cdata_add_or_sub(PyObject *v, PyObject *w, int sign) |
| { |
| Py_ssize_t i, itemsize; |
| CDataObject *cd; |
| CTypeDescrObject *ctptr; |
| |
| if (!CData_Check(v)) { |
| PyObject *swap; |
| assert(CData_Check(w)); |
| if (sign != 1) |
| goto not_implemented; |
| swap = v; |
| v = w; |
| w = swap; |
| } |
| |
| i = PyNumber_AsSsize_t(w, PyExc_OverflowError); |
| if (i == -1 && PyErr_Occurred()) |
| return NULL; |
| i *= sign; |
| |
| cd = (CDataObject *)v; |
| if (cd->c_type->ct_flags & CT_POINTER) |
| ctptr = cd->c_type; |
| else if (cd->c_type->ct_flags & CT_ARRAY) { |
| ctptr = (CTypeDescrObject *)cd->c_type->ct_stuff; |
| } |
| else { |
| PyErr_Format(PyExc_TypeError, "cannot add a cdata '%s' and a number", |
| cd->c_type->ct_name); |
| return NULL; |
| } |
| itemsize = ctptr->ct_itemdescr->ct_size; |
| if (itemsize < 0) { |
| if (ctptr->ct_flags & CT_IS_VOID_PTR) { |
| itemsize = 1; |
| } |
| else { |
| PyErr_Format(PyExc_TypeError, |
| "ctype '%s' points to items of unknown size", |
| cd->c_type->ct_name); |
| return NULL; |
| } |
| } |
| return new_simple_cdata(cd->c_data + i * itemsize, ctptr); |
| |
| not_implemented: |
| Py_INCREF(Py_NotImplemented); |
| return Py_NotImplemented; |
| } |
| |
| static PyObject * |
| cdata_add(PyObject *v, PyObject *w) |
| { |
| return _cdata_add_or_sub(v, w, +1); |
| } |
| |
| static PyObject * |
| cdata_sub(PyObject *v, PyObject *w) |
| { |
| if (CData_Check(v) && CData_Check(w)) { |
| CDataObject *cdv = (CDataObject *)v; |
| CDataObject *cdw = (CDataObject *)w; |
| CTypeDescrObject *ct = cdw->c_type; |
| Py_ssize_t diff, itemsize; |
| |
| if (ct->ct_flags & CT_ARRAY) /* ptr_to_T - array_of_T: ok */ |
| ct = (CTypeDescrObject *)ct->ct_stuff; |
| |
| if (ct != cdv->c_type || !(ct->ct_flags & CT_POINTER) || |
| (ct->ct_itemdescr->ct_size <= 0 && |
| !(ct->ct_flags & CT_IS_VOID_PTR))) { |
| PyErr_Format(PyExc_TypeError, |
| "cannot subtract cdata '%s' and cdata '%s'", |
| cdv->c_type->ct_name, ct->ct_name); |
| return NULL; |
| } |
| itemsize = ct->ct_itemdescr->ct_size; |
| diff = cdv->c_data - cdw->c_data; |
| if (itemsize > 1) { |
| if (diff % itemsize) { |
| PyErr_SetString(PyExc_ValueError, |
| "pointer subtraction: the distance between the two " |
| "pointers is not a multiple of the item size"); |
| return NULL; |
| } |
| diff = diff / itemsize; |
| } |
| #if PY_MAJOR_VERSION < 3 |
| return PyInt_FromSsize_t(diff); |
| #else |
| return PyLong_FromSsize_t(diff); |
| #endif |
| } |
| |
| return _cdata_add_or_sub(v, w, -1); |
| } |
| |
| static void |
| _cdata_attr_errmsg(char *errmsg, CDataObject *cd, PyObject *attr) |
| { |
| const char *text; |
| if (!PyErr_ExceptionMatches(PyExc_AttributeError)) |
| return; |
| PyErr_Clear(); |
| text = PyText_AsUTF8(attr); |
| if (text == NULL) |
| return; |
| PyErr_Format(PyExc_AttributeError, errmsg, cd->c_type->ct_name, text); |
| } |
| |
| static PyObject * |
| cdata_getattro(CDataObject *cd, PyObject *attr) |
| { |
| CFieldObject *cf; |
| CTypeDescrObject *ct = cd->c_type; |
| char *errmsg = "cdata '%s' has no attribute '%s'"; |
| PyObject *x; |
| |
| if (ct->ct_flags & CT_POINTER) |
| ct = ct->ct_itemdescr; |
| |
| if (ct->ct_flags & (CT_STRUCT|CT_UNION)) { |
| switch (force_lazy_struct(ct)) { |
| case 1: |
| cf = (CFieldObject *)PyDict_GetItem(ct->ct_stuff, attr); |
| if (cf != NULL) { |
| /* read the field 'cf' */ |
| char *data = cd->c_data + cf->cf_offset; |
| Py_ssize_t array_len, size; |
| |
| if (cf->cf_bitshift == BS_REGULAR) { |
| return convert_to_object(data, cf->cf_type); |
| } |
| else if (cf->cf_bitshift != BS_EMPTY_ARRAY) { |
| return convert_to_object_bitfield(data, cf); |
| } |
| |
| /* variable-length array: */ |
| /* if reading variable length array from variable length |
| struct, calculate array type from allocated length */ |
| size = _cdata_var_byte_size(cd) - cf->cf_offset; |
| if (size >= 0) { |
| array_len = size / cf->cf_type->ct_itemdescr->ct_size; |
| return new_sized_cdata(data, cf->cf_type, array_len); |
| } |
| return new_simple_cdata(data, |
| (CTypeDescrObject *)cf->cf_type->ct_stuff); |
| } |
| errmsg = "cdata '%s' has no field '%s'"; |
| break; |
| case -1: |
| return NULL; |
| default: |
| errmsg = "cdata '%s' points to an opaque type: cannot read fields"; |
| break; |
| } |
| } |
| x = PyObject_GenericGetAttr((PyObject *)cd, attr); |
| if (x == NULL) |
| _cdata_attr_errmsg(errmsg, cd, attr); |
| return x; |
| } |
| |
| static int |
| cdata_setattro(CDataObject *cd, PyObject *attr, PyObject *value) |
| { |
| CFieldObject *cf; |
| CTypeDescrObject *ct = cd->c_type; |
| char *errmsg = "cdata '%s' has no attribute '%s'"; |
| int x; |
| |
| if (ct->ct_flags & CT_POINTER) |
| ct = ct->ct_itemdescr; |
| |
| if (ct->ct_flags & (CT_STRUCT|CT_UNION)) { |
| switch (force_lazy_struct(ct)) { |
| case 1: |
| cf = (CFieldObject *)PyDict_GetItem(ct->ct_stuff, attr); |
| if (cf != NULL) { |
| /* write the field 'cf' */ |
| if (value != NULL) { |
| return convert_field_from_object(cd->c_data, cf, value); |
| } |
| else { |
| PyErr_SetString(PyExc_AttributeError, |
| "cannot delete struct field"); |
| return -1; |
| } |
| } |
| errmsg = "cdata '%s' has no field '%s'"; |
| break; |
| case -1: |
| return -1; |
| default: |
| errmsg = "cdata '%s' points to an opaque type: cannot write fields"; |
| break; |
| } |
| } |
| x = PyObject_GenericSetAttr((PyObject *)cd, attr, value); |
| if (x < 0) |
| _cdata_attr_errmsg(errmsg, cd, attr); |
| return x; |
| } |
| |
| static PyObject * |
| convert_struct_to_owning_object(char *data, CTypeDescrObject *ct); /*forward*/ |
| |
| static cif_description_t * |
| fb_prepare_cif(PyObject *fargs, CTypeDescrObject *, Py_ssize_t, ffi_abi); |
| /*forward*/ |
| |
| static PyObject *new_primitive_type(const char *name); /*forward*/ |
| |
| static CTypeDescrObject *_get_ct_int(void) |
| { |
| static CTypeDescrObject *ct_int = NULL; |
| if (ct_int == NULL) { |
| ct_int = (CTypeDescrObject *)new_primitive_type("int"); |
| } |
| return ct_int; |
| } |
| |
| static Py_ssize_t |
| _prepare_pointer_call_argument(CTypeDescrObject *ctptr, PyObject *init, |
| char **output_data) |
| { |
| /* 'ctptr' is here a pointer type 'ITEM *'. Accept as argument an |
| initializer for an array 'ITEM[]'. This includes the case of |
| passing a Python byte string to a 'char *' argument. |
| |
| This function returns -1 if an error occurred, |
| 0 if conversion succeeded (into *output_data), |
| or N > 0 if conversion would require N bytes of storage. |
| */ |
| Py_ssize_t length, datasize; |
| CTypeDescrObject *ctitem; |
| |
| if (CData_Check(init)) |
| goto convert_default; |
| |
| ctitem = ctptr->ct_itemdescr; |
| /* XXX some code duplication, how to avoid it? */ |
| if (PyBytes_Check(init)) { |
| /* from a string: just returning the string here is fine. |
| We assume that the C code won't modify the 'char *' data. */ |
| if ((ctptr->ct_flags & CT_IS_VOIDCHAR_PTR) || |
| ((ctitem->ct_flags & (CT_PRIMITIVE_SIGNED|CT_PRIMITIVE_UNSIGNED)) |
| && (ctitem->ct_size == sizeof(char)))) { |
| #if defined(CFFI_MEM_DEBUG) || defined(CFFI_MEM_LEAK) |
| length = PyBytes_GET_SIZE(init) + 1; |
| #else |
| *output_data = PyBytes_AS_STRING(init); |
| if (ctitem->ct_flags & CT_IS_BOOL) |
| if (must_be_array_of_zero_or_one(*output_data, |
| PyBytes_GET_SIZE(init)) < 0) |
| return -1; |
| return 0; |
| #endif |
| } |
| else |
| goto convert_default; |
| } |
| else if (PyList_Check(init) || PyTuple_Check(init)) { |
| length = PySequence_Fast_GET_SIZE(init); |
| } |
| else if (PyUnicode_Check(init)) { |
| /* from a unicode, we add the null terminator */ |
| if (ctitem->ct_size == 2) |
| length = _my_PyUnicode_SizeAsChar16(init); |
| else |
| length = _my_PyUnicode_SizeAsChar32(init); |
| length += 1; |
| } |
| else if ((ctitem->ct_flags & CT_IS_FILE) && PyFile_Check(init)) { |
| *output_data = (char *)PyFile_AsFile(init); |
| if (*output_data == NULL && PyErr_Occurred()) |
| return -1; |
| return 0; |
| } |
| else { |
| /* refuse to receive just an integer (and interpret it |
| as the array size) */ |
| goto convert_default; |
| } |
| |
| if (ctitem->ct_size <= 0) |
| goto convert_default; |
| datasize = MUL_WRAPAROUND(length, ctitem->ct_size); |
| if ((datasize / ctitem->ct_size) != length) { |
| PyErr_SetString(PyExc_OverflowError, |
| "array size would overflow a Py_ssize_t"); |
| return -1; |
| } |
| if (datasize <= 0) |
| datasize = 1; |
| return datasize; |
| |
| convert_default: |
| return convert_from_object((char *)output_data, ctptr, init); |
| } |
| |
| static PyObject* |
| cdata_call(CDataObject *cd, PyObject *args, PyObject *kwds) |
| { |
| char *buffer; |
| void** buffer_array; |
| cif_description_t *cif_descr; |
| Py_ssize_t i, nargs, nargs_declared; |
| PyObject *signature, *res = NULL, *fvarargs; |
| CTypeDescrObject *fresult; |
| char *resultdata; |
| char *errormsg; |
| struct freeme_s { |
| struct freeme_s *next; |
| union_alignment alignment; |
| } *freeme = NULL; |
| |
| if (!(cd->c_type->ct_flags & CT_FUNCTIONPTR)) { |
| PyErr_Format(PyExc_TypeError, "cdata '%s' is not callable", |
| cd->c_type->ct_name); |
| return NULL; |
| } |
| if (cd->c_data == NULL) { |
| PyErr_Format(PyExc_RuntimeError, |
| "cannot call null pointer pointer from cdata '%s'", |
| cd->c_type->ct_name); |
| return NULL; |
| } |
| if (kwds != NULL && PyDict_Size(kwds) != 0) { |
| PyErr_SetString(PyExc_TypeError, |
| "a cdata function cannot be called with keyword arguments"); |
| return NULL; |
| } |
| signature = cd->c_type->ct_stuff; |
| nargs = PyTuple_Size(args); |
| if (nargs < 0) |
| return NULL; |
| nargs_declared = PyTuple_GET_SIZE(signature) - 2; |
| fresult = (CTypeDescrObject *)PyTuple_GET_ITEM(signature, 1); |
| fvarargs = NULL; |
| buffer = NULL; |
| |
| cif_descr = (cif_description_t *)cd->c_type->ct_extra; |
| |
| if (cif_descr != NULL) { |
| /* regular case: this function does not take '...' arguments */ |
| if (nargs != nargs_declared) { |
| errormsg = "'%s' expects %zd arguments, got %zd"; |
| bad_number_of_arguments: |
| PyErr_Format(PyExc_TypeError, errormsg, |
| cd->c_type->ct_name, nargs_declared, nargs); |
| goto error; |
| } |
| } |
| else { |
| /* call of a variadic function */ |
| ffi_abi fabi; |
| if (nargs < nargs_declared) { |
| errormsg = "'%s' expects at least %zd arguments, got %zd"; |
| goto bad_number_of_arguments; |
| } |
| fvarargs = PyTuple_New(nargs); |
| if (fvarargs == NULL) |
| goto error; |
| for (i = 0; i < nargs_declared; i++) { |
| PyObject *o = PyTuple_GET_ITEM(signature, 2 + i); |
| Py_INCREF(o); |
| PyTuple_SET_ITEM(fvarargs, i, o); |
| } |
| for (i = nargs_declared; i < nargs; i++) { |
| PyObject *obj = PyTuple_GET_ITEM(args, i); |
| CTypeDescrObject *ct; |
| |
| if (CData_Check(obj)) { |
| ct = ((CDataObject *)obj)->c_type; |
| if (ct->ct_flags & (CT_PRIMITIVE_CHAR | CT_PRIMITIVE_UNSIGNED | |
| CT_PRIMITIVE_SIGNED)) { |
| if (ct->ct_size < (Py_ssize_t)sizeof(int)) { |
| ct = _get_ct_int(); |
| if (ct == NULL) |
| goto error; |
| } |
| } |
| else if (ct->ct_flags & CT_ARRAY) { |
| ct = (CTypeDescrObject *)ct->ct_stuff; |
| } |
| Py_INCREF(ct); |
| } |
| else { |
| PyErr_Format(PyExc_TypeError, |
| "argument %zd passed in the variadic part " |
| "needs to be a cdata object (got %.200s)", |
| i + 1, Py_TYPE(obj)->tp_name); |
| goto error; |
| } |
| PyTuple_SET_ITEM(fvarargs, i, (PyObject *)ct); |
| } |
| #if PY_MAJOR_VERSION < 3 |
| fabi = PyInt_AS_LONG(PyTuple_GET_ITEM(signature, 0)); |
| #else |
| fabi = PyLong_AS_LONG(PyTuple_GET_ITEM(signature, 0)); |
| #endif |
| cif_descr = fb_prepare_cif(fvarargs, fresult, nargs_declared, fabi); |
| if (cif_descr == NULL) |
| goto error; |
| } |
| |
| buffer = PyObject_Malloc(cif_descr->exchange_size); |
| if (buffer == NULL) { |
| PyErr_NoMemory(); |
| goto error; |
| } |
| |
| buffer_array = (void **)buffer; |
| |
| for (i=0; i<nargs; i++) { |
| CTypeDescrObject *argtype; |
| char *data = buffer + cif_descr->exchange_offset_arg[1 + i]; |
| PyObject *obj = PyTuple_GET_ITEM(args, i); |
| |
| buffer_array[i] = data; |
| |
| if (i < nargs_declared) |
| argtype = (CTypeDescrObject *)PyTuple_GET_ITEM(signature, 2 + i); |
| else |
| argtype = (CTypeDescrObject *)PyTuple_GET_ITEM(fvarargs, i); |
| |
| if (argtype->ct_flags & CT_POINTER) { |
| char *tmpbuf; |
| Py_ssize_t datasize = _prepare_pointer_call_argument( |
| argtype, obj, (char **)data); |
| if (datasize == 0) |
| ; /* successfully filled '*data' */ |
| else if (datasize < 0) |
| goto error; |
| else { |
| if (datasize <= 512) { |
| tmpbuf = alloca(datasize); |
| } |
| else { |
| struct freeme_s *fp = (struct freeme_s *)PyObject_Malloc( |
| offsetof(struct freeme_s, alignment) + |
| (size_t)datasize); |
| if (fp == NULL) { |
| PyErr_NoMemory(); |
| goto error; |
| } |
| fp->next = freeme; |
| freeme = fp; |
| tmpbuf = (char *)&fp->alignment; |
| } |
| memset(tmpbuf, 0, datasize); |
| *(char **)data = tmpbuf; |
| if (convert_array_from_object(tmpbuf, argtype, obj) < 0) |
| goto error; |
| } |
| } |
| else if (convert_from_object(data, argtype, obj) < 0) |
| goto error; |
| } |
| |
| resultdata = buffer + cif_descr->exchange_offset_arg[0]; |
| /*READ(cd->c_data, sizeof(void(*)(void)))*/ |
| |
| Py_BEGIN_ALLOW_THREADS |
| restore_errno(); |
| ffi_call(&cif_descr->cif, (void (*)(void))(cd->c_data), |
| resultdata, buffer_array); |
| save_errno(); |
| Py_END_ALLOW_THREADS |
| |
| if (fresult->ct_flags & (CT_PRIMITIVE_CHAR | CT_PRIMITIVE_SIGNED | |
| CT_PRIMITIVE_UNSIGNED)) { |
| #ifdef WORDS_BIGENDIAN |
| /* For results of precisely these types, libffi has a strange |
| rule that they will be returned as a whole 'ffi_arg' if they |
| are smaller. The difference only matters on big-endian. */ |
| if (fresult->ct_size < sizeof(ffi_arg)) |
| resultdata += (sizeof(ffi_arg) - fresult->ct_size); |
| #endif |
| res = convert_to_object(resultdata, fresult); |
| } |
| else if (fresult->ct_flags & CT_VOID) { |
| res = Py_None; |
| Py_INCREF(res); |
| } |
| else if (fresult->ct_flags & CT_STRUCT) { |
| res = convert_struct_to_owning_object(resultdata, fresult); |
| } |
| else { |
| res = convert_to_object(resultdata, fresult); |
| } |
| /* fall-through */ |
| |
| error: |
| while (freeme != NULL) { |
| void *p = (void *)freeme; |
| freeme = freeme->next; |
| PyObject_Free(p); |
| } |
| if (buffer) |
| PyObject_Free(buffer); |
| if (fvarargs != NULL) { |
| Py_DECREF(fvarargs); |
| if (cif_descr != NULL) /* but only if fvarargs != NULL, if variadic */ |
| PyObject_Free(cif_descr); |
| } |
| return res; |
| } |
| |
| static PyObject *cdata_dir(PyObject *cd, PyObject *noarg) |
| { |
| CTypeDescrObject *ct = ((CDataObject *)cd)->c_type; |
| |
| /* replace the type 'pointer-to-t' with just 't' */ |
| if (ct->ct_flags & CT_POINTER) { |
| ct = ct->ct_itemdescr; |
| } |
| if ((ct->ct_flags & (CT_STRUCT | CT_UNION)) && |
| !(ct->ct_flags & CT_IS_OPAQUE)) { |
| |
| /* for non-opaque structs or unions */ |
| if (force_lazy_struct(ct) < 0) |
| return NULL; |
| return PyDict_Keys(ct->ct_stuff); |
| } |
| else { |
| return PyList_New(0); /* empty list for the other cases */ |
| } |
| } |
| |
| static PyObject *cdata_complex(PyObject *cd_, PyObject *noarg) |
| { |
| CDataObject *cd = (CDataObject *)cd_; |
| |
| if (cd->c_type->ct_flags & CT_PRIMITIVE_COMPLEX) { |
| Py_complex value = read_raw_complex_data(cd->c_data, cd->c_type->ct_size); |
| PyObject *op = PyComplex_FromCComplex(value); |
| return op; |
| } |
| /* <cdata 'float'> or <cdata 'int'> cannot be directly converted by |
| calling complex(), just like <cdata 'int'> cannot be directly |
| converted by calling float() */ |
| |
| PyErr_Format(PyExc_TypeError, "complex() not supported on cdata '%s'", |
| cd->c_type->ct_name); |
| return NULL; |
| } |
| |
| static int explicit_release_case(PyObject *cd) |
| { |
| CTypeDescrObject *ct = ((CDataObject *)cd)->c_type; |
| if (Py_TYPE(cd) == &CDataOwning_Type) { |
| if ((ct->ct_flags & (CT_POINTER | CT_ARRAY)) != 0) /* ffi.new() */ |
| return 0; |
| } |
| else if (Py_TYPE(cd) == &CDataFromBuf_Type) { |
| return 1; /* ffi.from_buffer() */ |
| } |
| else if (Py_TYPE(cd) == &CDataGCP_Type) { |
| return 2; /* ffi.gc() */ |
| } |
| PyErr_SetString(PyExc_ValueError, |
| "only 'cdata' object from ffi.new(), ffi.gc(), ffi.from_buffer() " |
| "or ffi.new_allocator()() can be used with the 'with' keyword or " |
| "ffi.release()"); |
| return -1; |
| } |
| |
| static PyObject *cdata_enter(PyObject *cd, PyObject *noarg) |
| { |
| if (explicit_release_case(cd) < 0) /* only to check the ctype */ |
| return NULL; |
| Py_INCREF(cd); |
| return cd; |
| } |
| |
| static PyObject *cdata_exit(PyObject *cd, PyObject *args) |
| { |
| /* 'args' ignored */ |
| CTypeDescrObject *ct; |
| Py_buffer *view; |
| switch (explicit_release_case(cd)) |
| { |
| case 0: /* ffi.new() */ |
| /* no effect on CPython: raw memory is allocated with the |
| same malloc() as the object itself, so it can't be |
| released independently. If we use a custom allocator, |
| then it's implemented with ffi.gc(). */ |
| ct = ((CDataObject *)cd)->c_type; |
| if (ct->ct_flags & CT_IS_PTR_TO_OWNED) { |
| PyObject *x = ((CDataObject_own_structptr *)cd)->structobj; |
| if (Py_TYPE(x) == &CDataGCP_Type) { |
| /* this is a special case for |
| ffi.new_allocator()("struct-or-union *") */ |
| cdatagcp_finalize((CDataObject_gcp *)x); |
| } |
| } |
| break; |
| |
| case 1: /* ffi.from_buffer() */ |
| view = ((CDataObject_frombuf *)cd)->bufferview; |
| PyBuffer_Release(view); |
| break; |
| |
| case 2: /* ffi.gc() or ffi.new_allocator()("not-struct-nor-union") */ |
| /* call the destructor immediately */ |
| cdatagcp_finalize((CDataObject_gcp *)cd); |
| break; |
| |
| default: |
| return NULL; |
| } |
| Py_INCREF(Py_None); |
| return Py_None; |
| } |
| |
| static PyObject *cdata_iter(CDataObject *); |
| |
| static PyNumberMethods CData_as_number = { |
| (binaryfunc)cdata_add, /*nb_add*/ |
| (binaryfunc)cdata_sub, /*nb_subtract*/ |
| 0, /*nb_multiply*/ |
| #if PY_MAJOR_VERSION < 3 |
| 0, /*nb_divide*/ |
| #endif |
| 0, /*nb_remainder*/ |
| 0, /*nb_divmod*/ |
| 0, /*nb_power*/ |
| 0, /*nb_negative*/ |
| 0, /*nb_positive*/ |
| 0, /*nb_absolute*/ |
| (inquiry)cdata_nonzero, /*nb_nonzero*/ |
| 0, /*nb_invert*/ |
| 0, /*nb_lshift*/ |
| 0, /*nb_rshift*/ |
| 0, /*nb_and*/ |
| 0, /*nb_xor*/ |
| 0, /*nb_or*/ |
| #if PY_MAJOR_VERSION < 3 |
| 0, /*nb_coerce*/ |
| #endif |
| (unaryfunc)cdata_int, /*nb_int*/ |
| #if PY_MAJOR_VERSION < 3 |
| (unaryfunc)cdata_long, /*nb_long*/ |
| #else |
| 0, |
| #endif |
| (unaryfunc)cdata_float, /*nb_float*/ |
| 0, /*nb_oct*/ |
| 0, /*nb_hex*/ |
| }; |
| |
| static PyMappingMethods CData_as_mapping = { |
| (lenfunc)cdata_length, /*mp_length*/ |
| (binaryfunc)cdata_subscript, /*mp_subscript*/ |
| (objobjargproc)cdata_ass_sub, /*mp_ass_subscript*/ |
| }; |
| |
| static PyMappingMethods CDataOwn_as_mapping = { |
| (lenfunc)cdata_length, /*mp_length*/ |
| (binaryfunc)cdataowning_subscript, /*mp_subscript*/ |
| (objobjargproc)cdata_ass_sub, /*mp_ass_subscript*/ |
| }; |
| |
| static PyMethodDef cdata_methods[] = { |
| {"__dir__", cdata_dir, METH_NOARGS}, |
| {"__complex__", cdata_complex, METH_NOARGS}, |
| {"__enter__", cdata_enter, METH_NOARGS}, |
| {"__exit__", cdata_exit, METH_VARARGS}, |
| {NULL, NULL} /* sentinel */ |
| }; |
| |
| static PyTypeObject CData_Type = { |
| PyVarObject_HEAD_INIT(NULL, 0) |
| "_cffi_backend._CDataBase", |
| sizeof(CDataObject), |
| 0, |
| (destructor)cdata_dealloc, /* tp_dealloc */ |
| 0, /* tp_print */ |
| 0, /* tp_getattr */ |
| 0, /* tp_setattr */ |
| 0, /* tp_compare */ |
| (reprfunc)cdata_repr, /* tp_repr */ |
| &CData_as_number, /* tp_as_number */ |
| 0, /* tp_as_sequence */ |
| &CData_as_mapping, /* tp_as_mapping */ |
| cdata_hash, /* tp_hash */ |
| (ternaryfunc)cdata_call, /* tp_call */ |
| 0, /* tp_str */ |
| (getattrofunc)cdata_getattro, /* tp_getattro */ |
| (setattrofunc)cdata_setattro, /* tp_setattro */ |
| 0, /* tp_as_buffer */ |
| Py_TPFLAGS_DEFAULT | Py_TPFLAGS_CHECKTYPES, /* tp_flags */ |
| "The internal base type for CData objects. Use FFI.CData to access " |
| "it. Always check with isinstance(): subtypes are sometimes returned " |
| "on CPython, for performance reasons.", /* tp_doc */ |
| 0, /* tp_traverse */ |
| 0, /* tp_clear */ |
| cdata_richcompare, /* tp_richcompare */ |
| offsetof(CDataObject, c_weakreflist), /* tp_weaklistoffset */ |
| (getiterfunc)cdata_iter, /* tp_iter */ |
| 0, /* tp_iternext */ |
| cdata_methods, /* tp_methods */ |
| 0, /* tp_members */ |
| 0, /* tp_getset */ |
| 0, /* tp_base */ |
| 0, /* tp_dict */ |
| 0, /* tp_descr_get */ |
| 0, /* tp_descr_set */ |
| 0, /* tp_dictoffset */ |
| 0, /* tp_init */ |
| 0, /* tp_alloc */ |
| 0, /* tp_new */ |
| PyObject_Del, /* tp_free */ |
| }; |
| |
| static PyTypeObject CDataOwning_Type = { |
| PyVarObject_HEAD_INIT(NULL, 0) |
| "_cffi_backend.__CDataOwn", |
| sizeof(CDataObject), |
| 0, |
| (destructor)cdataowning_dealloc, /* tp_dealloc */ |
| 0, /* tp_print */ |
| 0, /* tp_getattr */ |
| 0, /* tp_setattr */ |
| 0, /* tp_compare */ |
| (reprfunc)cdataowning_repr, /* tp_repr */ |
| 0, /* inherited */ /* tp_as_number */ |
| 0, /* tp_as_sequence */ |
| &CDataOwn_as_mapping, /* tp_as_mapping */ |
| 0, /* inherited */ /* tp_hash */ |
| 0, /* inherited */ /* tp_call */ |
| 0, /* tp_str */ |
| 0, /* inherited */ /* tp_getattro */ |
| 0, /* inherited */ /* tp_setattro */ |
| 0, /* tp_as_buffer */ |
| Py_TPFLAGS_DEFAULT | Py_TPFLAGS_CHECKTYPES, /* tp_flags */ |
| "This is an internal subtype of _CDataBase for performance only on " |
| "CPython. Check with isinstance(x, ffi.CData).", /* tp_doc */ |
| 0, /* tp_traverse */ |
| 0, /* tp_clear */ |
| 0, /* inherited */ /* tp_richcompare */ |
| 0, /* inherited */ /* tp_weaklistoffset */ |
| 0, /* inherited */ /* tp_iter */ |
| 0, /* tp_iternext */ |
| 0, /* inherited */ /* tp_methods */ |
| 0, /* tp_members */ |
| 0, /* tp_getset */ |
| &CData_Type, /* tp_base */ |
| 0, /* tp_dict */ |
| 0, /* tp_descr_get */ |
| 0, /* tp_descr_set */ |
| 0, /* tp_dictoffset */ |
| 0, /* tp_init */ |
| 0, /* tp_alloc */ |
| 0, /* tp_new */ |
| free, /* tp_free */ |
| }; |
| |
| static PyTypeObject CDataOwningGC_Type = { |
| PyVarObject_HEAD_INIT(NULL, 0) |
| "_cffi_backend.__CDataOwnGC", |
| sizeof(CDataObject_own_structptr), |
| 0, |
| (destructor)cdataowninggc_dealloc, /* tp_dealloc */ |
| 0, /* tp_print */ |
| 0, /* tp_getattr */ |
| 0, /* tp_setattr */ |
| 0, /* tp_compare */ |
| (reprfunc)cdataowninggc_repr, /* tp_repr */ |
| 0, /* inherited */ /* tp_as_number */ |
| 0, /* tp_as_sequence */ |
| 0, /* inherited */ /* tp_as_mapping */ |
| 0, /* inherited */ /* tp_hash */ |
| 0, /* inherited */ /* tp_call */ |
| 0, /* tp_str */ |
| 0, /* inherited */ /* tp_getattro */ |
| 0, /* inherited */ /* tp_setattro */ |
| 0, /* tp_as_buffer */ |
| Py_TPFLAGS_DEFAULT | Py_TPFLAGS_CHECKTYPES /* tp_flags */ |
| | Py_TPFLAGS_HAVE_GC, |
| "This is an internal subtype of _CDataBase for performance only on " |
| "CPython. Check with isinstance(x, ffi.CData).", /* tp_doc */ |
| (traverseproc)cdataowninggc_traverse, /* tp_traverse */ |
| (inquiry)cdataowninggc_clear, /* tp_clear */ |
| 0, /* inherited */ /* tp_richcompare */ |
| 0, /* inherited */ /* tp_weaklistoffset */ |
| 0, /* inherited */ /* tp_iter */ |
| 0, /* tp_iternext */ |
| 0, /* inherited */ /* tp_methods */ |
| 0, /* tp_members */ |
| 0, /* tp_getset */ |
| &CDataOwning_Type, /* tp_base */ |
| 0, /* tp_dict */ |
| 0, /* tp_descr_get */ |
| 0, /* tp_descr_set */ |
| 0, /* tp_dictoffset */ |
| 0, /* tp_init */ |
| 0, /* tp_alloc */ |
| 0, /* tp_new */ |
| PyObject_GC_Del, /* tp_free */ |
| }; |
| |
| static PyTypeObject CDataFromBuf_Type = { |
| PyVarObject_HEAD_INIT(NULL, 0) |
| "_cffi_backend.__CDataFromBuf", |
| sizeof(CDataObject_frombuf), |
| 0, |
| (destructor)cdatafrombuf_dealloc, /* tp_dealloc */ |
| 0, /* tp_print */ |
| 0, /* tp_getattr */ |
| 0, /* tp_setattr */ |
| 0, /* tp_compare */ |
| (reprfunc)cdatafrombuf_repr, /* tp_repr */ |
| 0, /* inherited */ /* tp_as_number */ |
| 0, /* tp_as_sequence */ |
| 0, /* inherited */ /* tp_as_mapping */ |
| 0, /* inherited */ /* tp_hash */ |
| 0, /* inherited */ /* tp_call */ |
| 0, /* tp_str */ |
| 0, /* inherited */ /* tp_getattro */ |
| 0, /* inherited */ /* tp_setattro */ |
| 0, /* tp_as_buffer */ |
| Py_TPFLAGS_DEFAULT | Py_TPFLAGS_CHECKTYPES /* tp_flags */ |
| | Py_TPFLAGS_HAVE_GC, |
| "This is an internal subtype of _CDataBase for performance only on " |
| "CPython. Check with isinstance(x, ffi.CData).", /* tp_doc */ |
| (traverseproc)cdatafrombuf_traverse, /* tp_traverse */ |
| (inquiry)cdatafrombuf_clear, /* tp_clear */ |
| 0, /* inherited */ /* tp_richcompare */ |
| 0, /* inherited */ /* tp_weaklistoffset */ |
| 0, /* inherited */ /* tp_iter */ |
| 0, /* tp_iternext */ |
| 0, /* inherited */ /* tp_methods */ |
| 0, /* tp_members */ |
| 0, /* tp_getset */ |
| &CData_Type, /* tp_base */ |
| 0, /* tp_dict */ |
| 0, /* tp_descr_get */ |
| 0, /* tp_descr_set */ |
| 0, /* tp_dictoffset */ |
| 0, /* tp_init */ |
| 0, /* tp_alloc */ |
| 0, /* tp_new */ |
| PyObject_GC_Del, /* tp_free */ |
| }; |
| |
| static PyTypeObject CDataGCP_Type = { |
| PyVarObject_HEAD_INIT(NULL, 0) |
| "_cffi_backend.__CDataGCP", |
| sizeof(CDataObject_gcp), |
| 0, |
| (destructor)cdatagcp_dealloc, /* tp_dealloc */ |
| 0, /* tp_print */ |
| 0, /* tp_getattr */ |
| 0, /* tp_setattr */ |
| 0, /* tp_compare */ |
| 0, /* inherited */ /* tp_repr */ |
| 0, /* inherited */ /* tp_as_number */ |
| 0, /* tp_as_sequence */ |
| 0, /* inherited */ /* tp_as_mapping */ |
| 0, /* inherited */ /* tp_hash */ |
| 0, /* inherited */ /* tp_call */ |
| 0, /* tp_str */ |
| 0, /* inherited */ /* tp_getattro */ |
| 0, /* inherited */ /* tp_setattro */ |
| 0, /* tp_as_buffer */ |
| Py_TPFLAGS_DEFAULT | Py_TPFLAGS_CHECKTYPES /* tp_flags */ |
| #ifdef Py_TPFLAGS_HAVE_FINALIZE |
| | Py_TPFLAGS_HAVE_FINALIZE |
| #endif |
| | Py_TPFLAGS_HAVE_GC, |
| "This is an internal subtype of _CDataBase for performance only on " |
| "CPython. Check with isinstance(x, ffi.CData).", /* tp_doc */ |
| (traverseproc)cdatagcp_traverse, /* tp_traverse */ |
| 0, /* tp_clear */ |
| 0, /* inherited */ /* tp_richcompare */ |
| 0, /* inherited */ /* tp_weaklistoffset */ |
| 0, /* inherited */ /* tp_iter */ |
| 0, /* tp_iternext */ |
| 0, /* inherited */ /* tp_methods */ |
| 0, /* tp_members */ |
| 0, /* tp_getset */ |
| &CData_Type, /* tp_base */ |
| #ifdef Py_TPFLAGS_HAVE_FINALIZE /* CPython >= 3.4 */ |
| 0, /* tp_dict */ |
| 0, /* tp_descr_get */ |
| 0, /* tp_descr_set */ |
| 0, /* tp_dictoffset */ |
| 0, /* tp_init */ |
| 0, /* tp_alloc */ |
| 0, /* tp_new */ |
| 0, /* inherited */ /* tp_free */ |
| 0, /* tp_is_gc */ |
| 0, /* tp_bases */ |
| 0, /* tp_mro */ |
| 0, /* tp_cache */ |
| 0, /* tp_subclasses */ |
| 0, /* tp_weaklist */ |
| 0, /* tp_del */ |
| 0, /* version_tag */ |
| (destructor)cdatagcp_finalize, /* tp_finalize */ |
| #endif |
| }; |
| |
| /************************************************************/ |
| |
| typedef struct { |
| PyObject_HEAD |
| char *di_next, *di_stop; |
| CDataObject *di_object; |
| CTypeDescrObject *di_itemtype; |
| } CDataIterObject; |
| |
| static PyObject * |
| cdataiter_next(CDataIterObject *it) |
| { |
| char *result = it->di_next; |
| if (result != it->di_stop) { |
| it->di_next = result + it->di_itemtype->ct_size; |
| return convert_to_object(result, it->di_itemtype); |
| } |
| return NULL; |
| } |
| |
| static void |
| cdataiter_dealloc(CDataIterObject *it) |
| { |
| Py_DECREF(it->di_object); |
| PyObject_Del(it); |
| } |
| |
| static PyTypeObject CDataIter_Type = { |
| PyVarObject_HEAD_INIT(NULL, 0) |
| "_cffi_backend.__CData_iterator", /* tp_name */ |
| sizeof(CDataIterObject), /* tp_basicsize */ |
| 0, /* tp_itemsize */ |
| /* methods */ |
| (destructor)cdataiter_dealloc, /* tp_dealloc */ |
| 0, /* tp_print */ |
| 0, /* tp_getattr */ |
| 0, /* tp_setattr */ |
| 0, /* tp_compare */ |
| 0, /* tp_repr */ |
| 0, /* tp_as_number */ |
| 0, /* tp_as_sequence */ |
| 0, /* tp_as_mapping */ |
| 0, /* tp_hash */ |
| 0, /* tp_call */ |
| 0, /* tp_str */ |
| PyObject_GenericGetAttr, /* tp_getattro */ |
| 0, /* tp_setattro */ |
| 0, /* tp_as_buffer */ |
| Py_TPFLAGS_DEFAULT, /* tp_flags */ |
| 0, /* tp_doc */ |
| 0, /* tp_traverse */ |
| 0, /* tp_clear */ |
| 0, /* tp_richcompare */ |
| 0, /* tp_weaklistoffset */ |
| PyObject_SelfIter, /* tp_iter */ |
| (iternextfunc)cdataiter_next, /* tp_iternext */ |
| }; |
| |
| static PyObject * |
| cdata_iter(CDataObject *cd) |
| { |
| CDataIterObject *it; |
| |
| if (!(cd->c_type->ct_flags & CT_ARRAY)) { |
| PyErr_Format(PyExc_TypeError, "cdata '%s' does not support iteration", |
| cd->c_type->ct_name); |
| return NULL; |
| } |
| |
| it = PyObject_New(CDataIterObject, &CDataIter_Type); |
| if (it == NULL) |
| return NULL; |
| |
| Py_INCREF(cd); |
| it->di_object = cd; |
| it->di_itemtype = cd->c_type->ct_itemdescr; |
| it->di_next = cd->c_data; |
| it->di_stop = cd->c_data + get_array_length(cd) * it->di_itemtype->ct_size; |
| return (PyObject *)it; |
| } |
| |
| /************************************************************/ |
| |
| static CDataObject *allocate_owning_object(Py_ssize_t size, |
| CTypeDescrObject *ct, |
| int dont_clear) |
| { |
| /* note: objects with &CDataOwning_Type are always allocated with |
| either a plain malloc() or calloc(), and freed with free(). */ |
| CDataObject *cd; |
| if (dont_clear) |
| cd = malloc(size); |
| else |
| cd = calloc(size, 1); |
| if (PyObject_Init((PyObject *)cd, &CDataOwning_Type) == NULL) |
| return NULL; |
| |
| Py_INCREF(ct); |
| cd->c_type = ct; |
| cd->c_weakreflist = NULL; |
| return cd; |
| } |
| |
| static PyObject * |
| convert_struct_to_owning_object(char *data, CTypeDescrObject *ct) |
| { |
| /* also accepts unions, for the API mode */ |
| CDataObject *cd; |
| Py_ssize_t dataoffset = offsetof(CDataObject_own_nolength, alignment); |
| Py_ssize_t datasize = ct->ct_size; |
| |
| if (datasize < 0) { |
| PyErr_SetString(PyExc_TypeError, |
| "return type is an opaque structure or union"); |
| return NULL; |
| } |
| if (ct->ct_flags & CT_WITH_VAR_ARRAY) { |
| PyErr_SetString(PyExc_TypeError, |
| "return type is a struct/union with a varsize array member"); |
| return NULL; |
| } |
| cd = allocate_owning_object(dataoffset + datasize, ct, /*dont_clear=*/1); |
| if (cd == NULL) |
| return NULL; |
| cd->c_data = ((char *)cd) + dataoffset; |
| |
| memcpy(cd->c_data, data, datasize); |
| return (PyObject *)cd; |
| } |
| |
| static CDataObject *allocate_gcp_object(CDataObject *origobj, |
| CTypeDescrObject *ct, |
| PyObject *destructor) |
| { |
| CDataObject_gcp *cd = PyObject_GC_New(CDataObject_gcp, &CDataGCP_Type); |
| if (cd == NULL) |
| return NULL; |
| |
| Py_XINCREF(destructor); |
| Py_INCREF(origobj); |
| Py_INCREF(ct); |
| cd->head.c_data = origobj->c_data; |
| cd->head.c_type = ct; |
| cd->head.c_weakreflist = NULL; |
| cd->origobj = (PyObject *)origobj; |
| cd->destructor = destructor; |
| |
| PyObject_GC_Track(cd); |
| return (CDataObject *)cd; |
| } |
| |
| static CDataObject *allocate_with_allocator(Py_ssize_t basesize, |
| Py_ssize_t datasize, |
| CTypeDescrObject *ct, |
| const cffi_allocator_t *allocator) |
| { |
| CDataObject *cd; |
| |
| if (allocator->ca_alloc == NULL) { |
| cd = allocate_owning_object(basesize + datasize, ct, |
| allocator->ca_dont_clear); |
| if (cd == NULL) |
| return NULL; |
| cd->c_data = ((char *)cd) + basesize; |
| } |
| else { |
| PyObject *res = PyObject_CallFunction(allocator->ca_alloc, "n", datasize); |
| if (res == NULL) |
| return NULL; |
| |
| if (!CData_Check(res)) { |
| PyErr_Format(PyExc_TypeError, |
| "alloc() must return a cdata object (got %.200s)", |
| Py_TYPE(res)->tp_name); |
| Py_DECREF(res); |
| return NULL; |
| } |
| cd = (CDataObject *)res; |
| if (!(cd->c_type->ct_flags & (CT_POINTER|CT_ARRAY))) { |
| PyErr_Format(PyExc_TypeError, |
| "alloc() must return a cdata pointer, not '%s'", |
| cd->c_type->ct_name); |
| Py_DECREF(res); |
| return NULL; |
| } |
| if (!cd->c_data) { |
| PyErr_SetString(PyExc_MemoryError, "alloc() returned NULL"); |
| Py_DECREF(res); |
| return NULL; |
| } |
| |
| cd = allocate_gcp_object(cd, ct, allocator->ca_free); |
| Py_DECREF(res); |
| if (!allocator->ca_dont_clear) |
| memset(cd->c_data, 0, datasize); |
| } |
| return cd; |
| } |
| |
| static PyObject *direct_newp(CTypeDescrObject *ct, PyObject *init, |
| const cffi_allocator_t *allocator) |
| { |
| CTypeDescrObject *ctitem; |
| CDataObject *cd; |
| Py_ssize_t dataoffset, datasize, explicitlength; |
| |
| explicitlength = -1; |
| if (ct->ct_flags & CT_POINTER) { |
| dataoffset = offsetof(CDataObject_own_nolength, alignment); |
| ctitem = ct->ct_itemdescr; |
| datasize = ctitem->ct_size; |
| if (datasize < 0) { |
| PyErr_Format(PyExc_TypeError, |
| "cannot instantiate ctype '%s' of unknown size", |
| ctitem->ct_name); |
| return NULL; |
| } |
| if (ctitem->ct_flags & CT_PRIMITIVE_CHAR) |
| datasize *= 2; /* forcefully add another character: a null */ |
| |
| if (ctitem->ct_flags & (CT_STRUCT | CT_UNION)) { |
| if (force_lazy_struct(ctitem) < 0) /* for CT_WITH_VAR_ARRAY */ |
| return NULL; |
| |
| if (ctitem->ct_flags & CT_WITH_VAR_ARRAY) { |
| assert(ct->ct_flags & CT_IS_PTR_TO_OWNED); |
| dataoffset = offsetof(CDataObject_own_length, alignment); |
| |
| if (init != Py_None) { |
| Py_ssize_t optvarsize = datasize; |
| if (convert_struct_from_object(NULL, ctitem, init, |
| &optvarsize) < 0) |
| return NULL; |
| datasize = optvarsize; |
| } |
| } |
| } |
| } |
| else if (ct->ct_flags & CT_ARRAY) { |
| dataoffset = offsetof(CDataObject_own_nolength, alignment); |
| datasize = ct->ct_size; |
| if (datasize < 0) { |
| explicitlength = get_new_array_length(ct->ct_itemdescr, &init); |
| if (explicitlength < 0) |
| return NULL; |
| ctitem = ct->ct_itemdescr; |
| dataoffset = offsetof(CDataObject_own_length, alignment); |
| datasize = MUL_WRAPAROUND(explicitlength, ctitem->ct_size); |
| if (explicitlength > 0 && |
| (datasize / explicitlength) != ctitem->ct_size) { |
| PyErr_SetString(PyExc_OverflowError, |
| "array size would overflow a Py_ssize_t"); |
| return NULL; |
| } |
| } |
| } |
| else { |
| PyErr_Format(PyExc_TypeError, |
| "expected a pointer or array ctype, got '%s'", |
| ct->ct_name); |
| return NULL; |
| } |
| |
| if (ct->ct_flags & CT_IS_PTR_TO_OWNED) { |
| /* common case of ptr-to-struct (or ptr-to-union): for this case |
| we build two objects instead of one, with the memory-owning |
| one being really the struct (or union) and the returned one |
| having a strong reference to it */ |
| CDataObject *cds; |
| |
| cds = allocate_with_allocator(dataoffset, datasize, ct->ct_itemdescr, |
| allocator); |
| if (cds == NULL) |
| return NULL; |
| |
| cd = allocate_owning_object(sizeof(CDataObject_own_structptr), ct, |
| /*dont_clear=*/1); |
| if (cd == NULL) { |
| Py_DECREF(cds); |
| return NULL; |
| } |
| /* store the only reference to cds into cd */ |
| ((CDataObject_own_structptr *)cd)->structobj = (PyObject *)cds; |
| /* store information about the allocated size of the struct */ |
| if (dataoffset == offsetof(CDataObject_own_length, alignment)) { |
| ((CDataObject_own_length *)cds)->length = datasize; |
| } |
| assert(explicitlength < 0); |
| |
| cd->c_data = cds->c_data; |
| } |
| else { |
| cd = allocate_with_allocator(dataoffset, datasize, ct, allocator); |
| if (cd == NULL) |
| return NULL; |
| |
| if (explicitlength >= 0) |
| ((CDataObject_own_length*)cd)->length = explicitlength; |
| } |
| |
| if (init != Py_None) { |
| if (convert_from_object(cd->c_data, |
| (ct->ct_flags & CT_POINTER) ? ct->ct_itemdescr : ct, init) < 0) { |
| Py_DECREF(cd); |
| return NULL; |
| } |
| } |
| return (PyObject *)cd; |
| } |
| |
| static PyObject *b_newp(PyObject *self, PyObject *args) |
| { |
| CTypeDescrObject *ct; |
| PyObject *init = Py_None; |
| if (!PyArg_ParseTuple(args, "O!|O:newp", &CTypeDescr_Type, &ct, &init)) |
| return NULL; |
| return direct_newp(ct, init, &default_allocator); |
| } |
| |
| static int |
| _my_PyObject_AsBool(PyObject *ob) |
| { |
| /* convert and cast a Python object to a boolean. Accept an integer |
| or a float object, up to a CData 'long double'. */ |
| PyObject *io; |
| PyNumberMethods *nb; |
| int res; |
| |
| #if PY_MAJOR_VERSION < 3 |
| if (PyInt_Check(ob)) { |
| return PyInt_AS_LONG(ob) != 0; |
| } |
| else |
| #endif |
| if (PyLong_Check(ob)) { |
| return _PyLong_Sign(ob) != 0; |
| } |
| else if (PyFloat_Check(ob)) { |
| return PyFloat_AS_DOUBLE(ob) != 0.0; |
| } |
| else if (CData_Check(ob)) { |
| CDataObject *cd = (CDataObject *)ob; |
| if (cd->c_type->ct_flags & CT_PRIMITIVE_FLOAT) { |
| /*READ(cd->c_data, cd->c_type->ct_size)*/ |
| if (cd->c_type->ct_flags & CT_IS_LONGDOUBLE) { |
| /* 'long double' objects: return the answer directly */ |
| return read_raw_longdouble_data(cd->c_data) != 0.0; |
| } |
| else { |
| /* 'float'/'double' objects: return the answer directly */ |
| return read_raw_float_data(cd->c_data, |
| cd->c_type->ct_size) != 0.0; |
| } |
| } |
| } |
| nb = ob->ob_type->tp_as_number; |
| if (nb == NULL || (nb->nb_float == NULL && nb->nb_int == NULL)) { |
| PyErr_SetString(PyExc_TypeError, "integer/float expected"); |
| return -1; |
| } |
| if (nb->nb_float && !CData_Check(ob)) |
| io = (*nb->nb_float) (ob); |
| else |
| io = (*nb->nb_int) (ob); |
| if (io == NULL) |
| return -1; |
| |
| if (PyIntOrLong_Check(io) || PyFloat_Check(io)) { |
| res = _my_PyObject_AsBool(io); |
| } |
| else { |
| PyErr_SetString(PyExc_TypeError, "integer/float conversion failed"); |
| res = -1; |
| } |
| Py_DECREF(io); |
| return res; |
| } |
| |
| static CDataObject *_new_casted_primitive(CTypeDescrObject *ct) |
| { |
| int dataoffset = offsetof(CDataObject_casted_primitive, alignment); |
| CDataObject *cd = (CDataObject *)PyObject_Malloc(dataoffset + ct->ct_size); |
| if (PyObject_Init((PyObject *)cd, &CData_Type) == NULL) |
| return NULL; |
| Py_INCREF(ct); |
| cd->c_type = ct; |
| cd->c_data = ((char*)cd) + dataoffset; |
| cd->c_weakreflist = NULL; |
| return cd; |
| } |
| |
| static CDataObject *cast_to_integer_or_char(CTypeDescrObject *ct, PyObject *ob) |
| { |
| unsigned PY_LONG_LONG value; |
| CDataObject *cd; |
| |
| if (CData_Check(ob) && |
| ((CDataObject *)ob)->c_type->ct_flags & |
| (CT_POINTER|CT_FUNCTIONPTR|CT_ARRAY)) { |
| value = (Py_intptr_t)((CDataObject *)ob)->c_data; |
| } |
| #if PY_MAJOR_VERSION < 3 |
| else if (PyString_Check(ob)) { |
| if (PyString_GET_SIZE(ob) != 1) { |
| PyErr_Format(PyExc_TypeError, |
| "cannot cast string of length %zd to ctype '%s'", |
| PyString_GET_SIZE(ob), ct->ct_name); |
| return NULL; |
| } |
| value = (unsigned char)PyString_AS_STRING(ob)[0]; |
| } |
| #endif |
| else if (PyUnicode_Check(ob)) { |
| char err_buf[80]; |
| cffi_char32_t ordinal; |
| if (_my_PyUnicode_AsSingleChar32(ob, &ordinal, err_buf) < 0) { |
| PyErr_Format(PyExc_TypeError, |
| "cannot cast %s to ctype '%s'", err_buf, ct->ct_name); |
| return NULL; |
| } |
| /* the types char16_t and char32_t are both unsigned. However, |
| wchar_t might be signed. In theory it does not matter, |
| because 'ordinal' comes from a regular Python unicode. */ |
| #ifdef HAVE_WCHAR_H |
| if (ct->ct_flags & CT_IS_SIGNED_WCHAR) |
| value = (wchar_t)ordinal; |
| else |
| #endif |
| value = ordinal; |
| } |
| else if (PyBytes_Check(ob)) { |
| int res = _convert_to_char(ob); |
| if (res < 0) |
| return NULL; |
| value = (unsigned char)res; |
| } |
| else if (ct->ct_flags & CT_IS_BOOL) { |
| int res = _my_PyObject_AsBool(ob); |
| if (res < 0) |
| return NULL; |
| value = res; |
| } |
| else { |
| value = _my_PyLong_AsUnsignedLongLong(ob, 0); |
| if (value == (unsigned PY_LONG_LONG)-1 && PyErr_Occurred()) |
| return NULL; |
| } |
| if (ct->ct_flags & CT_IS_BOOL) |
| value = !!value; |
| cd = _new_casted_primitive(ct); |
| if (cd != NULL) |
| write_raw_integer_data(cd->c_data, value, ct->ct_size); |
| return cd; |
| } |
| |
| /* returns -1 if cannot cast, 0 if we don't get a value, 1 if we do */ |
| static int check_bytes_for_float_compatible(PyObject *io, double *out_value) |
| { |
| if (PyBytes_Check(io)) { |
| if (PyBytes_GET_SIZE(io) != 1) |
| goto error; |
| *out_value = (unsigned char)PyBytes_AS_STRING(io)[0]; |
| return 1; |
| } |
| else if (PyUnicode_Check(io)) { |
| char ignored[80]; |
| cffi_char32_t ordinal; |
| if (_my_PyUnicode_AsSingleChar32(io, &ordinal, ignored) < 0) |
| goto error; |
| /* the signness of the 32-bit version of wide chars should not |
| * matter here, because 'ordinal' comes from a normal Python |
| * unicode string */ |
| *out_value = ordinal; |
| return 1; |
| } |
| *out_value = 0; /* silence a gcc warning if this function is inlined */ |
| return 0; |
| |
| error: |
| Py_DECREF(io); |
| *out_value = 0; /* silence a gcc warning if this function is inlined */ |
| return -1; |
| } |
| |
| static PyObject *do_cast(CTypeDescrObject *ct, PyObject *ob) |
| { |
| CDataObject *cd; |
| |
| if (ct->ct_flags & (CT_POINTER|CT_FUNCTIONPTR|CT_ARRAY) && |
| ct->ct_size >= 0) { |
| /* cast to a pointer, to a funcptr, or to an array. |
| Note that casting to an array is an extension to the C language, |
| which seems to be necessary in order to sanely get a |
| <cdata 'int[3]'> at some address. */ |
| unsigned PY_LONG_LONG value; |
| |
| if (CData_Check(ob)) { |
| CDataObject *cdsrc = (CDataObject *)ob; |
| if (cdsrc->c_type->ct_flags & |
| (CT_POINTER|CT_FUNCTIONPTR|CT_ARRAY)) { |
| return new_simple_cdata(cdsrc->c_data, ct); |
| } |
| } |
| if ((ct->ct_flags & CT_POINTER) && |
| (ct->ct_itemdescr->ct_flags & CT_IS_FILE) && |
| PyFile_Check(ob)) { |
| FILE *f = PyFile_AsFile(ob); |
| if (f == NULL && PyErr_Occurred()) |
| return NULL; |
| return new_simple_cdata((char *)f, ct); |
| } |
| value = _my_PyLong_AsUnsignedLongLong(ob, 0); |
| if (value == (unsigned PY_LONG_LONG)-1 && PyErr_Occurred()) |
| return NULL; |
| return new_simple_cdata((char *)(Py_intptr_t)value, ct); |
| } |
| else if (ct->ct_flags & (CT_PRIMITIVE_SIGNED|CT_PRIMITIVE_UNSIGNED |
| |CT_PRIMITIVE_CHAR)) { |
| /* cast to an integer type or a char */ |
| return (PyObject *)cast_to_integer_or_char(ct, ob); |
| } |
| else if (ct->ct_flags & CT_PRIMITIVE_FLOAT) { |
| /* cast to a float */ |
| double value; |
| PyObject *io; |
| int res; |
| |
| if (CData_Check(ob)) { |
| CDataObject *cdsrc = (CDataObject *)ob; |
| |
| if (!(cdsrc->c_type->ct_flags & CT_PRIMITIVE_ANY)) |
| goto cannot_cast; |
| io = convert_to_object(cdsrc->c_data, cdsrc->c_type); |
| if (io == NULL) |
| return NULL; |
| } |
| else { |
| io = ob; |
| Py_INCREF(io); |
| } |
| |
| res = check_bytes_for_float_compatible(io, &value); |
| if (res == -1) |
| goto cannot_cast; |
| if (res == 0) { |
| if ((ct->ct_flags & CT_IS_LONGDOUBLE) && |
| CData_Check(io) && |
| (((CDataObject *)io)->c_type->ct_flags & CT_IS_LONGDOUBLE)) { |
| long double lvalue; |
| char *data = ((CDataObject *)io)->c_data; |
| /*READ(data, sizeof(long double)*/ |
| lvalue = read_raw_longdouble_data(data); |
| Py_DECREF(io); |
| cd = _new_casted_primitive(ct); |
| if (cd != NULL) |
| write_raw_longdouble_data(cd->c_data, lvalue); |
| return (PyObject *)cd; |
| } |
| value = PyFloat_AsDouble(io); |
| } |
| Py_DECREF(io); |
| if (value == -1.0 && PyErr_Occurred()) |
| return NULL; |
| |
| cd = _new_casted_primitive(ct); |
| if (cd != NULL) { |
| if (!(ct->ct_flags & CT_IS_LONGDOUBLE)) |
| write_raw_float_data(cd->c_data, value, ct->ct_size); |
| else |
| write_raw_longdouble_data(cd->c_data, (long double)value); |
| } |
| return (PyObject *)cd; |
| } |
| else if (ct->ct_flags & CT_PRIMITIVE_COMPLEX) { |
| /* cast to a complex */ |
| Py_complex value; |
| PyObject *io; |
| int res; |
| |
| if (CData_Check(ob)) { |
| CDataObject *cdsrc = (CDataObject *)ob; |
| |
| if (!(cdsrc->c_type->ct_flags & CT_PRIMITIVE_ANY)) |
| goto cannot_cast; |
| io = convert_to_object(cdsrc->c_data, cdsrc->c_type); |
| if (io == NULL) |
| return NULL; |
| } |
| else { |
| io = ob; |
| Py_INCREF(io); |
| } |
| |
| res = check_bytes_for_float_compatible(io, &value.real); |
| if (res == -1) |
| goto cannot_cast; |
| if (res == 1) { |
| // got it from string |
| value.imag = 0.0; |
| } else { |
| value = PyComplex_AsCComplex(io); |
| } |
| Py_DECREF(io); |
| if (PyErr_Occurred()) { |
| return NULL; |
| } |
| cd = _new_casted_primitive(ct); |
| if (cd != NULL) { |
| write_raw_complex_data(cd->c_data, value, ct->ct_size); |
| } |
| return (PyObject *)cd; |
| } |
| else { |
| PyErr_Format(PyExc_TypeError, "cannot cast to ctype '%s'", |
| ct->ct_name); |
| return NULL; |
| } |
| |
| cannot_cast: |
| if (CData_Check(ob)) |
| PyErr_Format(PyExc_TypeError, "cannot cast ctype '%s' to ctype '%s'", |
| ((CDataObject *)ob)->c_type->ct_name, ct->ct_name); |
| else |
| PyErr_Format(PyExc_TypeError, |
| "cannot cast %.200s object to ctype '%s'", |
| Py_TYPE(ob)->tp_name, ct->ct_name); |
| return NULL; |
| } |
| |
| static PyObject *b_cast(PyObject *self, PyObject *args) |
| { |
| CTypeDescrObject *ct; |
| PyObject *ob; |
| if (!PyArg_ParseTuple(args, "O!O:cast", &CTypeDescr_Type, &ct, &ob)) |
| return NULL; |
| |
| return do_cast(ct, ob); |
| } |
| |
| /************************************************************/ |
| |
| typedef struct { |
| PyObject_HEAD |
| void *dl_handle; |
| char *dl_name; |
| int dl_auto_close; |
| } DynLibObject; |
| |
| static void dl_dealloc(DynLibObject *dlobj) |
| { |
| if (dlobj->dl_handle != NULL && dlobj->dl_auto_close) |
| dlclose(dlobj->dl_handle); |
| free(dlobj->dl_name); |
| PyObject_Del(dlobj); |
| } |
| |
| static PyObject *dl_repr(DynLibObject *dlobj) |
| { |
| return PyText_FromFormat("<clibrary '%s'>", dlobj->dl_name); |
| } |
| |
| static int dl_check_closed(DynLibObject *dlobj) |
| { |
| if (dlobj->dl_handle == NULL) |
| { |
| PyErr_Format(PyExc_ValueError, "library '%s' has already been closed", |
| dlobj->dl_name); |
| return -1; |
| } |
| return 0; |
| } |
| |
| static PyObject *dl_load_function(DynLibObject *dlobj, PyObject *args) |
| { |
| CTypeDescrObject *ct; |
| char *funcname; |
| void *funcptr; |
| |
| if (!PyArg_ParseTuple(args, "O!s:load_function", |
| &CTypeDescr_Type, &ct, &funcname)) |
| return NULL; |
| |
| if (dl_check_closed(dlobj) < 0) |
| return NULL; |
| |
| if (!(ct->ct_flags & (CT_FUNCTIONPTR | CT_POINTER | CT_ARRAY))) { |
| PyErr_Format(PyExc_TypeError, |
| "function or pointer or array cdata expected, got '%s'", |
| ct->ct_name); |
| return NULL; |
| } |
| dlerror(); /* clear error condition */ |
| funcptr = dlsym(dlobj->dl_handle, funcname); |
| if (funcptr == NULL) { |
| const char *error = dlerror(); |
| PyErr_Format(PyExc_AttributeError, |
| "function/symbol '%s' not found in library '%s': %s", |
| funcname, dlobj->dl_name, error); |
| return NULL; |
| } |
| |
| if ((ct->ct_flags & CT_ARRAY) && ct->ct_length < 0) { |
| ct = (CTypeDescrObject *)ct->ct_stuff; |
| } |
| return new_simple_cdata(funcptr, ct); |
| } |
| |
| static PyObject *dl_read_variable(DynLibObject *dlobj, PyObject *args) |
| { |
| CTypeDescrObject *ct; |
| char *varname; |
| char *data; |
| |
| if (!PyArg_ParseTuple(args, "O!s:read_variable", |
| &CTypeDescr_Type, &ct, &varname)) |
| return NULL; |
| |
| if (dl_check_closed(dlobj) < 0) |
| return NULL; |
| |
| dlerror(); /* clear error condition */ |
| data = dlsym(dlobj->dl_handle, varname); |
| if (data == NULL) { |
| const char *error = dlerror(); |
| if (error != NULL) { |
| PyErr_Format(PyExc_KeyError, |
| "variable '%s' not found in library '%s': %s", |
| varname, dlobj->dl_name, error); |
| return NULL; |
| } |
| } |
| return convert_to_object(data, ct); |
| } |
| |
| static PyObject *dl_write_variable(DynLibObject *dlobj, PyObject *args) |
| { |
| CTypeDescrObject *ct; |
| PyObject *value; |
| char *varname; |
| char *data; |
| |
| if (!PyArg_ParseTuple(args, "O!sO:write_variable", |
| &CTypeDescr_Type, &ct, &varname, &value)) |
| return NULL; |
| |
| if (dl_check_closed(dlobj) < 0) |
| return NULL; |
| |
| dlerror(); /* clear error condition */ |
| data = dlsym(dlobj->dl_handle, varname); |
| if (data == NULL) { |
| const char *error = dlerror(); |
| PyErr_Format(PyExc_KeyError, |
| "variable '%s' not found in library '%s': %s", |
| varname, dlobj->dl_name, error); |
| return NULL; |
| } |
| if (convert_from_object(data, ct, value) < 0) |
| return NULL; |
| Py_INCREF(Py_None); |
| return Py_None; |
| } |
| |
| static PyObject *dl_close_lib(DynLibObject *dlobj, PyObject *no_args) |
| { |
| if (dlobj->dl_handle != NULL) |
| { |
| dlclose(dlobj->dl_handle); |
| dlobj->dl_handle = NULL; |
| } |
| Py_INCREF(Py_None); |
| return Py_None; |
| } |
| |
| static PyMethodDef dl_methods[] = { |
| {"load_function", (PyCFunction)dl_load_function, METH_VARARGS}, |
| {"read_variable", (PyCFunction)dl_read_variable, METH_VARARGS}, |
| {"write_variable", (PyCFunction)dl_write_variable, METH_VARARGS}, |
| {"close_lib", (PyCFunction)dl_close_lib, METH_NOARGS}, |
| {NULL, NULL} /* sentinel */ |
| }; |
| |
| static PyTypeObject dl_type = { |
| PyVarObject_HEAD_INIT(NULL, 0) |
| "_cffi_backend.CLibrary", /* tp_name */ |
| sizeof(DynLibObject), /* tp_basicsize */ |
| 0, /* tp_itemsize */ |
| /* methods */ |
| (destructor)dl_dealloc, /* tp_dealloc */ |
| 0, /* tp_print */ |
| 0, /* tp_getattr */ |
| 0, /* tp_setattr */ |
| 0, /* tp_compare */ |
| (reprfunc)dl_repr, /* tp_repr */ |
| 0, /* tp_as_number */ |
| 0, /* tp_as_sequence */ |
| 0, /* tp_as_mapping */ |
| 0, /* tp_hash */ |
| 0, /* tp_call */ |
| 0, /* tp_str */ |
| PyObject_GenericGetAttr, /* tp_getattro */ |
| 0, /* tp_setattro */ |
| 0, /* tp_as_buffer */ |
| Py_TPFLAGS_DEFAULT, /* tp_flags */ |
| 0, /* tp_doc */ |
| 0, /* tp_traverse */ |
| 0, /* tp_clear */ |
| 0, /* tp_richcompare */ |
| 0, /* tp_weaklistoffset */ |
| 0, /* tp_iter */ |
| 0, /* tp_iternext */ |
| dl_methods, /* tp_methods */ |
| }; |
| |
| static void *b_do_dlopen(PyObject *args, const char **p_printable_filename, |
| PyObject **p_temp, int *auto_close) |
| { |
| /* Logic to call the correct version of dlopen(). Returns NULL in case of error. |
| Otherwise, '*p_printable_filename' will point to a printable char version of |
| the filename (maybe utf-8-encoded). '*p_temp' will be set either to NULL or |
| to a temporary object that must be freed after looking at printable_filename. |
| */ |
| void *handle; |
| char *filename_or_null; |
| int flags = 0; |
| *p_temp = NULL; |
| *auto_close = 1; |
| |
| if (PyTuple_GET_SIZE(args) == 0 || PyTuple_GET_ITEM(args, 0) == Py_None) { |
| PyObject *dummy; |
| if (!PyArg_ParseTuple(args, "|Oi:load_library", |
| &dummy, &flags)) |
| return NULL; |
| filename_or_null = NULL; |
| *p_printable_filename = "<None>"; |
| } |
| else if (CData_Check(PyTuple_GET_ITEM(args, 0))) |
| { |
| CDataObject *cd; |
| if (!PyArg_ParseTuple(args, "O|i:load_library", &cd, &flags)) |
| return NULL; |
| /* 'flags' is accepted but ignored in this case */ |
| if ((cd->c_type->ct_flags & CT_IS_VOID_PTR) == 0) { |
| PyErr_Format(PyExc_TypeError, |
| "dlopen() takes a file name or 'void *' handle, not '%s'", |
| cd->c_type->ct_name); |
| return NULL; |
| } |
| handle = cd->c_data; |
| if (handle == NULL) { |
| PyErr_Format(PyExc_RuntimeError, "cannot call dlopen(NULL)"); |
| return NULL; |
| } |
| *p_temp = PyText_FromFormat("%p", handle); |
| *p_printable_filename = PyText_AsUTF8(*p_temp); |
| *auto_close = 0; |
| return handle; |
| } |
| else |
| { |
| PyObject *s = PyTuple_GET_ITEM(args, 0); |
| #ifdef MS_WIN32 |
| PyObject *filename_unicode; |
| if (PyArg_ParseTuple(args, "U|i:load_library", &filename_unicode, &flags)) |
| { |
| Py_ssize_t sz1; |
| wchar_t *w1; |
| #if PY_MAJOR_VERSION < 3 |
| s = PyUnicode_AsUTF8String(s); |
| if (s == NULL) |
| return NULL; |
| *p_temp = s; |
| #endif |
| *p_printable_filename = PyText_AsUTF8(s); |
| if (*p_printable_filename == NULL) |
| return NULL; |
| |
| sz1 = PyUnicode_GetSize(filename_unicode) + 1; |
| sz1 *= 2; /* should not be needed, but you never know */ |
| w1 = alloca(sizeof(wchar_t) * sz1); |
| sz1 = PyUnicode_AsWideChar((PyUnicodeObject *)filename_unicode, |
| w1, sz1 - 1); |
| if (sz1 < 0) |
| return NULL; |
| w1[sz1] = 0; |
| handle = dlopenW(w1); |
| goto got_handle; |
| } |
| PyErr_Clear(); |
| #endif |
| if (!PyArg_ParseTuple(args, "et|i:load_library", |
| Py_FileSystemDefaultEncoding, &filename_or_null, &flags)) |
| return NULL; |
| #if PY_MAJOR_VERSION < 3 |
| if (PyUnicode_Check(s)) |
| { |
| s = PyUnicode_AsUTF8String(s); |
| if (s == NULL) { |
| PyMem_Free(filename_or_null); |
| return NULL; |
| } |
| *p_temp = s; |
| } |
| #endif |
| *p_printable_filename = PyText_AsUTF8(s); |
| if (*p_printable_filename == NULL) { |
| PyMem_Free(filename_or_null); |
| return NULL; |
| } |
| } |
| if ((flags & (RTLD_NOW | RTLD_LAZY)) == 0) |
| flags |= RTLD_NOW; |
| |
| #ifdef MS_WIN32 |
| if (filename_or_null == NULL) { |
| PyErr_SetString(PyExc_OSError, "dlopen(None) not supported on Windows"); |
| return NULL; |
| } |
| #endif |
| |
| handle = dlopen(filename_or_null, flags); |
| PyMem_Free(filename_or_null); |
| |
| #ifdef MS_WIN32 |
| got_handle: |
| #endif |
| if (handle == NULL) { |
| const char *error = dlerror(); |
| PyErr_Format(PyExc_OSError, "cannot load library '%s': %s", |
| *p_printable_filename, error); |
| return NULL; |
| } |
| return handle; |
| } |
| |
| static PyObject *b_load_library(PyObject *self, PyObject *args) |
| { |
| const char *printable_filename; |
| PyObject *temp; |
| void *handle; |
| DynLibObject *dlobj = NULL; |
| int auto_close; |
| |
| handle = b_do_dlopen(args, &printable_filename, &temp, &auto_close); |
| if (handle == NULL) |
| goto error; |
| |
| dlobj = PyObject_New(DynLibObject, &dl_type); |
| if (dlobj == NULL) { |
| dlclose(handle); |
| goto error; |
| } |
| dlobj->dl_handle = handle; |
| dlobj->dl_name = strdup(printable_filename); |
| dlobj->dl_auto_close = auto_close; |
| |
| error: |
| Py_XDECREF(temp); |
| return (PyObject *)dlobj; |
| } |
| |
| /************************************************************/ |
| |
| static PyObject *get_unique_type(CTypeDescrObject *x, |
| const void *unique_key[], long keylength) |
| { |
| /* Replace the CTypeDescrObject 'x' with a standardized one. |
| This either just returns x, or x is decrefed and a new reference |
| to the already-existing equivalent is returned. |
| |
| In this function, 'x' always contains a reference that must be |
| either decrefed or returned. |
| |
| Keys: |
| void ["void"] |
| primitive [&static_struct] |
| pointer [ctype] |
| array [ctype, length] |
| funcptr [ctresult, ellipsis+abi, num_args, ctargs...] |
| */ |
| PyObject *key, *y; |
| void *pkey; |
| |
| key = PyBytes_FromStringAndSize(NULL, keylength * sizeof(void *)); |
| if (key == NULL) |
| goto error; |
| |
| pkey = PyBytes_AS_STRING(key); |
| memcpy(pkey, unique_key, keylength * sizeof(void *)); |
| |
| y = PyDict_GetItem(unique_cache, key); |
| if (y != NULL) { |
| Py_DECREF(key); |
| Py_INCREF(y); |
| Py_DECREF(x); |
| return y; |
| } |
| if (PyDict_SetItem(unique_cache, key, (PyObject *)x) < 0) { |
| Py_DECREF(key); |
| goto error; |
| } |
| /* Haaaack for our reference count hack: gcmodule.c must not see this |
| dictionary. The problem is that any PyDict_SetItem() notices that |
| 'x' is tracked and re-tracks the unique_cache dictionary. So here |
| we re-untrack it again... */ |
| PyObject_GC_UnTrack(unique_cache); |
| |
| assert(x->ct_unique_key == NULL); |
| x->ct_unique_key = key; /* the key will be freed in ctypedescr_dealloc() */ |
| /* the 'value' in unique_cache doesn't count as 1, but don't use |
| Py_DECREF(x) here because it will confuse debug builds into thinking |
| there was an extra DECREF in total. */ |
| ((PyObject *)x)->ob_refcnt--; |
| return (PyObject *)x; |
| |
| error: |
| Py_DECREF(x); |
| return NULL; |
| } |
| |
| /* according to the C standard, these types should be equivalent to the |
| _Complex types for the purposes of storage (not arguments in calls!) */ |
| typedef float cffi_float_complex_t[2]; |
| typedef double cffi_double_complex_t[2]; |
| |
| static PyObject *new_primitive_type(const char *name) |
| { |
| #define ENUM_PRIMITIVE_TYPES \ |
| EPTYPE(c, char, CT_PRIMITIVE_CHAR) \ |
| EPTYPE(s, short, CT_PRIMITIVE_SIGNED ) \ |
| EPTYPE(i, int, CT_PRIMITIVE_SIGNED ) \ |
| EPTYPE(l, long, CT_PRIMITIVE_SIGNED ) \ |
| EPTYPE(ll, long long, CT_PRIMITIVE_SIGNED ) \ |
| EPTYPE(sc, signed char, CT_PRIMITIVE_SIGNED ) \ |
| EPTYPE(uc, unsigned char, CT_PRIMITIVE_UNSIGNED ) \ |
| EPTYPE(us, unsigned short, CT_PRIMITIVE_UNSIGNED ) \ |
| EPTYPE(ui, unsigned int, CT_PRIMITIVE_UNSIGNED ) \ |
| EPTYPE(ul, unsigned long, CT_PRIMITIVE_UNSIGNED ) \ |
| EPTYPE(ull, unsigned long long, CT_PRIMITIVE_UNSIGNED ) \ |
| EPTYPE(f, float, CT_PRIMITIVE_FLOAT ) \ |
| EPTYPE(d, double, CT_PRIMITIVE_FLOAT ) \ |
| EPTYPE(ld, long double, CT_PRIMITIVE_FLOAT | CT_IS_LONGDOUBLE ) \ |
| EPTYPE2(fc, "float _Complex", cffi_float_complex_t, CT_PRIMITIVE_COMPLEX ) \ |
| EPTYPE2(dc, "double _Complex", cffi_double_complex_t, CT_PRIMITIVE_COMPLEX ) \ |
| ENUM_PRIMITIVE_TYPES_WCHAR \ |
| EPTYPE2(c16, "char16_t", cffi_char16_t, CT_PRIMITIVE_CHAR ) \ |
| EPTYPE2(c32, "char32_t", cffi_char32_t, CT_PRIMITIVE_CHAR ) \ |
| EPTYPE(b, _Bool, CT_PRIMITIVE_UNSIGNED | CT_IS_BOOL ) \ |
| /* the following types are not primitive in the C sense */ \ |
| EPTYPE(i8, int8_t, CT_PRIMITIVE_SIGNED) \ |
| EPTYPE(u8, uint8_t, CT_PRIMITIVE_UNSIGNED) \ |
| EPTYPE(i16, int16_t, CT_PRIMITIVE_SIGNED) \ |
| EPTYPE(u16, uint16_t, CT_PRIMITIVE_UNSIGNED) \ |
| EPTYPE(i32, int32_t, CT_PRIMITIVE_SIGNED) \ |
| EPTYPE(u32, uint32_t, CT_PRIMITIVE_UNSIGNED) \ |
| EPTYPE(i64, int64_t, CT_PRIMITIVE_SIGNED) \ |
| EPTYPE(u64, uint64_t, CT_PRIMITIVE_UNSIGNED) \ |
| EPTYPE(il8, int_least8_t, CT_PRIMITIVE_SIGNED) \ |
| EPTYPE(ul8, uint_least8_t, CT_PRIMITIVE_UNSIGNED) \ |
| EPTYPE(il16, int_least16_t, CT_PRIMITIVE_SIGNED) \ |
| EPTYPE(ul16, uint_least16_t, CT_PRIMITIVE_UNSIGNED) \ |
| EPTYPE(il32, int_least32_t, CT_PRIMITIVE_SIGNED) \ |
| EPTYPE(ul32, uint_least32_t, CT_PRIMITIVE_UNSIGNED) \ |
| EPTYPE(il64, int_least64_t, CT_PRIMITIVE_SIGNED) \ |
| EPTYPE(ul64, uint_least64_t, CT_PRIMITIVE_UNSIGNED) \ |
| EPTYPE(if8, int_fast8_t, CT_PRIMITIVE_SIGNED) \ |
| EPTYPE(uf8, uint_fast8_t, CT_PRIMITIVE_UNSIGNED) \ |
| EPTYPE(if16, int_fast16_t, CT_PRIMITIVE_SIGNED) \ |
| EPTYPE(uf16, uint_fast16_t, CT_PRIMITIVE_UNSIGNED) \ |
| EPTYPE(if32, int_fast32_t, CT_PRIMITIVE_SIGNED) \ |
| EPTYPE(uf32, uint_fast32_t, CT_PRIMITIVE_UNSIGNED) \ |
| EPTYPE(if64, int_fast64_t, CT_PRIMITIVE_SIGNED) \ |
| EPTYPE(uf64, uint_fast64_t, CT_PRIMITIVE_UNSIGNED) \ |
| EPTYPE(ip, intptr_t, CT_PRIMITIVE_SIGNED) \ |
| EPTYPE(up, uintptr_t, CT_PRIMITIVE_UNSIGNED) \ |
| EPTYPE(im, intmax_t, CT_PRIMITIVE_SIGNED) \ |
| EPTYPE(um, uintmax_t, CT_PRIMITIVE_UNSIGNED) \ |
| EPTYPE(pd, ptrdiff_t, CT_PRIMITIVE_SIGNED) \ |
| EPTYPE(sz, size_t, CT_PRIMITIVE_UNSIGNED) \ |
| EPTYPE2(ssz, "ssize_t", Py_ssize_t, CT_PRIMITIVE_SIGNED) |
| |
| #ifdef HAVE_WCHAR_H |
| # define ENUM_PRIMITIVE_TYPES_WCHAR \ |
| EPTYPE(wc, wchar_t, CT_PRIMITIVE_CHAR | \ |
| (((wchar_t)-1) > 0 ? 0 : CT_IS_SIGNED_WCHAR)) |
| #else |
| # define ENUM_PRIMITIVE_TYPES_WCHAR /* nothing */ |
| #endif |
| |
| #define EPTYPE(code, typename, flags) EPTYPE2(code, #typename, typename, flags) |
| |
| #define EPTYPE2(code, export_name, typename, flags) \ |
| struct aligncheck_##code { char x; typename y; }; |
| ENUM_PRIMITIVE_TYPES |
| #undef EPTYPE2 |
| |
| CTypeDescrObject *td; |
| static const struct descr_s { const char *name; int size, align, flags; } |
| types[] = { |
| #define EPTYPE2(code, export_name, typename, flags) \ |
| { export_name, \ |
| sizeof(typename), \ |
| offsetof(struct aligncheck_##code, y), \ |
| flags \ |
| }, |
| ENUM_PRIMITIVE_TYPES |
| #undef EPTYPE2 |
| #undef EPTYPE |
| #undef ENUM_PRIMITIVE_TYPES_WCHAR |
| #undef ENUM_PRIMITIVE_TYPES |
| { NULL } |
| }; |
| const struct descr_s *ptypes; |
| const void *unique_key[1]; |
| int name_size; |
| ffi_type *ffitype; |
| |
| for (ptypes=types; ; ptypes++) { |
| if (ptypes->name == NULL) { |
| #ifndef HAVE_WCHAR_H |
| if (strcmp(name, "wchar_t")) |
| PyErr_SetString(PyExc_NotImplementedError, name); |
| else |
| #endif |
| PyErr_SetString(PyExc_KeyError, name); |
| return NULL; |
| } |
| if (strcmp(name, ptypes->name) == 0) |
| break; |
| } |
| |
| if (ptypes->flags & CT_PRIMITIVE_SIGNED) { |
| switch (ptypes->size) { |
| case 1: ffitype = &ffi_type_sint8; break; |
| case 2: ffitype = &ffi_type_sint16; break; |
| case 4: ffitype = &ffi_type_sint32; break; |
| case 8: ffitype = &ffi_type_sint64; break; |
| default: goto bad_ffi_type; |
| } |
| } |
| else if (ptypes->flags & CT_PRIMITIVE_FLOAT) { |
| if (strcmp(ptypes->name, "float") == 0) |
| ffitype = &ffi_type_float; |
| else if (strcmp(ptypes->name, "double") == 0) |
| ffitype = &ffi_type_double; |
| else if (strcmp(ptypes->name, "long double") == 0) { |
| /* assume that if sizeof(double) == sizeof(long double), then |
| the two types are equivalent for C. libffi bugs on Win64 |
| if a function's return type is ffi_type_longdouble... */ |
| if (sizeof(double) == sizeof(long double)) |
| ffitype = &ffi_type_double; |
| else |
| ffitype = &ffi_type_longdouble; |
| } |
| else |
| goto bad_ffi_type; |
| } |
| else if (ptypes->flags & CT_PRIMITIVE_COMPLEX) { |
| /* As of March 2017, still no libffi support for complex. |
| It fails silently if we try to use ffi_type_complex_float |
| or ffi_type_complex_double. Better not use it at all. |
| */ |
| ffitype = NULL; |
| } |
| else { |
| switch (ptypes->size) { |
| case 1: ffitype = &ffi_type_uint8; break; |
| case 2: ffitype = &ffi_type_uint16; break; |
| case 4: ffitype = &ffi_type_uint32; break; |
| case 8: ffitype = &ffi_type_uint64; break; |
| default: goto bad_ffi_type; |
| } |
| } |
| |
| name_size = strlen(ptypes->name) + 1; |
| td = ctypedescr_new(name_size); |
| if (td == NULL) |
| return NULL; |
| |
| memcpy(td->ct_name, name, name_size); |
| td->ct_size = ptypes->size; |
| td->ct_length = ptypes->align; |
| td->ct_extra = ffitype; |
| td->ct_flags = ptypes->flags; |
| if (td->ct_flags & (CT_PRIMITIVE_SIGNED | CT_PRIMITIVE_CHAR)) { |
| if (td->ct_size <= (Py_ssize_t)sizeof(long)) |
| td->ct_flags |= CT_PRIMITIVE_FITS_LONG; |
| } |
| else if (td->ct_flags & CT_PRIMITIVE_UNSIGNED) { |
| if (td->ct_size < (Py_ssize_t)sizeof(long)) |
| td->ct_flags |= CT_PRIMITIVE_FITS_LONG; |
| } |
| td->ct_name_position = strlen(td->ct_name); |
| unique_key[0] = ptypes; |
| return get_unique_type(td, unique_key, 1); |
| |
| bad_ffi_type: |
| PyErr_Format(PyExc_NotImplementedError, |
| "primitive type '%s' has size %d; " |
| "the supported sizes are 1, 2, 4, 8", |
| name, (int)ptypes->size); |
| return NULL; |
| } |
| |
| static PyObject *b_new_primitive_type(PyObject *self, PyObject *args) |
| { |
| char *name; |
| if (!PyArg_ParseTuple(args, "s:new_primitive_type", &name)) |
| return NULL; |
| return new_primitive_type(name); |
| } |
| |
| static PyObject *new_pointer_type(CTypeDescrObject *ctitem) |
| { |
| CTypeDescrObject *td; |
| const char *extra; |
| const void *unique_key[1]; |
| |
| if (ctitem->ct_flags & CT_ARRAY) |
| extra = "(*)"; /* obscure case: see test_array_add */ |
| else |
| extra = " *"; |
| td = ctypedescr_new_on_top(ctitem, extra, 2); |
| if (td == NULL) |
| return NULL; |
| |
| td->ct_size = sizeof(void *); |
| td->ct_length = -1; |
| td->ct_flags = CT_POINTER; |
| if (ctitem->ct_flags & (CT_STRUCT|CT_UNION)) |
| td->ct_flags |= CT_IS_PTR_TO_OWNED; |
| if (ctitem->ct_flags & CT_VOID) |
| td->ct_flags |= CT_IS_VOID_PTR; |
| if ((ctitem->ct_flags & CT_VOID) || |
| ((ctitem->ct_flags & CT_PRIMITIVE_CHAR) && |
| ctitem->ct_size == sizeof(char))) |
| td->ct_flags |= CT_IS_VOIDCHAR_PTR; /* 'void *' or 'char *' only */ |
| unique_key[0] = ctitem; |
| return get_unique_type(td, unique_key, 1); |
| } |
| |
| static PyObject *b_new_pointer_type(PyObject *self, PyObject *args) |
| { |
| CTypeDescrObject *ctitem; |
| if (!PyArg_ParseTuple(args, "O!:new_pointer_type", |
| &CTypeDescr_Type, &ctitem)) |
| return NULL; |
| return new_pointer_type(ctitem); |
| } |
| |
| static PyObject *b_new_array_type(PyObject *self, PyObject *args) |
| { |
| PyObject *lengthobj; |
| Py_ssize_t length; |
| CTypeDescrObject *ctptr; |
| |
| if (!PyArg_ParseTuple(args, "O!O:new_array_type", |
| &CTypeDescr_Type, &ctptr, &lengthobj)) |
| return NULL; |
| |
| if (lengthobj == Py_None) { |
| length = -1; |
| } |
| else { |
| length = PyNumber_AsSsize_t(lengthobj, PyExc_OverflowError); |
| if (length < 0) { |
| if (!PyErr_Occurred()) |
| PyErr_SetString(PyExc_ValueError, "negative array length"); |
| return NULL; |
| } |
| } |
| return new_array_type(ctptr, length); |
| } |
| |
| static PyObject * |
| new_array_type(CTypeDescrObject *ctptr, Py_ssize_t length) |
| { |
| CTypeDescrObject *td, *ctitem; |
| char extra_text[32]; |
| Py_ssize_t arraysize; |
| int flags = CT_ARRAY; |
| const void *unique_key[2]; |
| |
| if (!(ctptr->ct_flags & CT_POINTER)) { |
| PyErr_SetString(PyExc_TypeError, "first arg must be a pointer ctype"); |
| return NULL; |
| } |
| ctitem = ctptr->ct_itemdescr; |
| if (ctitem->ct_size < 0) { |
| PyErr_Format(PyExc_ValueError, "array item of unknown size: '%s'", |
| ctitem->ct_name); |
| return NULL; |
| } |
| |
| if (length < 0) { |
| sprintf(extra_text, "[]"); |
| length = -1; |
| arraysize = -1; |
| } |
| else { |
| sprintf(extra_text, "[%llu]", (unsigned PY_LONG_LONG)length); |
| arraysize = MUL_WRAPAROUND(length, ctitem->ct_size); |
| if (length > 0 && (arraysize / length) != ctitem->ct_size) { |
| PyErr_SetString(PyExc_OverflowError, |
| "array size would overflow a Py_ssize_t"); |
| return NULL; |
| } |
| } |
| td = ctypedescr_new_on_top(ctitem, extra_text, 0); |
| if (td == NULL) |
| return NULL; |
| |
| Py_INCREF(ctptr); |
| td->ct_stuff = (PyObject *)ctptr; |
| td->ct_size = arraysize; |
| td->ct_length = length; |
| td->ct_flags = flags; |
| unique_key[0] = ctptr; |
| unique_key[1] = (void *)length; |
| return get_unique_type(td, unique_key, 2); |
| } |
| |
| static PyObject *new_void_type(void) |
| { |
| int name_size = strlen("void") + 1; |
| const void *unique_key[1]; |
| CTypeDescrObject *td = ctypedescr_new(name_size); |
| if (td == NULL) |
| return NULL; |
| |
| memcpy(td->ct_name, "void", name_size); |
| td->ct_size = -1; |
| td->ct_flags = CT_VOID | CT_IS_OPAQUE; |
| td->ct_name_position = strlen("void"); |
| unique_key[0] = "void"; |
| return get_unique_type(td, unique_key, 1); |
| } |
| |
| static PyObject *b_new_void_type(PyObject *self, PyObject *args) |
| { |
| return new_void_type(); |
| } |
| |
| static PyObject *new_struct_or_union_type(const char *name, int flag) |
| { |
| int namelen = strlen(name); |
| CTypeDescrObject *td = ctypedescr_new(namelen + 1); |
| if (td == NULL) |
| return NULL; |
| |
| td->ct_size = -1; |
| td->ct_length = -1; |
| td->ct_flags = flag | CT_IS_OPAQUE; |
| td->ct_extra = NULL; |
| memcpy(td->ct_name, name, namelen + 1); |
| td->ct_name_position = namelen; |
| return (PyObject *)td; |
| } |
| |
| static PyObject *b_new_struct_type(PyObject *self, PyObject *args) |
| { |
| char *name; |
| int flag; |
| if (!PyArg_ParseTuple(args, "s:new_struct_type", &name)) |
| return NULL; |
| |
| flag = CT_STRUCT; |
| if (strcmp(name, "struct _IO_FILE") == 0 || strcmp(name, "FILE") == 0) |
| flag |= CT_IS_FILE; |
| return new_struct_or_union_type(name, flag); |
| } |
| |
| static PyObject *b_new_union_type(PyObject *self, PyObject *args) |
| { |
| char *name; |
| if (!PyArg_ParseTuple(args, "s:new_union_type", &name)) |
| return NULL; |
| return new_struct_or_union_type(name, CT_UNION); |
| } |
| |
| static CFieldObject * |
| _add_field(PyObject *interned_fields, PyObject *fname, CTypeDescrObject *ftype, |
| Py_ssize_t offset, int bitshift, int fbitsize, int flags) |
| { |
| int err; |
| Py_ssize_t prev_size; |
| CFieldObject *cf = PyObject_New(CFieldObject, &CField_Type); |
| if (cf == NULL) |
| return NULL; |
| |
| Py_INCREF(ftype); |
| cf->cf_type = ftype; |
| cf->cf_offset = offset; |
| cf->cf_bitshift = bitshift; |
| cf->cf_bitsize = fbitsize; |
| cf->cf_flags = flags; |
| |
| Py_INCREF(fname); |
| PyText_InternInPlace(&fname); |
| prev_size = PyDict_Size(interned_fields); |
| err = PyDict_SetItem(interned_fields, fname, (PyObject *)cf); |
| Py_DECREF(fname); |
| Py_DECREF(cf); |
| if (err < 0) |
| return NULL; |
| |
| if (PyDict_Size(interned_fields) != prev_size + 1) { |
| PyErr_Format(PyExc_KeyError, "duplicate field name '%s'", |
| PyText_AS_UTF8(fname)); |
| return NULL; |
| } |
| return cf; /* borrowed reference */ |
| } |
| |
| #define SF_MSVC_BITFIELDS 0x01 |
| #define SF_GCC_ARM_BITFIELDS 0x02 |
| #define SF_GCC_X86_BITFIELDS 0x10 |
| |
| #define SF_GCC_BIG_ENDIAN 0x04 |
| #define SF_GCC_LITTLE_ENDIAN 0x40 |
| |
| #define SF_PACKED 0x08 |
| #define SF_STD_FIELD_POS 0x80 |
| |
| #ifdef MS_WIN32 |
| # define SF_DEFAULT_PACKING 8 |
| #else |
| # define SF_DEFAULT_PACKING 0x40000000 /* a huge power of two */ |
| #endif |
| |
| static int complete_sflags(int sflags) |
| { |
| /* add one of the SF_xxx_BITFIELDS flags if none is specified */ |
| if (!(sflags & (SF_MSVC_BITFIELDS | SF_GCC_ARM_BITFIELDS | |
| SF_GCC_X86_BITFIELDS))) { |
| #ifdef MS_WIN32 |
| sflags |= SF_MSVC_BITFIELDS; |
| #else |
| # if defined(__APPLE__) && defined(__arm64__) |
| sflags |= SF_GCC_X86_BITFIELDS; |
| # elif defined(__arm__) || defined(__aarch64__) |
| sflags |= SF_GCC_ARM_BITFIELDS; |
| # else |
| sflags |= SF_GCC_X86_BITFIELDS; |
| # endif |
| #endif |
| } |
| /* add one of SF_GCC_xx_ENDIAN if none is specified */ |
| if (!(sflags & (SF_GCC_BIG_ENDIAN | SF_GCC_LITTLE_ENDIAN))) { |
| int _check_endian = 1; |
| if (*(char *)&_check_endian == 0) |
| sflags |= SF_GCC_BIG_ENDIAN; |
| else |
| sflags |= SF_GCC_LITTLE_ENDIAN; |
| } |
| return sflags; |
| } |
| |
| static int detect_custom_layout(CTypeDescrObject *ct, int sflags, |
| Py_ssize_t cdef_value, |
| Py_ssize_t compiler_value, |
| const char *msg1, const char *txt, |
| const char *msg2) |
| { |
| if (compiler_value != cdef_value) { |
| if (sflags & SF_STD_FIELD_POS) { |
| PyErr_Format(FFIError, |
| "%s: %s%s%s (cdef says %zd, but C compiler says %zd)." |
| " fix it or use \"...;\" as the last field in the " |
| "cdef for %s to make it flexible", |
| ct->ct_name, msg1, txt, msg2, |
| cdef_value, compiler_value, |
| ct->ct_name); |
| return -1; |
| } |
| ct->ct_flags |= CT_CUSTOM_FIELD_POS; |
| } |
| return 0; |
| } |
| |
| #define ROUNDUP_BYTES(bytes, bits) ((bytes) + ((bits) > 0)) |
| |
| static PyObject *b_complete_struct_or_union(PyObject *self, PyObject *args) |
| { |
| CTypeDescrObject *ct; |
| PyObject *fields, *interned_fields, *ignored; |
| int is_union, alignment; |
| Py_ssize_t byteoffset, i, nb_fields, byteoffsetmax, alignedsize; |
| int bitoffset; |
| Py_ssize_t byteoffsetorg; |
| Py_ssize_t totalsize = -1; |
| int totalalignment = -1; |
| CFieldObject **previous; |
| int prev_bitfield_size, prev_bitfield_free; |
| int sflags = 0, fflags; |
| int pack = 0; |
| |
| if (!PyArg_ParseTuple(args, "O!O!|Oniii:complete_struct_or_union", |
| &CTypeDescr_Type, &ct, |
| &PyList_Type, &fields, |
| &ignored, &totalsize, &totalalignment, &sflags, |
| &pack)) |
| return NULL; |
| |
| sflags = complete_sflags(sflags); |
| if (sflags & SF_PACKED) |
| pack = 1; |
| else if (pack <= 0) |
| pack = SF_DEFAULT_PACKING; |
| else |
| sflags |= SF_PACKED; |
| |
| if ((ct->ct_flags & (CT_STRUCT|CT_IS_OPAQUE)) == |
| (CT_STRUCT|CT_IS_OPAQUE)) { |
| is_union = 0; |
| } |
| else if ((ct->ct_flags & (CT_UNION|CT_IS_OPAQUE)) == |
| (CT_UNION|CT_IS_OPAQUE)) { |
| is_union = 1; |
| } |
| else { |
| PyErr_SetString(PyExc_TypeError, |
| "first arg must be a non-initialized struct or union ctype"); |
| return NULL; |
| } |
| ct->ct_flags &= ~(CT_CUSTOM_FIELD_POS | CT_WITH_PACKED_CHANGE); |
| |
| alignment = 1; |
| byteoffset = 0; /* the real value is 'byteoffset+bitoffset*8', which */ |
| bitoffset = 0; /* counts the offset in bits */ |
| byteoffsetmax = 0; /* the maximum value of byteoffset-rounded-up-to-byte */ |
| prev_bitfield_size = 0; |
| prev_bitfield_free = 0; |
| nb_fields = PyList_GET_SIZE(fields); |
| interned_fields = PyDict_New(); |
| if (interned_fields == NULL) |
| return NULL; |
| |
| previous = (CFieldObject **)&ct->ct_extra; |
| |
| for (i=0; i<nb_fields; i++) { |
| PyObject *fname; |
| CTypeDescrObject *ftype; |
| int fbitsize = -1, falign, falignorg, do_align; |
| Py_ssize_t foffset = -1; |
| |
| if (!PyArg_ParseTuple(PyList_GET_ITEM(fields, i), "O!O!|in:list item", |
| &PyText_Type, &fname, |
| &CTypeDescr_Type, &ftype, |
| &fbitsize, &foffset)) |
| goto error; |
| |
| if (ftype->ct_size < 0) { |
| if ((ftype->ct_flags & CT_ARRAY) && fbitsize < 0 |
| && (i == nb_fields - 1 || foffset != -1)) { |
| ct->ct_flags |= CT_WITH_VAR_ARRAY; |
| } |
| else { |
| PyErr_Format(PyExc_TypeError, |
| "field '%s.%s' has ctype '%s' of unknown size", |
| ct->ct_name, PyText_AS_UTF8(fname), |
| ftype->ct_name); |
| goto error; |
| } |
| } |
| else if (ftype->ct_flags & (CT_STRUCT|CT_UNION)) { |
| if (force_lazy_struct(ftype) < 0) /* for CT_WITH_VAR_ARRAY */ |
| return NULL; |
| |
| /* GCC (or maybe C99) accepts var-sized struct fields that are not |
| the last field of a larger struct. That's why there is no |
| check here for "last field": we propagate the flag |
| CT_WITH_VAR_ARRAY to any struct that contains either an open- |
| ended array or another struct that recursively contains an |
| open-ended array. */ |
| if (ftype->ct_flags & CT_WITH_VAR_ARRAY) |
| ct->ct_flags |= CT_WITH_VAR_ARRAY; |
| } |
| |
| if (is_union) |
| byteoffset = bitoffset = 0; /* reset each field at offset 0 */ |
| |
| /* update the total alignment requirement, but skip it if the |
| field is an anonymous bitfield or if SF_PACKED */ |
| falignorg = get_alignment(ftype); |
| if (falignorg < 0) |
| goto error; |
| falign = (pack < falignorg) ? pack : falignorg; |
| |
| do_align = 1; |
| if (!(sflags & SF_GCC_ARM_BITFIELDS) && fbitsize >= 0) { |
| if (!(sflags & SF_MSVC_BITFIELDS)) { |
| /* GCC: anonymous bitfields (of any size) don't cause alignment */ |
| do_align = PyText_GetSize(fname) > 0; |
| } |
| else { |
| /* MSVC: zero-sized bitfields don't cause alignment */ |
| do_align = fbitsize > 0; |
| } |
| } |
| if (alignment < falign && do_align) |
| alignment = falign; |
| |
| fflags = (is_union && i > 0) ? BF_IGNORE_IN_CTOR : 0; |
| |
| if (fbitsize < 0) { |
| /* not a bitfield: common case */ |
| int bs_flag; |
| |
| if ((ftype->ct_flags & CT_ARRAY) && ftype->ct_length <= 0) |
| bs_flag = BS_EMPTY_ARRAY; |
| else |
| bs_flag = BS_REGULAR; |
| |
| /* align this field to its own 'falign' by inserting padding */ |
| |
| /* first, pad to the next byte, |
| * then pad to 'falign' or 'falignorg' bytes */ |
| byteoffset = ROUNDUP_BYTES(byteoffset, bitoffset); |
| bitoffset = 0; |
| byteoffsetorg = (byteoffset + falignorg-1) & ~(falignorg-1); |
| byteoffset = (byteoffset + falign-1) & ~(falign-1); |
| |
| if (byteoffsetorg != byteoffset) { |
| ct->ct_flags |= CT_WITH_PACKED_CHANGE; |
| } |
| |
| if (foffset >= 0) { |
| /* a forced field position: ignore the offset just computed, |
| except to know if we must set CT_CUSTOM_FIELD_POS */ |
| if (detect_custom_layout(ct, sflags, byteoffset, foffset, |
| "wrong offset for field '", |
| PyText_AS_UTF8(fname), "'") < 0) |
| goto error; |
| byteoffset = foffset; |
| } |
| |
| if (PyText_GetSize(fname) == 0 && |
| ftype->ct_flags & (CT_STRUCT|CT_UNION)) { |
| /* a nested anonymous struct or union */ |
| CFieldObject *cfsrc = (CFieldObject *)ftype->ct_extra; |
| for (; cfsrc != NULL; cfsrc = cfsrc->cf_next) { |
| /* broken complexity in the call to get_field_name(), |
| but we'll assume you never do that with nested |
| anonymous structures with thousand of fields */ |
| *previous = _add_field(interned_fields, |
| get_field_name(ftype, cfsrc), |
| cfsrc->cf_type, |
| byteoffset + cfsrc->cf_offset, |
| cfsrc->cf_bitshift, |
| cfsrc->cf_bitsize, |
| cfsrc->cf_flags | fflags); |
| if (*previous == NULL) |
| goto error; |
| previous = &(*previous)->cf_next; |
| } |
| /* always forbid such structures from being passed by value */ |
| ct->ct_flags |= CT_CUSTOM_FIELD_POS; |
| } |
| else { |
| *previous = _add_field(interned_fields, fname, ftype, |
| byteoffset, bs_flag, -1, fflags); |
| if (*previous == NULL) |
| goto error; |
| previous = &(*previous)->cf_next; |
| } |
| if (ftype->ct_size >= 0) |
| byteoffset += ftype->ct_size; |
| prev_bitfield_size = 0; |
| } |
| else { |
| /* this is the case of a bitfield */ |
| Py_ssize_t field_offset_bytes; |
| int bits_already_occupied, bitshift; |
| |
| if (foffset >= 0) { |
| PyErr_Format(PyExc_TypeError, |
| "field '%s.%s' is a bitfield, " |
| "but a fixed offset is specified", |
| ct->ct_name, PyText_AS_UTF8(fname)); |
| goto error; |
| } |
| |
| if (!(ftype->ct_flags & (CT_PRIMITIVE_SIGNED | |
| CT_PRIMITIVE_UNSIGNED | |
| CT_PRIMITIVE_CHAR))) { |
| PyErr_Format(PyExc_TypeError, |
| "field '%s.%s' declared as '%s' cannot be a bit field", |
| ct->ct_name, PyText_AS_UTF8(fname), |
| ftype->ct_name); |
| goto error; |
| } |
| if (fbitsize > 8 * ftype->ct_size) { |
| PyErr_Format(PyExc_TypeError, |
| "bit field '%s.%s' is declared '%s:%d', which " |
| "exceeds the width of the type", |
| ct->ct_name, PyText_AS_UTF8(fname), |
| ftype->ct_name, fbitsize); |
| goto error; |
| } |
| |
| /* compute the starting position of the theoretical field |
| that covers a complete 'ftype', inside of which we will |
| locate the real bitfield */ |
| field_offset_bytes = byteoffset; |
| field_offset_bytes &= ~(falign - 1); |
| |
| if (fbitsize == 0) { |
| if (PyText_GetSize(fname) > 0) { |
| PyErr_Format(PyExc_TypeError, |
| "field '%s.%s' is declared with :0", |
| ct->ct_name, PyText_AS_UTF8(fname)); |
| goto error; |
| } |
| if (!(sflags & SF_MSVC_BITFIELDS)) { |
| /* GCC's notion of "ftype :0;" */ |
| |
| /* pad byteoffset to a value aligned for "ftype" */ |
| if (ROUNDUP_BYTES(byteoffset, bitoffset) > field_offset_bytes) { |
| field_offset_bytes += falign; |
| assert(byteoffset < field_offset_bytes); |
| } |
| byteoffset = field_offset_bytes; |
| bitoffset = 0; |
| } |
| else { |
| /* MSVC's notion of "ftype :0;" */ |
| |
| /* Mostly ignored. It seems they only serve as |
| separator between other bitfields, to force them |
| into separate words. */ |
| } |
| prev_bitfield_size = 0; |
| } |
| else { |
| if (!(sflags & SF_MSVC_BITFIELDS)) { |
| /* GCC's algorithm */ |
| |
| /* Can the field start at the offset given by 'boffset'? It |
| can if it would entirely fit into an aligned ftype field. */ |
| bits_already_occupied = (byteoffset-field_offset_bytes) * 8 |
| + bitoffset; |
| |
| if (bits_already_occupied + fbitsize > 8 * ftype->ct_size) { |
| /* it would not fit, we need to start at the next |
| allowed position */ |
| if ((sflags & SF_PACKED) && |
| (bits_already_occupied & 7)) { |
| PyErr_Format(PyExc_NotImplementedError, |
| "with 'packed', gcc would compile field " |
| "'%s.%s' to reuse some bits in the previous " |
| "field", ct->ct_name, PyText_AS_UTF8(fname)); |
| goto error; |
| } |
| field_offset_bytes += falign; |
| assert(byteoffset < field_offset_bytes); |
| byteoffset = field_offset_bytes; |
| bitoffset = 0; |
| bitshift = 0; |
| } |
| else { |
| bitshift = bits_already_occupied; |
| assert(bitshift >= 0); |
| } |
| bitoffset += fbitsize; |
| byteoffset += (bitoffset >> 3); |
| bitoffset &= 7; |
| } |
| else { |
| /* MSVC's algorithm */ |
| |
| /* A bitfield is considered as taking the full width |
| of their declared type. It can share some bits |
| with the previous field only if it was also a |
| bitfield and used a type of the same size. */ |
| if (prev_bitfield_size == ftype->ct_size && |
| prev_bitfield_free >= fbitsize) { |
| /* yes: reuse */ |
| bitshift = 8 * prev_bitfield_size - prev_bitfield_free; |
| } |
| else { |
| /* no: start a new full field */ |
| byteoffset = ROUNDUP_BYTES(byteoffset, bitoffset); |
| bitoffset = 0; |
| /* align */ |
| byteoffset = (byteoffset + falign-1) & ~(falign-1); |
| byteoffset += ftype->ct_size; |
| bitshift = 0; |
| prev_bitfield_size = ftype->ct_size; |
| prev_bitfield_free = 8 * prev_bitfield_size; |
| } |
| prev_bitfield_free -= fbitsize; |
| field_offset_bytes = byteoffset - ftype->ct_size; |
| } |
| if (sflags & SF_GCC_BIG_ENDIAN) |
| bitshift = 8 * ftype->ct_size - fbitsize - bitshift; |
| |
| if (PyText_GetSize(fname) > 0) { |
| |
| *previous = _add_field(interned_fields, fname, ftype, |
| field_offset_bytes, bitshift, fbitsize, |
| fflags); |
| if (*previous == NULL) |
| goto error; |
| previous = &(*previous)->cf_next; |
| } |
| } |
| } |
| |
| assert(bitoffset == (bitoffset & 7)); |
| if (ROUNDUP_BYTES(byteoffset, bitoffset) > byteoffsetmax) |
| byteoffsetmax = ROUNDUP_BYTES(byteoffset, bitoffset); |
| } |
| *previous = NULL; |
| |
| /* Like C, if the size of this structure would be zero, we compute it |
| as 1 instead. But for ctypes support, we allow the manually- |
| specified totalsize to be zero in this case. */ |
| alignedsize = (byteoffsetmax + alignment - 1) & ~(alignment-1); |
| if (alignedsize == 0) |
| alignedsize = 1; |
| |
| if (totalsize < 0) { |
| totalsize = alignedsize; |
| } |
| else { |
| if (detect_custom_layout(ct, sflags, alignedsize, |
| totalsize, "wrong total size", "", "") < 0) |
| goto error; |
| if (totalsize < byteoffsetmax) { |
| PyErr_Format(PyExc_TypeError, |
| "%s cannot be of size %zd: there are fields at least " |
| "up to %zd", ct->ct_name, totalsize, byteoffsetmax); |
| goto error; |
| } |
| } |
| if (totalalignment < 0) { |
| totalalignment = alignment; |
| } |
| else { |
| if (detect_custom_layout(ct, sflags, alignment, totalalignment, |
| "wrong total alignment", "", "") < 0) |
| goto error; |
| } |
| |
| ct->ct_size = totalsize; |
| ct->ct_length = totalalignment; |
| ct->ct_stuff = interned_fields; |
| ct->ct_flags &= ~CT_IS_OPAQUE; |
| |
| Py_INCREF(Py_None); |
| return Py_None; |
| |
| error: |
| ct->ct_extra = NULL; |
| Py_DECREF(interned_fields); |
| return NULL; |
| } |
| |
| struct funcbuilder_s { |
| Py_ssize_t nb_bytes; |
| char *bufferp; |
| ffi_type **atypes; |
| ffi_type *rtype; |
| Py_ssize_t nargs; |
| CTypeDescrObject *fct; |
| }; |
| |
| static void *fb_alloc(struct funcbuilder_s *fb, Py_ssize_t size) |
| { |
| if (fb->bufferp == NULL) { |
| fb->nb_bytes += size; |
| return NULL; |
| } |
| else { |
| char *result = fb->bufferp; |
| fb->bufferp += size; |
| return result; |
| } |
| } |
| |
| #define SUPPORTED_IN_API_MODE \ |
| " are only supported as %s if the function is " \ |
| "'API mode' and non-variadic (i.e. declared inside ffibuilder" \ |
| ".cdef()+ffibuilder.set_source() and not taking a final '...' " \ |
| "argument)" |
| |
| static ffi_type *fb_unsupported(CTypeDescrObject *ct, const char *place, |
| const char *detail) |
| { |
| PyErr_Format(PyExc_NotImplementedError, |
| "ctype '%s' not supported as %s. %s. " |
| "Such structs" SUPPORTED_IN_API_MODE, |
| ct->ct_name, place, detail, place); |
| return NULL; |
| } |
| |
| static ffi_type *fb_fill_type(struct funcbuilder_s *fb, CTypeDescrObject *ct, |
| int is_result_type) |
| { |
| const char *place = is_result_type ? "return value" : "argument"; |
| |
| if (ct->ct_flags & (CT_PRIMITIVE_ANY & ~CT_PRIMITIVE_COMPLEX)) { |
| return (ffi_type *)ct->ct_extra; |
| } |
| else if (ct->ct_flags & (CT_POINTER|CT_FUNCTIONPTR)) { |
| return &ffi_type_pointer; |
| } |
| else if ((ct->ct_flags & CT_VOID) && is_result_type) { |
| return &ffi_type_void; |
| } |
| |
| if (ct->ct_size <= 0) { |
| PyErr_Format(PyExc_TypeError, |
| ct->ct_size < 0 ? "ctype '%s' has incomplete type" |
| : "ctype '%s' has size 0", |
| ct->ct_name); |
| return NULL; |
| } |
| if (ct->ct_flags & CT_STRUCT) { |
| ffi_type *ffistruct, *ffifield; |
| ffi_type **elements; |
| Py_ssize_t i, n, nflat; |
| CFieldObject *cf; |
| |
| /* We can't pass a struct that was completed by verify(). |
| Issue: assume verify() is given "struct { long b; ...; }". |
| Then it will complete it in the same way whether it is actually |
| "struct { long a, b; }" or "struct { double a; long b; }". |
| But on 64-bit UNIX, these two structs are passed by value |
| differently: e.g. on x86-64, "b" ends up in register "rsi" in |
| the first case and "rdi" in the second case. |
| |
| Another reason for CT_CUSTOM_FIELD_POS would be anonymous |
| nested structures: we lost the information about having it |
| here, so better safe (and forbid it) than sorry (and maybe |
| crash). Note: it seems we only get in this case with |
| ffi.verify(). |
| */ |
| if (force_lazy_struct(ct) < 0) |
| return NULL; |
| if (ct->ct_flags & CT_CUSTOM_FIELD_POS) { |
| /* these NotImplementedErrors may be caught and ignored until |
| a real call is made to a function of this type */ |
| return fb_unsupported(ct, place, |
| "It is a struct declared with \"...;\", but the C " |
| "calling convention may depend on the missing fields; " |
| "or, it contains anonymous struct/unions"); |
| } |
| /* Another reason: __attribute__((packed)) is not supported by libffi. |
| */ |
| if (ct->ct_flags & CT_WITH_PACKED_CHANGE) { |
| return fb_unsupported(ct, place, |
| "It is a 'packed' structure, with a different layout than " |
| "expected by libffi"); |
| } |
| |
| n = PyDict_Size(ct->ct_stuff); |
| nflat = 0; |
| |
| /* walk the fields, expanding arrays into repetitions; first, |
| only count how many flattened fields there are */ |
| cf = (CFieldObject *)ct->ct_extra; |
| for (i=0; i<n; i++) { |
| Py_ssize_t flat; |
| CTypeDescrObject *ct1; |
| assert(cf != NULL); |
| if (cf->cf_bitshift >= 0) { |
| return fb_unsupported(ct, place, |
| "It is a struct with bit fields, which libffi does not " |
| "support"); |
| } |
| flat = 1; |
| ct1 = cf->cf_type; |
| while (ct1->ct_flags & CT_ARRAY) { |
| flat *= ct1->ct_length; |
| ct1 = ct1->ct_itemdescr; |
| } |
| if (flat <= 0) { |
| return fb_unsupported(ct, place, |
| "It is a struct with a zero-length array, which libffi " |
| "does not support"); |
| } |
| nflat += flat; |
| cf = cf->cf_next; |
| } |
| assert(cf == NULL); |
| |
| /* next, allocate and fill the flattened list */ |
| elements = fb_alloc(fb, (nflat + 1) * sizeof(ffi_type*)); |
| nflat = 0; |
| cf = (CFieldObject *)ct->ct_extra; |
| for (i=0; i<n; i++) { |
| Py_ssize_t j, flat = 1; |
| CTypeDescrObject *ct = cf->cf_type; |
| while (ct->ct_flags & CT_ARRAY) { |
| flat *= ct->ct_length; |
| ct = ct->ct_itemdescr; |
| } |
| ffifield = fb_fill_type(fb, ct, 0); |
| if (PyErr_Occurred()) |
| return NULL; |
| if (elements != NULL) { |
| for (j=0; j<flat; j++) |
| elements[nflat++] = ffifield; |
| } |
| cf = cf->cf_next; |
| } |
| |
| /* finally, allocate the FFI_TYPE_STRUCT */ |
| ffistruct = fb_alloc(fb, sizeof(ffi_type)); |
| if (ffistruct != NULL) { |
| elements[nflat] = NULL; |
| ffistruct->size = ct->ct_size; |
| ffistruct->alignment = ct->ct_length; |
| ffistruct->type = FFI_TYPE_STRUCT; |
| ffistruct->elements = elements; |
| } |
| return ffistruct; |
| } |
| else if (ct->ct_flags & CT_UNION) { |
| PyErr_Format(PyExc_NotImplementedError, |
| "ctype '%s' not supported as %s by libffi. " |
| "Unions" SUPPORTED_IN_API_MODE, |
| ct->ct_name, place, place); |
| return NULL; |
| } |
| else { |
| char *extra = ""; |
| if (ct->ct_flags & CT_PRIMITIVE_COMPLEX) |
| extra = " (the support for complex types inside libffi " |
| "is mostly missing at this point, so CFFI only " |
| "supports complex types as arguments or return " |
| "value in API-mode functions)"; |
| |
| PyErr_Format(PyExc_NotImplementedError, |
| "ctype '%s' (size %zd) not supported as %s%s", |
| ct->ct_name, ct->ct_size, place, extra); |
| return NULL; |
| } |
| } |
| |
| #define ALIGN_ARG(n) ((n) + 7) & ~7 |
| |
| static int fb_build(struct funcbuilder_s *fb, PyObject *fargs, |
| CTypeDescrObject *fresult) |
| { |
| Py_ssize_t i, nargs = PyTuple_GET_SIZE(fargs); |
| Py_ssize_t exchange_offset; |
| cif_description_t *cif_descr; |
| |
| /* ffi buffer: start with a cif_description */ |
| cif_descr = fb_alloc(fb, sizeof(cif_description_t) + |
| nargs * sizeof(Py_ssize_t)); |
| |
| /* ffi buffer: next comes an array of 'ffi_type*', one per argument */ |
| fb->atypes = fb_alloc(fb, nargs * sizeof(ffi_type*)); |
| fb->nargs = nargs; |
| |
| /* ffi buffer: next comes the result type */ |
| fb->rtype = fb_fill_type(fb, fresult, 1); |
| if (PyErr_Occurred()) |
| return -1; |
| if (cif_descr != NULL) { |
| /* exchange data size */ |
| /* first, enough room for an array of 'nargs' pointers */ |
| exchange_offset = nargs * sizeof(void*); |
| exchange_offset = ALIGN_ARG(exchange_offset); |
| cif_descr->exchange_offset_arg[0] = exchange_offset; |
| /* then enough room for the result --- which means at least |
| sizeof(ffi_arg), according to the ffi docs */ |
| i = fb->rtype->size; |
| if (i < (Py_ssize_t)sizeof(ffi_arg)) |
| i = sizeof(ffi_arg); |
| exchange_offset += i; |
| } |
| else |
| exchange_offset = 0; /* not used */ |
| |
| /* loop over the arguments */ |
| for (i=0; i<nargs; i++) { |
| CTypeDescrObject *farg; |
| ffi_type *atype; |
| |
| farg = (CTypeDescrObject *)PyTuple_GET_ITEM(fargs, i); |
| /* convert arrays to pointers */ |
| if (farg->ct_flags & CT_ARRAY) |
| farg = (CTypeDescrObject *)farg->ct_stuff; |
| |
| /* ffi buffer: fill in the ffi for the i'th argument */ |
| assert(farg != NULL); |
| atype = fb_fill_type(fb, farg, 0); |
| if (PyErr_Occurred()) |
| return -1; |
| |
| if (fb->atypes != NULL) { |
| fb->atypes[i] = atype; |
| /* exchange data size */ |
| exchange_offset = ALIGN_ARG(exchange_offset); |
| cif_descr->exchange_offset_arg[1 + i] = exchange_offset; |
| exchange_offset += atype->size; |
| } |
| } |
| |
| if (cif_descr != NULL) { |
| /* exchange data size */ |
| /* we also align it to the next multiple of 8, in an attempt to |
| work around bugs(?) of libffi like #241 */ |
| cif_descr->exchange_size = ALIGN_ARG(exchange_offset); |
| } |
| return 0; |
| } |
| |
| #undef ALIGN_ARG |
| |
| static void fb_cat_name(struct funcbuilder_s *fb, const char *piece, |
| int piecelen) |
| { |
| if (fb->bufferp == NULL) { |
| fb->nb_bytes += piecelen; |
| } |
| else { |
| memcpy(fb->bufferp, piece, piecelen); |
| fb->bufferp += piecelen; |
| } |
| } |
| |
| static int fb_build_name(struct funcbuilder_s *fb, const char *repl, |
| CTypeDescrObject **pfargs, Py_ssize_t nargs, |
| CTypeDescrObject *fresult, int ellipsis) |
| { |
| Py_ssize_t i; |
| fb->nargs = nargs; |
| |
| /* name: the function type name we build here is, like in C, made |
| as follows: |
| |
| RESULT_TYPE_HEAD (*)(ARG_1_TYPE, ARG_2_TYPE, etc) RESULT_TYPE_TAIL |
| */ |
| fb_cat_name(fb, fresult->ct_name, fresult->ct_name_position); |
| if (repl[0] != '(' && |
| fresult->ct_name[fresult->ct_name_position - 1] != '*') |
| fb_cat_name(fb, " ", 1); /* add a space */ |
| fb_cat_name(fb, repl, strlen(repl)); |
| if (fb->fct) { |
| i = strlen(repl) - 1; /* between '(*' and ')' */ |
| assert(repl[i] == ')'); |
| fb->fct->ct_name_position = fresult->ct_name_position + i; |
| } |
| fb_cat_name(fb, "(", 1); |
| |
| /* loop over the arguments */ |
| for (i=0; i<nargs; i++) { |
| CTypeDescrObject *farg; |
| |
| farg = pfargs[i]; |
| if (!CTypeDescr_Check(farg)) { |
| PyErr_SetString(PyExc_TypeError, "expected a tuple of ctypes"); |
| return -1; |
| } |
| /* name: concatenate the name of the i'th argument's type */ |
| if (i > 0) |
| fb_cat_name(fb, ", ", 2); |
| fb_cat_name(fb, farg->ct_name, strlen(farg->ct_name)); |
| } |
| |
| /* name: add the '...' if needed */ |
| if (ellipsis) { |
| if (nargs > 0) |
| fb_cat_name(fb, ", ", 2); |
| fb_cat_name(fb, "...", 3); |
| } |
| |
| /* name: concatenate the tail of the result type */ |
| fb_cat_name(fb, ")", 1); |
| fb_cat_name(fb, fresult->ct_name + fresult->ct_name_position, |
| strlen(fresult->ct_name) - fresult->ct_name_position + 1); |
| return 0; |
| } |
| |
| static CTypeDescrObject *fb_prepare_ctype(struct funcbuilder_s *fb, |
| PyObject *fargs, |
| CTypeDescrObject *fresult, |
| int ellipsis, int fabi) |
| { |
| CTypeDescrObject *fct, **pfargs; |
| Py_ssize_t nargs; |
| char *repl = "(*)"; |
| |
| fb->nb_bytes = 0; |
| fb->bufferp = NULL; |
| fb->fct = NULL; |
| |
| pfargs = (CTypeDescrObject **)&PyTuple_GET_ITEM(fargs, 0); |
| nargs = PyTuple_GET_SIZE(fargs); |
| #if defined(MS_WIN32) && !defined(_WIN64) |
| if (fabi == FFI_STDCALL) |
| repl = "(__stdcall *)"; |
| #endif |
| |
| /* compute the total size needed for the name */ |
| if (fb_build_name(fb, repl, pfargs, nargs, fresult, ellipsis) < 0) |
| return NULL; |
| |
| /* allocate the function type */ |
| fct = ctypedescr_new(fb->nb_bytes); |
| if (fct == NULL) |
| return NULL; |
| fb->fct = fct; |
| |
| /* call again fb_build_name() to really build the ct_name */ |
| fb->bufferp = fct->ct_name; |
| if (fb_build_name(fb, repl, pfargs, nargs, fresult, ellipsis) < 0) |
| goto error; |
| assert(fb->bufferp == fct->ct_name + fb->nb_bytes); |
| |
| fct->ct_extra = NULL; |
| fct->ct_size = sizeof(void(*)(void)); |
| fct->ct_flags = CT_FUNCTIONPTR; |
| return fct; |
| |
| error: |
| Py_DECREF(fct); |
| return NULL; |
| } |
| |
| static cif_description_t *fb_prepare_cif(PyObject *fargs, |
| CTypeDescrObject *fresult, |
| Py_ssize_t variadic_nargs_declared, |
| ffi_abi fabi) |
| |
| { |
| char *buffer; |
| cif_description_t *cif_descr; |
| struct funcbuilder_s funcbuffer; |
| ffi_status status = (ffi_status)-1; |
| |
| funcbuffer.nb_bytes = 0; |
| funcbuffer.bufferp = NULL; |
| |
| /* compute the total size needed in the buffer for libffi */ |
| if (fb_build(&funcbuffer, fargs, fresult) < 0) |
| return NULL; |
| |
| /* allocate the buffer */ |
| buffer = PyObject_Malloc(funcbuffer.nb_bytes); |
| if (buffer == NULL) { |
| PyErr_NoMemory(); |
| return NULL; |
| } |
| |
| /* call again fb_build() to really build the libffi data structures */ |
| funcbuffer.bufferp = buffer; |
| if (fb_build(&funcbuffer, fargs, fresult) < 0) |
| goto error; |
| assert(funcbuffer.bufferp == buffer + funcbuffer.nb_bytes); |
| |
| cif_descr = (cif_description_t *)buffer; |
| |
| /* use `ffi_prep_cif_var` if necessary and available */ |
| #if CFFI_CHECK_FFI_PREP_CIF_VAR_MAYBE |
| if (variadic_nargs_declared >= 0) { |
| if (CFFI_CHECK_FFI_PREP_CIF_VAR) { |
| status = ffi_prep_cif_var(&cif_descr->cif, fabi, |
| variadic_nargs_declared, funcbuffer.nargs, |
| funcbuffer.rtype, funcbuffer.atypes); |
| } |
| } |
| #endif |
| |
| if (status == (ffi_status)-1) { |
| status = ffi_prep_cif(&cif_descr->cif, fabi, funcbuffer.nargs, |
| funcbuffer.rtype, funcbuffer.atypes); |
| } |
| |
| if (status != FFI_OK) { |
| PyErr_SetString(PyExc_SystemError, |
| "libffi failed to build this function type"); |
| goto error; |
| } |
| return cif_descr; |
| |
| error: |
| PyObject_Free(buffer); |
| return NULL; |
| } |
| |
| static PyObject *new_function_type(PyObject *fargs, /* tuple */ |
| CTypeDescrObject *fresult, |
| int ellipsis, int fabi) |
| { |
| PyObject *fabiobj; |
| CTypeDescrObject *fct; |
| struct funcbuilder_s funcbuilder; |
| Py_ssize_t i; |
| const void **unique_key; |
| |
| if ((fresult->ct_size < 0 && !(fresult->ct_flags & CT_VOID)) || |
| (fresult->ct_flags & CT_ARRAY)) { |
| char *msg; |
| if (fresult->ct_flags & CT_IS_OPAQUE) |
| msg = "result type '%s' is opaque"; |
| else |
| msg = "invalid result type: '%s'"; |
| PyErr_Format(PyExc_TypeError, msg, fresult->ct_name); |
| return NULL; |
| } |
| |
| fct = fb_prepare_ctype(&funcbuilder, fargs, fresult, ellipsis, fabi); |
| if (fct == NULL) |
| return NULL; |
| |
| if (!ellipsis) { |
| /* Functions with '...' varargs are stored without a cif_descr |
| at all. The cif is computed on every call from the actual |
| types passed in. For all other functions, the cif_descr |
| is computed here. */ |
| cif_description_t *cif_descr; |
| |
| cif_descr = fb_prepare_cif(fargs, fresult, -1, fabi); |
| if (cif_descr == NULL) { |
| if (PyErr_ExceptionMatches(PyExc_NotImplementedError)) { |
| PyErr_Clear(); /* will get the exception if we see an |
| actual call */ |
| } |
| else |
| goto error; |
| } |
| |
| fct->ct_extra = (char *)cif_descr; |
| } |
| |
| /* build the signature, given by a tuple of ctype objects */ |
| fct->ct_stuff = PyTuple_New(2 + funcbuilder.nargs); |
| if (fct->ct_stuff == NULL) |
| goto error; |
| fabiobj = PyInt_FromLong(fabi); |
| if (fabiobj == NULL) |
| goto error; |
| PyTuple_SET_ITEM(fct->ct_stuff, 0, fabiobj); |
| |
| Py_INCREF(fresult); |
| PyTuple_SET_ITEM(fct->ct_stuff, 1, (PyObject *)fresult); |
| for (i=0; i<funcbuilder.nargs; i++) { |
| PyObject *o = PyTuple_GET_ITEM(fargs, i); |
| /* convert arrays into pointers */ |
| if (((CTypeDescrObject *)o)->ct_flags & CT_ARRAY) |
| o = ((CTypeDescrObject *)o)->ct_stuff; |
| Py_INCREF(o); |
| PyTuple_SET_ITEM(fct->ct_stuff, 2 + i, o); |
| } |
| |
| /* [ctresult, ellipsis+abi, num_args, ctargs...] */ |
| unique_key = alloca((3 + funcbuilder.nargs) * sizeof(void *)); |
| unique_key[0] = fresult; |
| unique_key[1] = (const void *)(Py_ssize_t)((fabi << 1) | !!ellipsis); |
| unique_key[2] = (const void *)(Py_ssize_t)(funcbuilder.nargs); |
| for (i=0; i<funcbuilder.nargs; i++) |
| unique_key[3 + i] = PyTuple_GET_ITEM(fct->ct_stuff, 2 + i); |
| return get_unique_type(fct, unique_key, 3 + funcbuilder.nargs); |
| |
| error: |
| Py_DECREF(fct); |
| return NULL; |
| } |
| |
| static PyObject *b_new_function_type(PyObject *self, PyObject *args) |
| { |
| PyObject *fargs; |
| CTypeDescrObject *fresult; |
| int ellipsis = 0, fabi = FFI_DEFAULT_ABI; |
| |
| if (!PyArg_ParseTuple(args, "O!O!|ii:new_function_type", |
| &PyTuple_Type, &fargs, |
| &CTypeDescr_Type, &fresult, |
| &ellipsis, |
| &fabi)) |
| return NULL; |
| |
| return new_function_type(fargs, fresult, ellipsis, fabi); |
| } |
| |
| static int convert_from_object_fficallback(char *result, |
| CTypeDescrObject *ctype, |
| PyObject *pyobj, |
| int encode_result_for_libffi) |
| { |
| /* work work work around a libffi irregularity: for integer return |
| types we have to fill at least a complete 'ffi_arg'-sized result |
| buffer. */ |
| if (ctype->ct_size < (Py_ssize_t)sizeof(ffi_arg)) { |
| if (ctype->ct_flags & CT_VOID) { |
| if (pyobj == Py_None) { |
| return 0; |
| } |
| else { |
| PyErr_SetString(PyExc_TypeError, |
| "callback with the return type 'void' must return None"); |
| return -1; |
| } |
| } |
| if (!encode_result_for_libffi) |
| goto skip; |
| if (ctype->ct_flags & CT_PRIMITIVE_SIGNED) { |
| PY_LONG_LONG value; |
| /* It's probably fine to always zero-extend, but you never |
| know: maybe some code somewhere expects a negative |
| 'short' result to be returned into EAX as a 32-bit |
| negative number. Better safe than sorry. This code |
| is about that case. Let's ignore this for enums. |
| */ |
| /* do a first conversion only to detect overflows. This |
| conversion produces stuff that is otherwise ignored. */ |
| if (convert_from_object(result, ctype, pyobj) < 0) |
| return -1; |
| /* manual inlining and tweaking of convert_from_object() |
| in order to write a whole 'ffi_arg'. */ |
| value = _my_PyLong_AsLongLong(pyobj); |
| if (value == -1 && PyErr_Occurred()) |
| return -1; |
| write_raw_integer_data(result, value, sizeof(ffi_arg)); |
| return 0; |
| } |
| else if (ctype->ct_flags & (CT_PRIMITIVE_CHAR | CT_PRIMITIVE_SIGNED | |
| CT_PRIMITIVE_UNSIGNED | |
| CT_POINTER | CT_FUNCTIONPTR)) { |
| /* zero extension: fill the '*result' with zeros, and (on big- |
| endian machines) correct the 'result' pointer to write to. |
| We also do that for pointers, even though we're normally not |
| in this branch because ctype->ct_size == sizeof(ffi_arg) for |
| pointers---except on some architectures like x32 (issue #372). |
| */ |
| memset(result, 0, sizeof(ffi_arg)); |
| #ifdef WORDS_BIGENDIAN |
| result += (sizeof(ffi_arg) - ctype->ct_size); |
| #endif |
| } |
| } |
| skip: |
| return convert_from_object(result, ctype, pyobj); |
| } |
| |
| static void _my_PyErr_WriteUnraisable(PyObject *t, PyObject *v, PyObject *tb, |
| char *objdescr, PyObject *obj, |
| char *extra_error_line) |
| { |
| /* like PyErr_WriteUnraisable(), but write a full traceback */ |
| #ifdef USE_WRITEUNRAISABLEMSG |
| |
| /* PyErr_WriteUnraisable actually writes the full traceback anyway |
| from Python 3.4, but we can't really get the formatting of the |
| custom text to be what we want. We can do better from Python |
| 3.8 by calling the new _PyErr_WriteUnraisableMsg(). |
| Luckily it's also Python 3.8 that adds new functionality that |
| people might want: the new sys.unraisablehook(). |
| */ |
| PyObject *s; |
| int first_char; |
| assert(objdescr != NULL && objdescr[0] != 0); /* non-empty */ |
| first_char = objdescr[0]; |
| if (first_char >= 'A' && first_char <= 'Z') |
| first_char += 'a' - 'A'; /* lower() the very first character */ |
| if (extra_error_line == NULL) |
| extra_error_line = ""; |
| |
| if (obj != NULL) |
| s = PyUnicode_FromFormat("%c%s%R%s", |
| first_char, objdescr + 1, obj, extra_error_line); |
| else |
| s = PyUnicode_FromFormat("%c%s%s", |
| first_char, objdescr + 1, extra_error_line); |
| |
| PyErr_Restore(t, v, tb); |
| if (s != NULL) { |
| _PyErr_WriteUnraisableMsg(PyText_AS_UTF8(s), NULL); |
| Py_DECREF(s); |
| } |
| else |
| PyErr_WriteUnraisable(obj); /* best effort */ |
| PyErr_Clear(); |
| |
| #else |
| |
| /* version for Python 2.7 and < 3.8 */ |
| PyObject *f; |
| #if PY_MAJOR_VERSION >= 3 |
| /* jump through hoops to ensure the tb is attached to v, on Python 3 */ |
| PyErr_NormalizeException(&t, &v, &tb); |
| if (tb == NULL) { |
| tb = Py_None; |
| Py_INCREF(tb); |
| } |
| PyException_SetTraceback(v, tb); |
| #endif |
| f = PySys_GetObject("stderr"); |
| if (f != NULL) { |
| if (obj != NULL) { |
| PyFile_WriteString(objdescr, f); |
| PyFile_WriteObject(obj, f, 0); |
| PyFile_WriteString(":\n", f); |
| } |
| if (extra_error_line != NULL) |
| PyFile_WriteString(extra_error_line, f); |
| PyErr_Display(t, v, tb); |
| } |
| Py_XDECREF(t); |
| Py_XDECREF(v); |
| Py_XDECREF(tb); |
| |
| #endif |
| } |
| |
| static void general_invoke_callback(int decode_args_from_libffi, |
| void *result, char *args, void *userdata) |
| { |
| PyObject *cb_args = (PyObject *)userdata; |
| CTypeDescrObject *ct = (CTypeDescrObject *)PyTuple_GET_ITEM(cb_args, 0); |
| PyObject *signature = ct->ct_stuff; |
| PyObject *py_ob = PyTuple_GET_ITEM(cb_args, 1); |
| PyObject *py_args = NULL; |
| PyObject *py_res = NULL; |
| PyObject *py_rawerr; |
| PyObject *onerror_cb; |
| Py_ssize_t i, n; |
| char *extra_error_line = NULL; |
| |
| #define SIGNATURE(i) ((CTypeDescrObject *)PyTuple_GET_ITEM(signature, i)) |
| |
| Py_INCREF(cb_args); |
| |
| n = PyTuple_GET_SIZE(signature) - 2; |
| py_args = PyTuple_New(n); |
| if (py_args == NULL) |
| goto error; |
| |
| for (i=0; i<n; i++) { |
| char *a_src; |
| PyObject *a; |
| CTypeDescrObject *a_ct = SIGNATURE(2 + i); |
| |
| if (decode_args_from_libffi) { |
| a_src = ((void **)args)[i]; |
| } |
| else { |
| a_src = args + i * 8; |
| if (a_ct->ct_flags & (CT_IS_LONGDOUBLE | CT_STRUCT | CT_UNION)) |
| a_src = *(char **)a_src; |
| } |
| a = convert_to_object(a_src, a_ct); |
| if (a == NULL) |
| goto error; |
| PyTuple_SET_ITEM(py_args, i, a); |
| } |
| |
| py_res = PyObject_Call(py_ob, py_args, NULL); |
| if (py_res == NULL) |
| goto error; |
| if (convert_from_object_fficallback(result, SIGNATURE(1), py_res, |
| decode_args_from_libffi) < 0) { |
| #ifdef USE_WRITEUNRAISABLEMSG |
| extra_error_line = ", trying to convert the result back to C"; |
| #else |
| extra_error_line = "Trying to convert the result back to C:\n"; |
| #endif |
| goto error; |
| } |
| done: |
| Py_XDECREF(py_args); |
| Py_XDECREF(py_res); |
| Py_DECREF(cb_args); |
| return; |
| |
| error: |
| if (SIGNATURE(1)->ct_size > 0) { |
| py_rawerr = PyTuple_GET_ITEM(cb_args, 2); |
| memcpy(result, PyBytes_AS_STRING(py_rawerr), |
| PyBytes_GET_SIZE(py_rawerr)); |
| } |
| onerror_cb = PyTuple_GET_ITEM(cb_args, 3); |
| if (onerror_cb == Py_None) { |
| PyObject *ecap, *t, *v, *tb; |
| PyErr_Fetch(&t, &v, &tb); |
| ecap = _cffi_start_error_capture(); |
| _my_PyErr_WriteUnraisable(t, v, tb, "From cffi callback ", py_ob, |
| extra_error_line); |
| _cffi_stop_error_capture(ecap); |
| } |
| else { |
| PyObject *exc1, *val1, *tb1, *res1, *exc2, *val2, *tb2; |
| PyErr_Fetch(&exc1, &val1, &tb1); |
| PyErr_NormalizeException(&exc1, &val1, &tb1); |
| res1 = PyObject_CallFunctionObjArgs(onerror_cb, |
| exc1 ? exc1 : Py_None, |
| val1 ? val1 : Py_None, |
| tb1 ? tb1 : Py_None, |
| NULL); |
| if (res1 != NULL) { |
| if (res1 != Py_None) |
| convert_from_object_fficallback(result, SIGNATURE(1), res1, |
| decode_args_from_libffi); |
| Py_DECREF(res1); |
| } |
| if (!PyErr_Occurred()) { |
| Py_XDECREF(exc1); |
| Py_XDECREF(val1); |
| Py_XDECREF(tb1); |
| } |
| else { |
| /* double exception! print a double-traceback... */ |
| PyObject *ecap; |
| PyErr_Fetch(&exc2, &val2, &tb2); |
| ecap = _cffi_start_error_capture(); |
| _my_PyErr_WriteUnraisable(exc1, val1, tb1, |
| "From cffi callback ", py_ob, |
| extra_error_line); |
| #ifdef USE_WRITEUNRAISABLEMSG |
| _my_PyErr_WriteUnraisable(exc2, val2, tb2, |
| "during handling of the above exception by 'onerror'", |
| NULL, NULL); |
| #else |
| extra_error_line = ("\nDuring the call to 'onerror', " |
| "another exception occurred:\n\n"); |
| _my_PyErr_WriteUnraisable(exc2, val2, tb2, |
| NULL, NULL, extra_error_line); |
| #endif |
| _cffi_stop_error_capture(ecap); |
| } |
| } |
| goto done; |
| |
| #undef SIGNATURE |
| } |
| |
| static void invoke_callback(ffi_cif *cif, void *result, void **args, |
| void *userdata) |
| { |
| save_errno(); |
| { |
| PyGILState_STATE state = gil_ensure(); |
| general_invoke_callback(1, result, (char *)args, userdata); |
| gil_release(state); |
| } |
| restore_errno(); |
| } |
| |
| static PyObject *prepare_callback_info_tuple(CTypeDescrObject *ct, |
| PyObject *ob, |
| PyObject *error_ob, |
| PyObject *onerror_ob, |
| int decode_args_from_libffi) |
| { |
| CTypeDescrObject *ctresult; |
| PyObject *py_rawerr, *infotuple; |
| Py_ssize_t size; |
| |
| if (!(ct->ct_flags & CT_FUNCTIONPTR)) { |
| PyErr_Format(PyExc_TypeError, "expected a function ctype, got '%s'", |
| ct->ct_name); |
| return NULL; |
| } |
| if (!PyCallable_Check(ob)) { |
| PyErr_Format(PyExc_TypeError, |
| "expected a callable object, not %.200s", |
| Py_TYPE(ob)->tp_name); |
| return NULL; |
| } |
| if (onerror_ob != Py_None && !PyCallable_Check(onerror_ob)) { |
| PyErr_Format(PyExc_TypeError, |
| "expected a callable object for 'onerror', not %.200s", |
| Py_TYPE(onerror_ob)->tp_name); |
| return NULL; |
| } |
| |
| ctresult = (CTypeDescrObject *)PyTuple_GET_ITEM(ct->ct_stuff, 1); |
| size = ctresult->ct_size; |
| if (size < (Py_ssize_t)sizeof(ffi_arg)) |
| size = sizeof(ffi_arg); |
| py_rawerr = PyBytes_FromStringAndSize(NULL, size); |
| if (py_rawerr == NULL) |
| return NULL; |
| memset(PyBytes_AS_STRING(py_rawerr), 0, size); |
| if (error_ob != Py_None) { |
| if (convert_from_object_fficallback( |
| PyBytes_AS_STRING(py_rawerr), ctresult, error_ob, |
| decode_args_from_libffi) < 0) { |
| Py_DECREF(py_rawerr); |
| return NULL; |
| } |
| } |
| infotuple = Py_BuildValue("OOOO", ct, ob, py_rawerr, onerror_ob); |
| Py_DECREF(py_rawerr); |
| |
| #if defined(WITH_THREAD) && PY_VERSION_HEX < 0x03070000 |
| /* We must setup the GIL here, in case the callback is invoked in |
| some other non-Pythonic thread. This is the same as ctypes. |
| But PyEval_InitThreads() is always a no-op from CPython 3.7 |
| (the call from ctypes was removed some time later I think). */ |
| PyEval_InitThreads(); |
| #endif |
| |
| return infotuple; |
| } |
| |
| /* messily try to silence a gcc/clang deprecation warning for |
| ffi_prep_closure. Don't miss the "pragma pop" after the function. |
| This is done around the whole function because very old GCCs don't |
| support it inside a function. */ |
| #if defined(__clang__) |
| # pragma clang diagnostic push |
| # pragma clang diagnostic ignored "-Wdeprecated-declarations" |
| #elif defined(__GNUC__) |
| # pragma GCC diagnostic push |
| # pragma GCC diagnostic ignored "-Wdeprecated-declarations" |
| #endif |
| static PyObject *b_callback(PyObject *self, PyObject *args) |
| { |
| CTypeDescrObject *ct; |
| CDataObject_closure *cd; |
| PyObject *ob, *error_ob = Py_None, *onerror_ob = Py_None; |
| PyObject *infotuple; |
| cif_description_t *cif_descr; |
| ffi_closure *closure; |
| ffi_status status; |
| void *closure_exec; |
| |
| if (!PyArg_ParseTuple(args, "O!O|OO:callback", &CTypeDescr_Type, &ct, &ob, |
| &error_ob, &onerror_ob)) |
| return NULL; |
| |
| infotuple = prepare_callback_info_tuple(ct, ob, error_ob, onerror_ob, 1); |
| if (infotuple == NULL) |
| return NULL; |
| |
| #if CFFI_CHECK_FFI_CLOSURE_ALLOC_MAYBE |
| if (CFFI_CHECK_FFI_CLOSURE_ALLOC) { |
| closure = ffi_closure_alloc(sizeof(ffi_closure), &closure_exec); |
| } else |
| #endif |
| { |
| closure = cffi_closure_alloc(); |
| closure_exec = closure; |
| } |
| |
| if (closure == NULL) { |
| Py_DECREF(infotuple); |
| PyErr_SetString(PyExc_MemoryError, |
| "Cannot allocate write+execute memory for ffi.callback(). " |
| "You might be running on a system that prevents this. " |
| "For more information, see " |
| "https://cffi.readthedocs.io/en/latest/using.html#callbacks"); |
| return NULL; |
| } |
| cd = PyObject_GC_New(CDataObject_closure, &CDataOwningGC_Type); |
| if (cd == NULL) |
| goto error; |
| Py_INCREF(ct); |
| cd->head.c_type = ct; |
| cd->head.c_data = (char *)closure_exec; |
| cd->head.c_weakreflist = NULL; |
| closure->user_data = NULL; |
| cd->closure = closure; |
| |
| cif_descr = (cif_description_t *)ct->ct_extra; |
| if (cif_descr == NULL) { |
| PyErr_Format(PyExc_NotImplementedError, |
| "%s: callback with unsupported argument or " |
| "return type or with '...'", ct->ct_name); |
| goto error; |
| } |
| |
| #if CFFI_CHECK_FFI_PREP_CLOSURE_LOC_MAYBE |
| if (CFFI_CHECK_FFI_PREP_CLOSURE_LOC) { |
| status = ffi_prep_closure_loc(closure, &cif_descr->cif, |
| invoke_callback, infotuple, closure_exec); |
| } |
| else |
| #endif |
| { |
| #if defined(__APPLE__) && defined(FFI_AVAILABLE_APPLE) && !FFI_LEGACY_CLOSURE_API |
| PyErr_Format(PyExc_SystemError, "ffi_prep_closure_loc() is missing"); |
| goto error; |
| #else |
| status = ffi_prep_closure(closure, &cif_descr->cif, |
| invoke_callback, infotuple); |
| #endif |
| } |
| |
| if (status != FFI_OK) { |
| PyErr_SetString(PyExc_SystemError, |
| "libffi failed to build this callback"); |
| goto error; |
| } |
| |
| if (closure->user_data != infotuple) { |
| /* Issue #266. Should not occur, but could, if we are using |
| at runtime a version of libffi compiled with a different |
| 'ffi_closure' structure than the one we expect from ffi.h |
| (e.g. difference in details of the platform): a difference |
| in FFI_TRAMPOLINE_SIZE means that the 'user_data' field |
| ends up somewhere else, and so the test above fails. |
| */ |
| PyErr_SetString(PyExc_SystemError, |
| "ffi_prep_closure(): bad user_data (it seems that the " |
| "version of the libffi library seen at runtime is " |
| "different from the 'ffi.h' file seen at compile-time)"); |
| goto error; |
| } |
| PyObject_GC_Track(cd); |
| return (PyObject *)cd; |
| |
| error: |
| closure->user_data = NULL; |
| if (cd == NULL) { |
| #if CFFI_CHECK_FFI_CLOSURE_ALLOC_MAYBE |
| if (CFFI_CHECK_FFI_CLOSURE_ALLOC) { |
| ffi_closure_free(closure); |
| } |
| else |
| #endif |
| cffi_closure_free(closure); |
| } |
| else |
| Py_DECREF(cd); |
| Py_XDECREF(infotuple); |
| return NULL; |
| } |
| #if defined(__clang__) |
| # pragma clang diagnostic pop |
| #elif defined(__GNUC__) |
| # pragma GCC diagnostic pop |
| #endif |
| |
| static PyObject *b_new_enum_type(PyObject *self, PyObject *args) |
| { |
| char *ename; |
| PyObject *enumerators, *enumvalues; |
| PyObject *dict1 = NULL, *dict2 = NULL, *combined = NULL, *tmpkey = NULL; |
| int name_size; |
| CTypeDescrObject *td, *basetd; |
| Py_ssize_t i, n; |
| |
| if (!PyArg_ParseTuple(args, "sO!O!O!:new_enum_type", |
| &ename, |
| &PyTuple_Type, &enumerators, |
| &PyTuple_Type, &enumvalues, |
| &CTypeDescr_Type, &basetd)) |
| return NULL; |
| |
| n = PyTuple_GET_SIZE(enumerators); |
| if (n != PyTuple_GET_SIZE(enumvalues)) { |
| PyErr_SetString(PyExc_ValueError, |
| "tuple args must have the same size"); |
| return NULL; |
| } |
| |
| if (!(basetd->ct_flags & (CT_PRIMITIVE_SIGNED|CT_PRIMITIVE_UNSIGNED))) { |
| PyErr_SetString(PyExc_TypeError, |
| "expected a primitive signed or unsigned base type"); |
| return NULL; |
| } |
| |
| dict1 = PyDict_New(); |
| if (dict1 == NULL) |
| goto error; |
| dict2 = PyDict_New(); |
| if (dict2 == NULL) |
| goto error; |
| |
| for (i=n; --i >= 0; ) { |
| long long lvalue; |
| PyObject *value = PyTuple_GET_ITEM(enumvalues, i); |
| tmpkey = PyTuple_GET_ITEM(enumerators, i); |
| Py_INCREF(tmpkey); |
| if (!PyText_Check(tmpkey)) { |
| #if PY_MAJOR_VERSION < 3 |
| if (PyUnicode_Check(tmpkey)) { |
| const char *text = PyText_AsUTF8(tmpkey); |
| if (text == NULL) |
| goto error; |
| Py_DECREF(tmpkey); |
| tmpkey = PyString_FromString(text); |
| if (tmpkey == NULL) |
| goto error; |
| } |
| else |
| #endif |
| { |
| PyErr_SetString(PyExc_TypeError, |
| "enumerators must be a list of strings"); |
| goto error; |
| } |
| } |
| if (convert_from_object((char*)&lvalue, basetd, value) < 0) |
| goto error; /* out-of-range or badly typed 'value' */ |
| if (PyDict_SetItem(dict1, tmpkey, value) < 0) |
| goto error; |
| if (PyDict_SetItem(dict2, value, tmpkey) < 0) |
| goto error; |
| Py_DECREF(tmpkey); |
| tmpkey = NULL; |
| } |
| |
| combined = PyTuple_Pack(2, dict1, dict2); |
| if (combined == NULL) |
| goto error; |
| |
| Py_CLEAR(dict2); |
| Py_CLEAR(dict1); |
| |
| name_size = strlen(ename) + 1; |
| td = ctypedescr_new(name_size); |
| if (td == NULL) |
| goto error; |
| |
| memcpy(td->ct_name, ename, name_size); |
| td->ct_stuff = combined; |
| td->ct_size = basetd->ct_size; |
| td->ct_length = basetd->ct_length; /* alignment */ |
| td->ct_extra = basetd->ct_extra; /* ffi type */ |
| td->ct_flags = basetd->ct_flags | CT_IS_ENUM; |
| td->ct_name_position = name_size - 1; |
| return (PyObject *)td; |
| |
| error: |
| Py_XDECREF(tmpkey); |
| Py_XDECREF(combined); |
| Py_XDECREF(dict2); |
| Py_XDECREF(dict1); |
| return NULL; |
| } |
| |
| static PyObject *b_alignof(PyObject *self, PyObject *arg) |
| { |
| int align; |
| if (!CTypeDescr_Check(arg)) { |
| PyErr_SetString(PyExc_TypeError, "expected a 'ctype' object"); |
| return NULL; |
| } |
| align = get_alignment((CTypeDescrObject *)arg); |
| if (align < 0) |
| return NULL; |
| return PyInt_FromLong(align); |
| } |
| |
| static Py_ssize_t direct_sizeof_cdata(CDataObject *cd) |
| { |
| Py_ssize_t size; |
| if (cd->c_type->ct_flags & CT_ARRAY) |
| size = get_array_length(cd) * cd->c_type->ct_itemdescr->ct_size; |
| else { |
| size = -1; |
| if (cd->c_type->ct_flags & (CT_STRUCT | CT_UNION)) |
| size = _cdata_var_byte_size(cd); |
| if (size < 0) |
| size = cd->c_type->ct_size; |
| } |
| return size; |
| } |
| |
| static PyObject *b_sizeof(PyObject *self, PyObject *arg) |
| { |
| Py_ssize_t size; |
| |
| if (CData_Check(arg)) { |
| size = direct_sizeof_cdata((CDataObject *)arg); |
| } |
| else if (CTypeDescr_Check(arg)) { |
| size = ((CTypeDescrObject *)arg)->ct_size; |
| if (size < 0) { |
| PyErr_Format(PyExc_ValueError, "ctype '%s' is of unknown size", |
| ((CTypeDescrObject *)arg)->ct_name); |
| return NULL; |
| } |
| } |
| else { |
| PyErr_SetString(PyExc_TypeError, |
| "expected a 'cdata' or 'ctype' object"); |
| return NULL; |
| } |
| return PyInt_FromSsize_t(size); |
| } |
| |
| static PyObject *b_typeof(PyObject *self, PyObject *arg) |
| { |
| PyObject *res; |
| |
| if (!CData_Check(arg)) { |
| PyErr_SetString(PyExc_TypeError, "expected a 'cdata' object"); |
| return NULL; |
| } |
| res = (PyObject *)((CDataObject *)arg)->c_type; |
| Py_INCREF(res); |
| return res; |
| } |
| |
| static CTypeDescrObject *direct_typeoffsetof(CTypeDescrObject *ct, |
| PyObject *fieldname, |
| int following, Py_ssize_t *offset) |
| { |
| /* Does not return a new reference! */ |
| CTypeDescrObject *res; |
| CFieldObject *cf; |
| |
| if (PyTextAny_Check(fieldname)) { |
| if (!following && (ct->ct_flags & CT_POINTER)) |
| ct = ct->ct_itemdescr; |
| if (!(ct->ct_flags & (CT_STRUCT|CT_UNION))) { |
| PyErr_SetString(PyExc_TypeError, |
| "with a field name argument, expected a " |
| "struct or union ctype"); |
| return NULL; |
| } |
| if (force_lazy_struct(ct) <= 0) { |
| if (!PyErr_Occurred()) |
| PyErr_SetString(PyExc_TypeError, "struct/union is opaque"); |
| return NULL; |
| } |
| cf = (CFieldObject *)PyDict_GetItem(ct->ct_stuff, fieldname); |
| if (cf == NULL) { |
| PyErr_SetObject(PyExc_KeyError, fieldname); |
| return NULL; |
| } |
| if (cf->cf_bitshift >= 0) { |
| PyErr_SetString(PyExc_TypeError, "not supported for bitfields"); |
| return NULL; |
| } |
| res = cf->cf_type; |
| *offset = cf->cf_offset; |
| } |
| else { |
| Py_ssize_t index = PyInt_AsSsize_t(fieldname); |
| if (index < 0 && PyErr_Occurred()) { |
| PyErr_SetString(PyExc_TypeError, |
| "field name or array index expected"); |
| return NULL; |
| } |
| |
| if (!(ct->ct_flags & (CT_ARRAY|CT_POINTER)) || |
| ct->ct_itemdescr->ct_size < 0) { |
| PyErr_SetString(PyExc_TypeError, "with an integer argument, " |
| "expected an array ctype or a " |
| "pointer to non-opaque"); |
| return NULL; |
| } |
| res = ct->ct_itemdescr; |
| *offset = MUL_WRAPAROUND(index, ct->ct_itemdescr->ct_size); |
| if ((*offset / ct->ct_itemdescr->ct_size) != index) { |
| PyErr_SetString(PyExc_OverflowError, |
| "array offset would overflow a Py_ssize_t"); |
| return NULL; |
| } |
| } |
| return res; |
| } |
| |
| static PyObject *b_typeoffsetof(PyObject *self, PyObject *args) |
| { |
| PyObject *res, *fieldname; |
| CTypeDescrObject *ct; |
| Py_ssize_t offset; |
| int following = 0; |
| |
| if (!PyArg_ParseTuple(args, "O!O|i:typeoffsetof", |
| &CTypeDescr_Type, &ct, &fieldname, &following)) |
| return NULL; |
| |
| res = (PyObject *)direct_typeoffsetof(ct, fieldname, following, &offset); |
| if (res == NULL) |
| return NULL; |
| |
| return Py_BuildValue("(On)", res, offset); |
| } |
| |
| static PyObject *b_rawaddressof(PyObject *self, PyObject *args) |
| { |
| CTypeDescrObject *ct; |
| CDataObject *cd; |
| Py_ssize_t offset; |
| int accepted_flags; |
| |
| if (!PyArg_ParseTuple(args, "O!O!n:rawaddressof", |
| &CTypeDescr_Type, &ct, |
| &CData_Type, &cd, |
| &offset)) |
| return NULL; |
| |
| accepted_flags = CT_STRUCT | CT_UNION | CT_ARRAY | CT_POINTER; |
| if ((cd->c_type->ct_flags & accepted_flags) == 0) { |
| PyErr_SetString(PyExc_TypeError, |
| "expected a cdata struct/union/array/pointer object"); |
| return NULL; |
| } |
| if ((ct->ct_flags & CT_POINTER) == 0) { |
| PyErr_SetString(PyExc_TypeError, |
| "expected a pointer ctype"); |
| return NULL; |
| } |
| return new_simple_cdata(cd->c_data + offset, ct); |
| } |
| |
| static PyObject *b_getcname(PyObject *self, PyObject *args) |
| { |
| CTypeDescrObject *ct; |
| char *replace_with, *p, *s; |
| Py_ssize_t namelen, replacelen; |
| |
| if (!PyArg_ParseTuple(args, "O!s:getcname", |
| &CTypeDescr_Type, &ct, &replace_with)) |
| return NULL; |
| |
| namelen = strlen(ct->ct_name); |
| replacelen = strlen(replace_with); |
| s = p = alloca(namelen + replacelen + 1); |
| memcpy(p, ct->ct_name, ct->ct_name_position); |
| p += ct->ct_name_position; |
| memcpy(p, replace_with, replacelen); |
| p += replacelen; |
| memcpy(p, ct->ct_name + ct->ct_name_position, |
| namelen - ct->ct_name_position); |
| |
| return PyText_FromStringAndSize(s, namelen + replacelen); |
| } |
| |
| static PyObject *b_string(PyObject *self, PyObject *args, PyObject *kwds) |
| { |
| CDataObject *cd; |
| Py_ssize_t maxlen = -1; |
| static char *keywords[] = {"cdata", "maxlen", NULL}; |
| |
| if (!PyArg_ParseTupleAndKeywords(args, kwds, "O!|n:string", keywords, |
| &CData_Type, &cd, &maxlen)) |
| return NULL; |
| |
| if (cd->c_type->ct_itemdescr != NULL && |
| cd->c_type->ct_itemdescr->ct_flags & (CT_PRIMITIVE_CHAR | |
| CT_PRIMITIVE_SIGNED | |
| CT_PRIMITIVE_UNSIGNED) && |
| !(cd->c_type->ct_itemdescr->ct_flags & CT_IS_BOOL)) { |
| Py_ssize_t length = maxlen; |
| if (cd->c_data == NULL) { |
| PyObject *s = cdata_repr(cd); |
| if (s != NULL) { |
| PyErr_Format(PyExc_RuntimeError, |
| "cannot use string() on %s", |
| PyText_AS_UTF8(s)); |
| Py_DECREF(s); |
| } |
| return NULL; |
| } |
| if (length < 0 && cd->c_type->ct_flags & CT_ARRAY) { |
| length = get_array_length(cd); |
| } |
| if (cd->c_type->ct_itemdescr->ct_size == sizeof(char)) { |
| const char *start = cd->c_data; |
| if (length < 0) { |
| /*READ(start, 1)*/ |
| length = strlen(start); |
| /*READ(start, length)*/ |
| } |
| else { |
| const char *end; |
| /*READ(start, length)*/ |
| end = (const char *)memchr(start, 0, length); |
| if (end != NULL) |
| length = end - start; |
| } |
| return PyBytes_FromStringAndSize(start, length); |
| } |
| else if (cd->c_type->ct_itemdescr->ct_flags & CT_PRIMITIVE_CHAR) { |
| switch (cd->c_type->ct_itemdescr->ct_size) { |
| case 2: { |
| const cffi_char16_t *start = (cffi_char16_t *)cd->c_data; |
| if (length < 0) { |
| /*READ(start, 2)*/ |
| length = 0; |
| while (start[length]) |
| length++; |
| /*READ(start, 2 * length)*/ |
| } |
| else { |
| /*READ(start, 2 * length)*/ |
| maxlen = length; |
| length = 0; |
| while (length < maxlen && start[length]) |
| length++; |
| } |
| return _my_PyUnicode_FromChar16(start, length); |
| } |
| case 4: { |
| const cffi_char32_t *start = (cffi_char32_t *)cd->c_data; |
| if (length < 0) { |
| /*READ(start, 4)*/ |
| length = 0; |
| while (start[length]) |
| length++; |
| /*READ(start, 4 * length)*/ |
| } |
| else { |
| /*READ(start, 4 * length)*/ |
| maxlen = length; |
| length = 0; |
| while (length < maxlen && start[length]) |
| length++; |
| } |
| return _my_PyUnicode_FromChar32(start, length); |
| } |
| } |
| } |
| } |
| else if (cd->c_type->ct_flags & CT_IS_ENUM) { |
| return convert_cdata_to_enum_string(cd, 0); |
| } |
| else if (cd->c_type->ct_flags & CT_IS_BOOL) { |
| /* fall through to TypeError */ |
| } |
| else if (cd->c_type->ct_flags & (CT_PRIMITIVE_CHAR | |
| CT_PRIMITIVE_SIGNED | |
| CT_PRIMITIVE_UNSIGNED)) { |
| /*READ(cd->c_data, cd->c_type->ct_size)*/ |
| if (cd->c_type->ct_size == sizeof(char)) |
| return PyBytes_FromStringAndSize(cd->c_data, 1); |
| else if (cd->c_type->ct_flags & CT_PRIMITIVE_CHAR) { |
| switch (cd->c_type->ct_size) { |
| case 2: |
| return _my_PyUnicode_FromChar16((cffi_char16_t *)cd->c_data, 1); |
| case 4: |
| return _my_PyUnicode_FromChar32((cffi_char32_t *)cd->c_data, 1); |
| } |
| } |
| } |
| PyErr_Format(PyExc_TypeError, "string(): unexpected cdata '%s' argument", |
| cd->c_type->ct_name); |
| return NULL; |
| } |
| |
| static PyObject *b_unpack(PyObject *self, PyObject *args, PyObject *kwds) |
| { |
| CDataObject *cd; |
| CTypeDescrObject *ctitem; |
| Py_ssize_t i, length, itemsize; |
| PyObject *result; |
| char *src; |
| int casenum; |
| static char *keywords[] = {"cdata", "length", NULL}; |
| |
| if (!PyArg_ParseTupleAndKeywords(args, kwds, "O!n:unpack", keywords, |
| &CData_Type, &cd, &length)) |
| return NULL; |
| |
| if (!(cd->c_type->ct_flags & (CT_ARRAY|CT_POINTER))) { |
| PyErr_Format(PyExc_TypeError, |
| "expected a pointer or array, got '%s'", |
| cd->c_type->ct_name); |
| return NULL; |
| } |
| if (length < 0) { |
| PyErr_SetString(PyExc_ValueError, "'length' cannot be negative"); |
| return NULL; |
| } |
| if (cd->c_data == NULL) { |
| PyObject *s = cdata_repr(cd); |
| if (s != NULL) { |
| PyErr_Format(PyExc_RuntimeError, |
| "cannot use unpack() on %s", |
| PyText_AS_UTF8(s)); |
| Py_DECREF(s); |
| } |
| return NULL; |
| } |
| |
| /* byte- and unicode strings */ |
| ctitem = cd->c_type->ct_itemdescr; |
| if (ctitem->ct_flags & CT_PRIMITIVE_CHAR) { |
| switch (ctitem->ct_size) { |
| case sizeof(char): |
| return PyBytes_FromStringAndSize(cd->c_data, length); |
| case 2: |
| return _my_PyUnicode_FromChar16((cffi_char16_t *)cd->c_data,length); |
| case 4: |
| return _my_PyUnicode_FromChar32((cffi_char32_t *)cd->c_data,length); |
| } |
| } |
| |
| /* else, the result is a list. This implementation should be |
| equivalent to but much faster than '[p[i] for i in range(length)]'. |
| (Note that on PyPy, 'list(p[0:length])' should be equally fast, |
| but arguably, finding out that there *is* such an unexpected way |
| to write things down is the real problem.) |
| */ |
| result = PyList_New(length); |
| if (result == NULL) |
| return NULL; |
| |
| src = cd->c_data; |
| itemsize = ctitem->ct_size; |
| if (itemsize < 0) { |
| Py_DECREF(result); |
| PyErr_Format(PyExc_ValueError, "'%s' points to items of unknown size", |
| cd->c_type->ct_name); |
| return NULL; |
| } |
| |
| /* Determine some common fast-paths for the loop below. The case -1 |
| is the fall-back, which always gives the right answer. */ |
| |
| #define ALIGNMENT_CHECK(align) \ |
| (((align) & ((align) - 1)) == 0 && \ |
| (((uintptr_t)src) & ((align) - 1)) == 0) |
| |
| casenum = -1; |
| |
| if ((ctitem->ct_flags & CT_PRIMITIVE_ANY) && |
| ALIGNMENT_CHECK(ctitem->ct_length)) { |
| /* Source data is fully aligned; we can directly read without |
| memcpy(). The unaligned case is expected to be rare; in |
| this situation it is ok to fall back to the general |
| convert_to_object() in the loop. For now we also use this |
| fall-back for types that are too large. |
| */ |
| if (ctitem->ct_flags & CT_PRIMITIVE_SIGNED) { |
| if (itemsize == sizeof(long)) casenum = 3; |
| else if (itemsize == sizeof(int)) casenum = 2; |
| else if (itemsize == sizeof(short)) casenum = 1; |
| else if (itemsize == sizeof(signed char)) casenum = 0; |
| } |
| else if (ctitem->ct_flags & CT_PRIMITIVE_UNSIGNED) { |
| /* Note: we never pick case 6 if sizeof(int) == sizeof(long), |
| so that case 6 below can assume that the 'unsigned int' result |
| would always fit in a 'signed long'. */ |
| if (ctitem->ct_flags & CT_IS_BOOL) casenum = 11; |
| else if (itemsize == sizeof(unsigned long)) casenum = 7; |
| else if (itemsize == sizeof(unsigned int)) casenum = 6; |
| else if (itemsize == sizeof(unsigned short)) casenum = 5; |
| else if (itemsize == sizeof(unsigned char)) casenum = 4; |
| } |
| else if (ctitem->ct_flags & CT_PRIMITIVE_FLOAT) { |
| if (itemsize == sizeof(double)) casenum = 9; |
| else if (itemsize == sizeof(float)) casenum = 8; |
| } |
| } |
| else if (ctitem->ct_flags & (CT_POINTER | CT_FUNCTIONPTR)) { |
| casenum = 10; /* any pointer */ |
| } |
| #undef ALIGNMENT_CHECK |
| |
| for (i = 0; i < length; i++) { |
| PyObject *x; |
| switch (casenum) { |
| /* general case */ |
| default: x = convert_to_object(src, ctitem); break; |
| |
| /* special cases for performance only */ |
| case 0: x = PyInt_FromLong(*(signed char *)src); break; |
| case 1: x = PyInt_FromLong(*(short *)src); break; |
| case 2: x = PyInt_FromLong(*(int *)src); break; |
| case 3: x = PyInt_FromLong(*(long *)src); break; |
| case 4: x = PyInt_FromLong(*(unsigned char *)src); break; |
| case 5: x = PyInt_FromLong(*(unsigned short *)src); break; |
| case 6: x = PyInt_FromLong((long)*(unsigned int *)src); break; |
| case 7: x = PyLong_FromUnsignedLong(*(unsigned long *)src); break; |
| case 8: x = PyFloat_FromDouble(*(float *)src); break; |
| case 9: x = PyFloat_FromDouble(*(double *)src); break; |
| case 10: x = new_simple_cdata(*(char **)src, ctitem); break; |
| case 11: |
| switch (*(unsigned char *)src) { |
| case 0: x = Py_False; Py_INCREF(x); break; |
| case 1: x = Py_True; Py_INCREF(x); break; |
| default: x = convert_to_object(src, ctitem); /* error */ |
| } |
| break; |
| } |
| if (x == NULL) { |
| Py_DECREF(result); |
| return NULL; |
| } |
| PyList_SET_ITEM(result, i, x); |
| src += itemsize; |
| } |
| return result; |
| } |
| |
| static PyObject * |
| b_buffer_new(PyTypeObject *type, PyObject *args, PyObject *kwds) |
| { |
| /* this is the constructor of the type implemented in minibuffer.h */ |
| CDataObject *cd; |
| Py_ssize_t size = -1; |
| static char *keywords[] = {"cdata", "size", NULL}; |
| |
| if (!PyArg_ParseTupleAndKeywords(args, kwds, "O!|n:buffer", keywords, |
| &CData_Type, &cd, &size)) |
| return NULL; |
| |
| if (size < 0) |
| size = _cdata_var_byte_size(cd); |
| |
| if (cd->c_type->ct_flags & CT_POINTER) { |
| if (size < 0) |
| size = cd->c_type->ct_itemdescr->ct_size; |
| } |
| else if (cd->c_type->ct_flags & CT_ARRAY) { |
| if (size < 0) |
| size = get_array_length(cd) * cd->c_type->ct_itemdescr->ct_size; |
| } |
| else { |
| PyErr_Format(PyExc_TypeError, |
| "expected a pointer or array cdata, got '%s'", |
| cd->c_type->ct_name); |
| return NULL; |
| } |
| if (size < 0) { |
| PyErr_Format(PyExc_TypeError, |
| "don't know the size pointed to by '%s'", |
| cd->c_type->ct_name); |
| return NULL; |
| } |
| /*WRITE(cd->c_data, size)*/ |
| return minibuffer_new(cd->c_data, size, (PyObject *)cd); |
| } |
| |
| static PyObject *b_get_errno(PyObject *self, PyObject *noarg) |
| { |
| int err; |
| restore_errno_only(); |
| err = errno; |
| errno = 0; |
| return PyInt_FromLong(err); |
| } |
| |
| static PyObject *b_set_errno(PyObject *self, PyObject *arg) |
| { |
| long ival = PyInt_AsLong(arg); |
| if (ival == -1 && PyErr_Occurred()) |
| return NULL; |
| else if (ival < INT_MIN || ival > INT_MAX) { |
| PyErr_SetString(PyExc_OverflowError, "errno value too large"); |
| return NULL; |
| } |
| errno = (int)ival; |
| save_errno_only(); |
| errno = 0; |
| Py_INCREF(Py_None); |
| return Py_None; |
| } |
| |
| static PyObject *newp_handle(CTypeDescrObject *ct_voidp, PyObject *x) |
| { |
| CDataObject_own_structptr *cd; |
| cd = (CDataObject_own_structptr *)PyObject_GC_New(CDataObject_own_structptr, |
| &CDataOwningGC_Type); |
| if (cd == NULL) |
| return NULL; |
| Py_INCREF(ct_voidp); /* must be "void *" */ |
| cd->head.c_type = ct_voidp; |
| cd->head.c_data = (char *)cd; |
| cd->head.c_weakreflist = NULL; |
| Py_INCREF(x); |
| cd->structobj = x; |
| PyObject_GC_Track(cd); |
| return (PyObject *)cd; |
| } |
| |
| static PyObject *b_newp_handle(PyObject *self, PyObject *args) |
| { |
| CTypeDescrObject *ct; |
| PyObject *x; |
| if (!PyArg_ParseTuple(args, "O!O", &CTypeDescr_Type, &ct, &x)) |
| return NULL; |
| |
| if (!(ct->ct_flags & CT_IS_VOID_PTR)) { |
| PyErr_Format(PyExc_TypeError, "needs 'void *', got '%s'", ct->ct_name); |
| return NULL; |
| } |
| |
| return newp_handle(ct, x); |
| } |
| |
| static PyObject *b_from_handle(PyObject *self, PyObject *arg) |
| { |
| CTypeDescrObject *ct; |
| CDataObject_own_structptr *orgcd; |
| PyObject *x; |
| if (!CData_Check(arg)) { |
| PyErr_SetString(PyExc_TypeError, "expected a 'cdata' object"); |
| return NULL; |
| } |
| ct = ((CDataObject *)arg)->c_type; |
| if (!(ct->ct_flags & CT_IS_VOIDCHAR_PTR)) { |
| PyErr_Format(PyExc_TypeError, |
| "expected a 'cdata' object with a 'void *' out of " |
| "new_handle(), got '%s'", ct->ct_name); |
| return NULL; |
| } |
| orgcd = (CDataObject_own_structptr *)((CDataObject *)arg)->c_data; |
| if (!orgcd) { |
| PyErr_SetString(PyExc_RuntimeError, |
| "cannot use from_handle() on NULL pointer"); |
| return NULL; |
| } |
| if (Py_REFCNT(orgcd) <= 0 || Py_TYPE(orgcd) != &CDataOwningGC_Type) { |
| Py_FatalError("ffi.from_handle() detected that the address passed " |
| "points to garbage. If it is really the result of " |
| "ffi.new_handle(), then the Python object has already " |
| "been garbage collected"); |
| } |
| x = orgcd->structobj; |
| Py_INCREF(x); |
| return x; |
| } |
| |
| static int _my_PyObject_GetContiguousBuffer(PyObject *x, Py_buffer *view, |
| int writable_only) |
| { |
| #if PY_MAJOR_VERSION < 3 |
| /* Some objects only support the buffer interface and CPython doesn't |
| translate it into the memoryview interface, mess. Hack a very |
| minimal content for 'view'. Don't care if the other fields are |
| uninitialized: we only call PyBuffer_Release(), which only reads |
| 'view->obj'. */ |
| PyBufferProcs *pb = x->ob_type->tp_as_buffer; |
| if (pb && !pb->bf_releasebuffer) { |
| /* we used to try all three in some vaguely sensible order, |
| i.e. first the write. But trying to call the write on a |
| read-only buffer fails with TypeError. So we use a less- |
| sensible order now. See test_from_buffer_more_cases. |
| |
| If 'writable_only', we only try bf_getwritebuffer. |
| */ |
| readbufferproc proc = NULL; |
| if (!writable_only) { |
| proc = (readbufferproc)pb->bf_getreadbuffer; |
| if (!proc) |
| proc = (readbufferproc)pb->bf_getcharbuffer; |
| } |
| if (!proc) |
| proc = (readbufferproc)pb->bf_getwritebuffer; |
| |
| if (proc && pb->bf_getsegcount) { |
| if ((*pb->bf_getsegcount)(x, NULL) != 1) { |
| PyErr_SetString(PyExc_TypeError, |
| "expected a single-segment buffer object"); |
| return -1; |
| } |
| view->len = (*proc)(x, 0, &view->buf); |
| if (view->len < 0) |
| return -1; |
| view->obj = x; |
| Py_INCREF(x); |
| return 0; |
| } |
| } |
| #endif |
| |
| if (PyObject_GetBuffer(x, view, writable_only ? PyBUF_WRITABLE |
| : PyBUF_SIMPLE) < 0) |
| return -1; |
| |
| if (!PyBuffer_IsContiguous(view, 'A')) { |
| PyBuffer_Release(view); |
| PyErr_SetString(PyExc_TypeError, "contiguous buffer expected"); |
| return -1; |
| } |
| return 0; |
| } |
| |
| static PyObject *direct_from_buffer(CTypeDescrObject *ct, PyObject *x, |
| int require_writable) |
| { |
| CDataObject *cd; |
| Py_buffer *view; |
| Py_ssize_t arraylength, minimumlength = 0; |
| |
| if (!(ct->ct_flags & (CT_ARRAY | CT_POINTER))) { |
| PyErr_Format(PyExc_TypeError, |
| "expected a pointer or array ctype, got '%s'", |
| ct->ct_name); |
| return NULL; |
| } |
| |
| /* PyPy 5.7 can obtain buffers for string (python 2) |
| or bytes (python 3). from_buffer(u"foo") is disallowed. |
| */ |
| if (PyUnicode_Check(x)) { |
| PyErr_SetString(PyExc_TypeError, |
| "from_buffer() cannot return the address " |
| "of a unicode object"); |
| return NULL; |
| } |
| |
| view = PyObject_Malloc(sizeof(Py_buffer)); |
| if (view == NULL) { |
| PyErr_NoMemory(); |
| return NULL; |
| } |
| if (_my_PyObject_GetContiguousBuffer(x, view, require_writable) < 0) |
| goto error1; |
| |
| if (ct->ct_flags & CT_POINTER) |
| { |
| arraylength = view->len; /* number of bytes, not used so far */ |
| } |
| else { |
| /* ct->ct_flags & CT_ARRAY */ |
| if (ct->ct_length >= 0) { |
| /* it's an array with a fixed length; make sure that the |
| buffer contains enough bytes. */ |
| minimumlength = ct->ct_size; |
| arraylength = ct->ct_length; |
| } |
| else { |
| /* it's an open 'array[]' */ |
| if (ct->ct_itemdescr->ct_size == 1) { |
| /* fast path, performance only */ |
| arraylength = view->len; |
| } |
| else if (ct->ct_itemdescr->ct_size > 0) { |
| /* give it as many items as fit the buffer. Ignore a |
| partial last element. */ |
| arraylength = view->len / ct->ct_itemdescr->ct_size; |
| } |
| else { |
| /* it's an array 'empty[]'. Unsupported obscure case: |
| the problem is that setting the length of the result |
| to anything large (like SSIZE_T_MAX) is dangerous, |
| because if someone tries to loop over it, it will |
| turn effectively into an infinite loop. */ |
| PyErr_Format(PyExc_ZeroDivisionError, |
| "from_buffer('%s', ..): the actual length of the array " |
| "cannot be computed", ct->ct_name); |
| goto error2; |
| } |
| } |
| } |
| if (view->len < minimumlength) { |
| PyErr_Format(PyExc_ValueError, |
| "buffer is too small (%zd bytes) for '%s' (%zd bytes)", |
| view->len, ct->ct_name, minimumlength); |
| goto error2; |
| } |
| |
| cd = (CDataObject *)PyObject_GC_New(CDataObject_frombuf, |
| &CDataFromBuf_Type); |
| if (cd == NULL) |
| goto error2; |
| |
| Py_INCREF(ct); |
| cd->c_type = ct; |
| cd->c_data = view->buf; |
| cd->c_weakreflist = NULL; |
| ((CDataObject_frombuf *)cd)->length = arraylength; |
| ((CDataObject_frombuf *)cd)->bufferview = view; |
| PyObject_GC_Track(cd); |
| return (PyObject *)cd; |
| |
| error2: |
| PyBuffer_Release(view); |
| error1: |
| PyObject_Free(view); |
| return NULL; |
| } |
| |
| static PyObject *b_from_buffer(PyObject *self, PyObject *args) |
| { |
| CTypeDescrObject *ct; |
| PyObject *x; |
| int require_writable = 0; |
| |
| if (!PyArg_ParseTuple(args, "O!O|i", &CTypeDescr_Type, &ct, &x, |
| &require_writable)) |
| return NULL; |
| |
| return direct_from_buffer(ct, x, require_writable); |
| } |
| |
| static int _fetch_as_buffer(PyObject *x, Py_buffer *view, int writable_only) |
| { |
| if (CData_Check(x)) { |
| CTypeDescrObject *ct = ((CDataObject *)x)->c_type; |
| if (!(ct->ct_flags & (CT_POINTER|CT_ARRAY))) { |
| PyErr_Format(PyExc_TypeError, |
| "expected a pointer or array ctype, got '%s'", |
| ct->ct_name); |
| return -1; |
| } |
| view->buf = ((CDataObject *)x)->c_data; |
| view->obj = NULL; |
| return 0; |
| } |
| else { |
| return _my_PyObject_GetContiguousBuffer(x, view, writable_only); |
| } |
| } |
| |
| static PyObject *b_memmove(PyObject *self, PyObject *args, PyObject *kwds) |
| { |
| PyObject *dest_obj, *src_obj; |
| Py_buffer dest_view, src_view; |
| Py_ssize_t n; |
| static char *keywords[] = {"dest", "src", "n", NULL}; |
| |
| if (!PyArg_ParseTupleAndKeywords(args, kwds, "OOn", keywords, |
| &dest_obj, &src_obj, &n)) |
| return NULL; |
| if (n < 0) { |
| PyErr_SetString(PyExc_ValueError, "negative size"); |
| return NULL; |
| } |
| |
| if (_fetch_as_buffer(src_obj, &src_view, 0) < 0) { |
| return NULL; |
| } |
| if (_fetch_as_buffer(dest_obj, &dest_view, 1) < 0) { |
| PyBuffer_Release(&src_view); |
| return NULL; |
| } |
| |
| memmove(dest_view.buf, src_view.buf, n); |
| |
| PyBuffer_Release(&dest_view); |
| PyBuffer_Release(&src_view); |
| Py_INCREF(Py_None); |
| return Py_None; |
| } |
| |
| static PyObject *b__get_types(PyObject *self, PyObject *noarg) |
| { |
| return PyTuple_Pack(2, (PyObject *)&CData_Type, |
| (PyObject *)&CTypeDescr_Type); |
| } |
| |
| /* forward, in commontypes.c */ |
| static PyObject *b__get_common_types(PyObject *self, PyObject *arg); |
| |
| static PyObject *b_gcp(PyObject *self, PyObject *args, PyObject *kwds) |
| { |
| CDataObject *cd; |
| CDataObject *origobj; |
| PyObject *destructor; |
| Py_ssize_t ignored; /* for pypy */ |
| static char *keywords[] = {"cdata", "destructor", "size", NULL}; |
| |
| if (!PyArg_ParseTupleAndKeywords(args, kwds, "O!O|n:gc", keywords, |
| &CData_Type, &origobj, &destructor, |
| &ignored)) |
| return NULL; |
| |
| if (destructor == Py_None) { |
| if (!PyObject_TypeCheck(origobj, &CDataGCP_Type)) { |
| PyErr_SetString(PyExc_TypeError, |
| "Can remove destructor only on a object " |
| "previously returned by ffi.gc()"); |
| return NULL; |
| } |
| Py_CLEAR(((CDataObject_gcp *)origobj)->destructor); |
| Py_RETURN_NONE; |
| } |
| |
| cd = allocate_gcp_object(origobj, origobj->c_type, destructor); |
| return (PyObject *)cd; |
| } |
| |
| static PyObject *b_release(PyObject *self, PyObject *arg) |
| { |
| if (!CData_Check(arg)) { |
| PyErr_SetString(PyExc_TypeError, "expected a 'cdata' object"); |
| return NULL; |
| } |
| return cdata_exit(arg, NULL); |
| } |
| |
| /************************************************************/ |
| |
| static char _testfunc0(char a, char b) |
| { |
| return a + b; |
| } |
| static long _testfunc1(int a, long b) |
| { |
| return (long)a + b; |
| } |
| static PY_LONG_LONG _testfunc2(PY_LONG_LONG a, PY_LONG_LONG b) |
| { |
| return a + b; |
| } |
| static double _testfunc3(float a, double b) |
| { |
| return a + b; |
| } |
| static float _testfunc4(float a, double b) |
| { |
| return (float)(a + b); |
| } |
| static void _testfunc5(void) |
| { |
| errno = errno + 15; |
| } |
| static int *_testfunc6(int *x) |
| { |
| static int y; |
| y = *x - 1000; |
| return &y; |
| } |
| struct _testfunc7_s { unsigned char a1; short a2; }; |
| static short _testfunc7(struct _testfunc7_s inlined) |
| { |
| return inlined.a1 + inlined.a2; |
| } |
| static int _testfunc9(int num, ...) |
| { |
| va_list vargs; |
| int i, total = 0; |
| va_start(vargs, num); |
| for (i=0; i<num; i++) { |
| int value = va_arg(vargs, int); |
| if (value == 0) |
| value = -66666666; |
| total += value; |
| } |
| va_end(vargs); |
| return total; |
| } |
| |
| static struct _testfunc7_s _testfunc10(int n) |
| { |
| struct _testfunc7_s result; |
| result.a1 = n; |
| result.a2 = n * n; |
| return result; |
| } |
| |
| struct _testfunc11_s { int a1, a2; }; |
| static struct _testfunc11_s _testfunc11(int n) |
| { |
| struct _testfunc11_s result; |
| result.a1 = n; |
| result.a2 = n * n; |
| return result; |
| } |
| |
| struct _testfunc12_s { double a1; }; |
| static struct _testfunc12_s _testfunc12(int n) |
| { |
| struct _testfunc12_s result; |
| result.a1 = n; |
| return result; |
| } |
| |
| struct _testfunc13_s { int a1, a2, a3; }; |
| static struct _testfunc13_s _testfunc13(int n) |
| { |
| struct _testfunc13_s result; |
| result.a1 = n; |
| result.a2 = n * n; |
| result.a3 = n * n * n; |
| return result; |
| } |
| |
| struct _testfunc14_s { float a1; }; |
| static struct _testfunc14_s _testfunc14(int n) |
| { |
| struct _testfunc14_s result; |
| result.a1 = (float)n; |
| return result; |
| } |
| |
| struct _testfunc15_s { float a1; int a2; }; |
| static struct _testfunc15_s _testfunc15(int n) |
| { |
| struct _testfunc15_s result; |
| result.a1 = (float)n; |
| result.a2 = n * n; |
| return result; |
| } |
| |
| struct _testfunc16_s { float a1, a2; }; |
| static struct _testfunc16_s _testfunc16(int n) |
| { |
| struct _testfunc16_s result; |
| result.a1 = (float)n; |
| result.a2 = -(float)n; |
| return result; |
| } |
| |
| struct _testfunc17_s { int a1; float a2; }; |
| static struct _testfunc17_s _testfunc17(int n) |
| { |
| struct _testfunc17_s result; |
| result.a1 = n; |
| result.a2 = (float)n * (float)n; |
| return result; |
| } |
| |
| static int _testfunc18(struct _testfunc17_s *ptr) |
| { |
| return ptr->a1 + (int)ptr->a2; |
| } |
| |
| static long double _testfunc19(long double x, int count) |
| { |
| int i; |
| for (i=0; i<count; i++) { |
| x = 4*x - x*x; |
| } |
| return x; |
| } |
| |
| static short _testfunc20(struct _testfunc7_s *ptr) |
| { |
| return ptr->a1 + ptr->a2; |
| } |
| |
| struct _testfunc21_s { int a, b, c, d, e, f, g, h, i, j; }; |
| static int _testfunc21(struct _testfunc21_s inlined) |
| { |
| return ((inlined.a << 0) + |
| (inlined.b << 1) + |
| (inlined.c << 2) + |
| (inlined.d << 3) + |
| (inlined.e << 4) + |
| (inlined.f << 5) + |
| (inlined.g << 6) + |
| (inlined.h << 7) + |
| (inlined.i << 8) + |
| (inlined.j << 9)); |
| } |
| |
| struct _testfunc22_s { int a[10]; }; |
| static struct _testfunc22_s _testfunc22(struct _testfunc22_s s1, |
| struct _testfunc22_s s2) |
| { |
| struct _testfunc22_s result; |
| int i; |
| for (i=0; i<10; i++) |
| result.a[i] = s1.a[i] - s2.a[i]; |
| return result; |
| } |
| |
| static int _testfunc23(char *p) |
| { |
| if (p) |
| return 1000 * p[0]; |
| return -42; |
| } |
| |
| #if 0 /* libffi doesn't properly support complexes currently */ |
| /* also, MSVC might not support _Complex... */ |
| /* if this is enabled one day, remember to also add _Complex |
| * arguments in addition to return values. */ |
| static float _Complex _testfunc24(float a, float b) |
| { |
| return a + I*2.0*b; |
| } |
| static double _Complex _testfunc25(double a, double b) |
| { |
| return a + I*2.0*b; |
| } |
| #endif |
| |
| static PyObject *b__testfunc(PyObject *self, PyObject *args) |
| { |
| /* for testing only */ |
| int i; |
| void *f; |
| if (!PyArg_ParseTuple(args, "i:_testfunc", &i)) |
| return NULL; |
| switch (i) { |
| case 0: f = &_testfunc0; break; |
| case 1: f = &_testfunc1; break; |
| case 2: f = &_testfunc2; break; |
| case 3: f = &_testfunc3; break; |
| case 4: f = &_testfunc4; break; |
| case 5: f = &_testfunc5; break; |
| case 6: f = &_testfunc6; break; |
| case 7: f = &_testfunc7; break; |
| case 8: f = stderr; break; |
| case 9: f = &_testfunc9; break; |
| case 10: f = &_testfunc10; break; |
| case 11: f = &_testfunc11; break; |
| case 12: f = &_testfunc12; break; |
| case 13: f = &_testfunc13; break; |
| case 14: f = &_testfunc14; break; |
| case 15: f = &_testfunc15; break; |
| case 16: f = &_testfunc16; break; |
| case 17: f = &_testfunc17; break; |
| case 18: f = &_testfunc18; break; |
| case 19: f = &_testfunc19; break; |
| case 20: f = &_testfunc20; break; |
| case 21: f = &_testfunc21; break; |
| case 22: f = &_testfunc22; break; |
| case 23: f = &_testfunc23; break; |
| #if 0 |
| case 24: f = &_testfunc24; break; |
| case 25: f = &_testfunc25; break; |
| #endif |
| default: |
| PyErr_SetNone(PyExc_ValueError); |
| return NULL; |
| } |
| return PyLong_FromVoidPtr(f); |
| } |
| |
| #if PY_MAJOR_VERSION < 3 |
| static Py_ssize_t _test_segcountproc(PyObject *o, Py_ssize_t *ignored) |
| { |
| return 1; |
| } |
| static Py_ssize_t _test_getreadbuf(PyObject *o, Py_ssize_t i, void **r) |
| { |
| static char buf[] = "RDB"; |
| *r = buf; |
| return 3; |
| } |
| static Py_ssize_t _test_getwritebuf(PyObject *o, Py_ssize_t i, void **r) |
| { |
| static char buf[] = "WRB"; |
| *r = buf; |
| return 3; |
| } |
| static Py_ssize_t _test_getcharbuf(PyObject *o, Py_ssize_t i, char **r) |
| { |
| static char buf[] = "CHB"; |
| *r = buf; |
| return 3; |
| } |
| #endif |
| static int _test_getbuf(PyObject *self, Py_buffer *view, int flags) |
| { |
| static char buf[] = "GTB"; |
| return PyBuffer_FillInfo(view, self, buf, 3, /*readonly=*/0, flags); |
| } |
| static int _test_getbuf_ro(PyObject *self, Py_buffer *view, int flags) |
| { |
| static char buf[] = "ROB"; |
| return PyBuffer_FillInfo(view, self, buf, 3, /*readonly=*/1, flags); |
| } |
| |
| |
| static PyObject *b__testbuff(PyObject *self, PyObject *args) |
| { |
| /* for testing only */ |
| int methods; |
| PyTypeObject *obj; |
| if (!PyArg_ParseTuple(args, "O!i|_testbuff", &PyType_Type, &obj, &methods)) |
| return NULL; |
| |
| assert(obj->tp_as_buffer != NULL); |
| |
| #if PY_MAJOR_VERSION < 3 |
| obj->tp_as_buffer->bf_getsegcount = &_test_segcountproc; |
| obj->tp_flags |= Py_TPFLAGS_HAVE_GETCHARBUFFER; |
| obj->tp_flags |= Py_TPFLAGS_HAVE_NEWBUFFER; |
| if (methods & 1) obj->tp_as_buffer->bf_getreadbuffer = &_test_getreadbuf; |
| if (methods & 2) obj->tp_as_buffer->bf_getwritebuffer = &_test_getwritebuf; |
| if (methods & 4) obj->tp_as_buffer->bf_getcharbuffer = &_test_getcharbuf; |
| #endif |
| if (methods & 8) obj->tp_as_buffer->bf_getbuffer = &_test_getbuf; |
| if (methods & 16) obj->tp_as_buffer->bf_getbuffer = &_test_getbuf_ro; |
| |
| Py_INCREF(Py_None); |
| return Py_None; |
| } |
| |
| static PyObject *b_init_cffi_1_0_external_module(PyObject *, PyObject *); |
| /* forward, see cffi1_module.c */ |
| |
| |
| static PyMethodDef FFIBackendMethods[] = { |
| {"load_library", b_load_library, METH_VARARGS}, |
| {"new_primitive_type", b_new_primitive_type, METH_VARARGS}, |
| {"new_pointer_type", b_new_pointer_type, METH_VARARGS}, |
| {"new_array_type", b_new_array_type, METH_VARARGS}, |
| {"new_void_type", b_new_void_type, METH_NOARGS}, |
| {"new_struct_type", b_new_struct_type, METH_VARARGS}, |
| {"new_union_type", b_new_union_type, METH_VARARGS}, |
| {"complete_struct_or_union", b_complete_struct_or_union, METH_VARARGS}, |
| {"new_function_type", b_new_function_type, METH_VARARGS}, |
| {"new_enum_type", b_new_enum_type, METH_VARARGS}, |
| {"newp", b_newp, METH_VARARGS}, |
| {"cast", b_cast, METH_VARARGS}, |
| {"callback", b_callback, METH_VARARGS}, |
| {"alignof", b_alignof, METH_O}, |
| {"sizeof", b_sizeof, METH_O}, |
| {"typeof", b_typeof, METH_O}, |
| {"typeoffsetof", b_typeoffsetof, METH_VARARGS}, |
| {"rawaddressof", b_rawaddressof, METH_VARARGS}, |
| {"getcname", b_getcname, METH_VARARGS}, |
| {"string", (PyCFunction)b_string, METH_VARARGS | METH_KEYWORDS}, |
| {"unpack", (PyCFunction)b_unpack, METH_VARARGS | METH_KEYWORDS}, |
| {"get_errno", b_get_errno, METH_NOARGS}, |
| {"set_errno", b_set_errno, METH_O}, |
| {"newp_handle", b_newp_handle, METH_VARARGS}, |
| {"from_handle", b_from_handle, METH_O}, |
| {"from_buffer", b_from_buffer, METH_VARARGS}, |
| {"memmove", (PyCFunction)b_memmove, METH_VARARGS | METH_KEYWORDS}, |
| {"gcp", (PyCFunction)b_gcp, METH_VARARGS | METH_KEYWORDS}, |
| {"release", b_release, METH_O}, |
| #ifdef MS_WIN32 |
| {"getwinerror", (PyCFunction)b_getwinerror, METH_VARARGS | METH_KEYWORDS}, |
| #endif |
| {"_get_types", b__get_types, METH_NOARGS}, |
| {"_get_common_types", b__get_common_types, METH_O}, |
| {"_testfunc", b__testfunc, METH_VARARGS}, |
| {"_testbuff", b__testbuff, METH_VARARGS}, |
| {"_init_cffi_1_0_external_module", b_init_cffi_1_0_external_module, METH_O}, |
| {NULL, NULL} /* Sentinel */ |
| }; |
| |
| /************************************************************/ |
| /* Functions used by '_cffi_N.so', the generated modules */ |
| |
| #define _cffi_to_c_SIGNED_FN(RETURNTYPE, SIZE) \ |
| static RETURNTYPE _cffi_to_c_i##SIZE(PyObject *obj) { \ |
| PY_LONG_LONG tmp = _my_PyLong_AsLongLong(obj); \ |
| if ((tmp > (PY_LONG_LONG)((1ULL<<(SIZE-1)) - 1)) || \ |
| (tmp < (PY_LONG_LONG)(0ULL-(1ULL<<(SIZE-1))))) \ |
| if (!PyErr_Occurred()) \ |
| return (RETURNTYPE)_convert_overflow(obj, #SIZE "-bit int"); \ |
| return (RETURNTYPE)tmp; \ |
| } |
| |
| #define _cffi_to_c_UNSIGNED_FN(RETURNTYPE, SIZE) \ |
| static RETURNTYPE _cffi_to_c_u##SIZE(PyObject *obj) { \ |
| unsigned PY_LONG_LONG tmp = _my_PyLong_AsUnsignedLongLong(obj, 1); \ |
| if (tmp > ~(((unsigned PY_LONG_LONG)-2) << (SIZE-1))) \ |
| if (!PyErr_Occurred()) \ |
| return (RETURNTYPE)_convert_overflow(obj, \ |
| #SIZE "-bit unsigned int"); \ |
| return (RETURNTYPE)tmp; \ |
| } |
| |
| _cffi_to_c_SIGNED_FN(int, 8) |
| _cffi_to_c_SIGNED_FN(int, 16) |
| _cffi_to_c_SIGNED_FN(int, 32) |
| _cffi_to_c_SIGNED_FN(PY_LONG_LONG, 64) |
| _cffi_to_c_UNSIGNED_FN(int, 8) |
| _cffi_to_c_UNSIGNED_FN(int, 16) |
| _cffi_to_c_UNSIGNED_FN(unsigned int, 32) |
| _cffi_to_c_UNSIGNED_FN(unsigned PY_LONG_LONG, 64) |
| |
| static PyObject *_cffi_from_c_pointer(char *ptr, CTypeDescrObject *ct) |
| { |
| return convert_to_object((char *)&ptr, ct); |
| } |
| |
| static char *_cffi_to_c_pointer(PyObject *obj, CTypeDescrObject *ct) |
| { |
| char *result; |
| if (convert_from_object((char *)&result, ct, obj) < 0) { |
| if ((ct->ct_flags & CT_POINTER) && |
| (ct->ct_itemdescr->ct_flags & CT_IS_FILE) && |
| PyFile_Check(obj)) { |
| PyErr_Clear(); |
| return (char *)PyFile_AsFile(obj); |
| } |
| return NULL; |
| } |
| return result; |
| } |
| |
| static long double _cffi_to_c_long_double(PyObject *obj) |
| { |
| if (CData_Check(obj) && |
| (((CDataObject *)obj)->c_type->ct_flags & CT_IS_LONGDOUBLE)) { |
| char *data = ((CDataObject *)obj)->c_data; |
| /*READ(data, sizeof(long double))*/ |
| return read_raw_longdouble_data(data); |
| } |
| else |
| return PyFloat_AsDouble(obj); |
| } |
| |
| static _Bool _cffi_to_c__Bool(PyObject *obj) |
| { |
| PY_LONG_LONG tmp = _my_PyLong_AsLongLong(obj); |
| if (tmp == 0) |
| return 0; |
| else if (tmp == 1) |
| return 1; |
| else if (PyErr_Occurred()) |
| return (_Bool)-1; |
| else |
| return (_Bool)_convert_overflow(obj, "_Bool"); |
| } |
| |
| static PyObject *_cffi_get_struct_layout(Py_ssize_t nums[]) |
| { |
| PyObject *result; |
| int count = 0; |
| while (nums[count] >= 0) |
| count++; |
| |
| result = PyList_New(count); |
| if (result == NULL) |
| return NULL; |
| |
| while (--count >= 0) { |
| PyObject *o = PyInt_FromSsize_t(nums[count]); |
| if (o == NULL) { |
| Py_DECREF(result); |
| return NULL; |
| } |
| PyList_SET_ITEM(result, count, o); |
| } |
| return result; |
| } |
| |
| static PyObject *_cffi_from_c_char(char x) { |
| return PyBytes_FromStringAndSize(&x, 1); |
| } |
| |
| /* backward-compatibility hack: instead of _cffi_to_c_char16_t() and |
| * _cffi_to_c_char32_t(), we have _cffi_to_c_wchar_t() handling whatever |
| * size is wchar_t, and _cffi_to_c_wchar3216_t() handling the opposite. |
| */ |
| #ifdef HAVE_WCHAR_H |
| typedef wchar_t cffi_wchar_t; |
| #else |
| typedef uint16_t cffi_wchar_t; /* random pick... */ |
| #endif |
| |
| static cffi_wchar_t _cffi_to_c_wchar_t(PyObject *init) |
| { |
| if (sizeof(cffi_wchar_t) == 2) |
| return (cffi_wchar_t)_convert_to_char16_t(init); |
| else |
| return (cffi_wchar_t)_convert_to_char32_t(init); |
| } |
| static PyObject *_cffi_from_c_wchar_t(cffi_wchar_t x) { |
| if (sizeof(cffi_wchar_t) == 2) { |
| cffi_char16_t input = x; |
| return _my_PyUnicode_FromChar16(&input, 1); |
| } |
| else { |
| cffi_char32_t input = x; |
| return _my_PyUnicode_FromChar32(&input, 1); |
| } |
| } |
| static int _cffi_to_c_wchar3216_t(PyObject *init) |
| { |
| if (sizeof(cffi_wchar_t) == 4) |
| return (int)_convert_to_char16_t(init); |
| else |
| return (int)_convert_to_char32_t(init); |
| } |
| static PyObject *_cffi_from_c_wchar3216_t(int x) { |
| if (sizeof(cffi_wchar_t) == 4) { |
| cffi_char16_t input = x; |
| return _my_PyUnicode_FromChar16(&input, 1); |
| } |
| else { |
| cffi_char32_t input = x; |
| return _my_PyUnicode_FromChar32(&input, 1); |
| } |
| } |
| |
| struct _cffi_externpy_s; /* forward declaration */ |
| static void cffi_call_python(struct _cffi_externpy_s *, char *args); |
| |
| static void *cffi_exports[] = { |
| NULL, |
| _cffi_to_c_i8, |
| _cffi_to_c_u8, |
| _cffi_to_c_i16, |
| _cffi_to_c_u16, |
| _cffi_to_c_i32, |
| _cffi_to_c_u32, |
| _cffi_to_c_i64, |
| _cffi_to_c_u64, |
| _convert_to_char, |
| _cffi_from_c_pointer, |
| _cffi_to_c_pointer, |
| _cffi_get_struct_layout, |
| restore_errno, |
| save_errno, |
| _cffi_from_c_char, |
| convert_to_object, |
| convert_from_object, |
| convert_struct_to_owning_object, |
| _cffi_to_c_wchar_t, |
| _cffi_from_c_wchar_t, |
| _cffi_to_c_long_double, |
| _cffi_to_c__Bool, |
| _prepare_pointer_call_argument, |
| convert_array_from_object, |
| cffi_call_python, |
| _cffi_to_c_wchar3216_t, |
| _cffi_from_c_wchar3216_t, |
| }; |
| |
| static struct { const char *name; int value; } all_dlopen_flags[] = { |
| { "RTLD_LAZY", RTLD_LAZY }, |
| { "RTLD_NOW", RTLD_NOW }, |
| { "RTLD_GLOBAL", RTLD_GLOBAL }, |
| #ifdef RTLD_LOCAL |
| { "RTLD_LOCAL", RTLD_LOCAL }, |
| #else |
| { "RTLD_LOCAL", 0 }, |
| #endif |
| #ifdef RTLD_NODELETE |
| { "RTLD_NODELETE", RTLD_NODELETE }, |
| #endif |
| #ifdef RTLD_NOLOAD |
| { "RTLD_NOLOAD", RTLD_NOLOAD }, |
| #endif |
| #ifdef RTLD_DEEPBIND |
| { "RTLD_DEEPBIND", RTLD_DEEPBIND }, |
| #endif |
| { NULL, 0 } |
| }; |
| |
| |
| /************************************************************/ |
| |
| #include "cffi1_module.c" |
| |
| /************************************************************/ |
| |
| #if PY_MAJOR_VERSION >= 3 |
| static struct PyModuleDef FFIBackendModuleDef = { |
| PyModuleDef_HEAD_INIT, |
| "_cffi_backend", |
| NULL, |
| -1, |
| FFIBackendMethods, |
| NULL, NULL, NULL, NULL |
| }; |
| #define INITERROR return NULL |
| |
| PyMODINIT_FUNC |
| PyInit__cffi_backend(void) |
| #else |
| #define INITERROR return |
| |
| PyMODINIT_FUNC |
| init_cffi_backend(void) |
| #endif |
| { |
| PyObject *m, *v; |
| int i; |
| static char init_done = 0; |
| static PyTypeObject *all_types[] = { |
| &dl_type, |
| &CTypeDescr_Type, |
| &CField_Type, |
| &CData_Type, |
| &CDataOwning_Type, |
| &CDataOwningGC_Type, |
| &CDataFromBuf_Type, |
| &CDataGCP_Type, |
| &CDataIter_Type, |
| &MiniBuffer_Type, |
| &FFI_Type, |
| &Lib_Type, |
| &GlobSupport_Type, |
| NULL |
| }; |
| |
| v = PySys_GetObject("version"); |
| if (v == NULL || !PyText_Check(v) || |
| strncmp(PyText_AS_UTF8(v), PY_VERSION, 3) != 0) { |
| PyErr_Format(PyExc_ImportError, |
| "this module was compiled for Python %c%c%c", |
| PY_VERSION[0], PY_VERSION[1], PY_VERSION[2]); |
| INITERROR; |
| } |
| |
| #if PY_MAJOR_VERSION >= 3 |
| m = PyModule_Create(&FFIBackendModuleDef); |
| #else |
| m = Py_InitModule("_cffi_backend", FFIBackendMethods); |
| #endif |
| |
| if (m == NULL) |
| INITERROR; |
| |
| if (unique_cache == NULL) { |
| unique_cache = PyDict_New(); |
| if (unique_cache == NULL) |
| INITERROR; |
| } |
| |
| /* readify all types and add them to the module */ |
| for (i = 0; all_types[i] != NULL; i++) { |
| PyTypeObject *tp = all_types[i]; |
| PyObject *tpo = (PyObject *)tp; |
| if (strncmp(tp->tp_name, "_cffi_backend.", 14) != 0) { |
| PyErr_Format(PyExc_ImportError, |
| "'%s' is an ill-formed type name", tp->tp_name); |
| INITERROR; |
| } |
| if (PyType_Ready(tp) < 0) |
| INITERROR; |
| |
| Py_INCREF(tpo); |
| if (PyModule_AddObject(m, tp->tp_name + 14, tpo) < 0) |
| INITERROR; |
| } |
| |
| if (!init_done) { |
| v = PyText_FromString("_cffi_backend"); |
| if (v == NULL || PyDict_SetItemString(CData_Type.tp_dict, |
| "__module__", v) < 0) |
| INITERROR; |
| v = PyText_FromString("<cdata>"); |
| if (v == NULL || PyDict_SetItemString(CData_Type.tp_dict, |
| "__name__", v) < 0) |
| INITERROR; |
| init_done = 1; |
| } |
| |
| /* this is for backward compatibility only */ |
| v = PyCapsule_New((void *)cffi_exports, "cffi", NULL); |
| if (v == NULL || PyModule_AddObject(m, "_C_API", v) < 0) |
| INITERROR; |
| |
| v = PyText_FromString(CFFI_VERSION); |
| if (v == NULL || PyModule_AddObject(m, "__version__", v) < 0) |
| INITERROR; |
| |
| if (PyModule_AddIntConstant(m, "FFI_DEFAULT_ABI", FFI_DEFAULT_ABI) < 0 || |
| #if defined(MS_WIN32) && !defined(_WIN64) |
| PyModule_AddIntConstant(m, "FFI_STDCALL", FFI_STDCALL) < 0 || |
| #endif |
| PyModule_AddIntConstant(m, "FFI_CDECL", FFI_DEFAULT_ABI) < 0 || |
| |
| #ifdef MS_WIN32 |
| # ifdef _WIN64 |
| PyModule_AddIntConstant(m, "_WIN", 64) < 0 || /* win64 */ |
| # else |
| PyModule_AddIntConstant(m, "_WIN", 32) < 0 || /* win32 */ |
| # endif |
| #endif |
| 0) |
| INITERROR; |
| |
| for (i = 0; all_dlopen_flags[i].name != NULL; i++) { |
| if (PyModule_AddIntConstant(m, |
| all_dlopen_flags[i].name, |
| all_dlopen_flags[i].value) < 0) |
| INITERROR; |
| } |
| |
| init_cffi_tls(); |
| if (PyErr_Occurred()) |
| INITERROR; |
| init_cffi_tls_zombie(); |
| if (PyErr_Occurred()) |
| INITERROR; |
| |
| if (init_ffi_lib(m) < 0) |
| INITERROR; |
| |
| #if PY_MAJOR_VERSION >= 3 |
| if (init_file_emulator() < 0) |
| INITERROR; |
| return m; |
| #endif |
| } |