diff options
author | Donghee Na <donghee.na@python.org> | 2024-08-22 23:49:09 +0900 |
---|---|---|
committer | GitHub <noreply@github.com> | 2024-08-22 23:49:09 +0900 |
commit | 297f2e093ec95800ae2184330b8408c875523467 (patch) | |
tree | 5c1b6debe984d76e5874136146b854f98847cbb4 /Python/executor_cases.c.h | |
parent | 4abc1c1456413f3d2692257545a33bb16b24f900 (diff) | |
download | cpython-297f2e093ec95800ae2184330b8408c875523467.tar.gz cpython-297f2e093ec95800ae2184330b8408c875523467.zip |
gh-123083: Fix a potential use-after-free in ``STORE_ATTR_WITH_HINT`` (gh-123092)
Diffstat (limited to 'Python/executor_cases.c.h')
-rw-r--r-- | Python/executor_cases.c.h | 13 |
1 files changed, 7 insertions, 6 deletions
diff --git a/Python/executor_cases.c.h b/Python/executor_cases.c.h index 55b06a0e235..4274d51b3fa 100644 --- a/Python/executor_cases.c.h +++ b/Python/executor_cases.c.h @@ -2637,18 +2637,19 @@ UOP_STAT_INC(uopcode, miss); JUMP_TO_JUMP_TARGET(); } + /* Ensure dict is GC tracked if it needs to be */ + if (!_PyObject_GC_IS_TRACKED(dict) && _PyObject_GC_MAY_BE_TRACKED(PyStackRef_AsPyObjectBorrow(value))) { + _PyObject_GC_TRACK(dict); + } old_value = ep->me_value; PyDict_WatchEvent event = old_value == NULL ? PyDict_EVENT_ADDED : PyDict_EVENT_MODIFIED; new_version = _PyDict_NotifyEvent(tstate->interp, event, dict, name, PyStackRef_AsPyObjectBorrow(value)); ep->me_value = PyStackRef_AsPyObjectSteal(value); + dict->ma_version_tag = new_version; // PEP 509 + // old_value should be DECREFed after GC track checking is done, if not, it could raise a segmentation fault, + // when dict only holds the strong reference to value in ep->me_value. Py_XDECREF(old_value); STAT_INC(STORE_ATTR, hit); - /* Ensure dict is GC tracked if it needs to be */ - if (!_PyObject_GC_IS_TRACKED(dict) && _PyObject_GC_MAY_BE_TRACKED(PyStackRef_AsPyObjectBorrow(value))) { - _PyObject_GC_TRACK(dict); - } - /* PEP 509 */ - dict->ma_version_tag = new_version; PyStackRef_CLOSE(owner); stack_pointer += -2; assert(WITHIN_STACK_BOUNDS()); |