mirror of https://github.com/python/cpython
gh-124248: Fix crash in struct when processing 0p fields (#124251)
This commit is contained in:
parent
baa3550bc3
commit
63f196090f
|
@ -96,6 +96,13 @@ class StructTest(unittest.TestCase):
|
||||||
('10s', b'helloworld', b'helloworld', b'helloworld', 0),
|
('10s', b'helloworld', b'helloworld', b'helloworld', 0),
|
||||||
('11s', b'helloworld', b'helloworld\0', b'helloworld\0', 1),
|
('11s', b'helloworld', b'helloworld\0', b'helloworld\0', 1),
|
||||||
('20s', b'helloworld', b'helloworld'+10*b'\0', b'helloworld'+10*b'\0', 1),
|
('20s', b'helloworld', b'helloworld'+10*b'\0', b'helloworld'+10*b'\0', 1),
|
||||||
|
('0p', b'helloworld', b'', b'', 1),
|
||||||
|
('1p', b'helloworld', b'\x00', b'\x00', 1),
|
||||||
|
('2p', b'helloworld', b'\x01h', b'\x01h', 1),
|
||||||
|
('10p', b'helloworld', b'\x09helloworl', b'\x09helloworl', 1),
|
||||||
|
('11p', b'helloworld', b'\x0Ahelloworld', b'\x0Ahelloworld', 0),
|
||||||
|
('12p', b'helloworld', b'\x0Ahelloworld\0', b'\x0Ahelloworld\0', 1),
|
||||||
|
('20p', b'helloworld', b'\x0Ahelloworld'+9*b'\0', b'\x0Ahelloworld'+9*b'\0', 1),
|
||||||
('b', 7, b'\7', b'\7', 0),
|
('b', 7, b'\7', b'\7', 0),
|
||||||
('b', -7, b'\371', b'\371', 0),
|
('b', -7, b'\371', b'\371', 0),
|
||||||
('B', 7, b'\7', b'\7', 0),
|
('B', 7, b'\7', b'\7', 0),
|
||||||
|
@ -339,6 +346,7 @@ class StructTest(unittest.TestCase):
|
||||||
def test_p_code(self):
|
def test_p_code(self):
|
||||||
# Test p ("Pascal string") code.
|
# Test p ("Pascal string") code.
|
||||||
for code, input, expected, expectedback in [
|
for code, input, expected, expectedback in [
|
||||||
|
('0p', b'abc', b'', b''),
|
||||||
('p', b'abc', b'\x00', b''),
|
('p', b'abc', b'\x00', b''),
|
||||||
('1p', b'abc', b'\x00', b''),
|
('1p', b'abc', b'\x00', b''),
|
||||||
('2p', b'abc', b'\x01a', b'a'),
|
('2p', b'abc', b'\x01a', b'a'),
|
||||||
|
@ -580,6 +588,7 @@ class StructTest(unittest.TestCase):
|
||||||
self.check_sizeof('187s', 1)
|
self.check_sizeof('187s', 1)
|
||||||
self.check_sizeof('20p', 1)
|
self.check_sizeof('20p', 1)
|
||||||
self.check_sizeof('0s', 1)
|
self.check_sizeof('0s', 1)
|
||||||
|
self.check_sizeof('0p', 1)
|
||||||
self.check_sizeof('0c', 0)
|
self.check_sizeof('0c', 0)
|
||||||
|
|
||||||
def test_boundary_error_message(self):
|
def test_boundary_error_message(self):
|
||||||
|
|
|
@ -1653,6 +1653,7 @@ Scott Schram
|
||||||
Robin Schreiber
|
Robin Schreiber
|
||||||
Chad J. Schroeder
|
Chad J. Schroeder
|
||||||
Simon-Martin Schroeder
|
Simon-Martin Schroeder
|
||||||
|
Brian Schubert
|
||||||
Christian Schubert
|
Christian Schubert
|
||||||
Sam Schulenburg
|
Sam Schulenburg
|
||||||
Andreas Schwab
|
Andreas Schwab
|
||||||
|
|
|
@ -0,0 +1,2 @@
|
||||||
|
Fixed potential crash when using :mod:`struct` to process zero-width
|
||||||
|
'Pascal string' fields (``0p``).
|
|
@ -1669,9 +1669,16 @@ s_unpack_internal(PyStructObject *soself, const char *startfrom,
|
||||||
if (e->format == 's') {
|
if (e->format == 's') {
|
||||||
v = PyBytes_FromStringAndSize(res, code->size);
|
v = PyBytes_FromStringAndSize(res, code->size);
|
||||||
} else if (e->format == 'p') {
|
} else if (e->format == 'p') {
|
||||||
Py_ssize_t n = *(unsigned char*)res;
|
Py_ssize_t n;
|
||||||
if (n >= code->size)
|
if (code->size == 0) {
|
||||||
|
n = 0;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
n = *(unsigned char*)res;
|
||||||
|
if (n >= code->size) {
|
||||||
n = code->size - 1;
|
n = code->size - 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
v = PyBytes_FromStringAndSize(res + 1, n);
|
v = PyBytes_FromStringAndSize(res + 1, n);
|
||||||
} else {
|
} else {
|
||||||
v = e->unpack(state, res, e);
|
v = e->unpack(state, res, e);
|
||||||
|
@ -1982,8 +1989,12 @@ s_pack_internal(PyStructObject *soself, PyObject *const *args, int offset,
|
||||||
n = PyByteArray_GET_SIZE(v);
|
n = PyByteArray_GET_SIZE(v);
|
||||||
p = PyByteArray_AS_STRING(v);
|
p = PyByteArray_AS_STRING(v);
|
||||||
}
|
}
|
||||||
if (n > (code->size - 1))
|
if (code->size == 0) {
|
||||||
|
n = 0;
|
||||||
|
}
|
||||||
|
else if (n > (code->size - 1)) {
|
||||||
n = code->size - 1;
|
n = code->size - 1;
|
||||||
|
}
|
||||||
if (n > 0)
|
if (n > 0)
|
||||||
memcpy(res + 1, p, n);
|
memcpy(res + 1, p, n);
|
||||||
if (n > 255)
|
if (n > 255)
|
||||||
|
|
Loading…
Reference in New Issue