| /* Wrap void * pointers to be passed between C modules */ | 
 |  | 
 | #include "Python.h" | 
 |  | 
 | /* Internal structure of PyCapsule */ | 
 | typedef struct { | 
 |     PyObject_HEAD | 
 |     void *pointer; | 
 |     const char *name; | 
 |     void *context; | 
 |     PyCapsule_Destructor destructor; | 
 | } PyCapsule; | 
 |  | 
 |  | 
 |  | 
 | static int | 
 | _is_legal_capsule(PyCapsule *capsule, const char *invalid_capsule) | 
 | { | 
 |     if (!capsule || !PyCapsule_CheckExact(capsule) || capsule->pointer == NULL) { | 
 |         PyErr_SetString(PyExc_ValueError, invalid_capsule); | 
 |         return 0; | 
 |     } | 
 |     return 1; | 
 | } | 
 |  | 
 | #define is_legal_capsule(capsule, name) \ | 
 |     (_is_legal_capsule(capsule, \ | 
 |      name " called with invalid PyCapsule object")) | 
 |  | 
 |  | 
 | static int | 
 | name_matches(const char *name1, const char *name2) { | 
 |     /* if either is NULL, */ | 
 |     if (!name1 || !name2) { | 
 |         /* they're only the same if they're both NULL. */ | 
 |         return name1 == name2; | 
 |     } | 
 |     return !strcmp(name1, name2); | 
 | } | 
 |  | 
 |  | 
 |  | 
 | PyObject * | 
 | PyCapsule_New(void *pointer, const char *name, PyCapsule_Destructor destructor) | 
 | { | 
 |     PyCapsule *capsule; | 
 |  | 
 |     if (!pointer) { | 
 |         PyErr_SetString(PyExc_ValueError, "PyCapsule_New called with null pointer"); | 
 |         return NULL; | 
 |     } | 
 |  | 
 |     capsule = PyObject_NEW(PyCapsule, &PyCapsule_Type); | 
 |     if (capsule == NULL) { | 
 |         return NULL; | 
 |     } | 
 |  | 
 |     capsule->pointer = pointer; | 
 |     capsule->name = name; | 
 |     capsule->context = NULL; | 
 |     capsule->destructor = destructor; | 
 |  | 
 |     return (PyObject *)capsule; | 
 | } | 
 |  | 
 |  | 
 | int | 
 | PyCapsule_IsValid(PyObject *o, const char *name) | 
 | { | 
 |     PyCapsule *capsule = (PyCapsule *)o; | 
 |  | 
 |     return (capsule != NULL && | 
 |             PyCapsule_CheckExact(capsule) && | 
 |             capsule->pointer != NULL && | 
 |             name_matches(capsule->name, name)); | 
 | } | 
 |  | 
 |  | 
 | void * | 
 | PyCapsule_GetPointer(PyObject *o, const char *name) | 
 | { | 
 |     PyCapsule *capsule = (PyCapsule *)o; | 
 |  | 
 |     if (!is_legal_capsule(capsule, "PyCapsule_GetPointer")) { | 
 |         return NULL; | 
 |     } | 
 |  | 
 |     if (!name_matches(name, capsule->name)) { | 
 |         PyErr_SetString(PyExc_ValueError, "PyCapsule_GetPointer called with incorrect name"); | 
 |         return NULL; | 
 |     } | 
 |  | 
 |     return capsule->pointer; | 
 | } | 
 |  | 
 |  | 
 | const char * | 
 | PyCapsule_GetName(PyObject *o) | 
 | { | 
 |     PyCapsule *capsule = (PyCapsule *)o; | 
 |  | 
 |     if (!is_legal_capsule(capsule, "PyCapsule_GetName")) { | 
 |         return NULL; | 
 |     } | 
 |     return capsule->name; | 
 | } | 
 |  | 
 |  | 
 | PyCapsule_Destructor | 
 | PyCapsule_GetDestructor(PyObject *o) | 
 | { | 
 |     PyCapsule *capsule = (PyCapsule *)o; | 
 |  | 
 |     if (!is_legal_capsule(capsule, "PyCapsule_GetDestructor")) { | 
 |         return NULL; | 
 |     } | 
 |     return capsule->destructor; | 
 | } | 
 |  | 
 |  | 
 | void * | 
 | PyCapsule_GetContext(PyObject *o) | 
 | { | 
 |     PyCapsule *capsule = (PyCapsule *)o; | 
 |  | 
 |     if (!is_legal_capsule(capsule, "PyCapsule_GetContext")) { | 
 |         return NULL; | 
 |     } | 
 |     return capsule->context; | 
 | } | 
 |  | 
 |  | 
 | int | 
 | PyCapsule_SetPointer(PyObject *o, void *pointer) | 
 | { | 
 |     PyCapsule *capsule = (PyCapsule *)o; | 
 |  | 
 |     if (!pointer) { | 
 |         PyErr_SetString(PyExc_ValueError, "PyCapsule_SetPointer called with null pointer"); | 
 |         return -1; | 
 |     } | 
 |  | 
 |     if (!is_legal_capsule(capsule, "PyCapsule_SetPointer")) { | 
 |         return -1; | 
 |     } | 
 |  | 
 |     capsule->pointer = pointer; | 
 |     return 0; | 
 | } | 
 |  | 
 |  | 
 | int | 
 | PyCapsule_SetName(PyObject *o, const char *name) | 
 | { | 
 |     PyCapsule *capsule = (PyCapsule *)o; | 
 |  | 
 |     if (!is_legal_capsule(capsule, "PyCapsule_SetName")) { | 
 |         return -1; | 
 |     } | 
 |  | 
 |     capsule->name = name; | 
 |     return 0; | 
 | } | 
 |  | 
 |  | 
 | int | 
 | PyCapsule_SetDestructor(PyObject *o, PyCapsule_Destructor destructor) | 
 | { | 
 |     PyCapsule *capsule = (PyCapsule *)o; | 
 |  | 
 |     if (!is_legal_capsule(capsule, "PyCapsule_SetDestructor")) { | 
 |         return -1; | 
 |     } | 
 |  | 
 |     capsule->destructor = destructor; | 
 |     return 0; | 
 | } | 
 |  | 
 |  | 
 | int | 
 | PyCapsule_SetContext(PyObject *o, void *context) | 
 | { | 
 |     PyCapsule *capsule = (PyCapsule *)o; | 
 |  | 
 |     if (!is_legal_capsule(capsule, "PyCapsule_SetContext")) { | 
 |         return -1; | 
 |     } | 
 |  | 
 |     capsule->context = context; | 
 |     return 0; | 
 | } | 
 |  | 
 |  | 
 | void * | 
 | PyCapsule_Import(const char *name, int no_block) | 
 | { | 
 |     PyObject *object = NULL; | 
 |     void *return_value = NULL; | 
 |     char *trace; | 
 |     size_t name_length = (strlen(name) + 1) * sizeof(char); | 
 |     char *name_dup = (char *)PyMem_MALLOC(name_length); | 
 |  | 
 |     if (!name_dup) { | 
 |         return NULL; | 
 |     } | 
 |  | 
 |     memcpy(name_dup, name, name_length); | 
 |  | 
 |     trace = name_dup; | 
 |     while (trace) { | 
 |         char *dot = strchr(trace, '.'); | 
 |         if (dot) { | 
 |             *dot++ = '\0'; | 
 |         } | 
 |  | 
 |         if (object == NULL) { | 
 |             if (no_block) { | 
 |                 object = PyImport_ImportModuleNoBlock(trace); | 
 |             } else { | 
 |                 object = PyImport_ImportModule(trace); | 
 |                 if (!object) { | 
 |                     PyErr_Format(PyExc_ImportError, "PyCapsule_Import could not import module \"%s\"", trace); | 
 |                 } | 
 |             } | 
 |         } else { | 
 |             PyObject *object2 = PyObject_GetAttrString(object, trace); | 
 |             Py_DECREF(object); | 
 |             object = object2; | 
 |         } | 
 |         if (!object) { | 
 |             goto EXIT; | 
 |         } | 
 |  | 
 |         trace = dot; | 
 |     } | 
 |  | 
 |     /* compare attribute name to module.name by hand */ | 
 |     if (PyCapsule_IsValid(object, name)) { | 
 |         PyCapsule *capsule = (PyCapsule *)object; | 
 |         return_value = capsule->pointer; | 
 |     } else { | 
 |         PyErr_Format(PyExc_AttributeError, | 
 |             "PyCapsule_Import \"%s\" is not valid", | 
 |             name); | 
 |     } | 
 |  | 
 | EXIT: | 
 |     Py_XDECREF(object); | 
 |     if (name_dup) { | 
 |         PyMem_FREE(name_dup); | 
 |     } | 
 |     return return_value; | 
 | } | 
 |  | 
 |  | 
 | static void | 
 | capsule_dealloc(PyObject *o) | 
 | { | 
 |     PyCapsule *capsule = (PyCapsule *)o; | 
 |     if (capsule->destructor) { | 
 |         capsule->destructor(o); | 
 |     } | 
 |     PyObject_DEL(o); | 
 | } | 
 |  | 
 |  | 
 | static PyObject * | 
 | capsule_repr(PyObject *o) | 
 | { | 
 |     PyCapsule *capsule = (PyCapsule *)o; | 
 |     const char *name; | 
 |     const char *quote; | 
 |  | 
 |     if (capsule->name) { | 
 |         quote = "\""; | 
 |         name = capsule->name; | 
 |     } else { | 
 |         quote = ""; | 
 |         name = "NULL"; | 
 |     } | 
 |  | 
 |     return PyString_FromFormat("<capsule object %s%s%s at %p>", | 
 |         quote, name, quote, capsule); | 
 | } | 
 |  | 
 |  | 
 |  | 
 | PyDoc_STRVAR(PyCapsule_Type__doc__, | 
 | "Capsule objects let you wrap a C \"void *\" pointer in a Python\n\ | 
 | object.  They're a way of passing data through the Python interpreter\n\ | 
 | without creating your own custom type.\n\ | 
 | \n\ | 
 | Capsules are used for communication between extension modules.\n\ | 
 | They provide a way for an extension module to export a C interface\n\ | 
 | to other extension modules, so that extension modules can use the\n\ | 
 | Python import mechanism to link to one another.\n\ | 
 | "); | 
 |  | 
 | PyTypeObject PyCapsule_Type = { | 
 |     PyVarObject_HEAD_INIT(&PyType_Type, 0) | 
 |     "PyCapsule",		/*tp_name*/ | 
 |     sizeof(PyCapsule),		/*tp_basicsize*/ | 
 |     0,				/*tp_itemsize*/ | 
 |     /* methods */ | 
 |     capsule_dealloc, /*tp_dealloc*/ | 
 |     0,				/*tp_print*/ | 
 |     0,				/*tp_getattr*/ | 
 |     0,				/*tp_setattr*/ | 
 |     0,				/*tp_reserved*/ | 
 |     capsule_repr, /*tp_repr*/ | 
 |     0,				/*tp_as_number*/ | 
 |     0,				/*tp_as_sequence*/ | 
 |     0,				/*tp_as_mapping*/ | 
 |     0,				/*tp_hash*/ | 
 |     0,				/*tp_call*/ | 
 |     0,				/*tp_str*/ | 
 |     0,				/*tp_getattro*/ | 
 |     0,				/*tp_setattro*/ | 
 |     0,				/*tp_as_buffer*/ | 
 |     0,				/*tp_flags*/ | 
 |     PyCapsule_Type__doc__	/*tp_doc*/ | 
 | }; | 
 |  | 
 |  |