func - function to be unregistered"); static PyObject * atexit_unregister(PyObject *self, PyObject *func) { atexitmodule_state *modstate; atexit_callback *cb; int i, eq; modstate = GET_ATEXIT_STATE(self); for (i = 0; i < modstate->ncallbacks; i++) { cb = modstate->atexit_callbacks[i]; if (cb == NULL) continue; eq = PyObject_RichCompareBool(cb->func, func, Py_EQ); if (eq < 0) return NULL; if (eq) atexit_delete_cb(modstate, i); } Py_RETURN_NONE; }
static void atexit_free(PyObject *m) { atexitmodule_state *modstate; modstate = GET_ATEXIT_STATE(m); atexit_cleanup(modstate); PyMem_Free(modstate->atexit_callbacks); }
static int atexit_m_clear(PyObject *self) { atexitmodule_state *modstate; modstate = GET_ATEXIT_STATE(self); atexit_cleanup(modstate); return 0; }
static PyObject * atexit_ncallbacks(PyObject *self, PyObject *unused) { atexitmodule_state *modstate; modstate = GET_ATEXIT_STATE(self); return PyLong_FromSsize_t(modstate->ncallbacks); }
static PyObject * atexit_register(PyObject *self, PyObject *args, PyObject *kwargs) { atexitmodule_state *modstate; atexit_callback *new_callback; PyObject *func = NULL; modstate = GET_ATEXIT_STATE(self); if (modstate->ncallbacks >= modstate->callback_len) { atexit_callback **r; modstate->callback_len += 16; r = (atexit_callback**)PyMem_Realloc(modstate->atexit_callbacks, sizeof(atexit_callback*) * modstate->callback_len); if (r == NULL) return PyErr_NoMemory(); modstate->atexit_callbacks = r; } if (PyTuple_GET_SIZE(args) == 0) { PyErr_SetString(PyExc_TypeError, "register() takes at least 1 argument (0 given)"); return NULL; } func = PyTuple_GET_ITEM(args, 0); if (!PyCallable_Check(func)) { PyErr_SetString(PyExc_TypeError, "the first argument must be callable"); return NULL; } new_callback = PyMem_Malloc(sizeof(atexit_callback)); if (new_callback == NULL) return PyErr_NoMemory(); new_callback->args = PyTuple_GetSlice(args, 1, PyTuple_GET_SIZE(args)); if (new_callback->args == NULL) { PyMem_Free(new_callback); return NULL; } new_callback->func = func; new_callback->kwargs = kwargs; Py_INCREF(func); Py_XINCREF(kwargs); modstate->atexit_callbacks[modstate->ncallbacks++] = new_callback; Py_INCREF(func); return func; }
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 void atexit_callfuncs(void) { 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); if (modstate->ncallbacks == 0) return; for (i = modstate->ncallbacks - 1; i >= 0; i--) { cb = modstate->atexit_callbacks[i]; if (cb == NULL) continue; r = PyObject_Call(cb->func, cb->args, cb->kwargs); Py_XDECREF(r); if (r == NULL) { /* Maintain the last exception, but don't leak if there are multiple exceptions. */ if (exc_type) { Py_DECREF(exc_type); Py_XDECREF(exc_value); Py_XDECREF(exc_tb); } PyErr_Fetch(&exc_type, &exc_value, &exc_tb); if (!PyErr_ExceptionMatches(PyExc_SystemExit)) { PySys_WriteStderr("Error in atexit._run_exitfuncs:\n"); PyErr_NormalizeException(&exc_type, &exc_value, &exc_tb); PyErr_Display(exc_type, exc_value, exc_tb); } } } atexit_cleanup(modstate); if (exc_type) PyErr_Restore(exc_type, exc_value, exc_tb); }
static int atexit_m_traverse(PyObject *self, visitproc visit, void *arg) { int i; 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); } return 0; }
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; }
static PyObject * atexit_clear(PyObject *self, PyObject *unused) { atexit_cleanup(GET_ATEXIT_STATE(self)); Py_RETURN_NONE; }