| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217 |
- using System.Runtime.CompilerServices;
- using Lua.Internal;
- using Lua.Runtime;
- namespace Lua;
- public abstract class LuaThread
- {
- protected LuaThread() { }
- public virtual LuaThreadStatus GetStatus()
- {
- return LuaThreadStatus.Running;
- }
- public virtual void UnsafeSetStatus(LuaThreadStatus status) { }
- public virtual ValueTask<int> ResumeAsync(LuaFunctionExecutionContext context, CancellationToken cancellationToken = default)
- {
- return new(context.Return(false, "cannot resume non-suspended coroutine"));
- }
- public virtual ValueTask<int> YieldAsync(LuaFunctionExecutionContext context, CancellationToken cancellationToken = default)
- {
- throw new LuaRuntimeException(context.Thread, "attempt to yield from outside a coroutine");
- }
- protected class ThreadCoreData : IPoolNode<ThreadCoreData>
- {
- //internal LuaCoroutineData? coroutineData;
- internal readonly LuaStack Stack = new();
- internal FastStackCore<CallStackFrame> CallStack;
- public void Clear()
- {
- Stack.Clear();
- CallStack.Clear();
- }
- static LinkedPool<ThreadCoreData> pool;
- ThreadCoreData? nextNode;
- public ref ThreadCoreData? NextNode => ref nextNode;
- public static ThreadCoreData Create()
- {
- if (!pool.TryPop(out ThreadCoreData result))
- {
- result = new ThreadCoreData();
- }
- return result;
- }
- public void Release()
- {
- Clear();
- pool.TryPush(this);
- }
- }
- public LuaState State { get; protected set; } = null!;
- protected ThreadCoreData? CoreData = new();
- internal BitFlags2 LineAndCountHookMask;
- internal BitFlags2 CallOrReturnHookMask;
- internal bool IsInHook;
- internal int HookCount;
- internal int BaseHookCount;
- internal int LastPc;
- internal int LastVersion;
- internal int CurrentVersion;
- internal ILuaTracebackBuildable? CurrentException;
- internal readonly ReversedStack<CallStackFrame> ExceptionTrace = new();
-
- // internal bool CancelRequested;
- // internal CancellationToken CancellationToken;
- public bool IsRunning => CallStackFrameCount != 0;
- internal LuaFunction? Hook { get; set; }
- public LuaStack Stack => CoreData!.Stack;
- internal bool IsLineHookEnabled
- {
- get => LineAndCountHookMask.Flag0;
- set => LineAndCountHookMask.Flag0 = value;
- }
- internal bool IsCountHookEnabled
- {
- get => LineAndCountHookMask.Flag1;
- set => LineAndCountHookMask.Flag1 = value;
- }
- internal bool IsCallHookEnabled
- {
- get => CallOrReturnHookMask.Flag0;
- set => CallOrReturnHookMask.Flag0 = value;
- }
- internal bool IsReturnHookEnabled
- {
- get => CallOrReturnHookMask.Flag1;
- set => CallOrReturnHookMask.Flag1 = value;
- }
- public int CallStackFrameCount => CoreData == null ? 0 : CoreData!.CallStack.Count;
-
- internal LuaThreadAccess CurrentAccess => new(this, CurrentVersion);
- public LuaThreadAccess TopLevelAccess => new(this, 0);
- public ref readonly CallStackFrame GetCurrentFrame()
- {
- return ref CoreData!.CallStack.PeekRef();
- }
- public ReadOnlySpan<LuaValue> GetStackValues()
- {
- return CoreData == null ? default : CoreData!.Stack.AsSpan();
- }
- public ReadOnlySpan<CallStackFrame> GetCallStackFrames()
- {
- return CoreData == null ? default : CoreData!.CallStack.AsSpan();
- }
- void UpdateCurrentVersion(ref FastStackCore<CallStackFrame> callStack)
- {
- CurrentVersion = callStack.Count == 0 ? 0 : callStack.PeekRef().Version;
- }
- [MethodImpl(MethodImplOptions.AggressiveInlining)]
- internal LuaThreadAccess PushCallStackFrame(in CallStackFrame frame)
- {
- CurrentException?.BuildOrGet();
- CurrentException = null;
- ref var callStack = ref CoreData!.CallStack;
- callStack.Push(frame);
- callStack.PeekRef().Version = CurrentVersion = ++LastVersion;
- return new LuaThreadAccess(this, CurrentVersion);
- }
- [MethodImpl(MethodImplOptions.AggressiveInlining)]
- internal void PopCallStackFrameWithStackPop()
- {
- var coreData = CoreData!;
- ref var callStack = ref coreData.CallStack;
- var popFrame = callStack.Pop();
- UpdateCurrentVersion(ref callStack);
- if (CurrentException != null)
- {
- ExceptionTrace.Push(popFrame);
- }
- coreData.Stack.PopUntil(popFrame.ReturnBase);
- }
- [MethodImpl(MethodImplOptions.AggressiveInlining)]
- internal void PopCallStackFrameWithStackPop(int frameBase)
- {
- var coreData = CoreData!;
- ref var callStack = ref coreData.CallStack;
- var popFrame = callStack.Pop();
- UpdateCurrentVersion(ref callStack);
- if (CurrentException != null)
- {
- ExceptionTrace.Push(popFrame);
- }
- {
- coreData.Stack.PopUntil(frameBase);
- }
- }
- [MethodImpl(MethodImplOptions.AggressiveInlining)]
- internal void PopCallStackFrame()
- {
- var coreData = CoreData!;
- ref var callStack = ref coreData.CallStack;
- var popFrame = callStack.Pop();
- UpdateCurrentVersion(ref callStack);
- if (CurrentException != null)
- {
- ExceptionTrace.Push(popFrame);
- }
- }
- [MethodImpl(MethodImplOptions.AggressiveInlining)]
- internal void PopCallStackFrameUntil(int top)
- {
- var coreData = CoreData!;
- ref var callStack = ref coreData.CallStack;
- if (CurrentException != null)
- {
- ExceptionTrace.Push(callStack.AsSpan()[top..]);
- }
- callStack.PopUntil(top);
- UpdateCurrentVersion(ref callStack);
- }
- internal void DumpStackValues()
- {
- var span = GetStackValues();
- for (int i = 0; i < span.Length; i++)
- {
- Console.WriteLine($"LuaStack [{i}]\t{span[i]}");
- }
- }
- public Traceback GetTraceback()
- {
- return new(State, GetCallStackFrames());
- }
- }
|