-
-
Notifications
You must be signed in to change notification settings - Fork 31.5k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Race in stgdict PyCStructUnionType_update_stginfo under free-threading #128570
Comments
import ctypes
import threading
if __name__ == "__main__":
num_workers = 100
def closure(barrier, cls, i):
fields = [(f'i{i}', ctypes.c_int * i)]
barrier.wait()
try:
cls._fields_ = fields
except AttributeError as exc:
pass
while True:
class cls(ctypes.Structure):
pass
barrier = threading.Barrier(num_workers)
threads = []
for i in range(num_workers):
thread = threading.Thread(target=closure, args=(barrier, cls, i))
threads.append(thread)
thread.start()
for thread in threads:
thread.join()
print(cls._fields_) Results: $ ./python ../../free/check2.py
[('i24', <class '__main__.c_int_Array_24'>)]
[('i98', <class '__main__.c_int_Array_98'>)]
[('i97', <class '__main__.c_int_Array_97'>)]
[('i97', <class '__main__.c_int_Array_97'>)]
[('i98', <class '__main__.c_int_Array_98'>)]
mimalloc: error: double free detected of block 0x20000091050 with size 48
Segmentation fault (core dumped)
$ ./python ../../free/check2.py
[('i99', <class '__main__.c_int_Array_99'>)]
[('i78', <class '__main__.c_int_Array_78'>)]
[('i99', <class '__main__.c_int_Array_99'>)]
Debug memory block at address p=0x20000090b50: API '�'
16102724419182469211 bytes originally requested
The 7 pad bytes at p-7 are not all FORBIDDENBYTE (0xfd):
at p-7: 0xdd *** OUCH
at p-6: 0xdd *** OUCH
at p-5: 0xdd *** OUCH
at p-4: 0xdd *** OUCH
at p-3: 0xdd *** OUCH
at p-2: 0xdd *** OUCH
at p-1: 0xdd *** OUCH
Because memory is corrupted at the start, the count of bytes requested
may be bogus, and checking the trailing pad bytes may segfault.
The 8 pad bytes at tail=0xdf78608b196a2bab are Segmentation fault (core dumped) Also in some other functions like |
I haven't fully decided what to do about stgdict's yet. I'm leaning towards just slapping big critical sections on them, but I need to look through what fields are written so we can go for locks or atomics as needed. |
A mutex fixed the problem for me. Global instead of per-type because of possibility of $ git diff main
diff --git a/Modules/_ctypes/_ctypes.c b/Modules/_ctypes/_ctypes.c
index ede95bdf98..f6479df571 100644
--- a/Modules/_ctypes/_ctypes.c
+++ b/Modules/_ctypes/_ctypes.c
@@ -699,11 +699,14 @@ StructUnionType_init(PyObject *self, PyObject *args, PyObject *kwds, int isStruc
}
/* copy base info */
+ PyMutex_Lock(&st->mutex);
if (PyCStgInfo_clone(info, baseinfo) < 0) {
+ PyMutex_Unlock(&st->mutex);
return -1;
}
info->flags &= ~DICTFLAG_FINAL; /* clear the 'final' flag in the subclass info */
baseinfo->flags |= DICTFLAG_FINAL; /* set the 'final' flag in the baseclass info */
+ PyMutex_Unlock(&st->mutex);
}
return 0;
}
@@ -3138,7 +3141,9 @@ PyCData_FromBaseObj(ctypes_state *st,
return NULL;
}
+ PyMutex_Lock(&st->mutex);
info->flags |= DICTFLAG_FINAL;
+ PyMutex_Unlock(&st->mutex);
cmem = (CDataObject *)((PyTypeObject *)type)->tp_alloc((PyTypeObject *)type, 0);
if (cmem == NULL) {
return NULL;
@@ -3187,7 +3192,9 @@ PyCData_AtAddress(ctypes_state *st, PyObject *type, void *buf)
return NULL;
}
+ PyMutex_Lock(&st->mutex);
info->flags |= DICTFLAG_FINAL;
+ PyMutex_Unlock(&st->mutex);
pd = (CDataObject *)((PyTypeObject *)type)->tp_alloc((PyTypeObject *)type, 0);
if (!pd) {
@@ -3401,7 +3408,9 @@ generic_pycdata_new(ctypes_state *st,
return NULL;
}
+ PyMutex_Lock(&st->mutex);
info->flags |= DICTFLAG_FINAL;
+ PyMutex_Unlock(&st->mutex);
obj = (CDataObject *)type->tp_alloc(type, 0);
if (!obj)
diff --git a/Modules/_ctypes/ctypes.h b/Modules/_ctypes/ctypes.h
index 45e00a538f..82688f0084 100644
--- a/Modules/_ctypes/ctypes.h
+++ b/Modules/_ctypes/ctypes.h
@@ -75,6 +75,7 @@ typedef struct {
PyObject *error_object_name; // callproc.c
PyObject *PyExc_ArgError;
PyObject *swapped_suffix;
+ PyMutex mutex;
} ctypes_state; Not sure is in all needed places though or fits desired convention so I leave it there if u are messing with it. |
That's not safe, we need to use critical sections ( |
Bug report
Bug description:
I built cpython (3.13 branch) with free-threading and TSAN. The following python code reports TSAN warnings:
TSAN report extract:
Full traceback
Related to #127945
CPython versions tested on:
3.13
Operating systems tested on:
Linux
The text was updated successfully, but these errors were encountered: