Skip to content

Commit 39a39d4

Browse files
committed
Implement thread creation deletion event callback.
1 parent cd4091f commit 39a39d4

File tree

6 files changed

+93
-3
lines changed

6 files changed

+93
-3
lines changed

src/lib.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -113,7 +113,7 @@ pub use crate::traits::{
113113
FromLua, FromLuaMulti, IntoLua, IntoLuaMulti, LuaNativeFn, LuaNativeFnMut, ObjectLike,
114114
};
115115
pub use crate::types::{
116-
AppDataRef, AppDataRefMut, Either, Integer, LightUserData, MaybeSend, Number, RegistryKey, VmState,
116+
AppDataRef, AppDataRefMut, Either, Integer, LightUserData, MaybeSend, Number, RegistryKey, VmState, ThreadEventInfo
117117
};
118118
pub use crate::userdata::{
119119
AnyUserData, MetaMethod, UserData, UserDataFields, UserDataMetatable, UserDataMethods, UserDataRef,

src/prelude.rs

+1
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@ pub use crate::{
1414
UserDataMethods as LuaUserDataMethods, UserDataRef as LuaUserDataRef,
1515
UserDataRefMut as LuaUserDataRefMut, UserDataRegistry as LuaUserDataRegistry, Value as LuaValue,
1616
VmState as LuaVmState,
17+
ThreadEventInfo as LuaThreadEventInfo
1718
};
1819

1920
#[cfg(not(feature = "luau"))]

src/state.rs

+50-1
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,7 @@ use crate::thread::Thread;
2121
use crate::traits::{FromLua, FromLuaMulti, IntoLua, IntoLuaMulti};
2222
use crate::types::{
2323
AppDataRef, AppDataRefMut, ArcReentrantMutexGuard, Integer, LuaType, MaybeSend, Number, ReentrantMutex,
24-
ReentrantMutexGuard, RegistryKey, VmState, XRc, XWeak,
24+
ReentrantMutexGuard, RegistryKey, VmState, XRc, XWeak, ThreadEventInfo
2525
};
2626
use crate::userdata::{AnyUserData, UserData, UserDataProxy, UserDataRegistry, UserDataStorage};
2727
use crate::util::{
@@ -671,6 +671,55 @@ impl Lua {
671671
}
672672
}
673673

674+
/// Sets a callback that will be called by Luau whenever a thread is created/destroyed.
675+
///
676+
/// Often used for keeping track of threads.
677+
#[cfg(any(feature = "luau", doc))]
678+
#[cfg_attr(docsrs, doc(cfg(feature = "luau")))]
679+
pub fn set_thread_event_callback<F>(&self, callback: F)
680+
where
681+
F: Fn(&Lua, ThreadEventInfo) -> Result<()> + MaybeSend + 'static,
682+
{
683+
use std::rc::Rc;
684+
685+
unsafe extern "C-unwind" fn userthread_proc(parent: *mut ffi::lua_State, state: *mut ffi::lua_State) {
686+
let extra = ExtraData::get(state);
687+
let _guard = StateGuard::new((*extra).raw_lua(), state);
688+
let raw_lua: &RawLua = (*extra).raw_lua();
689+
raw_lua.push_ref_thread(parent).unwrap();
690+
callback_error_ext(state, ptr::null_mut(), move |extra, _| {
691+
let userthread_cb = (*extra).userthread_callback.clone();
692+
let userthread_cb = mlua_expect!(userthread_cb, "no userthread callback set in userthread_proc");
693+
let event_info = match (*extra).raw_lua().pop_value() {
694+
Value::Thread(thr) => ThreadEventInfo::Created(thr),
695+
Value::Nil => ThreadEventInfo::Destroying,
696+
_ => unreachable!()
697+
};
698+
userthread_cb((*extra).lua(), event_info)
699+
});
700+
}
701+
702+
// Set interrupt callback
703+
let lua = self.lock();
704+
unsafe {
705+
(*lua.extra.get()).userthread_callback = Some(Rc::new(callback));
706+
(*ffi::lua_callbacks(lua.main_state())).userthread = Some(userthread_proc);
707+
}
708+
}
709+
710+
/// Removes any thread event function previously set by `set_thread_event_callback`.
711+
///
712+
/// This function has no effect if a callback was not previously set.
713+
#[cfg(any(feature = "luau", doc))]
714+
#[cfg_attr(docsrs, doc(cfg(feature = "luau")))]
715+
pub fn remove_thread_event_callback(&self) {
716+
let lua = self.lock();
717+
unsafe {
718+
(*lua.extra.get()).userthread_callback = None;
719+
(*ffi::lua_callbacks(lua.main_state())).userthread = None;
720+
}
721+
}
722+
674723
/// Sets the warning function to be used by Lua to emit warnings.
675724
///
676725
/// Requires `feature = "lua54"`

src/state/extra.rs

+4
Original file line numberDiff line numberDiff line change
@@ -80,6 +80,8 @@ pub(crate) struct ExtraData {
8080
pub(super) warn_callback: Option<crate::types::WarnCallback>,
8181
#[cfg(feature = "luau")]
8282
pub(super) interrupt_callback: Option<crate::types::InterruptCallback>,
83+
#[cfg(feature = "luau")]
84+
pub(super) userthread_callback: Option<crate::types::ThreadEventCallback>,
8385

8486
#[cfg(feature = "luau")]
8587
pub(super) sandboxed: bool,
@@ -177,6 +179,8 @@ impl ExtraData {
177179
#[cfg(feature = "luau")]
178180
interrupt_callback: None,
179181
#[cfg(feature = "luau")]
182+
userthread_callback: None,
183+
#[cfg(feature = "luau")]
180184
sandboxed: false,
181185
#[cfg(feature = "luau")]
182186
compiler: None,

src/state/raw.rs

+18-1
Original file line numberDiff line numberDiff line change
@@ -64,7 +64,8 @@ impl Drop for RawLua {
6464
}
6565

6666
let mem_state = MemoryState::get(self.main_state());
67-
67+
#[cfg(feature = "luau")] // Fixes a crash during shutdown
68+
{ (*ffi::lua_callbacks(self.main_state())).userthread = None; }
6869
ffi::lua_close(self.main_state());
6970

7071
// Deallocate `MemoryState`
@@ -556,6 +557,22 @@ impl RawLua {
556557
value.push_into_stack(self)
557558
}
558559

560+
pub(crate) unsafe fn push_ref_thread(&self, ref_thread: *mut ffi::lua_State) -> Result<()> {
561+
let state = self.state();
562+
let _sg = StackGuard::new(state);
563+
check_stack(state, 2)?;
564+
let _sg = StackGuard::new(ref_thread);
565+
check_stack(ref_thread, 1)?;
566+
567+
if self.unlikely_memory_error() {
568+
ffi::lua_pushthread(ref_thread)
569+
} else {
570+
protect_lua!(ref_thread, 0, 1, |ref_thread| ffi::lua_pushthread(ref_thread))?
571+
};
572+
ffi::lua_xmove(ref_thread, self.ref_thread(), 1);
573+
Ok(())
574+
}
575+
559576
/// Pushes a `Value` (by reference) onto the Lua stack.
560577
///
561578
/// Uses 2 stack spaces, does not call `checkstack`.

src/types.rs

+19
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ use crate::error::Result;
66
#[cfg(not(feature = "luau"))]
77
use crate::hook::Debug;
88
use crate::state::{ExtraData, Lua, RawLua};
9+
use crate::thread::Thread;
910

1011
// Re-export mutex wrappers
1112
pub(crate) use sync::{ArcReentrantMutexGuard, ReentrantMutex, ReentrantMutexGuard, XRc, XWeak};
@@ -73,6 +74,17 @@ pub enum VmState {
7374
Yield,
7475
}
7576

77+
/// Information about a thread event.
78+
///
79+
/// For creating a thread, it contains the thread that created it.
80+
///
81+
/// This is useful when it is required for tracking all threads and where they come from.
82+
#[cfg(feature = "luau")]
83+
pub enum ThreadEventInfo {
84+
Created(Thread),
85+
Destroying
86+
}
87+
7688
#[cfg(all(feature = "send", not(feature = "luau")))]
7789
pub(crate) type HookCallback = Rc<dyn Fn(&Lua, Debug) -> Result<VmState> + Send>;
7890

@@ -85,6 +97,13 @@ pub(crate) type InterruptCallback = Rc<dyn Fn(&Lua) -> Result<VmState> + Send>;
8597
#[cfg(all(not(feature = "send"), feature = "luau"))]
8698
pub(crate) type InterruptCallback = Rc<dyn Fn(&Lua) -> Result<VmState>>;
8799

100+
#[cfg(all(feature = "send", feature = "luau"))]
101+
pub(crate) type ThreadEventCallback = Rc<dyn Fn(&Lua, ThreadEventInfo) -> Result<()> + Send>;
102+
103+
#[cfg(all(not(feature = "send"), feature = "luau"))]
104+
pub(crate) type ThreadEventCallback = Rc<dyn Fn(&Lua, ThreadEventInfo) -> Result<()>>;
105+
106+
88107
#[cfg(all(feature = "send", feature = "lua54"))]
89108
pub(crate) type WarnCallback = Box<dyn Fn(&Lua, &str, bool) -> Result<()> + Send>;
90109

0 commit comments

Comments
 (0)