summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--Lib/test/test_asyncio/test_eager_task_factory.py19
-rw-r--r--Misc/NEWS.d/next/Library/2023-06-22-15-21-11.gh-issue-105987.T7Kzrb.rst1
-rw-r--r--Modules/_asynciomodule.c11
3 files changed, 27 insertions, 4 deletions
diff --git a/Lib/test/test_asyncio/test_eager_task_factory.py b/Lib/test/test_asyncio/test_eager_task_factory.py
index 49d5dbb..fc9ad8e 100644
--- a/Lib/test/test_asyncio/test_eager_task_factory.py
+++ b/Lib/test/test_asyncio/test_eager_task_factory.py
@@ -7,6 +7,8 @@ import unittest
from unittest import mock
from asyncio import tasks
from test.test_asyncio import utils as test_utils
+import test.support
+from test.support.script_helper import assert_python_ok
MOCK_ANY = mock.ANY
@@ -222,6 +224,23 @@ class PyEagerTaskFactoryLoopTests(EagerTaskFactoryLoopTests, test_utils.TestCase
class CEagerTaskFactoryLoopTests(EagerTaskFactoryLoopTests, test_utils.TestCase):
Task = getattr(tasks, '_CTask', None)
+ def test_issue105987(self):
+ code = """if 1:
+ from _asyncio import _swap_current_task
+
+ class DummyTask:
+ pass
+
+ class DummyLoop:
+ pass
+
+ l = DummyLoop()
+ _swap_current_task(l, DummyTask())
+ t = _swap_current_task(l, None)
+ """
+
+ _, out, err = assert_python_ok("-c", code)
+ self.assertFalse(err)
class AsyncTaskCounter:
def __init__(self, loop, *, task_class, eager):
diff --git a/Misc/NEWS.d/next/Library/2023-06-22-15-21-11.gh-issue-105987.T7Kzrb.rst b/Misc/NEWS.d/next/Library/2023-06-22-15-21-11.gh-issue-105987.T7Kzrb.rst
new file mode 100644
index 0000000..0bc97da
--- /dev/null
+++ b/Misc/NEWS.d/next/Library/2023-06-22-15-21-11.gh-issue-105987.T7Kzrb.rst
@@ -0,0 +1 @@
+Fix crash due to improper reference counting in :mod:`asyncio` eager task factory internal routines.
diff --git a/Modules/_asynciomodule.c b/Modules/_asynciomodule.c
index 4b84c1d..5b28f2d 100644
--- a/Modules/_asynciomodule.c
+++ b/Modules/_asynciomodule.c
@@ -2047,20 +2047,23 @@ swap_current_task(asyncio_state *state, PyObject *loop, PyObject *task)
}
prev_task = Py_None;
}
+ Py_INCREF(prev_task);
if (task == Py_None) {
if (_PyDict_DelItem_KnownHash(state->current_tasks, loop, hash) == -1) {
- return NULL;
+ goto error;
}
} else {
if (_PyDict_SetItem_KnownHash(state->current_tasks, loop, task, hash) == -1) {
- return NULL;
+ goto error;
}
}
- Py_INCREF(prev_task);
-
return prev_task;
+
+error:
+ Py_DECREF(prev_task);
+ return NULL;
}
/* ----- Task */