diff options
author | Jeremy Hylton <jeremy@alum.mit.edu> | 2004-08-07 19:20:05 (GMT) |
---|---|---|
committer | Jeremy Hylton <jeremy@alum.mit.edu> | 2004-08-07 19:20:05 (GMT) |
commit | 4c989ddc9c8bd38ab14c9f42511eb567ed219604 (patch) | |
tree | 9e91f6cf6dd52b386920c4123132f846b94e4817 /Objects/stringobject.c | |
parent | cbd81556bb975b72697d2989eccfbc9f3e39b3de (diff) | |
download | cpython-4c989ddc9c8bd38ab14c9f42511eb567ed219604.zip cpython-4c989ddc9c8bd38ab14c9f42511eb567ed219604.tar.gz cpython-4c989ddc9c8bd38ab14c9f42511eb567ed219604.tar.bz2 |
Subclasses of string can no longer be interned. The semantics of
interning were not clear here -- a subclass could be mutable, for
example -- and had bugs. Explicitly interning a subclass of string
via intern() will raise a TypeError. Internal operations that attempt
to intern a string subclass will have no effect.
Added a few tests to test_builtin that includes the old buggy code and
verifies that calls like PyObject_SetAttr() don't fail. Perhaps these
tests should have gone in test_string.
Diffstat (limited to 'Objects/stringobject.c')
-rw-r--r-- | Objects/stringobject.c | 34 |
1 files changed, 12 insertions, 22 deletions
diff --git a/Objects/stringobject.c b/Objects/stringobject.c index 29562a9..f29929b 100644 --- a/Objects/stringobject.c +++ b/Objects/stringobject.c @@ -4313,6 +4313,10 @@ PyString_InternInPlace(PyObject **p) PyObject *t; if (s == NULL || !PyString_Check(s)) Py_FatalError("PyString_InternInPlace: strings only please!"); + /* If it's a string subclass, we don't really know what putting + it in the interned dict might do. */ + if (!PyString_CheckExact(s)) + return; if (PyString_CHECK_INTERNED(s)) return; if (interned == NULL) { @@ -4322,36 +4326,22 @@ PyString_InternInPlace(PyObject **p) return; } } - if ((t = PyDict_GetItem(interned, (PyObject *)s)) != NULL) { + t = PyDict_GetItem(interned, (PyObject *)s); + if (t) { Py_INCREF(t); Py_DECREF(*p); *p = t; return; } - /* Ensure that only true string objects appear in the intern dict */ - if (!PyString_CheckExact(s)) { - t = PyString_FromStringAndSize(PyString_AS_STRING(s), - PyString_GET_SIZE(s)); - if (t == NULL) { - PyErr_Clear(); - return; - } - } else { - t = (PyObject*) s; - Py_INCREF(t); - } - if (PyDict_SetItem(interned, t, t) == 0) { - /* The two references in interned are not counted by - refcnt. The string deallocator will take care of this */ - ((PyObject *)t)->ob_refcnt-=2; - PyString_CHECK_INTERNED(t) = SSTATE_INTERNED_MORTAL; - Py_DECREF(*p); - *p = t; + if (PyDict_SetItem(interned, s, s) < 0) { + PyErr_Clear(); return; } - Py_DECREF(t); - PyErr_Clear(); + /* The two references in interned are not counted by refcnt. + The string deallocator will take care of this */ + (*p)->ob_refcnt -= 2; + PyString_CHECK_INTERNED(s) = SSTATE_INTERNED_MORTAL; } void |