finally:
_testcapi.clear_dict_watcher(wid)
+ def test_racing_split_dict_clear_and_lookup(self):
+ class C:
+ pass
+
+ keys = [f"a{i}" for i in range(16)]
+
+ def make_split_nonembedded():
+ inst = C()
+ for key in keys:
+ setattr(inst, key, keys.index(key))
+ # dict.copy() of a split instance dict yields a split table
+ # with non-embedded values
+ return inst.__dict__.copy()
+
+ d = make_split_nonembedded()
+
+ def clearer():
+ for _ in range(1000):
+ d.clear()
+ d.update(make_split_nonembedded())
+
+ def reader():
+ for _ in range(1000):
+ for k in keys:
+ d.get(k)
+
+ threading_helper.run_concurrently([clearer, reader, reader])
+
def test_racing_dict_update_and_method_lookup(self):
# gh-144295: test race between dict modifications and method lookups.
# Uses BytesIO because the race requires a type without Py_TPFLAGS_INLINE_VALUES
set_keys(mp, Py_EMPTY_KEYS);
n = oldkeys->dk_nentries;
for (i = 0; i < n; i++) {
- Py_CLEAR(oldvalues->values[i]);
+ PyObject *tmp = oldvalues->values[i];
+ FT_ATOMIC_STORE_PTR_RELEASE(oldvalues->values[i], NULL);
+ Py_XDECREF(tmp);
}
free_values(oldvalues, IS_DICT_SHARED(mp));
- dictkeys_decref(oldkeys, false);
+ dictkeys_decref(oldkeys, IS_DICT_SHARED(mp));
}
ASSERT_CONSISTENT(mp);
}