220 lines
6.0 KiB
C
220 lines
6.0 KiB
C
|
/* 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 = "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,
|
||
|
};
|