Skip to content

Commit 12db452

Browse files
authored
gh-130851: Only intern constants of types generated by the compiler (#130901)
The free-threading build interns and immortalizes most constants generated by the bytecode compiler. However, users can construct their own code objects with arbitrary constants. We should not intern or immortalize these objects if they are not of a type that we know how to handle. This change fixes a reference leak failure in the recently added `test_code.test_unusual_constants` test. It also addresses a potential crash that could occur when attempting to destroy an immortalized object during interpreter shutdown.
1 parent c476410 commit 12db452

File tree

1 file changed

+41
-2
lines changed

1 file changed

+41
-2
lines changed

Objects/codeobject.c

+41-2
Original file line numberDiff line numberDiff line change
@@ -135,6 +135,44 @@ should_intern_string(PyObject *o)
135135

136136
#ifdef Py_GIL_DISABLED
137137
static PyObject *intern_one_constant(PyObject *op);
138+
139+
// gh-130851: In the free threading build, we intern and immortalize most
140+
// constants, except code objects. However, users can generate code objects
141+
// with arbitrary co_consts. We don't want to immortalize or intern unexpected
142+
// constants or tuples/sets containing unexpected constants.
143+
static int
144+
should_immortalize_constant(PyObject *v)
145+
{
146+
// Only immortalize containers if we've already immortalized all their
147+
// elements.
148+
if (PyTuple_CheckExact(v)) {
149+
for (Py_ssize_t i = PyTuple_GET_SIZE(v); --i >= 0; ) {
150+
if (!_Py_IsImmortal(PyTuple_GET_ITEM(v, i))) {
151+
return 0;
152+
}
153+
}
154+
return 1;
155+
}
156+
else if (PyFrozenSet_CheckExact(v)) {
157+
PyObject *item;
158+
Py_hash_t hash;
159+
Py_ssize_t pos = 0;
160+
while (_PySet_NextEntry(v, &pos, &item, &hash)) {
161+
if (!_Py_IsImmortal(item)) {
162+
return 0;
163+
}
164+
}
165+
return 1;
166+
}
167+
else if (PySlice_Check(v)) {
168+
PySliceObject *slice = (PySliceObject *)v;
169+
return (_Py_IsImmortal(slice->start) &&
170+
_Py_IsImmortal(slice->stop) &&
171+
_Py_IsImmortal(slice->step));
172+
}
173+
return (PyLong_CheckExact(v) || PyFloat_CheckExact(v) ||
174+
PyComplex_Check(v) || PyBytes_CheckExact(v));
175+
}
138176
#endif
139177

140178
static int
@@ -241,8 +279,9 @@ intern_constants(PyObject *tuple, int *modified)
241279

242280
// Intern non-string constants in the free-threaded build
243281
_PyThreadStateImpl *tstate = (_PyThreadStateImpl *)_PyThreadState_GET();
244-
if (!_Py_IsImmortal(v) && !PyCode_Check(v) &&
245-
!PyUnicode_CheckExact(v) && !tstate->suppress_co_const_immortalization)
282+
if (!_Py_IsImmortal(v) && !PyUnicode_CheckExact(v) &&
283+
should_immortalize_constant(v) &&
284+
!tstate->suppress_co_const_immortalization)
246285
{
247286
PyObject *interned = intern_one_constant(v);
248287
if (interned == NULL) {

0 commit comments

Comments
 (0)