Skip to content

Commit

Permalink
pythongh-119525: Fix deadlock with _PyType_Lookup and the GIL
Browse files Browse the repository at this point in the history
The deadlock only affected the free-threaded build and only occurred
when the GIL was enabled at runtime. The `Py_DECREF(old_name)` call
might temporarily release the GIL while holding the type seqlock.
Another thread may spin trying to acquire the seqlock while holding the
GIL.

The deadlock occurred roughly 1 in ~1,000 runs of `pool_in_threads.py`
from `test_multiprocessing_pool_circular_import`.
  • Loading branch information
colesbury committed May 24, 2024
1 parent 81d6336 commit d50ae9f
Show file tree
Hide file tree
Showing 2 changed files with 9 additions and 4 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
Fix deadlock involving ``_PyType_Lookup()`` cache in the free-threaded build
when the GIL is dynamically enabled at runtime.
11 changes: 7 additions & 4 deletions Objects/typeobject.c
Original file line number Diff line number Diff line change
Expand Up @@ -5165,7 +5165,7 @@ is_dunder_name(PyObject *name)
return 0;
}

static void
static PyObject *
update_cache(struct type_cache_entry *entry, PyObject *name, unsigned int version_tag, PyObject *value)
{
_Py_atomic_store_uint32_relaxed(&entry->version, version_tag);
Expand All @@ -5176,7 +5176,7 @@ update_cache(struct type_cache_entry *entry, PyObject *name, unsigned int versio
// exact unicode object or Py_None so it's safe to do so.
PyObject *old_name = entry->name;
_Py_atomic_store_ptr_relaxed(&entry->name, Py_NewRef(name));
Py_DECREF(old_name);
return old_name;
}

#if Py_GIL_DISABLED
Expand All @@ -5196,10 +5196,12 @@ update_cache_gil_disabled(struct type_cache_entry *entry, PyObject *name,
return;
}

update_cache(entry, name, version_tag, value);
PyObject *old_value = update_cache(entry, name, version_tag, value);

// Then update sequence to the next valid value
_PySeqLock_UnlockWrite(&entry->sequence);

Py_DECREF(old_value);
}

#endif
Expand Down Expand Up @@ -5311,7 +5313,8 @@ _PyType_LookupRef(PyTypeObject *type, PyObject *name)
#if Py_GIL_DISABLED
update_cache_gil_disabled(entry, name, version, res);
#else
update_cache(entry, name, version, res);
PyObject *old_value = update_cache(entry, name, version, res);
Py_DECREF(old_value);
#endif
}
return res;
Expand Down

0 comments on commit d50ae9f

Please sign in to comment.