Issue #14203: Remove obsolete support for view==NULL in bytesiobuf_getbuffer()

and array_buffer_getbuf().
This commit is contained in:
Stefan Krah 2015-02-03 21:43:23 +01:00
parent 38c30e6c8e
commit 650c1e818d
6 changed files with 48 additions and 17 deletions

View File

@ -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):

View File

@ -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):
# #

View File

@ -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

View File

@ -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

View File

@ -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},

View File

@ -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;
} }