| 
									
										
										
										
											2019-05-26 17:10:09 +02:00
										 |  |  | /* PickleBuffer object implementation */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "Python.h"
 | 
					
						
							| 
									
										
										
										
											2025-06-30 19:14:31 +08:00
										 |  |  | #include "pycore_weakref.h"     // FT_CLEAR_WEAKREFS()
 | 
					
						
							| 
									
										
										
										
											2019-05-26 17:10:09 +02:00
										 |  |  | #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 | 
					
						
							| 
									
										
										
										
											2025-03-20 12:27:03 +01:00
										 |  |  | picklebuf_traverse(PyObject *op, visitproc visit, void *arg) | 
					
						
							| 
									
										
										
										
											2019-05-26 17:10:09 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-03-20 12:27:03 +01:00
										 |  |  |     PyPickleBufferObject *self = (PyPickleBufferObject*)op; | 
					
						
							| 
									
										
										
										
											2019-05-26 17:10:09 +02:00
										 |  |  |     Py_VISIT(self->view.obj); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2025-03-20 12:27:03 +01:00
										 |  |  | picklebuf_clear(PyObject *op) | 
					
						
							| 
									
										
										
										
											2019-05-26 17:10:09 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-03-20 12:27:03 +01:00
										 |  |  |     PyPickleBufferObject *self = (PyPickleBufferObject*)op; | 
					
						
							| 
									
										
										
										
											2019-05-26 17:10:09 +02:00
										 |  |  |     PyBuffer_Release(&self->view); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2025-03-20 12:27:03 +01:00
										 |  |  | picklebuf_dealloc(PyObject *op) | 
					
						
							| 
									
										
										
										
											2019-05-26 17:10:09 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-03-20 12:27:03 +01:00
										 |  |  |     PyPickleBufferObject *self = (PyPickleBufferObject*)op; | 
					
						
							| 
									
										
										
										
											2019-05-26 17:10:09 +02:00
										 |  |  |     PyObject_GC_UnTrack(self); | 
					
						
							| 
									
										
										
										
											2025-06-30 19:14:31 +08:00
										 |  |  |     FT_CLEAR_WEAKREFS(op, self->weakreflist); | 
					
						
							| 
									
										
										
										
											2019-05-26 17:10:09 +02:00
										 |  |  |     PyBuffer_Release(&self->view); | 
					
						
							|  |  |  |     Py_TYPE(self)->tp_free((PyObject *) self); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Buffer API */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2025-03-20 12:27:03 +01:00
										 |  |  | picklebuf_getbuf(PyObject *op, Py_buffer *view, int flags) | 
					
						
							| 
									
										
										
										
											2019-05-26 17:10:09 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-03-20 12:27:03 +01:00
										 |  |  |     PyPickleBufferObject *self = (PyPickleBufferObject*)op; | 
					
						
							| 
									
										
										
										
											2019-05-26 17:10:09 +02:00
										 |  |  |     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 | 
					
						
							| 
									
										
										
										
											2025-03-20 12:27:03 +01:00
										 |  |  | picklebuf_releasebuf(PyObject *self, Py_buffer *view) | 
					
						
							| 
									
										
										
										
											2019-05-26 17:10:09 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     /* 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 = { | 
					
						
							| 
									
										
										
										
											2025-03-20 12:27:03 +01:00
										 |  |  |     .bf_getbuffer = picklebuf_getbuf, | 
					
						
							|  |  |  |     .bf_releasebuffer = picklebuf_releasebuf, | 
					
						
							| 
									
										
										
										
											2019-05-26 17:10:09 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Methods */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2025-03-20 12:27:03 +01:00
										 |  |  | picklebuf_raw(PyObject *op, PyObject *Py_UNUSED(ignored)) | 
					
						
							| 
									
										
										
										
											2019-05-26 17:10:09 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-03-20 12:27:03 +01:00
										 |  |  |     PyPickleBufferObject *self = (PyPickleBufferObject*)op; | 
					
						
							| 
									
										
										
										
											2019-05-26 17:10:09 +02:00
										 |  |  |     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 * | 
					
						
							| 
									
										
										
										
											2025-03-20 12:27:03 +01:00
										 |  |  | picklebuf_release(PyObject *op, PyObject *Py_UNUSED(ignored)) | 
					
						
							| 
									
										
										
										
											2019-05-26 17:10:09 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-03-20 12:27:03 +01:00
										 |  |  |     PyPickleBufferObject *self = (PyPickleBufferObject*)op; | 
					
						
							| 
									
										
										
										
											2019-05-26 17:10:09 +02:00
										 |  |  |     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[] = { | 
					
						
							| 
									
										
										
										
											2025-03-20 12:27:03 +01:00
										 |  |  |     {"raw",     picklebuf_raw,     METH_NOARGS, picklebuf_raw_doc}, | 
					
						
							|  |  |  |     {"release", picklebuf_release, METH_NOARGS, picklebuf_release_doc}, | 
					
						
							| 
									
										
										
										
											2019-05-26 17:10:09 +02:00
										 |  |  |     {NULL,      NULL} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyTypeObject PyPickleBuffer_Type = { | 
					
						
							|  |  |  |     PyVarObject_HEAD_INIT(NULL, 0) | 
					
						
							|  |  |  |     .tp_name = "pickle.PickleBuffer", | 
					
						
							| 
									
										
										
										
											2022-04-18 06:39:32 +03:00
										 |  |  |     .tp_doc = PyDoc_STR("Wrapper for potentially out-of-band buffers"), | 
					
						
							| 
									
										
										
										
											2019-05-26 17:10:09 +02:00
										 |  |  |     .tp_basicsize = sizeof(PyPickleBufferObject), | 
					
						
							|  |  |  |     .tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC, | 
					
						
							|  |  |  |     .tp_new = picklebuf_new, | 
					
						
							| 
									
										
										
										
											2025-03-20 12:27:03 +01:00
										 |  |  |     .tp_dealloc = picklebuf_dealloc, | 
					
						
							|  |  |  |     .tp_traverse = picklebuf_traverse, | 
					
						
							|  |  |  |     .tp_clear = picklebuf_clear, | 
					
						
							| 
									
										
										
										
											2019-05-26 17:10:09 +02:00
										 |  |  |     .tp_weaklistoffset = offsetof(PyPickleBufferObject, weakreflist), | 
					
						
							|  |  |  |     .tp_as_buffer = &picklebuf_as_buffer, | 
					
						
							|  |  |  |     .tp_methods = picklebuf_methods, | 
					
						
							|  |  |  | }; |