Skip to content

Commit db96327

Browse files
vstinnerencukou
andauthored
gh-121654: Add PyType_Freeze() function (#122457)
Co-authored-by: Petr Viktorin <encukou@gmail.com>
1 parent da8673d commit db96327

File tree

11 files changed

+162
-13
lines changed

11 files changed

+162
-13
lines changed

Doc/c-api/type.rst

+14
Original file line numberDiff line numberDiff line change
@@ -413,6 +413,20 @@ The following functions and structs are used to create
413413
Creating classes whose metaclass overrides
414414
:c:member:`~PyTypeObject.tp_new` is no longer allowed.
415415
416+
.. c:function:: int PyType_Freeze(PyTypeObject *type)
417+
418+
Make a type immutable: set the :c:macro:`Py_TPFLAGS_IMMUTABLETYPE` flag.
419+
420+
All base classes of *type* must be immutable.
421+
422+
On success, return ``0``.
423+
On error, set an exception and return ``-1``.
424+
425+
The type must not be used before it's made immutable. For example, type
426+
instances must not be created before the type is made immutable.
427+
428+
.. versionadded:: 3.14
429+
416430
.. raw:: html
417431
418432
<!-- Keep old URL fragments working (see gh-97908) -->

Doc/data/stable_abi.dat

+1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

Doc/whatsnew/3.14.rst

+3
Original file line numberDiff line numberDiff line change
@@ -777,6 +777,9 @@ New features
777777
(Contributed by Victor Stinner in :gh:`124502`.)
778778

779779

780+
* Add :c:func:`PyType_Freeze` function to make a type immutable.
781+
(Contributed by Victor Stinner in :gh:`121654`.)
782+
780783
Porting to Python 3.14
781784
----------------------
782785

Include/object.h

+4
Original file line numberDiff line numberDiff line change
@@ -796,6 +796,10 @@ static inline int PyType_CheckExact(PyObject *op) {
796796
PyAPI_FUNC(PyObject *) PyType_GetModuleByDef(PyTypeObject *, PyModuleDef *);
797797
#endif
798798

799+
#if !defined(Py_LIMITED_API) || Py_LIMITED_API+0 >= 0x030e0000
800+
PyAPI_FUNC(int) PyType_Freeze(PyTypeObject *type);
801+
#endif
802+
799803
#ifdef __cplusplus
800804
}
801805
#endif

Lib/test/test_capi/test_type.py

+66
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,66 @@
1+
from test.support import import_helper
2+
import unittest
3+
4+
_testcapi = import_helper.import_module('_testcapi')
5+
6+
7+
class TypeTests(unittest.TestCase):
8+
def test_freeze(self):
9+
# test PyType_Freeze()
10+
type_freeze = _testcapi.type_freeze
11+
12+
# simple case, no inherante
13+
class MyType:
14+
pass
15+
MyType.attr = "mutable"
16+
17+
type_freeze(MyType)
18+
err_msg = "cannot set 'attr' attribute of immutable type 'MyType'"
19+
with self.assertRaisesRegex(TypeError, err_msg):
20+
# the class is now immutable
21+
MyType.attr = "immutable"
22+
23+
# test MRO: PyType_Freeze() requires base classes to be immutable
24+
class A: pass
25+
class B: pass
26+
class C(B): pass
27+
class D(A, C): pass
28+
29+
self.assertEqual(D.mro(), [D, A, C, B, object])
30+
with self.assertRaises(TypeError):
31+
type_freeze(D)
32+
33+
type_freeze(A)
34+
type_freeze(B)
35+
type_freeze(C)
36+
# all parent classes are now immutable, so D can be made immutable
37+
# as well
38+
type_freeze(D)
39+
40+
def test_freeze_meta(self):
41+
"""test PyType_Freeze() with overridden MRO"""
42+
type_freeze = _testcapi.type_freeze
43+
44+
class Base:
45+
value = 1
46+
47+
class Meta(type):
48+
def mro(cls):
49+
return (cls, Base, object)
50+
51+
class FreezeThis(metaclass=Meta):
52+
"""This has `Base` in the MRO, but not tp_bases"""
53+
54+
self.assertEqual(FreezeThis.value, 1)
55+
56+
with self.assertRaises(TypeError):
57+
type_freeze(FreezeThis)
58+
59+
Base.value = 2
60+
self.assertEqual(FreezeThis.value, 2)
61+
62+
type_freeze(Base)
63+
with self.assertRaises(TypeError):
64+
Base.value = 3
65+
type_freeze(FreezeThis)
66+
self.assertEqual(FreezeThis.value, 2)

Lib/test/test_stable_abi_ctypes.py

+1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
Add :c:func:`PyType_Freeze` function to make a type immutable. Patch by
2+
Victor Stinner.

Misc/stable_abi.toml

+2
Original file line numberDiff line numberDiff line change
@@ -2538,3 +2538,5 @@
25382538
added = '3.14'
25392539
[function.PyUnicode_Equal]
25402540
added = '3.14'
2541+
[function.PyType_Freeze]
2542+
added = '3.14'

Modules/_testcapimodule.c

+16
Original file line numberDiff line numberDiff line change
@@ -3310,6 +3310,7 @@ test_critical_sections(PyObject *module, PyObject *Py_UNUSED(args))
33103310
Py_RETURN_NONE;
33113311
}
33123312

