Skip to content

Commit 965851b

Browse files
committed
Add yielding support.
1 parent cd4091f commit 965851b

File tree

2 files changed

+24
-1
lines changed

2 files changed

+24
-1
lines changed

src/error.rs

+10
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
use super::MultiValue;
12
use std::error::Error as StdError;
23
use std::fmt;
34
use std::io::Error as IoError;
@@ -205,6 +206,12 @@ pub enum Error {
205206
/// Underlying error.
206207
cause: Arc<Error>,
207208
},
209+
/// Yield.
210+
///
211+
/// Not an error.
212+
/// Returning `Err(Yielding(...))` from a Rust callback will yield the value as a Lua value.
213+
/// If it cannot yield, it will raise an error.
214+
Yielding(MultiValue),
208215
}
209216

210217
/// A specialized `Result` type used by `mlua`'s API.
@@ -321,6 +328,9 @@ impl fmt::Display for Error {
321328
Error::WithContext { context, cause } => {
322329
writeln!(fmt, "{context}")?;
323330
write!(fmt, "{cause}")
331+
},
332+
Error::Yielding(_) => {
333+
write!(fmt, "yield across Rust/Lua boundary")
324334
}
325335
}
326336
}

src/state/util.rs

+14-1
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ use std::sync::Arc;
66
use crate::error::{Error, Result};
77
use crate::state::{ExtraData, RawLua};
88
use crate::util::{self, get_internal_metatable, WrappedFailure};
9+
use crate::IntoLuaMulti;
910

1011
pub(super) struct StateGuard<'a>(&'a RawLua, *mut ffi::lua_State);
1112

@@ -107,7 +108,19 @@ where
107108
prealloc_failure.release(state, extra);
108109
r
109110
}
110-
Ok(Err(err)) => {
111+
Ok(Err(mut err)) => {
112+
if let Error::Yielding(tuple) = err {
113+
let raw = extra.as_ref().unwrap_unchecked().raw_lua();
114+
match tuple.push_into_stack_multi(raw) {
115+
Ok(nargs) => {
116+
ffi::lua_yield(state, nargs);
117+
unreachable!();
118+
}
119+
Err(new_err) => {
120+
err = new_err;
121+
}
122+
}
123+
}
111124
let wrapped_error = prealloc_failure.r#use(state, extra);
112125

113126
// Build `CallbackError` with traceback

0 commit comments

Comments
 (0)