Commit f9b2aa2e authored by Serhiy Storchaka's avatar Serhiy Storchaka

Issue #22193: Fixed integer overflow error in sys.getsizeof().

Fixed an error in _PySys_GetSizeOf declaration.
parent af627902
...@@ -23,7 +23,7 @@ PyAPI_FUNC(void) PySys_ResetWarnOptions(void); ...@@ -23,7 +23,7 @@ PyAPI_FUNC(void) PySys_ResetWarnOptions(void);
PyAPI_FUNC(void) PySys_AddWarnOption(char *); PyAPI_FUNC(void) PySys_AddWarnOption(char *);
PyAPI_FUNC(int) PySys_HasWarnOptions(void); PyAPI_FUNC(int) PySys_HasWarnOptions(void);
PyAPI_DATA(size_t) _PySys_GetSizeOf(PyObject *); PyAPI_FUNC(size_t) _PySys_GetSizeOf(PyObject *);
#ifdef __cplusplus #ifdef __cplusplus
} }
......
...@@ -494,6 +494,31 @@ class SizeofTest(unittest.TestCase): ...@@ -494,6 +494,31 @@ class SizeofTest(unittest.TestCase):
# but lists are # but lists are
self.assertEqual(sys.getsizeof([]), size('P PP') + gc_header_size) self.assertEqual(sys.getsizeof([]), size('P PP') + gc_header_size)
def test_errors(self):
class BadSizeof(object):
def __sizeof__(self):
raise ValueError
self.assertRaises(ValueError, sys.getsizeof, BadSizeof())
class InvalidSizeof(object):
def __sizeof__(self):
return None
self.assertRaises(TypeError, sys.getsizeof, InvalidSizeof())
sentinel = ["sentinel"]
self.assertIs(sys.getsizeof(InvalidSizeof(), sentinel), sentinel)
class OverflowSizeof(int):
def __sizeof__(self):
return int(self)
self.assertEqual(sys.getsizeof(OverflowSizeof(sys.maxsize)),
sys.maxsize + self.gc_headsize)
with self.assertRaises(OverflowError):
sys.getsizeof(OverflowSizeof(sys.maxsize + 1))
with self.assertRaises(ValueError):
sys.getsizeof(OverflowSizeof(-1))
with self.assertRaises((ValueError, OverflowError)):
sys.getsizeof(OverflowSizeof(-sys.maxsize - 1))
def test_default(self): def test_default(self):
size = test.test_support.calcobjsize size = test.test_support.calcobjsize
self.assertEqual(sys.getsizeof(True, -1), size('l')) self.assertEqual(sys.getsizeof(True, -1), size('l'))
......
...@@ -689,7 +689,7 @@ _PySys_GetSizeOf(PyObject *o) ...@@ -689,7 +689,7 @@ _PySys_GetSizeOf(PyObject *o)
{ {
static PyObject *str__sizeof__ = NULL; static PyObject *str__sizeof__ = NULL;
PyObject *res = NULL; PyObject *res = NULL;
size_t size; Py_ssize_t size;
/* Make sure the type is initialized. float gets initialized late */ /* Make sure the type is initialized. float gets initialized late */
if (PyType_Ready(Py_TYPE(o)) < 0) if (PyType_Ready(Py_TYPE(o)) < 0)
...@@ -718,14 +718,19 @@ _PySys_GetSizeOf(PyObject *o) ...@@ -718,14 +718,19 @@ _PySys_GetSizeOf(PyObject *o)
size = (size_t)PyInt_AsSsize_t(res); size = (size_t)PyInt_AsSsize_t(res);
Py_DECREF(res); Py_DECREF(res);
if (size == (size_t)-1 && PyErr_Occurred()) if (size == -1 && PyErr_Occurred())
return (size_t)-1;
}
if (size < 0) {
PyErr_SetString(PyExc_ValueError, "__sizeof__() should return >= 0");
return (size_t)-1; return (size_t)-1;
} }
/* add gc_head size */ /* add gc_head size */
if (PyObject_IS_GC(o)) if (PyObject_IS_GC(o))
size += sizeof(PyGC_Head); return ((size_t)size) + sizeof(PyGC_Head);
return size; return (size_t)size;
} }
static PyObject * static PyObject *
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment