2023-02-11 11:39:39 +00:00
|
|
|
use std::time::{Duration, Instant};
|
2023-01-21 18:33:33 +00:00
|
|
|
|
2023-01-23 02:31:55 +00:00
|
|
|
use mlua::prelude::*;
|
2023-02-03 19:15:20 +00:00
|
|
|
use tokio::time;
|
2023-01-23 07:38:32 +00:00
|
|
|
|
2023-01-23 23:52:31 +00:00
|
|
|
use crate::utils::{
|
|
|
|
table::TableBuilder,
|
|
|
|
task::{run_registered_task, TaskRunMode},
|
|
|
|
};
|
2023-01-23 07:38:32 +00:00
|
|
|
|
2023-01-24 00:59:35 +00:00
|
|
|
const MINIMUM_WAIT_OR_DELAY_DURATION: f32 = 10.0 / 1_000.0; // 10ms
|
|
|
|
|
2023-02-11 11:39:39 +00:00
|
|
|
pub fn create(lua: &'static Lua) -> LuaResult<LuaTable> {
|
2023-01-23 22:34:43 +00:00
|
|
|
// HACK: There is no way to call coroutine.close directly from the mlua
|
2023-01-23 23:52:31 +00:00
|
|
|
// crate, so we need to fetch the function and store it in the registry
|
2023-01-23 22:34:43 +00:00
|
|
|
let coroutine: LuaTable = lua.globals().raw_get("coroutine")?;
|
|
|
|
let close: LuaFunction = coroutine.raw_get("close")?;
|
|
|
|
lua.set_named_registry_value("coroutine.close", close)?;
|
2023-01-23 22:39:16 +00:00
|
|
|
// HACK: coroutine.resume has some weird scheduling issues, but our custom
|
|
|
|
// task.spawn implementation is more or less a replacement for it, so we
|
|
|
|
// overwrite the original coroutine.resume function with it to fix that
|
|
|
|
coroutine.raw_set("resume", lua.create_async_function(task_spawn)?)?;
|
|
|
|
// Rest of the task library is normal, just async functions, no metatable
|
2023-02-10 11:14:28 +00:00
|
|
|
TableBuilder::new(lua)?
|
|
|
|
.with_async_function("cancel", task_cancel)?
|
|
|
|
.with_async_function("delay", task_delay)?
|
|
|
|
.with_async_function("defer", task_defer)?
|
|
|
|
.with_async_function("spawn", task_spawn)?
|
|
|
|
.with_async_function("wait", task_wait)?
|
|
|
|
.build_readonly()
|
2023-01-22 21:26:45 +00:00
|
|
|
}
|
2023-01-23 07:38:32 +00:00
|
|
|
|
2023-02-11 11:39:39 +00:00
|
|
|
fn tof_to_thread<'a>(
|
|
|
|
lua: &'static Lua,
|
|
|
|
thread_or_function: LuaValue<'a>,
|
|
|
|
) -> LuaResult<LuaThread<'a>> {
|
|
|
|
match thread_or_function {
|
2023-01-23 07:38:32 +00:00
|
|
|
LuaValue::Thread(t) => Ok(t),
|
|
|
|
LuaValue::Function(f) => Ok(lua.create_thread(f)?),
|
|
|
|
value => Err(LuaError::RuntimeError(format!(
|
|
|
|
"Argument must be a thread or function, got {}",
|
|
|
|
value.type_name()
|
|
|
|
))),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-11 11:39:39 +00:00
|
|
|
async fn task_cancel<'a>(lua: &'static Lua, thread: LuaThread<'a>) -> LuaResult<()> {
|
2023-01-23 22:34:43 +00:00
|
|
|
let close: LuaFunction = lua.named_registry_value("coroutine.close")?;
|
|
|
|
close.call_async::<_, LuaMultiValue>(thread).await?;
|
2023-01-23 21:18:20 +00:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-01-24 17:10:43 +00:00
|
|
|
async fn task_defer<'a>(
|
2023-02-11 11:39:39 +00:00
|
|
|
lua: &'static Lua,
|
2023-01-24 17:24:57 +00:00
|
|
|
(tof, args): (LuaValue<'a>, LuaMultiValue<'a>),
|
2023-01-24 17:10:43 +00:00
|
|
|
) -> LuaResult<LuaThread<'a>> {
|
2023-01-23 21:18:20 +00:00
|
|
|
let task_thread = tof_to_thread(lua, tof)?;
|
|
|
|
let task_thread_key = lua.create_registry_value(task_thread)?;
|
2023-01-24 17:24:57 +00:00
|
|
|
let task_args_key = lua.create_registry_value(args.into_vec())?;
|
2023-01-23 21:18:20 +00:00
|
|
|
let lua_thread_to_return = lua.registry_value(&task_thread_key)?;
|
2023-01-23 23:52:31 +00:00
|
|
|
run_registered_task(lua, TaskRunMode::Deferred, async move {
|
2023-02-11 11:39:39 +00:00
|
|
|
let thread: LuaThread = lua.registry_value(&task_thread_key)?;
|
|
|
|
let argsv: Vec<LuaValue> = lua.registry_value(&task_args_key)?;
|
2023-01-24 17:24:57 +00:00
|
|
|
let args = LuaMultiValue::from_vec(argsv);
|
2023-01-23 23:52:31 +00:00
|
|
|
if thread.status() == LuaThreadStatus::Resumable {
|
2023-01-24 17:24:57 +00:00
|
|
|
let _: LuaMultiValue = thread.into_async(args).await?;
|
2023-01-23 23:52:31 +00:00
|
|
|
}
|
2023-02-11 11:39:39 +00:00
|
|
|
lua.remove_registry_value(task_thread_key)?;
|
|
|
|
lua.remove_registry_value(task_args_key)?;
|
2023-01-23 23:52:31 +00:00
|
|
|
Ok(())
|
|
|
|
})
|
2023-01-23 21:18:20 +00:00
|
|
|
.await?;
|
|
|
|
Ok(lua_thread_to_return)
|
2023-01-23 07:38:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
async fn task_delay<'a>(
|
2023-02-11 11:39:39 +00:00
|
|
|
lua: &'static Lua,
|
2023-01-24 17:24:57 +00:00
|
|
|
(duration, tof, args): (Option<f32>, LuaValue<'a>, LuaMultiValue<'a>),
|
2023-01-23 07:38:32 +00:00
|
|
|
) -> LuaResult<LuaThread<'a>> {
|
2023-01-23 21:18:20 +00:00
|
|
|
let task_thread = tof_to_thread(lua, tof)?;
|
|
|
|
let task_thread_key = lua.create_registry_value(task_thread)?;
|
2023-01-24 17:24:57 +00:00
|
|
|
let task_args_key = lua.create_registry_value(args.into_vec())?;
|
2023-01-23 21:18:20 +00:00
|
|
|
let lua_thread_to_return = lua.registry_value(&task_thread_key)?;
|
2023-01-23 23:52:31 +00:00
|
|
|
run_registered_task(lua, TaskRunMode::Deferred, async move {
|
2023-02-11 11:39:39 +00:00
|
|
|
task_wait(lua, duration).await?;
|
|
|
|
let thread: LuaThread = lua.registry_value(&task_thread_key)?;
|
|
|
|
let argsv: Vec<LuaValue> = lua.registry_value(&task_args_key)?;
|
2023-01-24 17:24:57 +00:00
|
|
|
let args = LuaMultiValue::from_vec(argsv);
|
2023-01-23 23:52:31 +00:00
|
|
|
if thread.status() == LuaThreadStatus::Resumable {
|
2023-01-24 17:24:57 +00:00
|
|
|
let _: LuaMultiValue = thread.into_async(args).await?;
|
2023-01-23 23:52:31 +00:00
|
|
|
}
|
2023-02-11 11:39:39 +00:00
|
|
|
lua.remove_registry_value(task_thread_key)?;
|
|
|
|
lua.remove_registry_value(task_args_key)?;
|
2023-01-23 23:52:31 +00:00
|
|
|
Ok(())
|
|
|
|
})
|
2023-01-23 21:18:20 +00:00
|
|
|
.await?;
|
|
|
|
Ok(lua_thread_to_return)
|
2023-01-23 07:38:32 +00:00
|
|
|
}
|
|
|
|
|
2023-01-24 17:10:43 +00:00
|
|
|
async fn task_spawn<'a>(
|
2023-02-11 11:39:39 +00:00
|
|
|
lua: &'static Lua,
|
2023-01-24 17:24:57 +00:00
|
|
|
(tof, args): (LuaValue<'a>, LuaMultiValue<'a>),
|
2023-01-24 17:10:43 +00:00
|
|
|
) -> LuaResult<LuaThread<'a>> {
|
2023-01-23 21:18:20 +00:00
|
|
|
let task_thread = tof_to_thread(lua, tof)?;
|
|
|
|
let task_thread_key = lua.create_registry_value(task_thread)?;
|
2023-01-24 17:24:57 +00:00
|
|
|
let task_args_key = lua.create_registry_value(args.into_vec())?;
|
2023-01-23 21:18:20 +00:00
|
|
|
let lua_thread_to_return = lua.registry_value(&task_thread_key)?;
|
2023-01-23 23:52:31 +00:00
|
|
|
run_registered_task(lua, TaskRunMode::Instant, async move {
|
2023-02-11 11:39:39 +00:00
|
|
|
let thread: LuaThread = lua.registry_value(&task_thread_key)?;
|
|
|
|
let argsv: Vec<LuaValue> = lua.registry_value(&task_args_key)?;
|
2023-01-24 17:24:57 +00:00
|
|
|
let args = LuaMultiValue::from_vec(argsv);
|
2023-01-23 23:52:31 +00:00
|
|
|
if thread.status() == LuaThreadStatus::Resumable {
|
2023-01-24 17:24:57 +00:00
|
|
|
let _: LuaMultiValue = thread.into_async(args).await?;
|
2023-01-23 23:52:31 +00:00
|
|
|
}
|
2023-02-11 11:39:39 +00:00
|
|
|
lua.remove_registry_value(task_thread_key)?;
|
|
|
|
lua.remove_registry_value(task_args_key)?;
|
2023-01-23 23:52:31 +00:00
|
|
|
Ok(())
|
|
|
|
})
|
2023-01-23 21:18:20 +00:00
|
|
|
.await?;
|
|
|
|
Ok(lua_thread_to_return)
|
2023-01-23 07:38:32 +00:00
|
|
|
}
|
|
|
|
|
2023-02-11 11:39:39 +00:00
|
|
|
async fn task_wait(lua: &'static Lua, duration: Option<f32>) -> LuaResult<f32> {
|
2023-01-23 07:38:32 +00:00
|
|
|
let start = Instant::now();
|
2023-01-23 23:52:31 +00:00
|
|
|
run_registered_task(lua, TaskRunMode::Blocking, async move {
|
2023-02-03 19:15:20 +00:00
|
|
|
time::sleep(Duration::from_secs_f32(
|
2023-01-23 23:52:31 +00:00
|
|
|
duration
|
2023-01-24 00:59:35 +00:00
|
|
|
.map(|d| d.max(MINIMUM_WAIT_OR_DELAY_DURATION))
|
|
|
|
.unwrap_or(MINIMUM_WAIT_OR_DELAY_DURATION),
|
|
|
|
))
|
2023-01-23 23:52:31 +00:00
|
|
|
.await;
|
|
|
|
Ok(())
|
|
|
|
})
|
2023-01-23 21:18:20 +00:00
|
|
|
.await?;
|
2023-01-23 07:38:32 +00:00
|
|
|
let end = Instant::now();
|
|
|
|
Ok((end - start).as_secs_f32())
|
|
|
|
}
|