mirror of
				https://github.com/python/cpython.git
				synced 2025-10-31 21:51:50 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			219 lines
		
	
	
	
		
			6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			219 lines
		
	
	
	
		
			6 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,
 | |
| };
 | 
