mirror of
https://github.com/python/cpython.git
synced 2025-08-30 13:38:43 +00:00
gh-104341: Adjust tstate_must_exit() to Respect Interpreter Finalization (gh-104437)
With the move to a per-interpreter GIL, this check slipped through the cracks.
This commit is contained in:
parent
cb88ae635e
commit
26baa747c2
11 changed files with 56 additions and 10 deletions
|
@ -198,7 +198,7 @@ get_warnings_attr(PyInterpreterState *interp, PyObject *attr, int try_import)
|
|||
PyObject *warnings_module, *obj;
|
||||
|
||||
/* don't try to import after the start of the Python finallization */
|
||||
if (try_import && !_Py_IsFinalizing()) {
|
||||
if (try_import && !_Py_IsInterpreterFinalizing(interp)) {
|
||||
warnings_module = PyImport_Import(&_Py_ID(warnings));
|
||||
if (warnings_module == NULL) {
|
||||
/* Fallback to the C implementation if we cannot get
|
||||
|
|
|
@ -332,6 +332,9 @@ tstate_must_exit(PyThreadState *tstate)
|
|||
After Py_Finalize() has been called, tstate can be a dangling pointer:
|
||||
point to PyThreadState freed memory. */
|
||||
PyThreadState *finalizing = _PyRuntimeState_GetFinalizing(&_PyRuntime);
|
||||
if (finalizing == NULL) {
|
||||
finalizing = _PyInterpreterState_GetFinalizing(tstate->interp);
|
||||
}
|
||||
return (finalizing != NULL && finalizing != tstate);
|
||||
}
|
||||
|
||||
|
|
|
@ -1788,6 +1788,7 @@ Py_FinalizeEx(void)
|
|||
|
||||
/* Remaining daemon threads will automatically exit
|
||||
when they attempt to take the GIL (ex: PyEval_RestoreThread()). */
|
||||
_PyInterpreterState_SetFinalizing(tstate->interp, tstate);
|
||||
_PyRuntimeState_SetFinalizing(runtime, tstate);
|
||||
runtime->initialized = 0;
|
||||
runtime->core_initialized = 0;
|
||||
|
@ -2142,6 +2143,10 @@ Py_EndInterpreter(PyThreadState *tstate)
|
|||
Py_FatalError("not the last thread");
|
||||
}
|
||||
|
||||
/* Remaining daemon threads will automatically exit
|
||||
when they attempt to take the GIL (ex: PyEval_RestoreThread()). */
|
||||
_PyInterpreterState_SetFinalizing(interp, tstate);
|
||||
|
||||
// XXX Call something like _PyImport_Disable() here?
|
||||
|
||||
_PyImport_FiniExternal(tstate->interp);
|
||||
|
@ -2152,6 +2157,18 @@ Py_EndInterpreter(PyThreadState *tstate)
|
|||
finalize_interp_delete(tstate->interp);
|
||||
}
|
||||
|
||||
int
|
||||
_Py_IsInterpreterFinalizing(PyInterpreterState *interp)
|
||||
{
|
||||
/* We check the runtime first since, in a daemon thread,
|
||||
interp might be dangling pointer. */
|
||||
PyThreadState *finalizing = _PyRuntimeState_GetFinalizing(&_PyRuntime);
|
||||
if (finalizing == NULL) {
|
||||
finalizing = _PyInterpreterState_GetFinalizing(interp);
|
||||
}
|
||||
return finalizing != NULL;
|
||||
}
|
||||
|
||||
/* Add the __main__ module */
|
||||
|
||||
static PyStatus
|
||||
|
|
|
@ -1436,11 +1436,13 @@ PyThreadState_Clear(PyThreadState *tstate)
|
|||
|
||||
if (verbose && tstate->cframe->current_frame != NULL) {
|
||||
/* bpo-20526: After the main thread calls
|
||||
_PyRuntimeState_SetFinalizing() in Py_FinalizeEx(), threads must
|
||||
exit when trying to take the GIL. If a thread exit in the middle of
|
||||
_PyEval_EvalFrameDefault(), tstate->frame is not reset to its
|
||||
previous value. It is more likely with daemon threads, but it can
|
||||
happen with regular threads if threading._shutdown() fails
|
||||
_PyInterpreterState_SetFinalizing() in Py_FinalizeEx()
|
||||
(or in Py_EndInterpreter() for subinterpreters),
|
||||
threads must exit when trying to take the GIL.
|
||||
If a thread exit in the middle of _PyEval_EvalFrameDefault(),
|
||||
tstate->frame is not reset to its previous value.
|
||||
It is more likely with daemon threads, but it can happen
|
||||
with regular threads if threading._shutdown() fails
|
||||
(ex: interrupted by CTRL+C). */
|
||||
fprintf(stderr,
|
||||
"PyThreadState_Clear: warning: thread still has a frame\n");
|
||||
|
|
|
@ -332,6 +332,7 @@ _PySys_ClearAuditHooks(PyThreadState *ts)
|
|||
}
|
||||
|
||||
_PyRuntimeState *runtime = ts->interp->runtime;
|
||||
/* The hooks are global so we have to check for runtime finalization. */
|
||||
PyThreadState *finalizing = _PyRuntimeState_GetFinalizing(runtime);
|
||||
assert(finalizing == ts);
|
||||
if (finalizing != ts) {
|
||||
|
@ -2039,6 +2040,9 @@ sys__clear_type_cache_impl(PyObject *module)
|
|||
Py_RETURN_NONE;
|
||||
}
|
||||
|
||||
/* Note that, for now, we do not have a per-interpreter equivalent
|
||||
for sys.is_finalizing(). */
|
||||
|
||||
/*[clinic input]
|
||||
sys.is_finalizing
|
||||
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue