luna-core 2.2.0

Pure-Rust Lua runtime (interpreter only, zero third-party dependencies). The JIT-equipped variant lives in the `luna-jit` crate.
Documentation
//! Coroutine (thread) objects (P05). A coroutine owns a full execution context
//! — value stack, call frames, open upvalues, to-be-closed slots and stack top
//! — that is swapped into the running `Vm` while it is active and saved back
//! here while it is suspended.

use crate::runtime::Upvalue;
use crate::runtime::function::{CallFrame, ContKind};
use crate::runtime::heap::{Gc, GcHeader, Marker};
use crate::runtime::table::Table;
use crate::runtime::value::Value;

/// Lua coroutine status (PUC `coroutine.status`).
#[derive(Clone, Copy, PartialEq, Eq, Debug)]
pub enum CoroStatus {
    /// created or yielded — resumable
    Suspended,
    /// currently executing (the running thread)
    Running,
    /// resumed another coroutine and is waiting for it
    Normal,
    /// finished or errored — not resumable
    Dead,
}

/// A Lua coroutine (`thread`) — one independent execution context plus its
/// saved value/frame stacks and resume linkage.
#[repr(C)]
pub struct Coro {
    pub(crate) hdr: GcHeader,
    /// Resume state (suspended / running / normal / dead).
    pub status: CoroStatus,
    /// the body function, kept for the first resume (and as a GC root)
    pub body: Value,
    /// whether the body frame has been pushed yet (first resume vs. continue)
    pub started: bool,
    /// the coroutine that resumed this one (to restore on yield/return and for
    /// `coroutine.running`); `None` once suspended/dead
    pub resumer: Option<Gc<Coro>>,
    /// where execution suspended on `yield`: the call slot and result count to
    /// finish that call with the next resume's arguments
    pub resume_at: Option<(u32, i32)>,
    /// the error object a coroutine died with (when it errored rather than
    /// returned); `coroutine.close` reports it once, then clears it
    pub error_value: Option<Value>,
    /// snapshot of the traceback at the error point — captured before the
    /// dying coroutine's frames are unwound, so `debug.traceback(co)` on a
    /// dead-with-error coroutine still shows the error site (PUC's
    /// `luaG_errormsg` flow plus a per-thread `errfunc` snapshot).
    pub error_traceback: Option<Vec<u8>>,
    // ---- saved execution context (valid while suspended/normal) ----
    /// Saved value stack.
    pub stack: Vec<Value>,
    /// Saved frame stack (Lua frames + native continuations).
    pub frames: Vec<CallFrame>,
    /// Open-upvalue list — `(stack slot, upvalue cell)` pairs.
    pub open_upvals: Vec<(u32, Gc<Upvalue>)>,
    /// Stack indices of registered `<close>` slots (5.4+).
    pub tbc: Vec<u32>,
    /// Saved stack top.
    pub top: u32,
    /// live pcall/xpcall continuation count (PUC nCcalls portion); see Vm
    pub pcall_depth: u32,
    /// this thread's debug hook state (PUC per-thread hook/hookmask)
    pub hook: crate::vm::exec::HookState,
    /// PUC `L->l_gt` — the thread's own globals table. Captured from the
    /// resuming thread at create time, then swapped with `Vm.globals` on
    /// every resume/yield boundary so a `setfenv(0, env)` inside the
    /// coroutine only retunes *this* thread (5.1 closure.lua :177 pins
    /// this — yielding `getfenv()` after the rewire must see the
    /// coroutine's own per-closure env, not the caller's).
    pub globals: Gc<Table>,
}

impl Coro {
    pub(crate) fn trace(&self, m: &mut Marker) {
        m.value(self.body);
        for &v in self.stack.iter() {
            m.value(v);
        }
        for cf in self.frames.iter() {
            match cf {
                CallFrame::Lua(f) => {
                    m.header(f.closure.as_ptr() as *mut GcHeader);
                }
                CallFrame::Cont(nc) => {
                    if let ContKind::Xpcall { handler } = nc.kind {
                        m.value(handler);
                    }
                }
            }
        }
        for &(_, uv) in self.open_upvals.iter() {
            m.header(uv.as_ptr() as *mut GcHeader);
        }
        if let Some(r) = self.resumer {
            m.header(r.as_ptr() as *mut GcHeader);
        }
        if let Some(e) = self.error_value {
            m.value(e);
        }
        if let Some(h) = self.hook.func {
            m.value(h);
        }
        m.value(Value::Table(self.globals));
    }
}