diff options
author | Guido van Rossum <guido@python.org> | 2001-12-04 16:23:42 (GMT) |
---|---|---|
committer | Guido van Rossum <guido@python.org> | 2001-12-04 16:23:42 (GMT) |
commit | 6b70599450777a8b911f0eff44b18cd22f1c1e1e (patch) | |
tree | 10bb79dc54602e3226dd68774994e15a6d1af2b9 | |
parent | ebca9fc1ba23e58d734b9ab422da6a64fa4d72e2 (diff) | |
download | cpython-6b70599450777a8b911f0eff44b18cd22f1c1e1e.zip cpython-6b70599450777a8b911f0eff44b18cd22f1c1e1e.tar.gz cpython-6b70599450777a8b911f0eff44b18cd22f1c1e1e.tar.bz2 |
Fix SF bug #486144: Uninitialized __slot__ vrbl is None.
There's now a new structmember code, T_OBJECT_EX, which is used for
all __slot__ variables (except __weakref__, which has special behavior
anyway). This new code raises AttributeError when the variable is
NULL rather than converting NULL to None.
-rw-r--r-- | Include/structmember.h | 4 | ||||
-rw-r--r-- | Lib/test/test_descr.py | 12 | ||||
-rw-r--r-- | Python/structmember.c | 9 |
3 files changed, 19 insertions, 6 deletions
diff --git a/Include/structmember.h b/Include/structmember.h index 2c9f41d..98eccc0 100644 --- a/Include/structmember.h +++ b/Include/structmember.h @@ -68,6 +68,10 @@ typedef struct PyMemberDef { #define T_PSTRING_INPLACE 15 #endif /* macintosh */ +#define T_OBJECT_EX 16 /* Like T_OBJECT, but raises AttributeError + when the value is NULL, instead of + converting to None. */ + /* Flags */ #define READONLY 1 #define RO READONLY /* Shorthand */ diff --git a/Lib/test/test_descr.py b/Lib/test/test_descr.py index d76013e..48d1138 100644 --- a/Lib/test/test_descr.py +++ b/Lib/test/test_descr.py @@ -978,19 +978,21 @@ def slots(): __slots__ = ['a'] x = C1() verify(not hasattr(x, "__dict__")) - vereq(x.a, None) + verify(not hasattr(x, "a")) x.a = 1 vereq(x.a, 1) + x.a = None + veris(x.a, None) del x.a - vereq(x.a, None) + verify(not hasattr(x, "a")) class C3(object): __slots__ = ['a', 'b', 'c'] x = C3() verify(not hasattr(x, "__dict__")) - verify(x.a is None) - verify(x.b is None) - verify(x.c is None) + verify(not hasattr(x, 'a')) + verify(not hasattr(x, 'b')) + verify(not hasattr(x, 'c')) x.a = 1 x.b = 2 x.c = 3 diff --git a/Python/structmember.c b/Python/structmember.c index aa0ea9f..34310b8 100644 --- a/Python/structmember.c +++ b/Python/structmember.c @@ -129,6 +129,12 @@ PyMember_GetOne(char *addr, PyMemberDef *l) v = Py_None; Py_INCREF(v); break; + case T_OBJECT_EX: + v = *(PyObject **)addr; + if (v == NULL) + PyErr_SetString(PyExc_AttributeError, l->name); + Py_XINCREF(v); + break; default: PyErr_SetString(PyExc_SystemError, "bad memberdescr type"); v = NULL; @@ -175,7 +181,7 @@ PyMember_SetOne(char *addr, PyMemberDef *l, PyObject *v) PyErr_SetString(PyExc_RuntimeError, "restricted attribute"); return -1; } - if (v == NULL && l->type != T_OBJECT) { + if (v == NULL && l->type != T_OBJECT_EX && l->type != T_OBJECT) { PyErr_SetString(PyExc_TypeError, "can't delete numeric/char attribute"); return -1; @@ -246,6 +252,7 @@ PyMember_SetOne(char *addr, PyMemberDef *l, PyObject *v) } break; case T_OBJECT: + case T_OBJECT_EX: Py_XINCREF(v); oldv = *(PyObject **)addr; *(PyObject **)addr = v; |