diff options
author | T. Wouters <thomas@python.org> | 2021-07-15 22:40:57 (GMT) |
---|---|---|
committer | GitHub <noreply@github.com> | 2021-07-15 22:40:57 (GMT) |
commit | 074e7659f208051b6b973f7fdb654dd22b93aaa2 (patch) | |
tree | aacfbcb0a7c3a783f9263812e92ad194d7c0d4bf | |
parent | 82b218f36ce6ef910bda5af227a9fd5be613c94f (diff) | |
download | cpython-074e7659f208051b6b973f7fdb654dd22b93aaa2.zip cpython-074e7659f208051b6b973f7fdb654dd22b93aaa2.tar.gz cpython-074e7659f208051b6b973f7fdb654dd22b93aaa2.tar.bz2 |
bpo-44184: Apply GH-26274 to the non-GC-type branch of subtype_dealloc (GH-27165)
The non-GC-type branch of subtype_dealloc is using the type of an object after freeing in the same unsafe way as GH-26274 fixes. (I believe the old news entry covers this change well enough.)
https://bugs.python.org/issue44184
-rw-r--r-- | Objects/typeobject.c | 14 |
1 files changed, 11 insertions, 3 deletions
diff --git a/Objects/typeobject.c b/Objects/typeobject.c index 02ea618..3331fee 100644 --- a/Objects/typeobject.c +++ b/Objects/typeobject.c @@ -1368,14 +1368,22 @@ subtype_dealloc(PyObject *self) /* Extract the type again; tp_del may have changed it */ type = Py_TYPE(self); + // Don't read type memory after calling basedealloc() since basedealloc() + // can deallocate the type and free its memory. + int type_needs_decref = (type->tp_flags & Py_TPFLAGS_HEAPTYPE + && !(base->tp_flags & Py_TPFLAGS_HEAPTYPE)); + /* Call the base tp_dealloc() */ assert(basedealloc); basedealloc(self); - /* Only decref if the base type is not already a heap allocated type. - Otherwise, basedealloc should have decref'd it already */ - if (type->tp_flags & Py_TPFLAGS_HEAPTYPE && !(base->tp_flags & Py_TPFLAGS_HEAPTYPE)) + /* Can't reference self beyond this point. It's possible tp_del switched + our type from a HEAPTYPE to a non-HEAPTYPE, so be careful about + reference counting. Only decref if the base type is not already a heap + allocated type. Otherwise, basedealloc should have decref'd it already */ + if (type_needs_decref) { Py_DECREF(type); + } /* Done */ return; |