Skip to content
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
12 changes: 11 additions & 1 deletion Include/cpython/object.h
Original file line number Diff line number Diff line change
Expand Up @@ -221,17 +221,27 @@ struct _typeobject {
PyObject *tp_weaklist; /* not used for static builtin types */
destructor tp_del;

/* Type attribute cache version tag. Added in version 2.6 */
/* Type attribute cache version tag. Added in version 2.6.
* If zero, the cache is invalid and must be initialized.
*/
unsigned int tp_version_tag;

destructor tp_finalize;
vectorcallfunc tp_vectorcall;

/* bitset of which type-watchers care about this type */
unsigned char tp_watched;

/* Number of tp_version_tag values used.
* Set to _Py_ATTR_CACHE_UNUSED if the attribute cache is
* disabled for this type (e.g. due to custom MRO entries).
* Otherwise, limited to MAX_VERSIONS_PER_CLASS (defined elsewhere).
*/
uint16_t tp_versions_used;
};

#define _Py_ATTR_CACHE_UNUSED (30000) // (see tp_versions_used)

/* This struct is used by the specializer
* It should be treated as an opaque blob
* by code other than the specializer and interpreter. */
Expand Down
27 changes: 27 additions & 0 deletions Lib/test/test_metaclass.py
Original file line number Diff line number Diff line change
Expand Up @@ -254,6 +254,33 @@
[...]
test.test_metaclass.ObscureException

Test setting attributes with a non-base type in mro() (gh-127773).

>>> class Base:
... value = 1
...
>>> class Meta(type):
... def mro(cls):
... return (cls, Base, object)
...
>>> class WeirdClass(metaclass=Meta):
... pass
...
>>> Base.value
1
>>> WeirdClass.value
1
>>> Base.value = 2
>>> Base.value
2
>>> WeirdClass.value
2
>>> Base.value = 3
>>> Base.value
3
>>> WeirdClass.value
3

"""

import sys
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Do not use the type attribute cache for types with incompatible MRO.
11 changes: 10 additions & 1 deletion Objects/typeobject.c
Original file line number Diff line number Diff line change
Expand Up @@ -992,6 +992,7 @@
set_version_unlocked(PyTypeObject *tp, unsigned int version)
{
ASSERT_TYPE_LOCK_HELD();
assert(version == 0 || (tp->tp_versions_used != _Py_ATTR_CACHE_UNUSED));
#ifndef Py_GIL_DISABLED
PyInterpreterState *interp = _PyInterpreterState_GET();
// lookup the old version and set to null
Expand Down Expand Up @@ -1148,6 +1149,10 @@
PyObject *b = PyTuple_GET_ITEM(bases, i);
PyTypeObject *cls = _PyType_CAST(b);

if (cls->tp_versions_used >= _Py_ATTR_CACHE_UNUSED) {
goto clear;
}

if (!is_subtype_with_mro(lookup_tp_mro(type), type, cls)) {
goto clear;
}
Expand All @@ -1156,7 +1161,8 @@

clear:
assert(!(type->tp_flags & _Py_TPFLAGS_STATIC_BUILTIN));
set_version_unlocked(type, 0); /* 0 is not a valid version tag */
set_version_unlocked(type, 0); /* 0 is not a valid version tag */
type->tp_versions_used = _Py_ATTR_CACHE_UNUSED;
if (PyType_HasFeature(type, Py_TPFLAGS_HEAPTYPE)) {
// This field *must* be invalidated if the type is modified (see the
// comment on struct _specialization_cache):
Expand Down Expand Up @@ -1208,6 +1214,8 @@


#define MAX_VERSIONS_PER_CLASS 1000
_Static_assert(MAX_VERSIONS_PER_CLASS < _Py_ATTR_CACHE_UNUSED,

Check failure on line 1217 in Objects/typeobject.c

View workflow job for this annotation

GitHub Actions / Windows / build and test (arm64)

syntax error: missing ')' before 'constant' [D:\a\cpython\cpython\PCbuild\_freeze_module.vcxproj]

Check failure on line 1217 in Objects/typeobject.c

View workflow job for this annotation

GitHub Actions / Windows / build and test (arm64)

syntax error: missing '{' before 'constant' [D:\a\cpython\cpython\PCbuild\_freeze_module.vcxproj]

Check failure on line 1217 in Objects/typeobject.c

View workflow job for this annotation

GitHub Actions / Windows / build and test (arm64)

syntax error: 'constant' [D:\a\cpython\cpython\PCbuild\_freeze_module.vcxproj]

Check failure on line 1217 in Objects/typeobject.c

View workflow job for this annotation

GitHub Actions / Windows (free-threading) / build and test (arm64)

syntax error: missing ')' before 'constant' [D:\a\cpython\cpython\PCbuild\_freeze_module.vcxproj]

Check failure on line 1217 in Objects/typeobject.c

View workflow job for this annotation

GitHub Actions / Windows (free-threading) / build and test (arm64)

syntax error: missing '{' before 'constant' [D:\a\cpython\cpython\PCbuild\_freeze_module.vcxproj]

Check failure on line 1217 in Objects/typeobject.c

View workflow job for this annotation

GitHub Actions / Windows (free-threading) / build and test (arm64)

syntax error: 'constant' [D:\a\cpython\cpython\PCbuild\_freeze_module.vcxproj]

Check failure on line 1217 in Objects/typeobject.c

View workflow job for this annotation

GitHub Actions / Windows / build and test (x64)

syntax error: missing ')' before 'constant' [D:\a\cpython\cpython\PCbuild\_freeze_module.vcxproj]

Check failure on line 1217 in Objects/typeobject.c

View workflow job for this annotation

GitHub Actions / Windows / build and test (x64)

syntax error: missing '{' before 'constant' [D:\a\cpython\cpython\PCbuild\_freeze_module.vcxproj]

Check failure on line 1217 in Objects/typeobject.c

View workflow job for this annotation

GitHub Actions / Windows / build and test (x64)

syntax error: 'constant' [D:\a\cpython\cpython\PCbuild\_freeze_module.vcxproj]

Check failure on line 1217 in Objects/typeobject.c

View workflow job for this annotation

GitHub Actions / Windows (free-threading) / build and test (x64)

syntax error: missing ')' before 'constant' [D:\a\cpython\cpython\PCbuild\_freeze_module.vcxproj]

Check failure on line 1217 in Objects/typeobject.c

View workflow job for this annotation

GitHub Actions / Windows (free-threading) / build and test (x64)

syntax error: missing '{' before 'constant' [D:\a\cpython\cpython\PCbuild\_freeze_module.vcxproj]

Check failure on line 1217 in Objects/typeobject.c

View workflow job for this annotation

GitHub Actions / Windows (free-threading) / build and test (x64)

syntax error: 'constant' [D:\a\cpython\cpython\PCbuild\_freeze_module.vcxproj]
"_Py_ATTR_CACHE_UNUSED must be bigger than max");

static int
assign_version_tag(PyInterpreterState *interp, PyTypeObject *type)
Expand All @@ -1225,6 +1233,7 @@
return 0;
}
if (type->tp_versions_used >= MAX_VERSIONS_PER_CLASS) {
/* (this includes `tp_versions_used == _Py_ATTR_CACHE_UNUSED`) */
return 0;
}

Expand Down
Loading