summaryrefslogtreecommitdiffstats
path: root/Python
diff options
context:
space:
mode:
authorSam Gross <colesbury@gmail.com>2024-04-08 16:11:36 (GMT)
committerGitHub <noreply@github.com>2024-04-08 16:11:36 (GMT)
commit1a6594f66166206b08f24c3ba633c85f86f99a56 (patch)
tree961c4d6e2f694e482636643993cd17ebbf9ea2fd /Python
parent2067da25796ea3254d0edf61a39bcc0326c4f71d (diff)
downloadcpython-1a6594f66166206b08f24c3ba633c85f86f99a56.zip
cpython-1a6594f66166206b08f24c3ba633c85f86f99a56.tar.gz
cpython-1a6594f66166206b08f24c3ba633c85f86f99a56.tar.bz2
gh-117439: Make refleak checking thread-safe without the GIL (#117469)
This keeps track of the per-thread total reference count operations in PyThreadState in the free-threaded builds. The count is merged into the interpreter's total when the thread exits.
Diffstat (limited to 'Python')
-rw-r--r--Python/gc_free_threading.c4
-rw-r--r--Python/pystate.c8
2 files changed, 10 insertions, 2 deletions
diff --git a/Python/gc_free_threading.c b/Python/gc_free_threading.c
index 7e4137a..111632f 100644
--- a/Python/gc_free_threading.c
+++ b/Python/gc_free_threading.c
@@ -168,7 +168,7 @@ merge_refcount(PyObject *op, Py_ssize_t extra)
refcount += extra;
#ifdef Py_REF_DEBUG
- _Py_AddRefTotal(_PyInterpreterState_GET(), extra);
+ _Py_AddRefTotal(_PyThreadState_GET(), extra);
#endif
// No atomics necessary; all other threads in this interpreter are paused.
@@ -307,7 +307,7 @@ merge_queued_objects(_PyThreadStateImpl *tstate, struct collection_state *state)
// decref and deallocate the object once we start the world again.
op->ob_ref_shared += (1 << _Py_REF_SHARED_SHIFT);
#ifdef Py_REF_DEBUG
- _Py_IncRefTotal(_PyInterpreterState_GET());
+ _Py_IncRefTotal(_PyThreadState_GET());
#endif
worklist_push(&state->objs_to_decref, op);
}
diff --git a/Python/pystate.c b/Python/pystate.c
index cee481c..4a52f64 100644
--- a/Python/pystate.c
+++ b/Python/pystate.c
@@ -1698,6 +1698,14 @@ tstate_delete_common(PyThreadState *tstate)
decrement_stoptheworld_countdown(&runtime->stoptheworld);
}
}
+
+#if defined(Py_REF_DEBUG) && defined(Py_GIL_DISABLED)
+ // Add our portion of the total refcount to the interpreter's total.
+ _PyThreadStateImpl *tstate_impl = (_PyThreadStateImpl *)tstate;
+ tstate->interp->object_state.reftotal += tstate_impl->reftotal;
+ tstate_impl->reftotal = 0;
+#endif
+
HEAD_UNLOCK(runtime);
#ifdef Py_GIL_DISABLED