mirror of
https://github.com/denoland/deno.git
synced 2025-01-11 16:42:21 -05:00
fix(napi): allow cleanup hook to remove itself (#17402)
This commit fixes "cleanup hooks" in NAPI integration in two ways: - don't hold to RefCell's borrow while iterating over hooks - allow a hook to remove itself when being called
This commit is contained in:
parent
1b207796aa
commit
d464df3a25
2 changed files with 31 additions and 4 deletions
|
@ -343,11 +343,32 @@ pub struct NapiState {
|
||||||
|
|
||||||
impl Drop for NapiState {
|
impl Drop for NapiState {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
let mut hooks = self.env_cleanup_hooks.borrow_mut();
|
let hooks = {
|
||||||
|
let h = self.env_cleanup_hooks.borrow_mut();
|
||||||
|
h.clone()
|
||||||
|
};
|
||||||
|
|
||||||
// Hooks are supposed to be run in LIFO order
|
// Hooks are supposed to be run in LIFO order
|
||||||
let hooks = hooks.drain(..).rev();
|
let hooks_to_run = hooks.into_iter().rev();
|
||||||
for (fn_ptr, data) in hooks {
|
|
||||||
(fn_ptr)(data);
|
for hook in hooks_to_run {
|
||||||
|
// This hook might have been removed by a previous hook, in such case skip it here.
|
||||||
|
if !self
|
||||||
|
.env_cleanup_hooks
|
||||||
|
.borrow()
|
||||||
|
.iter()
|
||||||
|
.any(|pair| pair.0 == hook.0 && pair.1 == hook.1)
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
(hook.0)(hook.1);
|
||||||
|
{
|
||||||
|
self
|
||||||
|
.env_cleanup_hooks
|
||||||
|
.borrow_mut()
|
||||||
|
.retain(|pair| !(pair.0 == hook.0 && pair.1 == hook.1));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -66,6 +66,11 @@ extern "C" fn cleanup(arg: *mut c_void) {
|
||||||
println!("cleanup({})", arg as i64);
|
println!("cleanup({})", arg as i64);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
extern "C" fn remove_this_hook(arg: *mut c_void) {
|
||||||
|
let env = arg as napi_env;
|
||||||
|
unsafe { napi_remove_env_cleanup_hook(env, Some(remove_this_hook), arg) };
|
||||||
|
}
|
||||||
|
|
||||||
static SECRET: i64 = 42;
|
static SECRET: i64 = 42;
|
||||||
static WRONG_SECRET: i64 = 17;
|
static WRONG_SECRET: i64 = 17;
|
||||||
static THIRD_SECRET: i64 = 18;
|
static THIRD_SECRET: i64 = 18;
|
||||||
|
@ -81,6 +86,7 @@ extern "C" fn install_cleanup_hook(
|
||||||
napi_add_env_cleanup_hook(env, Some(cleanup), WRONG_SECRET as *mut c_void);
|
napi_add_env_cleanup_hook(env, Some(cleanup), WRONG_SECRET as *mut c_void);
|
||||||
napi_add_env_cleanup_hook(env, Some(cleanup), SECRET as *mut c_void);
|
napi_add_env_cleanup_hook(env, Some(cleanup), SECRET as *mut c_void);
|
||||||
napi_add_env_cleanup_hook(env, Some(cleanup), THIRD_SECRET as *mut c_void);
|
napi_add_env_cleanup_hook(env, Some(cleanup), THIRD_SECRET as *mut c_void);
|
||||||
|
napi_add_env_cleanup_hook(env, Some(remove_this_hook), env as *mut c_void);
|
||||||
napi_remove_env_cleanup_hook(
|
napi_remove_env_cleanup_hook(
|
||||||
env,
|
env,
|
||||||
Some(cleanup),
|
Some(cleanup),
|
||||||
|
|
Loading…
Reference in a new issue