Skip to content

Commit cdd6a99

Browse files
committed
Support Thread::resume_error call (Luau)
This method uses Luau-specific C API extension `lua_resumerror` that allow to throw an error immediately when resuming a thead. Closes #500 #513
1 parent 21d39a0 commit cdd6a99

File tree

3 files changed

+76
-0
lines changed

3 files changed

+76
-0
lines changed

mlua-sys/src/luau/compat.rs

+15
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,8 @@ use super::lauxlib::*;
1010
use super::lua::*;
1111
use super::luacode::*;
1212

13+
pub const LUA_RESUMEERROR: c_int = -1;
14+
1315
unsafe fn compat53_reverse(L: *mut lua_State, mut a: c_int, mut b: c_int) {
1416
while a < b {
1517
lua_pushvalue(L, a);
@@ -284,6 +286,19 @@ pub unsafe fn lua_resume(L: *mut lua_State, from: *mut lua_State, narg: c_int, n
284286
ret
285287
}
286288

289+
#[inline(always)]
290+
pub unsafe fn lua_resumex(L: *mut lua_State, from: *mut lua_State, narg: c_int, nres: *mut c_int) -> c_int {
291+
let ret = if narg == LUA_RESUMEERROR {
292+
lua_resumeerror(L, from)
293+
} else {
294+
lua_resume_(L, from, narg)
295+
};
296+
if (ret == LUA_OK || ret == LUA_YIELD) && !(nres.is_null()) {
297+
*nres = lua_gettop(L);
298+
}
299+
ret
300+
}
301+
287302
//
288303
// lauxlib ported functions
289304
//

src/thread.rs

+36
Original file line numberDiff line numberDiff line change
@@ -173,14 +173,50 @@ impl Thread {
173173
}
174174
}
175175

176+
/// Resumes execution of this thread, immediately raising an error.
177+
///
178+
/// This is a Luau specific extension.
179+
#[cfg(feature = "luau")]
180+
#[cfg_attr(docsrs, doc(cfg(feature = "luau")))]
181+
pub fn resume_error<R>(&self, error: impl crate::IntoLua) -> Result<R>
182+
where
183+
R: FromLuaMulti,
184+
{
185+
let lua = self.0.lua.lock();
186+
match self.status_inner(&lua) {
187+
ThreadStatusInner::New(_) | ThreadStatusInner::Yielded(_) => {}
188+
_ => return Err(Error::CoroutineUnresumable),
189+
};
190+
191+
let state = lua.state();
192+
let thread_state = self.state();
193+
unsafe {
194+
let _sg = StackGuard::new(state);
195+
let _thread_sg = StackGuard::with_top(thread_state, 0);
196+
197+
check_stack(state, 1)?;
198+
error.push_into_stack(&lua)?;
199+
ffi::lua_xmove(state, thread_state, 1);
200+
201+
let (_, nresults) = self.resume_inner(&lua, ffi::LUA_RESUMEERROR)?;
202+
check_stack(state, nresults + 1)?;
203+
ffi::lua_xmove(thread_state, state, nresults);
204+
205+
R::from_stack_multi(nresults, &lua)
206+
}
207+
}
208+
176209
/// Resumes execution of this thread.
177210
///
178211
/// It's similar to `resume()` but leaves `nresults` values on the thread stack.
179212
unsafe fn resume_inner(&self, lua: &RawLua, nargs: c_int) -> Result<(ThreadStatusInner, c_int)> {
180213
let state = lua.state();
181214
let thread_state = self.state();
182215
let mut nresults = 0;
216+
#[cfg(not(feature = "luau"))]
183217
let ret = ffi::lua_resume(thread_state, state, nargs, &mut nresults as *mut c_int);
218+
#[cfg(feature = "luau")]
219+
let ret = ffi::lua_resumex(thread_state, state, nargs, &mut nresults as *mut c_int);
184220
match ret {
185221
ffi::LUA_OK => Ok((ThreadStatusInner::Finished, nresults)),
186222
ffi::LUA_YIELD => Ok((ThreadStatusInner::Yielded(0), nresults)),

tests/thread.rs

+25
Original file line numberDiff line numberDiff line change
@@ -227,3 +227,28 @@ fn test_thread_pointer() -> Result<()> {
227227

228228
Ok(())
229229
}
230+
231+
#[test]
232+
#[cfg(feature = "luau")]
233+
fn test_thread_resume_error() -> Result<()> {
234+
let lua = Lua::new();
235+
236+
let thread = lua
237+
.load(
238+
r#"
239+
coroutine.create(function()
240+
local ok, err = pcall(coroutine.yield, 123)
241+
assert(not ok, "yield should fail")
242+
assert(err == "myerror", "unexpected error: " .. tostring(err))
243+
return "success"
244+
end)
245+
"#,
246+
)
247+
.eval::<Thread>()?;
248+
249+
assert_eq!(thread.resume::<i64>(())?, 123);
250+
let status = thread.resume_error::<String>("myerror").unwrap();
251+
assert_eq!(status, "success");
252+
253+
Ok(())
254+
}

0 commit comments

Comments
 (0)