3313+
33133314
// Used by `finalize_thread_hang`.
33143315
#ifdef _POSIX_THREADS
33153316
static void finalize_thread_hang_cleanup_callback(void *Py_UNUSED(arg)) {
@@ -3339,6 +3340,20 @@ finalize_thread_hang(PyObject *self, PyObject *callback)
33393340
}
33403341

33413342

3343+
static PyObject *
3344+
type_freeze(PyObject *module, PyObject *args)
3345+
{
3346+
PyTypeObject *type;
3347+
if (!PyArg_ParseTuple(args, "O!", &PyType_Type, &type)) {
3348+
return NULL;
3349+
}
3350+
if (PyType_Freeze(type) < 0) {
3351+
return NULL;
3352+
}
3353+
Py_RETURN_NONE;
3354+
}
3355+
3356+
33423357
static PyMethodDef TestMethods[] = {
33433358
{"set_errno", set_errno, METH_VARARGS},
33443359
{"test_config", test_config, METH_NOARGS},
@@ -3479,6 +3494,7 @@ static PyMethodDef TestMethods[] = {
34793494
{"function_set_warning", function_set_warning, METH_NOARGS},
34803495
{"test_critical_sections", test_critical_sections, METH_NOARGS},
34813496
{"finalize_thread_hang", finalize_thread_hang, METH_O, NULL},
3497+
{"type_freeze", type_freeze, METH_VARARGS},
34823498
{NULL, NULL} /* sentinel */
34833499
};
34843500

Objects/typeobject.c

+52-13
Original file line numberDiff line numberDiff line change
@@ -4637,6 +4637,32 @@ check_basicsize_includes_size_and_offsets(PyTypeObject* type)
46374637
return 1;
46384638
}
46394639

4640+
static int
4641+
check_immutable_bases(const char *type_name, PyObject *bases, int skip_first)
4642+
{
4643+
Py_ssize_t i = 0;
4644+
if (skip_first) {
4645+
// When testing the MRO, skip the type itself
4646+
i = 1;
4647+
}
4648+
for (; i<PyTuple_GET_SIZE(bases); i++) {
4649+
PyTypeObject *b = (PyTypeObject*)PyTuple_GET_ITEM(bases, i);
4650+
if (!b) {
4651+
return -1;
4652+
}
4653+
if (!_PyType_HasFeature(b, Py_TPFLAGS_IMMUTABLETYPE)) {
4654+
PyErr_Format(
4655+
PyExc_TypeError,
4656+
"Creating immutable type %s from mutable base %N",
4657+
type_name, b
4658+
);
4659+
return -1;
4660+
}
4661+
}
4662+
return 0;
4663+
}
4664+
4665+
46404666
/* Set *dest to the offset specified by a special "__*offset__" member.
46414667
* Return 0 on success, -1 on failure.
46424668
*/
@@ -4820,19 +4846,8 @@ PyType_FromMetaclass(
48204846
* and only heap types can be mutable.)
48214847
*/
48224848
if (spec->flags & Py_TPFLAGS_IMMUTABLETYPE) {
4823-
for (int i=0; i<PyTuple_GET_SIZE(bases); i++) {
4824-
PyTypeObject *b = (PyTypeObject*)PyTuple_GET_ITEM(bases, i);
4825-
if (!b) {
4826-
goto finally;
4827-
}
4828-
if (!_PyType_HasFeature(b, Py_TPFLAGS_IMMUTABLETYPE)) {
4829-
PyErr_Format(
4830-
PyExc_TypeError,
4831-
"Creating immutable type %s from mutable base %N",
4832-
spec->name, b
4833-
);
4834-
goto finally;
4835-
}
4849+
if (check_immutable_bases(spec->name, bases, 0) < 0) {
4850+
goto finally;
48364851
}
48374852
}
48384853

@@ -11319,6 +11334,30 @@ add_operators(PyTypeObject *type)
1131911334
}
1132011335

1132111336

11337+
int
11338+
PyType_Freeze(PyTypeObject *type)
11339+
{
11340+
// gh-121654: Check the __mro__ instead of __bases__
11341+
PyObject *mro = type_get_mro(type, NULL);
11342+
if (!PyTuple_Check(mro)) {
11343+
Py_DECREF(mro);
11344+
PyErr_SetString(PyExc_TypeError, "unable to get the type MRO");
11345+
return -1;
11346+
}
11347+
11348+
int check = check_immutable_bases(type->tp_name, mro, 1);
11349+
Py_DECREF(mro);
11350+
if (check < 0) {
11351+
return -1;
11352+
}
11353+
11354+
type->tp_flags |= Py_TPFLAGS_IMMUTABLETYPE;
11355+
PyType_Modified(type);
11356+
11357+
return 0;
11358+
}
11359+
11360+
1132211361
/* Cooperative 'super' */
1132311362

1132411363
typedef struct {

PC/python3dll.c

+1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

0 commit comments

Comments
 (0)