use boa_gc::{Finalize, Gc, Trace};
use crate::{Context, JsResult, JsValue};
#[derive(Trace, Finalize)]
#[boa_gc(unsafe_no_drop)]
pub(crate) enum CoroutineState {
Yielded(JsValue),
Done,
}
trait TraceableCoroutine: Trace {
fn call(&self, value: JsResult<JsValue>, context: &mut Context) -> CoroutineState;
}
#[derive(Trace, Finalize)]
struct Coroutine<F, T>
where
F: Fn(JsResult<JsValue>, &T, &mut Context) -> CoroutineState,
T: Trace,
{
#[unsafe_ignore_trace]
f: F,
captures: T,
}
impl<F, T> TraceableCoroutine for Coroutine<F, T>
where
F: Fn(JsResult<JsValue>, &T, &mut Context) -> CoroutineState,
T: Trace,
{
fn call(&self, result: JsResult<JsValue>, context: &mut Context) -> CoroutineState {
(self.f)(result, &self.captures, context)
}
}
#[derive(Clone, Trace, Finalize)]
pub(crate) struct NativeCoroutine {
inner: Gc<dyn TraceableCoroutine>,
}
impl std::fmt::Debug for NativeCoroutine {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.debug_struct("NativeCoroutine").finish_non_exhaustive()
}
}
impl NativeCoroutine {
pub(crate) fn from_copy_closure_with_captures<F, T>(closure: F, captures: T) -> Self
where
F: Fn(JsResult<JsValue>, &T, &mut Context) -> CoroutineState + Copy + 'static,
T: Trace + 'static,
{
unsafe { Self::from_closure_with_captures(closure, captures) }
}
pub(crate) unsafe fn from_closure_with_captures<F, T>(closure: F, captures: T) -> Self
where
F: Fn(JsResult<JsValue>, &T, &mut Context) -> CoroutineState + 'static,
T: Trace + 'static,
{
let ptr = Gc::into_raw(Gc::new(Coroutine {
f: closure,
captures,
}));
unsafe {
Self {
inner: Gc::from_raw(ptr),
}
}
}
#[inline]
pub(crate) fn call(&self, result: JsResult<JsValue>, context: &mut Context) -> CoroutineState {
self.inner.call(result, context)
}
}