|
|
@ -78,16 +78,16 @@ struct Thread
|
|
|
|
// temporary data off the stack.
|
|
|
|
// temporary data off the stack.
|
|
|
|
SharedType idleData;
|
|
|
|
SharedType idleData;
|
|
|
|
|
|
|
|
|
|
|
|
// The contents of idleObj's extra data for the idle's owner class.
|
|
|
|
// The contents of the idle object's extra data for the idle's owner
|
|
|
|
|
|
|
|
// class.
|
|
|
|
SharedType extraData;
|
|
|
|
SharedType extraData;
|
|
|
|
|
|
|
|
|
|
|
|
// Set to true whenever we are running from state code. If we are
|
|
|
|
|
|
|
|
// inside the state itself, this will be true and 'next' will be 1.
|
|
|
|
|
|
|
|
bool isActive;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Set to true when a state change is in progress. Only used when
|
|
|
|
// Set to true when a state change is in progress. Only used when
|
|
|
|
// state is changed from within a function in active code.
|
|
|
|
// state is changed from within a function in active code.
|
|
|
|
bool stateChange;
|
|
|
|
bool shouldExit;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Function stack for this thread
|
|
|
|
|
|
|
|
FunctionStack fstack;
|
|
|
|
|
|
|
|
|
|
|
|
/*******************************************************
|
|
|
|
/*******************************************************
|
|
|
|
* *
|
|
|
|
* *
|
|
|
@ -96,21 +96,11 @@ struct Thread
|
|
|
|
*******************************************************/
|
|
|
|
*******************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
private:
|
|
|
|
// Temporarily needed since we need a state and an object to push on
|
|
|
|
NodeList *list; // List owning this thread
|
|
|
|
// the stack to return to state code. This'll change soon (we won't
|
|
|
|
|
|
|
|
// need to push anything to reenter, since the function stack will
|
|
|
|
|
|
|
|
// already be set up for us.)
|
|
|
|
|
|
|
|
MonsterObject * theObj;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Function *idle;
|
|
|
|
|
|
|
|
MonsterObject *idleObj; // Object owning the idle function
|
|
|
|
|
|
|
|
NodeList * list; // List owning this thread
|
|
|
|
|
|
|
|
int retPos; // Return position in byte code.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Stored copy of the stack. Used when the thread is not running.
|
|
|
|
// Stored copy of the stack. Used when the thread is not running.
|
|
|
|
int[] sstack;
|
|
|
|
int[] sstack;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
public:
|
|
|
|
/*******************************************************
|
|
|
|
/*******************************************************
|
|
|
|
* *
|
|
|
|
* *
|
|
|
@ -118,91 +108,120 @@ struct Thread
|
|
|
|
* *
|
|
|
|
* *
|
|
|
|
*******************************************************/
|
|
|
|
*******************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
// Get a new thread. It starts in the 'unused' list.
|
|
|
|
// Get a new thread. It starts in the 'transient' list.
|
|
|
|
static Thread* getNew(MonsterObject *obj = null)
|
|
|
|
static Thread* getNew()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
auto cn = scheduler.unused.getNew();
|
|
|
|
auto cn = scheduler.transient.getNew();
|
|
|
|
cn.list = &scheduler.unused;
|
|
|
|
cn.list = &scheduler.transient;
|
|
|
|
|
|
|
|
|
|
|
|
with(*cn)
|
|
|
|
with(*cn)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
theObj = obj;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Initialize other variables
|
|
|
|
// Initialize other variables
|
|
|
|
idle = null;
|
|
|
|
shouldExit = false;
|
|
|
|
idleObj = null;
|
|
|
|
|
|
|
|
isActive = false;
|
|
|
|
|
|
|
|
stateChange = false;
|
|
|
|
|
|
|
|
retPos = -1;
|
|
|
|
|
|
|
|
sstack = null;
|
|
|
|
sstack = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
if(obj !is null)
|
|
|
|
|
|
|
|
writefln("Got a new state thread");
|
|
|
|
|
|
|
|
else
|
|
|
|
|
|
|
|
writefln("Got a new non-state thread");
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return cn;
|
|
|
|
return cn;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Unschedule this node from the runlist or waitlist it belongs to,
|
|
|
|
// Stop the thread and return it to the freelist
|
|
|
|
// but don't kill it. Any idle function connected to this node is
|
|
|
|
void kill()
|
|
|
|
// aborted.
|
|
|
|
|
|
|
|
void cancel()
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if(idle !is null)
|
|
|
|
stop();
|
|
|
|
|
|
|
|
assert(fstack.isEmpty);
|
|
|
|
|
|
|
|
list.remove(this);
|
|
|
|
|
|
|
|
list = null;
|
|
|
|
|
|
|
|
assert(isDead);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Stop the execution of a thread and cancel any scheduling.
|
|
|
|
|
|
|
|
void stop()
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
assert(!isDead);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// TODO: We also have to handle (forbid) cases where we are
|
|
|
|
|
|
|
|
// the caller of another thread.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if(isRunning)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
fstack.pushIdle(idle, idleObj);
|
|
|
|
// We are running.
|
|
|
|
idle.idleFunc.abort(this);
|
|
|
|
assert(sstack.length == 0);
|
|
|
|
fstack.pop();
|
|
|
|
|
|
|
|
idle = null;
|
|
|
|
// Forbid stopping the thread if there are native functions on
|
|
|
|
|
|
|
|
// the function stack.
|
|
|
|
|
|
|
|
if(fstack.hasNatives)
|
|
|
|
|
|
|
|
fail("Cannot stop thread, there are native functions on the stack.");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Kill the stack tell execute() to stop running
|
|
|
|
|
|
|
|
stack.reset();
|
|
|
|
|
|
|
|
shouldExit = true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
// We are not running
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Free the stack buffers
|
|
|
|
|
|
|
|
if(sstack.length)
|
|
|
|
|
|
|
|
Buffers.free(sstack);
|
|
|
|
|
|
|
|
sstack = null;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Abort any idle function
|
|
|
|
|
|
|
|
if(fstack.isIdle)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
// Abort the idle function and pop it
|
|
|
|
|
|
|
|
getIdle().abort(this);
|
|
|
|
|
|
|
|
fstack.pop();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(!fstack.hasNatives);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
retPos = -1;
|
|
|
|
|
|
|
|
moveTo(&scheduler.unused);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Kill the function stack
|
|
|
|
|
|
|
|
fstack.killAll();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Move to the transient list (signalling that the thread is
|
|
|
|
|
|
|
|
// unused.)
|
|
|
|
|
|
|
|
moveTo(&scheduler.transient);
|
|
|
|
assert(!isScheduled);
|
|
|
|
assert(!isScheduled);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Remove the thread comletely
|
|
|
|
// Schedule this thread to run state code the next frame
|
|
|
|
void kill()
|
|
|
|
void scheduleState(MonsterObject *obj, int offs)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
assert(!isDead);
|
|
|
|
if(theObj is null)
|
|
|
|
assert(!isScheduled,
|
|
|
|
writefln("Killing non-state thread");
|
|
|
|
"cannot schedule an already scheduled thread");
|
|
|
|
else
|
|
|
|
assert(!fstack.isIdle);
|
|
|
|
writefln("Killing state thread");
|
|
|
|
assert(fstack.isEmpty);
|
|
|
|
*/
|
|
|
|
assert(offs >= 0);
|
|
|
|
|
|
|
|
assert(obj !is null);
|
|
|
|
|
|
|
|
|
|
|
|
cancel();
|
|
|
|
assert(isRunning == shouldExit);
|
|
|
|
list.remove(this);
|
|
|
|
|
|
|
|
list = null;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if(sstack.length)
|
|
|
|
// Move to the runlist
|
|
|
|
Buffers.free(sstack);
|
|
|
|
moveTo(scheduler.runNext);
|
|
|
|
sstack = null;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
writefln("Thread lists:");
|
|
|
|
|
|
|
|
writefln(" run: ", scheduler.run.length);
|
|
|
|
|
|
|
|
writefln(" runNext: ", scheduler.runNext.length);
|
|
|
|
|
|
|
|
writefln(" wait: ", scheduler.wait.length);
|
|
|
|
|
|
|
|
writefln(" unused: ", scheduler.unused.length);
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool isDead() { return list is null; }
|
|
|
|
// Set up the function stack
|
|
|
|
|
|
|
|
fstack.push(obj.state, obj);
|
|
|
|
|
|
|
|
fstack.cur.code.jump(offs);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Schedule this thread to run next frame
|
|
|
|
// Push a function and pause the thread
|
|
|
|
void schedule(int offs)
|
|
|
|
void pushFunc(Function *fn, MonsterObject *obj)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
assert(!isDead);
|
|
|
|
assert(!isScheduled,
|
|
|
|
assert(!isScheduled,
|
|
|
|
"cannot schedule an already scheduled thread");
|
|
|
|
"cannot schedule an already scheduled thread");
|
|
|
|
|
|
|
|
assert(fstack.isEmpty);
|
|
|
|
|
|
|
|
assert(!fstack.isIdle);
|
|
|
|
|
|
|
|
assert(fn !is null);
|
|
|
|
|
|
|
|
assert(obj !is null);
|
|
|
|
|
|
|
|
assert(!fn.isIdle);
|
|
|
|
|
|
|
|
|
|
|
|
retPos = offs;
|
|
|
|
// Set up the function stack
|
|
|
|
assert(offs >= 0);
|
|
|
|
assert(fn.owner.parentOf(obj));
|
|
|
|
moveTo(scheduler.runNext);
|
|
|
|
fstack.push(fn, obj);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
moveTo(&scheduler.paused);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Are we currently scheduled?
|
|
|
|
// Are we currently scheduled?
|
|
|
@ -211,17 +230,15 @@ struct Thread
|
|
|
|
// The node is per definition scheduled if it is in one of these
|
|
|
|
// The node is per definition scheduled if it is in one of these
|
|
|
|
// lists
|
|
|
|
// lists
|
|
|
|
return
|
|
|
|
return
|
|
|
|
list is &scheduler.wait ||
|
|
|
|
!isDead && (list is &scheduler.wait ||
|
|
|
|
list is scheduler.run ||
|
|
|
|
list is scheduler.run ||
|
|
|
|
list is scheduler.runNext;
|
|
|
|
list is scheduler.runNext);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool isUnused()
|
|
|
|
bool isTransient() { return list is &scheduler.transient; }
|
|
|
|
{
|
|
|
|
bool isRunning() { return cthread is this; }
|
|
|
|
return list is &scheduler.unused;
|
|
|
|
bool isDead() { return list is null; }
|
|
|
|
}
|
|
|
|
bool isPaused() { return list is &scheduler.paused; }
|
|
|
|
|
|
|
|
|
|
|
|
bool isIdle() { return idle !is null; }
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Get the next node in the freelist
|
|
|
|
// Get the next node in the freelist
|
|
|
|
Thread* getNext()
|
|
|
|
Thread* getNext()
|
|
|
@ -237,120 +254,122 @@ struct Thread
|
|
|
|
// Reenter this thread to the point where it was previously stopped.
|
|
|
|
// Reenter this thread to the point where it was previously stopped.
|
|
|
|
void reenter()
|
|
|
|
void reenter()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
assert(theObj !is null,
|
|
|
|
assert(!isDead);
|
|
|
|
"cannot reenter a non-state thread yet");
|
|
|
|
assert(cthread is null,
|
|
|
|
|
|
|
|
"cannot reenter when another thread is running");
|
|
|
|
// Most if not all of these checks will have to be removed in the
|
|
|
|
assert(!isRunning,
|
|
|
|
// future
|
|
|
|
"reenter cannot be called when thread is already running");
|
|
|
|
assert(theObj.state !is null, "attempted to call the empty state");
|
|
|
|
assert(isScheduled || isPaused);
|
|
|
|
assert(!isActive,
|
|
|
|
assert(!fstack.isEmpty);
|
|
|
|
"reenter cannot be called when object is already active");
|
|
|
|
|
|
|
|
assert(fstack.isEmpty,
|
|
|
|
// Put the thread in the foreground
|
|
|
|
"can only reenter at the bottom of the function stack");
|
|
|
|
foreground();
|
|
|
|
assert(isScheduled);
|
|
|
|
assert(isRunning);
|
|
|
|
|
|
|
|
|
|
|
|
if(isIdle)
|
|
|
|
// Notify the idle function, if any
|
|
|
|
|
|
|
|
if(fstack.isIdle)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
assert(idle !is null);
|
|
|
|
|
|
|
|
assert(idleObj !is null || idle.isStatic);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Tell the idle function that we we are reentering
|
|
|
|
// Tell the idle function that we we are reentering
|
|
|
|
fstack.pushIdle(idle, idleObj);
|
|
|
|
assert(fstack.isIdle);
|
|
|
|
idle.idleFunc.reentry(this);
|
|
|
|
getIdle().reentry(this);
|
|
|
|
fstack.pop();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// We're no longer idle
|
|
|
|
// Remove the idle function
|
|
|
|
idle = null;
|
|
|
|
fstack.pop();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Set the active flat to indicate that we are now actively
|
|
|
|
assert(fstack.cur.isNormal,
|
|
|
|
// running. (Might not be needed in the future)
|
|
|
|
"can only reenter script code");
|
|
|
|
isActive = true;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Set the thread
|
|
|
|
|
|
|
|
assert(cthread is null);
|
|
|
|
|
|
|
|
cthread = this;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Remove the current thread from the run list
|
|
|
|
// Remove the current thread from the run list
|
|
|
|
moveTo(&scheduler.unused);
|
|
|
|
moveTo(&scheduler.transient);
|
|
|
|
|
|
|
|
|
|
|
|
// Restore the stack
|
|
|
|
|
|
|
|
restoreStack();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Set up the code stack for state code.
|
|
|
|
|
|
|
|
fstack.push(theObj.state, theObj);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Set the position
|
|
|
|
|
|
|
|
assert(retPos >= 0);
|
|
|
|
|
|
|
|
fstack.cur.code.jump(retPos);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Run the code
|
|
|
|
// Run the code
|
|
|
|
execute();
|
|
|
|
execute();
|
|
|
|
|
|
|
|
|
|
|
|
// Reset the thread
|
|
|
|
// Exit immediately if the thread committed suicide.
|
|
|
|
cthread = null;
|
|
|
|
if(isDead) return;
|
|
|
|
|
|
|
|
|
|
|
|
fstack.pop();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// We are no longer active
|
|
|
|
shouldExit = false;
|
|
|
|
isActive = false;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
assert(stack.getPos == 0,
|
|
|
|
// Background the thread
|
|
|
|
format("Stack not returned to zero after state code, __STACK__=",
|
|
|
|
background();
|
|
|
|
stack.getPos));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if(!isUnused)
|
|
|
|
|
|
|
|
// Store the stack
|
|
|
|
|
|
|
|
acquireStack();
|
|
|
|
|
|
|
|
else
|
|
|
|
|
|
|
|
// If the thread is not used for anything, might as well kill it
|
|
|
|
|
|
|
|
kill();
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Make a copy of the stack and store it for later. Reset the global
|
|
|
|
// Put this thread in the background. Acquires the stack and
|
|
|
|
// stack.
|
|
|
|
// function stack.
|
|
|
|
void acquireStack()
|
|
|
|
void background()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
assert(!isUnused(),
|
|
|
|
assert(!isDead);
|
|
|
|
"unused threads should never need to aquire the stack");
|
|
|
|
|
|
|
|
assert(sstack.length == 0,
|
|
|
|
assert(sstack.length == 0,
|
|
|
|
"Thread already has a stack");
|
|
|
|
"Thread already has a stack");
|
|
|
|
assert(fstack.isEmpty);
|
|
|
|
assert(isRunning,
|
|
|
|
|
|
|
|
"cannot put a non-running thread in the background");
|
|
|
|
|
|
|
|
assert(!fstack.hasNatives);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// We're no longer the current thread
|
|
|
|
|
|
|
|
cthread = null;
|
|
|
|
|
|
|
|
|
|
|
|
// This can be optimized later
|
|
|
|
if(isTransient)
|
|
|
|
int len = stack.getPos();
|
|
|
|
|
|
|
|
if(len)
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
writefln("acquiring %s ints", len);
|
|
|
|
// The thread is not scheduled and will not be used
|
|
|
|
|
|
|
|
// anymore. Might as well kill it.
|
|
|
|
|
|
|
|
assert(!isRunning);
|
|
|
|
|
|
|
|
kill();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
// The thread will possibly be restored at some point. Store
|
|
|
|
|
|
|
|
// the stack and fstack for later.
|
|
|
|
|
|
|
|
|
|
|
|
// Get a new buffer, and copy the stack
|
|
|
|
// Stack.
|
|
|
|
sstack = Buffers.getInt(len);
|
|
|
|
int len = stack.getPos();
|
|
|
|
sstack[] = stack.popInts(len);
|
|
|
|
if(len)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
// Get a new buffer, and copy the stack
|
|
|
|
|
|
|
|
sstack = Buffers.getInt(len);
|
|
|
|
|
|
|
|
sstack[] = stack.popInts(len);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Clear out our stack values
|
|
|
|
stack.reset();
|
|
|
|
stack.reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
assert(!isRunning);
|
|
|
|
/*******************************************************
|
|
|
|
}
|
|
|
|
* *
|
|
|
|
|
|
|
|
* Private helper functions *
|
|
|
|
|
|
|
|
* *
|
|
|
|
|
|
|
|
*******************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void restoreStack()
|
|
|
|
// Put the thread in the foreground. Restore any stored stack
|
|
|
|
|
|
|
|
// values.
|
|
|
|
|
|
|
|
void foreground()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
assert(stack.getPos() == 0,
|
|
|
|
assert(!isDead);
|
|
|
|
"cannot restore into a non-empty stack");
|
|
|
|
assert(cthread is null,
|
|
|
|
|
|
|
|
"cannot restore thread, another thread is running");
|
|
|
|
|
|
|
|
assert(!isRunning,
|
|
|
|
|
|
|
|
"cannot restore thread, it is already running");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
assert((isTransient && fstack.isEmpty) ||
|
|
|
|
|
|
|
|
stack.getPos() == 0,
|
|
|
|
|
|
|
|
"only empty transient threads kan restore into a non-empty stack");
|
|
|
|
|
|
|
|
|
|
|
|
if(sstack.length)
|
|
|
|
if(sstack.length)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
assert(stack.getPos() == 0,
|
|
|
|
|
|
|
|
"cannot restore into a non-empty stack");
|
|
|
|
|
|
|
|
assert(!isTransient,
|
|
|
|
|
|
|
|
"cannot restore a transent thread with stack");
|
|
|
|
|
|
|
|
|
|
|
|
// Push the values back, and free the buffer
|
|
|
|
// Push the values back, and free the buffer
|
|
|
|
stack.pushInts(sstack);
|
|
|
|
stack.pushInts(sstack);
|
|
|
|
Buffers.free(sstack);
|
|
|
|
Buffers.free(sstack);
|
|
|
|
assert(stack.getPos == sstack.length);
|
|
|
|
assert(stack.getPos == sstack.length);
|
|
|
|
sstack = null;
|
|
|
|
sstack = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Restore the stack frame pointer
|
|
|
|
|
|
|
|
fstack.restoreFrame();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Set ourselves as the running thread
|
|
|
|
|
|
|
|
cthread = this;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Move this node to another list.
|
|
|
|
// Move this node to another list.
|
|
|
@ -361,6 +380,21 @@ struct Thread
|
|
|
|
list = to;
|
|
|
|
list = to;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
/*******************************************************
|
|
|
|
|
|
|
|
* *
|
|
|
|
|
|
|
|
* Private helper functions *
|
|
|
|
|
|
|
|
* *
|
|
|
|
|
|
|
|
*******************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
IdleFunction getIdle()
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
assert(fstack.isIdle);
|
|
|
|
|
|
|
|
assert(fstack.cur.func !is null);
|
|
|
|
|
|
|
|
assert(fstack.cur.func.idleFunc !is null);
|
|
|
|
|
|
|
|
return fstack.cur.func.idleFunc;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void fail(char[] msg)
|
|
|
|
void fail(char[] msg)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
Floc fl;
|
|
|
|
Floc fl;
|
|
|
@ -371,61 +405,72 @@ struct Thread
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Parse the BC.CallIdle instruction parameters and schedule the
|
|
|
|
// Parse the BC.CallIdle instruction parameters and schedule the
|
|
|
|
// given idle function.
|
|
|
|
// given idle function. Return true if we should exit execute()
|
|
|
|
void callIdle(MonsterObject *iObj)
|
|
|
|
bool callIdle(MonsterObject *iObj)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
assert(isActive && fstack.isStateCode,
|
|
|
|
assert(isRunning);
|
|
|
|
"Byte code attempted to call an idle function outside of state code."); assert(!isScheduled, "Thread is already scheduled");
|
|
|
|
assert(!isScheduled, "Thread is already scheduled");
|
|
|
|
|
|
|
|
assert(iObj !is null);
|
|
|
|
|
|
|
|
|
|
|
|
CodeStream *code = &fstack.cur.code;
|
|
|
|
if(fstack.hasNatives)
|
|
|
|
|
|
|
|
fail("Cannot run idle function: there are native functions on the stack");
|
|
|
|
|
|
|
|
|
|
|
|
// Store the object
|
|
|
|
CodeStream *code = &fstack.cur.code;
|
|
|
|
idleObj = iObj;
|
|
|
|
|
|
|
|
assert(idleObj !is null);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Get the class from the index
|
|
|
|
// Get the class from the index
|
|
|
|
auto cls = iObj.cls.upcast(code.getInt());
|
|
|
|
auto cls = iObj.cls.upcast(code.getInt());
|
|
|
|
|
|
|
|
|
|
|
|
// Get the function
|
|
|
|
// Get the function
|
|
|
|
idle = cls.findFunction(code.getInt());
|
|
|
|
Function *idle = cls.findFunction(code.getInt());
|
|
|
|
assert(idle !is null && idle.isIdle);
|
|
|
|
assert(idle !is null && idle.isIdle);
|
|
|
|
assert(cls is idle.owner);
|
|
|
|
assert(cls is idle.owner);
|
|
|
|
assert(idleObj.cls.childOf(cls));
|
|
|
|
assert(iObj.cls.childOf(cls));
|
|
|
|
|
|
|
|
|
|
|
|
// The IdleFunction object bound to this function is stored in
|
|
|
|
// The IdleFunction object bound to this function is stored in
|
|
|
|
// idle.idleFunc
|
|
|
|
// idle.idleFunc
|
|
|
|
if(idle.idleFunc is null)
|
|
|
|
if(idle.idleFunc is null)
|
|
|
|
fail("Called unimplemented idle function '" ~ idle.name.str ~ "'");
|
|
|
|
fail("Called unimplemented idle function '" ~ idle.name.str ~ "'");
|
|
|
|
|
|
|
|
|
|
|
|
// Set the return position
|
|
|
|
|
|
|
|
retPos = fstack.cur.code.getPos();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Set up extraData
|
|
|
|
// Set up extraData
|
|
|
|
extraData = *idleObj.getExtra(idle.owner);
|
|
|
|
extraData = *iObj.getExtra(idle.owner);
|
|
|
|
|
|
|
|
|
|
|
|
// Notify the idle function
|
|
|
|
// Push the idle function on the stack, with iObj as the 'this'
|
|
|
|
fstack.pushIdle(idle, idleObj);
|
|
|
|
// object.
|
|
|
|
if(idle.idleFunc.initiate(this))
|
|
|
|
fstack.pushIdle(idle, iObj);
|
|
|
|
moveTo(&scheduler.wait);
|
|
|
|
|
|
|
|
fstack.pop();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool shouldExit()
|
|
|
|
// Notify the idle function that it was called
|
|
|
|
{
|
|
|
|
auto res = idle.idleFunc.initiate(this);
|
|
|
|
if(fstack.isStateCode && stateChange)
|
|
|
|
//writefln("Called %s, result was %s", idle.name, res);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if(res == IS.Poll)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
assert(isActive);
|
|
|
|
moveTo(&scheduler.wait);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// The state was changed while in state code. Abort the code.
|
|
|
|
if(res == IS.Return)
|
|
|
|
stateChange = false;
|
|
|
|
{
|
|
|
|
|
|
|
|
// If we're returning, call reenter immediately
|
|
|
|
|
|
|
|
idle.idleFunc.reentry(this);
|
|
|
|
|
|
|
|
|
|
|
|
// There might be dangling stack values
|
|
|
|
// The function is done, pop it back of the stack
|
|
|
|
stack.reset();
|
|
|
|
fstack.pop();
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
|
|
// 'false' means continue running
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
assert(res == IS.Manual || res == IS.Kill);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// The only difference between Manual and Kill is what list the
|
|
|
|
|
|
|
|
// thread ends in. If the thread is in the transient list, it will
|
|
|
|
|
|
|
|
// be killed automatically when it's no longer running.
|
|
|
|
|
|
|
|
assert( (res == IS.Kill) == isTransient,
|
|
|
|
|
|
|
|
res == IS.Manual ? "Manually scheduled threads must be moved to another list." : "Killed threads cannot be moved to another list.");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// 'true' means exit execute()
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*******************************************************
|
|
|
|
/*******************************************************
|
|
|
|
* *
|
|
|
|
* *
|
|
|
@ -444,10 +489,12 @@ struct Thread
|
|
|
|
// an infinite loop.
|
|
|
|
// an infinite loop.
|
|
|
|
static const long limit = 10000000;
|
|
|
|
static const long limit = 10000000;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
assert(!isDead);
|
|
|
|
assert(fstack.cur !is null,
|
|
|
|
assert(fstack.cur !is null,
|
|
|
|
"Thread.execute called but there is no code on the function stack.");
|
|
|
|
"Thread.execute called but there is no code on the function stack.");
|
|
|
|
|
|
|
|
assert(fstack.cur.isNormal,
|
|
|
|
assert(cthread == this,
|
|
|
|
"execute() can only run script code");
|
|
|
|
|
|
|
|
assert(isRunning,
|
|
|
|
"can only run the current thread");
|
|
|
|
"can only run the current thread");
|
|
|
|
|
|
|
|
|
|
|
|
// Get some values from the function stack
|
|
|
|
// Get some values from the function stack
|
|
|
@ -544,78 +591,97 @@ struct Thread
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
case BC.Exit:
|
|
|
|
case BC.Exit:
|
|
|
|
// Exit execute(). The function stack cleanup is handled
|
|
|
|
// Step down once on the function stack
|
|
|
|
// by Our Glorious Caller.
|
|
|
|
fstack.pop();
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
case BC.Call:
|
|
|
|
if(!fstack.isNormal())
|
|
|
|
{
|
|
|
|
// The current function isn't a script function, so
|
|
|
|
// Get the correct function from the virtual table
|
|
|
|
// exit.
|
|
|
|
val = code.getInt(); // Class index
|
|
|
|
return;
|
|
|
|
auto fn = obj.cls.findVirtualFunc(val, code.getInt());
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Finally, call
|
|
|
|
assert(!shouldExit);
|
|
|
|
fn.call(obj);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if(shouldExit()) return;
|
|
|
|
// Set up the variables and continue running.
|
|
|
|
break;
|
|
|
|
assert(fstack.cur !is null);
|
|
|
|
}
|
|
|
|
cls = fstack.cur.getCls();
|
|
|
|
|
|
|
|
code = &fstack.cur.code;
|
|
|
|
|
|
|
|
obj = fstack.cur.obj;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case BC.CallFar:
|
|
|
|
// Start a block so these variables are local
|
|
|
|
{
|
|
|
|
{
|
|
|
|
auto mo = stack.popObject();
|
|
|
|
MonsterObject *mo;
|
|
|
|
|
|
|
|
Function *fn;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
case BC.Call:
|
|
|
|
|
|
|
|
mo = obj;
|
|
|
|
|
|
|
|
goto CallCommon;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
case BC.CallFar:
|
|
|
|
|
|
|
|
mo = stack.popObject();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CallCommon:
|
|
|
|
|
|
|
|
|
|
|
|
// Get the correct function from the virtual table
|
|
|
|
// Get the correct function from the virtual table
|
|
|
|
val = code.getInt(); // Class index
|
|
|
|
val = code.getInt(); // Class index
|
|
|
|
auto fn = mo.cls.findVirtualFunc(val, code.getInt());
|
|
|
|
fn = mo.cls.findVirtualFunc(val, code.getInt());
|
|
|
|
|
|
|
|
|
|
|
|
// Call the function
|
|
|
|
|
|
|
|
fn.call(mo);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Exit state code if the state was changed
|
|
|
|
if(fn.isNormal)
|
|
|
|
if(shouldExit()) return;
|
|
|
|
{
|
|
|
|
|
|
|
|
// Normal (script) function. We don't need to exit
|
|
|
|
|
|
|
|
// execute(), just change the function stack and the
|
|
|
|
|
|
|
|
// cls and code pointers. Then keep running.
|
|
|
|
|
|
|
|
fstack.push(fn, mo);
|
|
|
|
|
|
|
|
cls = fstack.cur.getCls();
|
|
|
|
|
|
|
|
code = &fstack.cur.code;
|
|
|
|
|
|
|
|
obj = mo;
|
|
|
|
|
|
|
|
assert(obj is fstack.cur.obj);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
// Native function. Let Function handle it.
|
|
|
|
|
|
|
|
assert(fn.isNative);
|
|
|
|
|
|
|
|
fn.call(mo);
|
|
|
|
|
|
|
|
if(shouldExit) return;
|
|
|
|
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
case BC.CallIdle:
|
|
|
|
case BC.CallIdle:
|
|
|
|
callIdle(obj);
|
|
|
|
if(callIdle(obj))
|
|
|
|
return;
|
|
|
|
return;
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case BC.CallIdleFar:
|
|
|
|
case BC.CallIdleFar:
|
|
|
|
callIdle(stack.popObject());
|
|
|
|
if(callIdle(stack.popObject()))
|
|
|
|
return;
|
|
|
|
return;
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case BC.Return:
|
|
|
|
case BC.Return:
|
|
|
|
// Remove the given number of bytes from the stack, and
|
|
|
|
// Remove the given number of bytes from the stack, and
|
|
|
|
// exit the function.
|
|
|
|
// exit the function.
|
|
|
|
stack.pop(code.getInt());
|
|
|
|
stack.pop(code.getInt());
|
|
|
|
return;
|
|
|
|
goto case BC.Exit;
|
|
|
|
|
|
|
|
|
|
|
|
case BC.ReturnVal:
|
|
|
|
case BC.ReturnVal:
|
|
|
|
stack.pop(code.getInt(), 1);
|
|
|
|
stack.pop(code.getInt(), 1);
|
|
|
|
return;
|
|
|
|
goto case BC.Exit;
|
|
|
|
|
|
|
|
|
|
|
|
case BC.ReturnValN:
|
|
|
|
case BC.ReturnValN:
|
|
|
|
val = code.getInt(); // Get the value first, since order
|
|
|
|
val = code.getInt(); // Get the value first, since order
|
|
|
|
// of evaluation is important.
|
|
|
|
// of evaluation is important.
|
|
|
|
stack.pop(val, code.getInt());
|
|
|
|
stack.pop(val, code.getInt());
|
|
|
|
return;
|
|
|
|
goto case BC.Exit;
|
|
|
|
|
|
|
|
|
|
|
|
case BC.State:
|
|
|
|
case BC.State:
|
|
|
|
val = code.getInt(); // State index
|
|
|
|
val = code.getInt(); // State index
|
|
|
|
val2 = code.getInt(); // Label index
|
|
|
|
val2 = code.getInt(); // Label index
|
|
|
|
// Get the class index and let setState handle everything
|
|
|
|
// Get the class index and let setState handle everything
|
|
|
|
obj.setState(val, val2, code.getInt());
|
|
|
|
obj.setState(val, val2, code.getInt());
|
|
|
|
if(shouldExit()) return;
|
|
|
|
if(shouldExit) return;
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case BC.Halt:
|
|
|
|
|
|
|
|
// A halt statement was encountered, and we should stop
|
|
|
|
|
|
|
|
// execution. This is only allowed in state code.
|
|
|
|
|
|
|
|
assert(isActive && fstack.isStateCode,
|
|
|
|
|
|
|
|
"halt command encountered outside state code.");
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
case BC.New:
|
|
|
|
case BC.New:
|
|
|
|
// Create a new object. Look up the class index in the
|
|
|
|
// Create a new object. Look up the class index in the
|
|
|
|
// global class table, and create an object from it.
|
|
|
|
// global class table, and create an object from it.
|
|
|
@ -1372,9 +1438,14 @@ struct Scheduler
|
|
|
|
// Waiting list - idle threads that are actively checked each frame.
|
|
|
|
// Waiting list - idle threads that are actively checked each frame.
|
|
|
|
NodeList wait;
|
|
|
|
NodeList wait;
|
|
|
|
|
|
|
|
|
|
|
|
// List of unused nodes. Any thread in this list (that is not
|
|
|
|
// List of transient nodes. Any thread in this list (that is not
|
|
|
|
// actively running) can and will be deleted eventually.
|
|
|
|
// actively running) can and will be deleted when it goes into the
|
|
|
|
NodeList unused;
|
|
|
|
// background.
|
|
|
|
|
|
|
|
NodeList transient;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// List of threads that are not running or scheduled, but should not
|
|
|
|
|
|
|
|
// be deleted.
|
|
|
|
|
|
|
|
NodeList paused;
|
|
|
|
|
|
|
|
|
|
|
|
// The run lists for this and the next round. We use pointers to the
|
|
|
|
// The run lists for this and the next round. We use pointers to the
|
|
|
|
// actual lists, since we want to swap them easily.
|
|
|
|
// actual lists, since we want to swap them easily.
|
|
|
@ -1413,6 +1484,8 @@ struct Scheduler
|
|
|
|
// good.) But all this falls in the "optimization" category.
|
|
|
|
// good.) But all this falls in the "optimization" category.
|
|
|
|
void doFrame()
|
|
|
|
void doFrame()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
assert(cthread is null,
|
|
|
|
|
|
|
|
"cannot run doFrame while another thread is running");
|
|
|
|
checkConditions();
|
|
|
|
checkConditions();
|
|
|
|
dispatch();
|
|
|
|
dispatch();
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -1436,15 +1509,12 @@ struct Scheduler
|
|
|
|
// way, ie to change object states or interact with the
|
|
|
|
// way, ie to change object states or interact with the
|
|
|
|
// scheduler. In fact, hasFinished() should do as little as
|
|
|
|
// scheduler. In fact, hasFinished() should do as little as
|
|
|
|
// possible.
|
|
|
|
// possible.
|
|
|
|
if(cn.isIdle)
|
|
|
|
if(cn.fstack.isIdle)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
fstack.pushIdle(cn.idle, cn.idleObj);
|
|
|
|
if(cn.getIdle().hasFinished(cn))
|
|
|
|
if(cn.idle.idleFunc.hasFinished(cn))
|
|
|
|
|
|
|
|
// Schedule the code to start running again this round. We
|
|
|
|
// Schedule the code to start running again this round. We
|
|
|
|
// move it from the wait list to the run list.
|
|
|
|
// move it from the wait list to the run list.
|
|
|
|
cn.moveTo(runNext);
|
|
|
|
cn.moveTo(runNext);
|
|
|
|
|
|
|
|
|
|
|
|
fstack.pop();
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Set the next item
|
|
|
|
// Set the next item
|
|
|
|
cn = next;
|
|
|
|
cn = next;
|
|
|
@ -1473,9 +1543,6 @@ struct Scheduler
|
|
|
|
// Execute
|
|
|
|
// Execute
|
|
|
|
cn.reenter();
|
|
|
|
cn.reenter();
|
|
|
|
|
|
|
|
|
|
|
|
// The function stack should now be at zero
|
|
|
|
|
|
|
|
assert(fstack.isEmpty());
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Get the next item.
|
|
|
|
// Get the next item.
|
|
|
|
cn = run.getHead();
|
|
|
|
cn = run.getHead();
|
|
|
|
}
|
|
|
|
}
|
|
|
|