mlua/tests/thread.rs

142 lines
4.1 KiB
Rust
Raw Normal View History

2019-10-16 05:56:44 -04:00
#![cfg_attr(
all(feature = "luajit", target_os = "macos", target_arch = "x86_64"),
feature(link_args)
)]
#[cfg_attr(
all(feature = "luajit", target_os = "macos", target_arch = "x86_64"),
link_args = "-pagezero_size 10000 -image_base 100000000"
)]
extern "system" {}
2018-02-09 23:52:05 -05:00
use std::panic::catch_unwind;
2019-10-14 17:21:30 -04:00
use mlua::{Error, Function, Lua, Result, Thread, ThreadStatus};
2018-02-09 23:52:05 -05:00
#[test]
2019-09-28 10:23:17 -04:00
fn test_thread() -> Result<()> {
2019-10-14 17:21:30 -04:00
let lua = Lua::new();
2019-09-28 10:23:17 -04:00
let thread = lua.create_thread(
lua.load(
r#"
2018-02-09 23:52:05 -05:00
function (s)
local sum = s
for i = 1,4 do
sum = sum + coroutine.yield(sum)
end
return sum
end
"#,
2019-09-27 12:38:24 -04:00
)
2019-09-28 10:23:17 -04:00
.eval()?,
)?;
2018-02-09 23:52:05 -05:00
assert_eq!(thread.status(), ThreadStatus::Resumable);
2019-09-28 10:23:17 -04:00
assert_eq!(thread.resume::<_, i64>(0)?, 0);
2018-02-09 23:52:05 -05:00
assert_eq!(thread.status(), ThreadStatus::Resumable);
2019-09-28 10:23:17 -04:00
assert_eq!(thread.resume::<_, i64>(1)?, 1);
2018-02-09 23:52:05 -05:00
assert_eq!(thread.status(), ThreadStatus::Resumable);
2019-09-28 10:23:17 -04:00
assert_eq!(thread.resume::<_, i64>(2)?, 3);
2018-02-09 23:52:05 -05:00
assert_eq!(thread.status(), ThreadStatus::Resumable);
2019-09-28 10:23:17 -04:00
assert_eq!(thread.resume::<_, i64>(3)?, 6);
2018-02-09 23:52:05 -05:00
assert_eq!(thread.status(), ThreadStatus::Resumable);
2019-09-28 10:23:17 -04:00
assert_eq!(thread.resume::<_, i64>(4)?, 10);
2018-02-09 23:52:05 -05:00
assert_eq!(thread.status(), ThreadStatus::Unresumable);
2019-09-28 10:23:17 -04:00
let accumulate = lua.create_thread(
lua.load(
r#"
2018-02-09 23:52:05 -05:00
function (sum)
while true do
sum = sum + coroutine.yield(sum)
end
end
"#,
2019-09-27 12:38:24 -04:00
)
2019-09-28 10:23:17 -04:00
.eval::<Function>()?,
)?;
2018-02-09 23:52:05 -05:00
for i in 0..4 {
2019-09-28 10:23:17 -04:00
accumulate.resume::<_, ()>(i)?;
2018-02-09 23:52:05 -05:00
}
2019-09-28 10:23:17 -04:00
assert_eq!(accumulate.resume::<_, i64>(4)?, 10);
2018-02-09 23:52:05 -05:00
assert_eq!(accumulate.status(), ThreadStatus::Resumable);
assert!(accumulate.resume::<_, ()>("error").is_err());
assert_eq!(accumulate.status(), ThreadStatus::Error);
2018-09-24 22:13:42 -04:00
let thread = lua
2019-09-28 10:23:17 -04:00
.load(
2018-08-05 09:51:39 -04:00
r#"
2019-09-28 10:23:17 -04:00
coroutine.create(function ()
while true do
coroutine.yield(42)
end
end)
"#,
2019-09-27 12:38:24 -04:00
)
2019-09-28 10:23:17 -04:00
.eval::<Thread>()?;
2018-02-09 23:52:05 -05:00
assert_eq!(thread.status(), ThreadStatus::Resumable);
2019-09-28 10:23:17 -04:00
assert_eq!(thread.resume::<_, i64>(())?, 42);
2018-02-09 23:52:05 -05:00
2018-09-24 22:13:42 -04:00
let thread: Thread = lua
2019-09-28 10:23:17 -04:00
.load(
2018-08-05 09:51:39 -04:00
r#"
2019-09-28 10:23:17 -04:00
coroutine.create(function(arg)
assert(arg == 42)
local yieldarg = coroutine.yield(123)
assert(yieldarg == 43)
return 987
end)
"#,
2019-09-27 12:38:24 -04:00
)
2019-09-28 10:23:17 -04:00
.eval()?;
2018-02-09 23:52:05 -05:00
2019-09-28 10:23:17 -04:00
assert_eq!(thread.resume::<_, u32>(42)?, 123);
assert_eq!(thread.resume::<_, u32>(43)?, 987);
2018-02-09 23:52:05 -05:00
match thread.resume::<_, u32>(()) {
Err(Error::CoroutineInactive) => {}
Err(_) => panic!("resuming dead coroutine error is not CoroutineInactive kind"),
_ => panic!("resuming dead coroutine did not return error"),
}
2019-09-28 10:23:17 -04:00
Ok(())
2018-02-09 23:52:05 -05:00
}
#[test]
2019-09-28 10:23:17 -04:00
fn coroutine_from_closure() -> Result<()> {
2019-10-14 17:21:30 -04:00
let lua = Lua::new();
2019-09-28 10:23:17 -04:00
let thrd_main = lua.create_function(|_, ()| Ok(()))?;
lua.globals().set("main", thrd_main)?;
2019-10-14 17:21:30 -04:00
2019-11-29 08:26:30 -05:00
#[cfg(any(feature = "lua53", feature = "lua52", feature = "luajit"))]
2019-09-28 10:23:17 -04:00
let thrd: Thread = lua.load("coroutine.create(main)").eval()?;
#[cfg(feature = "lua51")]
2019-10-14 17:21:30 -04:00
let thrd: Thread = lua
.load("coroutine.create(function(...) return main(unpack(arg)) end)")
.eval()?;
2019-09-28 10:23:17 -04:00
thrd.resume::<_, ()>(())?;
Ok(())
2018-02-09 23:52:05 -05:00
}
#[test]
fn coroutine_panic() {
match catch_unwind(|| -> Result<()> {
// check that coroutines propagate panics correctly
2019-10-14 17:21:30 -04:00
let lua = Lua::new();
let thrd_main = lua.create_function(|_, ()| -> Result<()> {
panic!("test_panic");
})?;
lua.globals().set("main", thrd_main.clone())?;
let thrd: Thread = lua.create_thread(thrd_main)?;
thrd.resume(())
}) {
2018-02-09 23:52:05 -05:00
Ok(r) => panic!("coroutine panic not propagated, instead returned {:?}", r),
Err(p) => assert!(*p.downcast::<&str>().unwrap() == "test_panic"),
2018-02-09 23:52:05 -05:00
}
}