mirror of
https://github.com/python/cpython.git
synced 2025-09-26 18:29:57 +00:00
gh-120198: Fix race condition when editing __class__ with an audit hook active (GH-120195)
Co-authored-by: Nadeshiko Manju <me@manjusaka.me>
This commit is contained in:
parent
0315fdc24d
commit
e8a9f0c9e7
3 changed files with 37 additions and 2 deletions
|
@ -1,9 +1,10 @@
|
|||
"""Unit tests for zero-argument super() & related machinery."""
|
||||
|
||||
import textwrap
|
||||
import threading
|
||||
import unittest
|
||||
from unittest.mock import patch
|
||||
from test.support import import_helper
|
||||
from test.support import import_helper, threading_helper
|
||||
|
||||
|
||||
ADAPTIVE_WARMUP_DELAY = 2
|
||||
|
@ -478,6 +479,38 @@ class TestSuper(unittest.TestCase):
|
|||
for _ in range(ADAPTIVE_WARMUP_DELAY):
|
||||
C.some(C)
|
||||
|
||||
@threading_helper.requires_working_threading()
|
||||
def test___class___modification_multithreaded(self):
|
||||
""" Note: this test isn't actually testing anything on its own.
|
||||
It requires a sys audithook to be set to crash on older Python.
|
||||
This should be the case anyways as our test suite sets
|
||||
an audit hook.
|
||||
"""
|
||||
class Foo:
|
||||
pass
|
||||
|
||||
class Bar:
|
||||
pass
|
||||
|
||||
thing = Foo()
|
||||
def work():
|
||||
foo = thing
|
||||
for _ in range(5000):
|
||||
foo.__class__ = Bar
|
||||
type(foo)
|
||||
foo.__class__ = Foo
|
||||
type(foo)
|
||||
|
||||
|
||||
threads = []
|
||||
for _ in range(6):
|
||||
thread = threading.Thread(target=work)
|
||||
thread.start()
|
||||
threads.append(thread)
|
||||
|
||||
for thread in threads:
|
||||
thread.join()
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
unittest.main()
|
||||
|
|
|
@ -0,0 +1 @@
|
|||
Fix a crash when multiple threads read and write to the same ``__class__`` of an object concurrently.
|
|
@ -5709,7 +5709,6 @@ differs:
|
|||
static int
|
||||
object_set_class(PyObject *self, PyObject *value, void *closure)
|
||||
{
|
||||
PyTypeObject *oldto = Py_TYPE(self);
|
||||
|
||||
if (value == NULL) {
|
||||
PyErr_SetString(PyExc_TypeError,
|
||||
|
@ -5729,6 +5728,8 @@ object_set_class(PyObject *self, PyObject *value, void *closure)
|
|||
return -1;
|
||||
}
|
||||
|
||||
PyTypeObject *oldto = Py_TYPE(self);
|
||||
|
||||
/* In versions of CPython prior to 3.5, the code in
|
||||
compatible_for_assignment was not set up to correctly check for memory
|
||||
layout / slot / etc. compatibility for non-HEAPTYPE classes, so we just
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue