Skip to content

Commit b73e3b6

Browse files
authored
GH-95589: Dont crash when subclassing extension classes with multiple inheritance (GH-96028)
* Treat tp_weakref and tp_dictoffset like other opaque slots for multiple inheritance. * Document Py_TPFLAGS_MANAGED_DICT and Py_TPFLAGS_MANAGED_WEAKREF in what's new.
1 parent 0f2b469 commit b73e3b6

File tree

4 files changed

+67
-43
lines changed

4 files changed

+67
-43
lines changed

Doc/whatsnew/3.12.rst

+17
Original file line numberDiff line numberDiff line change
@@ -450,6 +450,11 @@ New Features
450450
inherit the ``Py_TPFLAGS_HAVE_VECTORCALL`` flag.
451451
(Contributed by Petr Viktorin in :gh:`93274`.)
452452

453+
The :const:`Py_TPFLAGS_MANAGED_DICT` and :const:`Py_TPFLAGS_MANAGED_WEAKREF`
454+
flags have been added. This allows extensions classes to support object
455+
``__dict__`` and weakrefs with less bookkeeping,
456+
using less memory and with faster access.
457+
453458
Porting to Python 3.12
454459
----------------------
455460

@@ -486,6 +491,18 @@ Porting to Python 3.12
486491
:c:func:`PyUnicode_FromFormatV`.
487492
(Contributed by Philip Georgi in :gh:`95504`.)
488493

494+
* Extension classes wanting to add a ``__dict__`` or weak reference slot
495+
should use :const:`Py_TPFLAGS_MANAGED_DICT` and
496+
:const:`Py_TPFLAGS_MANAGED_WEAKREF` instead of ``tp_dictoffset`` and
497+
``tp_weaklistoffset``, respectively.
498+
The use of ``tp_dictoffset`` and ``tp_weaklistoffset`` is still
499+
supported, but does not fully support multiple inheritance
500+
(:gh: `95589`), and performance may be worse.
501+
Classes declaring :const:`Py_TPFLAGS_MANAGED_DICT` should call
502+
:c:func:`_PyObject_VisitManagedDict` and :c:func:`_PyObject_ClearManagedDict`
503+
to traverse and clear their instance's dictionaries.
504+
To clear weakrefs, call :c:func:`PyObject_ClearWeakRefs`, as before.
505+
489506
Deprecated
490507
----------
491508

Lib/test/test_capi.py

+33-11
Original file line numberDiff line numberDiff line change
@@ -677,21 +677,43 @@ def test_heaptype_with_custom_metaclass(self):
677677

678678
def test_multiple_inheritance_ctypes_with_weakref_or_dict(self):
679679

680-
class Both1(_testcapi.HeapCTypeWithWeakref, _testcapi.HeapCTypeWithDict):
680+
with self.assertRaises(TypeError):
681+
class Both1(_testcapi.HeapCTypeWithWeakref, _testcapi.HeapCTypeWithDict):
682+
pass
683+
with self.assertRaises(TypeError):
684+
class Both2(_testcapi.HeapCTypeWithDict, _testcapi.HeapCTypeWithWeakref):
685+
pass
686+
687+
def test_multiple_inheritance_ctypes_with_weakref_or_dict_and_other_builtin(self):
688+
689+
with self.assertRaises(TypeError):
690+
class C1(_testcapi.HeapCTypeWithDict, list):
691+
pass
692+
693+
with self.assertRaises(TypeError):
694+
class C2(_testcapi.HeapCTypeWithWeakref, list):
695+
pass
696+
697+
class C3(_testcapi.HeapCTypeWithManagedDict, list):
681698
pass
682-
class Both2(_testcapi.HeapCTypeWithDict, _testcapi.HeapCTypeWithWeakref):
699+
class C4(_testcapi.HeapCTypeWithManagedWeakref, list):
683700
pass
684701

