diff options
author | Marcel Plch <gmarcel.plch@gmail.com> | 2017-12-20 10:17:58 (GMT) |
---|---|---|
committer | Antoine Pitrou <pitrou@free.fr> | 2017-12-20 10:17:58 (GMT) |
commit | 776407fe893fd42972c7e3f71423d9d86741d07c (patch) | |
tree | 2d2a5781d83709c56d27e5815af2d7a2fc5726c0 /Modules/atexitmodule.c | |
parent | 19760863623b636a63ccf649107d9504c6465a92 (diff) | |
download | cpython-776407fe893fd42972c7e3f71423d9d86741d07c.zip cpython-776407fe893fd42972c7e3f71423d9d86741d07c.tar.gz cpython-776407fe893fd42972c7e3f71423d9d86741d07c.tar.bz2 |
bpo-31901: atexit callbacks should be run at subinterpreter shutdown (#4611)
Change atexit behavior and PEP-489 multiphase init support.
Diffstat (limited to 'Modules/atexitmodule.c')
-rw-r--r-- | Modules/atexitmodule.c | 72 |
1 files changed, 40 insertions, 32 deletions
diff --git a/Modules/atexitmodule.c b/Modules/atexitmodule.c index 35ebf08..afa1cfa 100644 --- a/Modules/atexitmodule.c +++ b/Modules/atexitmodule.c @@ -63,15 +63,13 @@ atexit_cleanup(atexitmodule_state *modstate) /* Installed into pylifecycle.c's atexit mechanism */ static void -atexit_callfuncs(void) +atexit_callfuncs(PyObject *module) { PyObject *exc_type = NULL, *exc_value, *exc_tb, *r; atexit_callback *cb; - PyObject *module; atexitmodule_state *modstate; int i; - module = PyState_FindModule(&atexitmodule); if (module == NULL) return; modstate = GET_ATEXIT_STATE(module); @@ -185,7 +183,7 @@ Run all registered exit functions."); static PyObject * atexit_run_exitfuncs(PyObject *self, PyObject *unused) { - atexit_callfuncs(); + atexit_callfuncs(self); if (PyErr_Occurred()) return NULL; Py_RETURN_NONE; @@ -225,13 +223,15 @@ atexit_m_traverse(PyObject *self, visitproc visit, void *arg) atexitmodule_state *modstate; modstate = GET_ATEXIT_STATE(self); - for (i = 0; i < modstate->ncallbacks; i++) { - atexit_callback *cb = modstate->atexit_callbacks[i]; - if (cb == NULL) - continue; - Py_VISIT(cb->func); - Py_VISIT(cb->args); - Py_VISIT(cb->kwargs); + if (modstate != NULL) { + for (i = 0; i < modstate->ncallbacks; i++) { + atexit_callback *cb = modstate->atexit_callbacks[i]; + if (cb == NULL) + continue; + Py_VISIT(cb->func); + Py_VISIT(cb->args); + Py_VISIT(cb->kwargs); + } } return 0; } @@ -241,7 +241,9 @@ atexit_m_clear(PyObject *self) { atexitmodule_state *modstate; modstate = GET_ATEXIT_STATE(self); - atexit_cleanup(modstate); + if (modstate != NULL) { + atexit_cleanup(modstate); + } return 0; } @@ -250,8 +252,10 @@ atexit_free(PyObject *m) { atexitmodule_state *modstate; modstate = GET_ATEXIT_STATE(m); - atexit_cleanup(modstate); - PyMem_Free(modstate->atexit_callbacks); + if (modstate != NULL) { + atexit_cleanup(modstate); + PyMem_Free(modstate->atexit_callbacks); + } } PyDoc_STRVAR(atexit_unregister__doc__, @@ -310,6 +314,26 @@ upon normal program termination.\n\ Two public functions, register and unregister, are defined.\n\ "); +static int +atexit_exec(PyObject *m) { + atexitmodule_state *modstate; + + modstate = GET_ATEXIT_STATE(m); + modstate->callback_len = 32; + modstate->ncallbacks = 0; + modstate->atexit_callbacks = PyMem_New(atexit_callback*, + modstate->callback_len); + if (modstate->atexit_callbacks == NULL) + return -1; + + _Py_PyAtExit(atexit_callfuncs, m); + return 0; +} + +static PyModuleDef_Slot atexit_slots[] = { + {Py_mod_exec, atexit_exec}, + {0, NULL} +}; static struct PyModuleDef atexitmodule = { PyModuleDef_HEAD_INIT, @@ -317,7 +341,7 @@ static struct PyModuleDef atexitmodule = { atexit__doc__, sizeof(atexitmodule_state), atexit_methods, - NULL, + atexit_slots, atexit_m_traverse, atexit_m_clear, (freefunc)atexit_free @@ -326,21 +350,5 @@ static struct PyModuleDef atexitmodule = { PyMODINIT_FUNC PyInit_atexit(void) { - PyObject *m; - atexitmodule_state *modstate; - - m = PyModule_Create(&atexitmodule); - if (m == NULL) - return NULL; - - modstate = GET_ATEXIT_STATE(m); - modstate->callback_len = 32; - modstate->ncallbacks = 0; - modstate->atexit_callbacks = PyMem_New(atexit_callback*, - modstate->callback_len); - if (modstate->atexit_callbacks == NULL) - return NULL; - - _Py_PyAtExit(atexit_callfuncs); - return m; + return PyModuleDef_Init(&atexitmodule); } |