mirror of
https://github.com/python/cpython.git
synced 2025-07-07 19:35:27 +00:00
gh-133931: Introduce _PyObject_XSetRefDelayed to replace Py_XSETREF (gh-134377)
Some checks are pending
Tests / Change detection (push) Waiting to run
Tests / Docs (push) Blocked by required conditions
Tests / Check if Autoconf files are up to date (push) Blocked by required conditions
Tests / Check if generated files are up to date (push) Blocked by required conditions
Tests / (push) Blocked by required conditions
Tests / Windows MSI (push) Blocked by required conditions
Tests / Ubuntu SSL tests with OpenSSL (push) Blocked by required conditions
Tests / WASI (push) Blocked by required conditions
Tests / Hypothesis tests on Ubuntu (push) Blocked by required conditions
Tests / Address sanitizer (push) Blocked by required conditions
Tests / Cross build Linux (push) Blocked by required conditions
Tests / CIFuzz (push) Blocked by required conditions
Tests / All required checks pass (push) Blocked by required conditions
Lint / lint (push) Waiting to run
mypy / Run mypy on Lib/_pyrepl (push) Waiting to run
mypy / Run mypy on Lib/test/libregrtest (push) Waiting to run
mypy / Run mypy on Lib/tomllib (push) Waiting to run
mypy / Run mypy on Tools/build (push) Waiting to run
mypy / Run mypy on Tools/cases_generator (push) Waiting to run
mypy / Run mypy on Tools/clinic (push) Waiting to run
mypy / Run mypy on Tools/jit (push) Waiting to run
mypy / Run mypy on Tools/peg_generator (push) Waiting to run
Some checks are pending
Tests / Change detection (push) Waiting to run
Tests / Docs (push) Blocked by required conditions
Tests / Check if Autoconf files are up to date (push) Blocked by required conditions
Tests / Check if generated files are up to date (push) Blocked by required conditions
Tests / (push) Blocked by required conditions
Tests / Windows MSI (push) Blocked by required conditions
Tests / Ubuntu SSL tests with OpenSSL (push) Blocked by required conditions
Tests / WASI (push) Blocked by required conditions
Tests / Hypothesis tests on Ubuntu (push) Blocked by required conditions
Tests / Address sanitizer (push) Blocked by required conditions
Tests / Cross build Linux (push) Blocked by required conditions
Tests / CIFuzz (push) Blocked by required conditions
Tests / All required checks pass (push) Blocked by required conditions
Lint / lint (push) Waiting to run
mypy / Run mypy on Lib/_pyrepl (push) Waiting to run
mypy / Run mypy on Lib/test/libregrtest (push) Waiting to run
mypy / Run mypy on Lib/tomllib (push) Waiting to run
mypy / Run mypy on Tools/build (push) Waiting to run
mypy / Run mypy on Tools/cases_generator (push) Waiting to run
mypy / Run mypy on Tools/clinic (push) Waiting to run
mypy / Run mypy on Tools/jit (push) Waiting to run
mypy / Run mypy on Tools/peg_generator (push) Waiting to run
This commit is contained in:
parent
cb39410111
commit
52be7f445e
6 changed files with 104 additions and 21 deletions
|
@ -767,6 +767,27 @@ _Py_TryIncref(PyObject *op)
|
|||
#endif
|
||||
}
|
||||
|
||||
// Enqueue an object to be freed possibly after some delay
|
||||
#ifdef Py_GIL_DISABLED
|
||||
PyAPI_FUNC(void) _PyObject_XDecRefDelayed(PyObject *obj);
|
||||
#else
|
||||
static inline void _PyObject_XDecRefDelayed(PyObject *obj)
|
||||
{
|
||||
Py_XDECREF(obj);
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef Py_GIL_DISABLED
|
||||
// Same as `Py_XSETREF` but in free-threading, it stores the object atomically
|
||||
// and queues the old object to be decrefed at a safe point using QSBR.
|
||||
PyAPI_FUNC(void) _PyObject_XSetRefDelayed(PyObject **p_obj, PyObject *obj);
|
||||
#else
|
||||
static inline void _PyObject_XSetRefDelayed(PyObject **p_obj, PyObject *obj)
|
||||
{
|
||||
Py_XSETREF(*p_obj, obj);
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef Py_REF_DEBUG
|
||||
extern void _PyInterpreterState_FinalizeRefTotal(PyInterpreterState *);
|
||||
extern void _Py_FinalizeRefTotal(_PyRuntimeState *);
|
||||
|
|
|
@ -90,16 +90,6 @@ extern int _PyMem_DebugEnabled(void);
|
|||
// Enqueue a pointer to be freed possibly after some delay.
|
||||
extern void _PyMem_FreeDelayed(void *ptr);
|
||||
|
||||
// Enqueue an object to be freed possibly after some delay
|
||||
#ifdef Py_GIL_DISABLED
|
||||
PyAPI_FUNC(void) _PyObject_XDecRefDelayed(PyObject *obj);
|
||||
#else
|
||||
static inline void _PyObject_XDecRefDelayed(PyObject *obj)
|
||||
{
|
||||
Py_XDECREF(obj);
|
||||
}
|
||||
#endif
|
||||
|
||||
// Periodically process delayed free requests.
|
||||
extern void _PyMem_ProcessDelayed(PyThreadState *tstate);
|
||||
|
||||
|
|
51
Lib/test/test_free_threading/test_generators.py
Normal file
51
Lib/test/test_free_threading/test_generators.py
Normal file
|
@ -0,0 +1,51 @@
|
|||
import concurrent.futures
|
||||
import unittest
|
||||
from threading import Barrier
|
||||
from unittest import TestCase
|
||||
import random
|
||||
import time
|
||||
|
||||
from test.support import threading_helper, Py_GIL_DISABLED
|
||||
|
||||
threading_helper.requires_working_threading(module=True)
|
||||
|
||||
|
||||
def random_sleep():
|
||||
delay_us = random.randint(50, 100)
|
||||
time.sleep(delay_us * 1e-6)
|
||||
|
||||
def random_string():
|
||||
return ''.join(random.choice('0123456789ABCDEF') for _ in range(10))
|
||||
|
||||
def set_gen_name(g, b):
|
||||
b.wait()
|
||||
random_sleep()
|
||||
g.__name__ = random_string()
|
||||
return g.__name__
|
||||
|
||||
def set_gen_qualname(g, b):
|
||||
b.wait()
|
||||
random_sleep()
|
||||
g.__qualname__ = random_string()
|
||||
return g.__qualname__
|
||||
|
||||
|
||||
@unittest.skipUnless(Py_GIL_DISABLED, "Enable only in FT build")
|
||||
class TestFTGenerators(TestCase):
|
||||
NUM_THREADS = 4
|
||||
|
||||
def concurrent_write_with_func(self, func):
|
||||
gen = (x for x in range(42))
|
||||
for j in range(1000):
|
||||
with concurrent.futures.ThreadPoolExecutor(max_workers=self.NUM_THREADS) as executor:
|
||||
b = Barrier(self.NUM_THREADS)
|
||||
futures = {executor.submit(func, gen, b): i for i in range(self.NUM_THREADS)}
|
||||
for fut in concurrent.futures.as_completed(futures):
|
||||
gen_name = fut.result()
|
||||
self.assertEqual(len(gen_name), 10)
|
||||
|
||||
def test_concurrent_write(self):
|
||||
with self.subTest(func=set_gen_name):
|
||||
self.concurrent_write_with_func(func=set_gen_name)
|
||||
with self.subTest(func=set_gen_qualname):
|
||||
self.concurrent_write_with_func(func=set_gen_qualname)
|
|
@ -704,7 +704,8 @@ static PyObject *
|
|||
gen_get_name(PyObject *self, void *Py_UNUSED(ignored))
|
||||
{
|
||||
PyGenObject *op = _PyGen_CAST(self);
|
||||
return Py_NewRef(op->gi_name);
|
||||
PyObject *name = FT_ATOMIC_LOAD_PTR_ACQUIRE(op->gi_name);
|
||||
return Py_NewRef(name);
|
||||
}
|
||||
|
||||
static int
|
||||
|
@ -718,7 +719,11 @@ gen_set_name(PyObject *self, PyObject *value, void *Py_UNUSED(ignored))
|
|||
"__name__ must be set to a string object");
|
||||
return -1;
|
||||
}
|
||||
Py_XSETREF(op->gi_name, Py_NewRef(value));
|
||||
Py_BEGIN_CRITICAL_SECTION(self);
|
||||
// gh-133931: To prevent use-after-free from other threads that reference
|
||||
// the gi_name.
|
||||
_PyObject_XSetRefDelayed(&op->gi_name, Py_NewRef(value));
|
||||
Py_END_CRITICAL_SECTION();
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
@ -726,7 +731,8 @@ static PyObject *
|
|||
gen_get_qualname(PyObject *self, void *Py_UNUSED(ignored))
|
||||
{
|
||||
PyGenObject *op = _PyGen_CAST(self);
|
||||
return Py_NewRef(op->gi_qualname);
|
||||
PyObject *qualname = FT_ATOMIC_LOAD_PTR_ACQUIRE(op->gi_qualname);
|
||||
return Py_NewRef(qualname);
|
||||
}
|
||||
|
||||
static int
|
||||
|
@ -740,7 +746,11 @@ gen_set_qualname(PyObject *self, PyObject *value, void *Py_UNUSED(ignored))
|
|||
"__qualname__ must be set to a string object");
|
||||
return -1;
|
||||
}
|
||||
Py_XSETREF(op->gi_qualname, Py_NewRef(value));
|
||||
Py_BEGIN_CRITICAL_SECTION(self);
|
||||
// gh-133931: To prevent use-after-free from other threads that reference
|
||||
// the gi_qualname.
|
||||
_PyObject_XSetRefDelayed(&op->gi_qualname, Py_NewRef(value));
|
||||
Py_END_CRITICAL_SECTION();
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
|
|
@ -1231,6 +1231,21 @@ _PyObject_XDecRefDelayed(PyObject *ptr)
|
|||
}
|
||||
#endif
|
||||
|
||||
#ifdef Py_GIL_DISABLED
|
||||
void
|
||||
_PyObject_XSetRefDelayed(PyObject **ptr, PyObject *value)
|
||||
{
|
||||
PyObject *old = *ptr;
|
||||
FT_ATOMIC_STORE_PTR_RELEASE(*ptr, value);
|
||||
if (old == NULL) {
|
||||
return;
|
||||
}
|
||||
if (!_Py_IsImmortal(old)) {
|
||||
_PyObject_XDecRefDelayed(old);
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
static struct _mem_work_chunk *
|
||||
work_queue_first(struct llist_node *head)
|
||||
{
|
||||
|
|
|
@ -3967,13 +3967,9 @@ _PyObject_SetDict(PyObject *obj, PyObject *value)
|
|||
return -1;
|
||||
}
|
||||
Py_BEGIN_CRITICAL_SECTION(obj);
|
||||
PyObject *olddict = *dictptr;
|
||||
FT_ATOMIC_STORE_PTR_RELEASE(*dictptr, Py_NewRef(value));
|
||||
#ifdef Py_GIL_DISABLED
|
||||
_PyObject_XDecRefDelayed(olddict);
|
||||
#else
|
||||
Py_XDECREF(olddict);
|
||||
#endif
|
||||
// gh-133980: To prevent use-after-free from other threads that reference
|
||||
// the __dict__
|
||||
_PyObject_XSetRefDelayed(dictptr, Py_NewRef(value));
|
||||
Py_END_CRITICAL_SECTION();
|
||||
return 0;
|
||||
}
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue