mirror of
https://github.com/python/cpython.git
synced 2026-04-14 15:50:50 +00:00
[3.14] gh-144438: Fix false sharing between QSBR and tlbc_index (gh-144554) (#144923)
Align the QSBR thread state array to a 64-byte cache line boundary
and add padding at the end of _PyThreadStateImpl. Depending on heap
layout, the QSBR array could end up sharing a cache line with a
thread's tlbc_index, causing QSBR quiescent state updates to contend
with reads of tlbc_index in RESUME_CHECK. This is sensitive to
earlier allocations during interpreter init and can appear or
disappear with seemingly unrelated changes.
Either change alone is sufficient to fix the specific issue, but both
are worthwhile to avoid similar problems in the future.
(cherry picked from commit 6577d870b0)
This commit is contained in:
parent
25b48b84b8
commit
6ea4f842fb
5 changed files with 1657 additions and 1600 deletions
|
|
@ -84,22 +84,29 @@ grow_thread_array(struct _qsbr_shared *shared)
|
|||
new_size = MIN_ARRAY_SIZE;
|
||||
}
|
||||
|
||||
struct _qsbr_pad *array = PyMem_RawCalloc(new_size, sizeof(*array));
|
||||
if (array == NULL) {
|
||||
// Overallocate by 63 bytes so we can align to a 64-byte boundary.
|
||||
// This avoids potential false sharing between the first entry and other
|
||||
// allocations.
|
||||
size_t alignment = 64;
|
||||
size_t alloc_size = (size_t)new_size * sizeof(struct _qsbr_pad) + alignment - 1;
|
||||
void *raw = PyMem_RawCalloc(1, alloc_size);
|
||||
if (raw == NULL) {
|
||||
return -1;
|
||||
}
|
||||
struct _qsbr_pad *array = _Py_ALIGN_UP(raw, alignment);
|
||||
|
||||
struct _qsbr_pad *old = shared->array;
|
||||
if (old != NULL) {
|
||||
void *old_raw = shared->array_raw;
|
||||
if (shared->array != NULL) {
|
||||
memcpy(array, shared->array, shared->size * sizeof(*array));
|
||||
}
|
||||
|
||||
shared->array = array;
|
||||
shared->array_raw = raw;
|
||||
shared->size = new_size;
|
||||
shared->freelist = NULL;
|
||||
initialize_new_array(shared);
|
||||
|
||||
PyMem_RawFree(old);
|
||||
PyMem_RawFree(old_raw);
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
|
@ -256,8 +263,9 @@ void
|
|||
_Py_qsbr_fini(PyInterpreterState *interp)
|
||||
{
|
||||
struct _qsbr_shared *shared = &interp->qsbr;
|
||||
PyMem_RawFree(shared->array);
|
||||
PyMem_RawFree(shared->array_raw);
|
||||
shared->array = NULL;
|
||||
shared->array_raw = NULL;
|
||||
shared->size = 0;
|
||||
shared->freelist = NULL;
|
||||
}
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue