summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorPeter Bierma <zintensitydev@gmail.com>2024-09-26 10:29:43 (GMT)
committerGitHub <noreply@github.com>2024-09-26 10:29:43 (GMT)
commitf923605658a29ff9af5a62edc1fc10191977627b (patch)
treefa26fb24c84fc965468c0f68254451f78fa30314
parent0387c34f7c91428681ca8a4ba4e3d22b9acffde4 (diff)
downloadcpython-f923605658a29ff9af5a62edc1fc10191977627b.zip
cpython-f923605658a29ff9af5a62edc1fc10191977627b.tar.gz
cpython-f923605658a29ff9af5a62edc1fc10191977627b.tar.bz2
gh-124538: Fix crash when using `gc.get_referents` on an untracked capsule object (#124559)
-rw-r--r--Lib/test/test_gc.py18
-rw-r--r--Misc/NEWS.d/next/Library/2024-09-25-18-34-48.gh-issue-124538.nXZk4R.rst1
-rw-r--r--Objects/capsule.c10
3 files changed, 26 insertions, 3 deletions
diff --git a/Lib/test/test_gc.py b/Lib/test/test_gc.py
index 906f988..bb7df1f 100644
--- a/Lib/test/test_gc.py
+++ b/Lib/test/test_gc.py
@@ -1048,6 +1048,24 @@ class GCTests(unittest.TestCase):
callback.assert_not_called()
gc.enable()
+ @cpython_only
+ def test_get_referents_on_capsule(self):
+ # gh-124538: Calling gc.get_referents() on an untracked capsule must not crash.
+ import _datetime
+ import _socket
+ untracked_capsule = _datetime.datetime_CAPI
+ tracked_capsule = _socket.CAPI
+
+ # For whoever sees this in the future: if this is failing
+ # after making datetime's capsule tracked, that's fine -- this isn't something
+ # users are relying on. Just find a different capsule that is untracked.
+ self.assertFalse(gc.is_tracked(untracked_capsule))
+ self.assertTrue(gc.is_tracked(tracked_capsule))
+
+ self.assertEqual(len(gc.get_referents(untracked_capsule)), 0)
+ gc.get_referents(tracked_capsule)
+
+
class IncrementalGCTests(unittest.TestCase):
diff --git a/Misc/NEWS.d/next/Library/2024-09-25-18-34-48.gh-issue-124538.nXZk4R.rst b/Misc/NEWS.d/next/Library/2024-09-25-18-34-48.gh-issue-124538.nXZk4R.rst
new file mode 100644
index 0000000..33ae037
--- /dev/null
+++ b/Misc/NEWS.d/next/Library/2024-09-25-18-34-48.gh-issue-124538.nXZk4R.rst
@@ -0,0 +1 @@
+Fixed crash when using :func:`gc.get_referents` on a capsule object.
diff --git a/Objects/capsule.c b/Objects/capsule.c
index 555979d..28965e0 100644
--- a/Objects/capsule.c
+++ b/Objects/capsule.c
@@ -317,10 +317,14 @@ static int
capsule_traverse(PyCapsule *capsule, visitproc visit, void *arg)
{
// Capsule object is only tracked by the GC
- // if _PyCapsule_SetTraverse() is called
- assert(capsule->traverse_func != NULL);
+ // if _PyCapsule_SetTraverse() is called, but
+ // this can still be manually triggered by gc.get_referents()
+
+ if (capsule->traverse_func != NULL) {
+ return capsule->traverse_func((PyObject*)capsule, visit, arg);
+ }
- return capsule->traverse_func((PyObject*)capsule, visit, arg);
+ return 0;
}