Issue #14203: Remove obsolete support for view==NULL in bytesiobuf_getbuffer()
and array_buffer_getbuf().
This commit is contained in:
parent
38c30e6c8e
commit
650c1e818d
|
@ -1041,6 +1041,11 @@ class BaseTest:
|
||||||
a = array.array(self.typecode, "foo")
|
a = array.array(self.typecode, "foo")
|
||||||
a = array.array(self.typecode, array.array('u', 'foo'))
|
a = array.array(self.typecode, array.array('u', 'foo'))
|
||||||
|
|
||||||
|
@support.cpython_only
|
||||||
|
def test_obsolete_write_lock(self):
|
||||||
|
from _testcapi import getbuffer_with_null_view
|
||||||
|
a = array.array('B', b"")
|
||||||
|
self.assertRaises(BufferError, getbuffer_with_null_view, a)
|
||||||
|
|
||||||
class StringTest(BaseTest):
|
class StringTest(BaseTest):
|
||||||
|
|
||||||
|
|
|
@ -1224,6 +1224,10 @@ class ByteArrayTest(BaseBytesTest, unittest.TestCase):
|
||||||
self.assertRaises(BufferError, delslice)
|
self.assertRaises(BufferError, delslice)
|
||||||
self.assertEqual(b, orig)
|
self.assertEqual(b, orig)
|
||||||
|
|
||||||
|
@test.support.cpython_only
|
||||||
|
def test_obsolete_write_lock(self):
|
||||||
|
from _testcapi import getbuffer_with_null_view
|
||||||
|
self.assertRaises(BufferError, getbuffer_with_null_view, bytearray())
|
||||||
|
|
||||||
class AssortedBytesTest(unittest.TestCase):
|
class AssortedBytesTest(unittest.TestCase):
|
||||||
#
|
#
|
||||||
|
|
|
@ -1560,9 +1560,9 @@ Build
|
||||||
C API
|
C API
|
||||||
-----
|
-----
|
||||||
|
|
||||||
- Issue #14203: Remove obsolete support for view==NULL in PyBuffer_FillInfo()
|
- Issue #14203: Remove obsolete support for view==NULL in PyBuffer_FillInfo(),
|
||||||
and bytearray_getbuffer(). Both functions now raise BufferError in that
|
bytearray_getbuffer(), bytesiobuf_getbuffer() and array_buffer_getbuf().
|
||||||
case.
|
All functions now raise BufferError in that case.
|
||||||
|
|
||||||
- Issue #22445: PyBuffer_IsContiguous() now implements precise contiguity
|
- Issue #22445: PyBuffer_IsContiguous() now implements precise contiguity
|
||||||
tests, compatible with NumPy's NPY_RELAXED_STRIDES_CHECKING compilation
|
tests, compatible with NumPy's NPY_RELAXED_STRIDES_CHECKING compilation
|
||||||
|
|
|
@ -1028,23 +1028,24 @@ PyTypeObject PyBytesIO_Type = {
|
||||||
static int
|
static int
|
||||||
bytesiobuf_getbuffer(bytesiobuf *obj, Py_buffer *view, int flags)
|
bytesiobuf_getbuffer(bytesiobuf *obj, Py_buffer *view, int flags)
|
||||||
{
|
{
|
||||||
int ret;
|
|
||||||
bytesio *b = (bytesio *) obj->source;
|
bytesio *b = (bytesio *) obj->source;
|
||||||
|
|
||||||
|
if (view == NULL) {
|
||||||
|
PyErr_SetString(PyExc_BufferError,
|
||||||
|
"bytesiobuf_getbuffer: view==NULL argument is obsolete");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
if (SHARED_BUF(b)) {
|
if (SHARED_BUF(b)) {
|
||||||
if (unshare_buffer(b, b->string_size) < 0)
|
if (unshare_buffer(b, b->string_size) < 0)
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
if (view == NULL) {
|
|
||||||
b->exports++;
|
/* cannot fail if view != NULL and readonly == 0 */
|
||||||
return 0;
|
(void)PyBuffer_FillInfo(view, (PyObject*)obj,
|
||||||
}
|
|
||||||
ret = PyBuffer_FillInfo(view, (PyObject*)obj,
|
|
||||||
PyBytes_AS_STRING(b->buf), b->string_size,
|
PyBytes_AS_STRING(b->buf), b->string_size,
|
||||||
0, flags);
|
0, flags);
|
||||||
if (ret >= 0) {
|
|
||||||
b->exports++;
|
b->exports++;
|
||||||
}
|
return 0;
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
|
|
@ -2519,20 +2519,25 @@ test_from_contiguous(PyObject* self, PyObject *noargs)
|
||||||
Py_RETURN_NONE;
|
Py_RETURN_NONE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
extern PyTypeObject _PyBytesIOBuffer_Type;
|
||||||
|
|
||||||
static PyObject *
|
static PyObject *
|
||||||
test_pep3118_obsolete_write_locks(PyObject* self, PyObject *noargs)
|
test_pep3118_obsolete_write_locks(PyObject* self, PyObject *noargs)
|
||||||
{
|
{
|
||||||
|
PyTypeObject *type = &_PyBytesIOBuffer_Type;
|
||||||
PyObject *b;
|
PyObject *b;
|
||||||
char *dummy[1];
|
char *dummy[1];
|
||||||
int ret, match;
|
int ret, match;
|
||||||
|
|
||||||
|
/* PyBuffer_FillInfo() */
|
||||||
ret = PyBuffer_FillInfo(NULL, NULL, dummy, 1, 0, PyBUF_SIMPLE);
|
ret = PyBuffer_FillInfo(NULL, NULL, dummy, 1, 0, PyBUF_SIMPLE);
|
||||||
match = PyErr_Occurred() && PyErr_ExceptionMatches(PyExc_BufferError);
|
match = PyErr_Occurred() && PyErr_ExceptionMatches(PyExc_BufferError);
|
||||||
PyErr_Clear();
|
PyErr_Clear();
|
||||||
if (ret != -1 || match == 0)
|
if (ret != -1 || match == 0)
|
||||||
goto error;
|
goto error;
|
||||||
|
|
||||||
b = PyByteArray_FromStringAndSize("", 0);
|
/* bytesiobuf_getbuffer() */
|
||||||
|
b = type->tp_alloc(type, 0);
|
||||||
if (b == NULL) {
|
if (b == NULL) {
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
@ -2552,6 +2557,18 @@ error:
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* This tests functions that historically supported write locks. It is
|
||||||
|
wrong to call getbuffer() with view==NULL and a compliant getbufferproc
|
||||||
|
is entitled to segfault in that case. */
|
||||||
|
static PyObject *
|
||||||
|
getbuffer_with_null_view(PyObject* self, PyObject *obj)
|
||||||
|
{
|
||||||
|
if (PyObject_GetBuffer(obj, NULL, PyBUF_SIMPLE) < 0)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
Py_RETURN_NONE;
|
||||||
|
}
|
||||||
|
|
||||||
/* Test that the fatal error from not having a current thread doesn't
|
/* Test that the fatal error from not having a current thread doesn't
|
||||||
cause an infinite loop. Run via Lib/test/test_capi.py */
|
cause an infinite loop. Run via Lib/test/test_capi.py */
|
||||||
static PyObject *
|
static PyObject *
|
||||||
|
@ -3213,6 +3230,7 @@ static PyMethodDef TestMethods[] = {
|
||||||
{"test_capsule", (PyCFunction)test_capsule, METH_NOARGS},
|
{"test_capsule", (PyCFunction)test_capsule, METH_NOARGS},
|
||||||
{"test_from_contiguous", (PyCFunction)test_from_contiguous, METH_NOARGS},
|
{"test_from_contiguous", (PyCFunction)test_from_contiguous, METH_NOARGS},
|
||||||
{"test_pep3118_obsolete_write_locks", (PyCFunction)test_pep3118_obsolete_write_locks, METH_NOARGS},
|
{"test_pep3118_obsolete_write_locks", (PyCFunction)test_pep3118_obsolete_write_locks, METH_NOARGS},
|
||||||
|
{"getbuffer_with_null_view", getbuffer_with_null_view, METH_O},
|
||||||
{"getargs_tuple", getargs_tuple, METH_VARARGS},
|
{"getargs_tuple", getargs_tuple, METH_VARARGS},
|
||||||
{"getargs_keywords", (PyCFunction)getargs_keywords,
|
{"getargs_keywords", (PyCFunction)getargs_keywords,
|
||||||
METH_VARARGS|METH_KEYWORDS},
|
METH_VARARGS|METH_KEYWORDS},
|
||||||
|
|
|
@ -2530,7 +2530,11 @@ static const void *emptybuf = "";
|
||||||
static int
|
static int
|
||||||
array_buffer_getbuf(arrayobject *self, Py_buffer *view, int flags)
|
array_buffer_getbuf(arrayobject *self, Py_buffer *view, int flags)
|
||||||
{
|
{
|
||||||
if (view==NULL) goto finish;
|
if (view == NULL) {
|
||||||
|
PyErr_SetString(PyExc_BufferError,
|
||||||
|
"array_buffer_getbuf: view==NULL argument is obsolete");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
view->buf = (void *)self->ob_item;
|
view->buf = (void *)self->ob_item;
|
||||||
view->obj = (PyObject*)self;
|
view->obj = (PyObject*)self;
|
||||||
|
@ -2560,7 +2564,6 @@ array_buffer_getbuf(arrayobject *self, Py_buffer *view, int flags)
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
finish:
|
|
||||||
self->ob_exports++;
|
self->ob_exports++;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue