pub use atexit::_run_exitfuncs;
pub(crate) use atexit::module_def;
#[pymodule]
mod atexit {
use crate::{AsObject, PyObjectRef, PyResult, VirtualMachine, function::FuncArgs};
#[pyfunction]
fn register(func: PyObjectRef, args: FuncArgs, vm: &VirtualMachine) -> PyObjectRef {
vm.state
.atexit_funcs
.lock()
.insert(0, Box::new((func.clone(), args)));
func
}
#[pyfunction]
fn _clear(vm: &VirtualMachine) {
vm.state.atexit_funcs.lock().clear();
}
#[pyfunction]
fn unregister(func: PyObjectRef, vm: &VirtualMachine) -> PyResult<()> {
let mut i = {
let funcs = vm.state.atexit_funcs.lock();
funcs.len() as isize - 1
};
while i >= 0 {
let (cb, entry_ptr) = {
let funcs = vm.state.atexit_funcs.lock();
if i as usize >= funcs.len() {
i = funcs.len() as isize;
i -= 1;
continue;
}
let entry = &funcs[i as usize];
(entry.0.clone(), &**entry as *const (PyObjectRef, FuncArgs))
};
let eq = vm.bool_eq(&func, &cb)?;
if eq {
let mut funcs = vm.state.atexit_funcs.lock();
let mut j = (funcs.len() as isize - 1).min(i);
while j >= 0 {
if core::ptr::eq(&**funcs.get(j as usize).unwrap(), entry_ptr) {
funcs.remove(j as usize);
i = j;
break;
}
j -= 1;
}
}
{
let funcs = vm.state.atexit_funcs.lock();
if i as usize >= funcs.len() {
i = funcs.len() as isize;
}
}
i -= 1;
}
Ok(())
}
#[pyfunction]
pub fn _run_exitfuncs(vm: &VirtualMachine) {
let funcs: Vec<_> = core::mem::take(&mut *vm.state.atexit_funcs.lock());
for entry in funcs.into_iter() {
let (func, args) = *entry;
if let Err(e) = func.call(args, vm) {
let exit = e.fast_isinstance(vm.ctx.exceptions.system_exit);
let msg = func
.repr(vm)
.ok()
.map(|r| format!("Exception ignored in atexit callback {}", r.as_wtf8()));
vm.run_unraisable(e, msg, vm.ctx.none());
if exit {
break;
}
}
}
}
#[pyfunction]
fn _ncallbacks(vm: &VirtualMachine) -> usize {
vm.state.atexit_funcs.lock().len()
}
}