SF patch 514641 (Naofumi Honda) - Negative ob_size of LongObjects

Due to the bizarre definition of _PyLong_Copy(), creating an instance
of a subclass of long with a negative value could cause core dumps
later on.  Unfortunately it looks like the behavior of _PyLong_Copy()
is quite intentional, so the fix is more work than feels comfortable.

This fix is almost, but not quite, the code that Naofumi Honda added;
in addition, I added a test case.
This commit is contained in:
Guido van Rossum 2002-03-01 22:24:49 +00:00
parent 6f33250ef9
commit 2eb0b87d14
4 changed files with 23 additions and 4 deletions

View File

@ -1748,6 +1748,10 @@ def inherits():
verify((a + 0).__class__ is long) verify((a + 0).__class__ is long)
verify((0 + a).__class__ is long) verify((0 + a).__class__ is long)
# Check that negative clones don't segfault
a = longclone(-1)
vereq(a.__dict__, {})
class precfloat(float): class precfloat(float):
__slots__ = ['prec'] __slots__ = ['prec']
def __init__(self, value=0.0, prec=12): def __init__(self, value=0.0, prec=12):

View File

@ -203,6 +203,7 @@ Chris Hoffman
Albert Hofkamp Albert Hofkamp
Gerrit Holl Gerrit Holl
Philip Homburg Philip Homburg
Naofumi Honda
Jeffrey Honig Jeffrey Honig
Rob Hooft Rob Hooft
Brian Hooper Brian Hooper

View File

@ -933,8 +933,16 @@ PyNumber_Long(PyObject *o)
Py_INCREF(o); Py_INCREF(o);
return o; return o;
} }
if (PyLong_Check(o)) if (PyLong_Check(o)) {
return _PyLong_Copy((PyLongObject *)o); PyObject *res;
res = _PyLong_Copy((PyLongObject *)o);
if (res != NULL)
((PyLongObject *)res)->ob_size =
((PyLongObject *)o)->ob_size;
return res;
}
if (PyString_Check(o)) if (PyString_Check(o))
/* need to do extra error checking that PyLong_FromString() /* need to do extra error checking that PyLong_FromString()
* doesn't do. In particular long('9.5') must raise an * doesn't do. In particular long('9.5') must raise an

View File

@ -1191,8 +1191,14 @@ _PyObject_GetDictPtr(PyObject *obj)
if (dictoffset == 0) if (dictoffset == 0)
return NULL; return NULL;
if (dictoffset < 0) { if (dictoffset < 0) {
const size_t size = _PyObject_VAR_SIZE(tp, int tsize;
((PyVarObject *)obj)->ob_size); size_t size;
tsize = ((PyVarObject *)obj)->ob_size;
if (tsize < 0)
tsize = -tsize;
size = _PyObject_VAR_SIZE(tp, tsize);
dictoffset += (long)size; dictoffset += (long)size;
assert(dictoffset > 0); assert(dictoffset > 0);
assert(dictoffset % SIZEOF_VOID_P == 0); assert(dictoffset % SIZEOF_VOID_P == 0);