685-
for cls in (_testcapi.HeapCTypeWithDict, _testcapi.HeapCTypeWithDict2,
686-
_testcapi.HeapCTypeWithWeakref, _testcapi.HeapCTypeWithWeakref2):
687-
for cls2 in (_testcapi.HeapCTypeWithDict, _testcapi.HeapCTypeWithDict2,
688-
_testcapi.HeapCTypeWithWeakref, _testcapi.HeapCTypeWithWeakref2):
689-
if cls is not cls2:
690-
class S(cls, cls2):
691-
pass
692-
class B1(Both1, cls):
702+
inst = C3()
703+
inst.append(0)
704+
str(inst.__dict__)
705+
706+
inst = C4()
707+
inst.append(0)
708+
str(inst.__weakref__)
709+
710+
for cls in (_testcapi.HeapCTypeWithManagedDict, _testcapi.HeapCTypeWithManagedWeakref):
711+
for cls2 in (_testcapi.HeapCTypeWithDict, _testcapi.HeapCTypeWithWeakref):
712+
class S(cls, cls2):
713+
pass
714+
class B1(C3, cls):
693715
pass
694-
class B2(Both1, cls):
716+
class B2(C4, cls):
695717
pass
696718

697719
def test_pytype_fromspec_with_repeated_slots(self):
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
Extensions classes that set ``tp_dictoffset`` and ``tp_weaklistoffset``
2+
lose the support for multiple inheritance, but are now safe. Extension
3+
classes should use :const:`Py_TPFLAGS_MANAGED_DICT` and
4+
:const:`Py_TPFLAGS_MANAGED_WEAKREF` instead.

Objects/typeobject.c

+13-32
Original file line numberDiff line numberDiff line change
@@ -2330,51 +2330,32 @@ best_base(PyObject *bases)
23302330
return base;
23312331
}
23322332

2333-
#define ADDED_FIELD_AT_OFFSET(name, offset) \
2334-
(type->tp_ ## name && (base->tp_ ##name == 0) && \
2335-
type->tp_ ## name + sizeof(PyObject *) == (offset) && \
2336-
type->tp_flags & Py_TPFLAGS_HEAPTYPE)
2337-
23382333
static int
2339-
extra_ivars(PyTypeObject *type, PyTypeObject *base)
2334+
shape_differs(PyTypeObject *t1, PyTypeObject *t2)
23402335
{
2341-
size_t t_size = type->tp_basicsize;
2342-
size_t b_size = base->tp_basicsize;
2343-
2344-
assert(t_size >= b_size); /* Else type smaller than base! */
2345-
if (type->tp_itemsize || base->tp_itemsize) {
2346-
/* If itemsize is involved, stricter rules */
2347-
return t_size != b_size ||
2348-
type->tp_itemsize != base->tp_itemsize;
2349-
}
2350-
/* Check for __dict__ and __weakrefs__ slots in either order */
2351-
if (ADDED_FIELD_AT_OFFSET(weaklistoffset, t_size)) {
2352-
t_size -= sizeof(PyObject *);
2353-
}
2354-
if ((type->tp_flags & Py_TPFLAGS_MANAGED_DICT) == 0 &&
2355-
ADDED_FIELD_AT_OFFSET(dictoffset, t_size)) {
2356-
t_size -= sizeof(PyObject *);
2357-
}
2358-
/* Check __weakrefs__ again, in case it precedes __dict__ */
2359-
if (ADDED_FIELD_AT_OFFSET(weaklistoffset, t_size)) {
2360-
t_size -= sizeof(PyObject *);
2361-
}
2362-
return t_size != b_size;
2336+
return (
2337+
t1->tp_basicsize != t2->tp_basicsize ||
2338+
t1->tp_itemsize != t2->tp_itemsize
2339+
);
23632340
}
23642341

23652342
static PyTypeObject *
23662343
solid_base(PyTypeObject *type)
23672344
{
23682345
PyTypeObject *base;
23692346

2370-
if (type->tp_base)
2347+
if (type->tp_base) {
23712348
base = solid_base(type->tp_base);
2372-
else
2349+
}
2350+
else {
23732351
base = &PyBaseObject_Type;
2374-
if (extra_ivars(type, base))
2352+
}
2353+
if (shape_differs(type, base)) {
23752354
return type;
2376-
else
2355+
}
2356+
else {
23772357
return base;
2358+
}
23782359
}
23792360

23802361
static void object_dealloc(PyObject *);

0 commit comments

Comments
 (0)