123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219 |
- /* PickleBuffer object implementation */
- #define PY_SSIZE_T_CLEAN
- #include "Python.h"
- #include <stddef.h>
- typedef struct {
- PyObject_HEAD
- /* The view exported by the original object */
- Py_buffer view;
- PyObject *weakreflist;
- } PyPickleBufferObject;
- /* C API */
- PyObject *
- PyPickleBuffer_FromObject(PyObject *base)
- {
- PyTypeObject *type = &PyPickleBuffer_Type;
- PyPickleBufferObject *self;
- self = (PyPickleBufferObject *) type->tp_alloc(type, 0);
- if (self == NULL) {
- return NULL;
- }
- self->view.obj = NULL;
- self->weakreflist = NULL;
- if (PyObject_GetBuffer(base, &self->view, PyBUF_FULL_RO) < 0) {
- Py_DECREF(self);
- return NULL;
- }
- return (PyObject *) self;
- }
- const Py_buffer *
- PyPickleBuffer_GetBuffer(PyObject *obj)
- {
- PyPickleBufferObject *self = (PyPickleBufferObject *) obj;
- if (!PyPickleBuffer_Check(obj)) {
- PyErr_Format(PyExc_TypeError,
- "expected PickleBuffer, %.200s found",
- Py_TYPE(obj)->tp_name);
- return NULL;
- }
- if (self->view.obj == NULL) {
- PyErr_SetString(PyExc_ValueError,
- "operation forbidden on released PickleBuffer object");
- return NULL;
- }
- return &self->view;
- }
- int
- PyPickleBuffer_Release(PyObject *obj)
- {
- PyPickleBufferObject *self = (PyPickleBufferObject *) obj;
- if (!PyPickleBuffer_Check(obj)) {
- PyErr_Format(PyExc_TypeError,
- "expected PickleBuffer, %.200s found",
- Py_TYPE(obj)->tp_name);
- return -1;
- }
- PyBuffer_Release(&self->view);
- return 0;
- }
- static PyObject *
- picklebuf_new(PyTypeObject *type, PyObject *args, PyObject *kwds)
- {
- PyPickleBufferObject *self;
- PyObject *base;
- char *keywords[] = {"", NULL};
- if (!PyArg_ParseTupleAndKeywords(args, kwds, "O:PickleBuffer",
- keywords, &base)) {
- return NULL;
- }
- self = (PyPickleBufferObject *) type->tp_alloc(type, 0);
- if (self == NULL) {
- return NULL;
- }
- self->view.obj = NULL;
- self->weakreflist = NULL;
- if (PyObject_GetBuffer(base, &self->view, PyBUF_FULL_RO) < 0) {
- Py_DECREF(self);
- return NULL;
- }
- return (PyObject *) self;
- }
- static int
- picklebuf_traverse(PyPickleBufferObject *self, visitproc visit, void *arg)
- {
- Py_VISIT(self->view.obj);
- return 0;
- }
- static int
- picklebuf_clear(PyPickleBufferObject *self)
- {
- PyBuffer_Release(&self->view);
- return 0;
- }
- static void
- picklebuf_dealloc(PyPickleBufferObject *self)
- {
- PyObject_GC_UnTrack(self);
- if (self->weakreflist != NULL)
- PyObject_ClearWeakRefs((PyObject *) self);
- PyBuffer_Release(&self->view);
- Py_TYPE(self)->tp_free((PyObject *) self);
- }
- /* Buffer API */
- static int
- picklebuf_getbuf(PyPickleBufferObject *self, Py_buffer *view, int flags)
- {
- if (self->view.obj == NULL) {
- PyErr_SetString(PyExc_ValueError,
- "operation forbidden on released PickleBuffer object");
- return -1;
- }
- return PyObject_GetBuffer(self->view.obj, view, flags);
- }
- static void
- picklebuf_releasebuf(PyPickleBufferObject *self, Py_buffer *view)
- {
- /* Since our bf_getbuffer redirects to the original object, this
- * implementation is never called. It only exists to signal that
- * buffers exported by PickleBuffer have non-trivial releasing
- * behaviour (see check in Python/getargs.c).
- */
- }
- static PyBufferProcs picklebuf_as_buffer = {
- .bf_getbuffer = (getbufferproc) picklebuf_getbuf,
- .bf_releasebuffer = (releasebufferproc) picklebuf_releasebuf,
- };
- /* Methods */
- static PyObject *
- picklebuf_raw(PyPickleBufferObject *self, PyObject *Py_UNUSED(ignored))
- {
- if (self->view.obj == NULL) {
- PyErr_SetString(PyExc_ValueError,
- "operation forbidden on released PickleBuffer object");
- return NULL;
- }
- if (self->view.suboffsets != NULL
- || !PyBuffer_IsContiguous(&self->view, 'A')) {
- PyErr_SetString(PyExc_BufferError,
- "cannot extract raw buffer from non-contiguous buffer");
- return NULL;
- }
- PyObject *m = PyMemoryView_FromObject((PyObject *) self);
- if (m == NULL) {
- return NULL;
- }
- PyMemoryViewObject *mv = (PyMemoryViewObject *) m;
- assert(mv->view.suboffsets == NULL);
- /* Mutate memoryview instance to make it a "raw" memoryview */
- mv->view.format = "B";
- mv->view.ndim = 1;
- mv->view.itemsize = 1;
- /* shape = (length,) */
- mv->view.shape = &mv->view.len;
- /* strides = (1,) */
- mv->view.strides = &mv->view.itemsize;
- /* Fix memoryview state flags */
- /* XXX Expose memoryobject.c's init_flags() instead? */
- mv->flags = _Py_MEMORYVIEW_C | _Py_MEMORYVIEW_FORTRAN;
- return m;
- }
- PyDoc_STRVAR(picklebuf_raw_doc,
- "raw($self, /)\n--\n\
- \n\
- Return a memoryview of the raw memory underlying this buffer.\n\
- Will raise BufferError is the buffer isn't contiguous.");
- static PyObject *
- picklebuf_release(PyPickleBufferObject *self, PyObject *Py_UNUSED(ignored))
- {
- PyBuffer_Release(&self->view);
- Py_RETURN_NONE;
- }
- PyDoc_STRVAR(picklebuf_release_doc,
- "release($self, /)\n--\n\
- \n\
- Release the underlying buffer exposed by the PickleBuffer object.");
- static PyMethodDef picklebuf_methods[] = {
- {"raw", (PyCFunction) picklebuf_raw, METH_NOARGS, picklebuf_raw_doc},
- {"release", (PyCFunction) picklebuf_release, METH_NOARGS, picklebuf_release_doc},
- {NULL, NULL}
- };
- PyTypeObject PyPickleBuffer_Type = {
- PyVarObject_HEAD_INIT(NULL, 0)
- .tp_name = "pickle.PickleBuffer",
- .tp_doc = PyDoc_STR("Wrapper for potentially out-of-band buffers"),
- .tp_basicsize = sizeof(PyPickleBufferObject),
- .tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC,
- .tp_new = picklebuf_new,
- .tp_dealloc = (destructor) picklebuf_dealloc,
- .tp_traverse = (traverseproc) picklebuf_traverse,
- .tp_clear = (inquiry) picklebuf_clear,
- .tp_weaklistoffset = offsetof(PyPickleBufferObject, weakreflist),
- .tp_as_buffer = &picklebuf_as_buffer,
- .tp_methods = picklebuf_methods,
- };
|