scummvm/engines/saga2/script.h

350 lines
11 KiB
C
Raw Normal View History

2021-05-17 18:47:39 +00:00
/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
2021-05-17 18:47:39 +00:00
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
2021-05-17 18:47:39 +00:00
*
*
* Based on the original sources
* Faery Tale II -- The Halls of the Dead
* (c) 1993-1996 The Wyrmkeep Entertainment Co.
*/
#ifndef SAGA2_SCRIPT_H
#define SAGA2_SCRIPT_H
#include "saga2/objects.h"
#include "saga2/calendar.h"
2021-05-17 18:47:39 +00:00
namespace Saga2 {
typedef int16 ThreadID;
// Various result codes returned from runScript
enum scriptResult {
// Code returned when attempt to run a non-existent script
2022-10-29 12:15:21 +00:00
kScriptResultNoScript = 0,
2021-05-17 18:47:39 +00:00
// Code returned when script was aborted before completion
2022-10-29 12:15:21 +00:00
kScriptResultAborted,
2021-05-17 18:47:39 +00:00
// Code returned when script finished
2022-10-29 12:15:21 +00:00
kScriptResultFinished,
2021-05-17 18:47:39 +00:00
// Script spun off as async thread; no answer available.
2022-10-29 12:15:21 +00:00
kScriptResultAsync
2021-05-17 18:47:39 +00:00
};
// Variables specific to a thread
struct scriptCallFrame {
// ID of object who's method is being called (which can be the same
// as one of the other objects below).
ObjectID invokedObject;
ActiveItemID invokedTAI;
// ID of the objects in the interaction.
ObjectID enactor, // actor who caused interaction
directObject, // the object being acted on
indirectObject; // the object being used
// with the other one
ActiveItemID directTAI, // the tile activity instance
// being used
indirectTAI; // the tile activity instance
// upon which the object is being
// used
int16 responseType; // used with knowledge package
int16 methodNum; // which method being invoked
// Misc fields used in passing parameters to scripts.
int16 idNum; // a misc. id number
int16 value; // a misc. parameter value
TilePoint coords; // a misc. tilepoint
int16 returnVal; // return value of script
};
// Standard return codes from scripts in the "returnVal" field
enum {
// Code returned by script when script decides requested
// action is not possible, and the calling C-code should
// take action to inform user
2022-10-29 12:15:21 +00:00
kActionResultFailure = 0,
2021-05-17 18:47:39 +00:00
// Code returned by script when script completes the action
// successfully and C-code should not complete the action
2022-10-29 12:15:21 +00:00
kActionResultSuccess,
2021-05-17 18:47:39 +00:00
// Code returned by script when requested action should complete
// the action
2022-10-29 12:15:21 +00:00
kActionResultNotDone
2021-05-17 18:47:39 +00:00
};
// Method used to refer to a SAGA object
struct SegmentRef {
uint16 segment; // segment / resource number
uint16 offset; // offset within segment
};
// Segment numbers of "builtin" SAGA data structures,
// such as actors and TAGS
enum builtinTypes {
2022-10-29 12:15:21 +00:00
kBuiltinTypeObject = -1,
kBuiltinTypeTAG = -2,
kBuiltinAbstract = -3,
kBuiltinTypeMission = -4
2021-05-17 18:47:39 +00:00
};
/* ===================================================================== *
SAGA management functions
* ===================================================================== */
// Load the SAGA data segment from the resource file
2021-09-11 09:13:35 +00:00
void initSAGADataSeg();
2021-05-17 18:47:39 +00:00
void saveSAGADataSeg(Common::OutSaveFile *outS);
void loadSAGADataSeg(Common::InSaveFile *in);
2021-05-17 18:47:39 +00:00
// Dispose of the SAGA data segment -- do nothing
2021-09-11 09:13:35 +00:00
inline void cleanupSAGADataSeg() {}
2021-05-17 18:47:39 +00:00
/* ===================================================================== *
Thread management functions
* ===================================================================== */
class Thread;
// Initialize the SAGA thread list
2021-09-11 09:13:35 +00:00
void initSAGAThreads();
2021-05-17 18:47:39 +00:00
void saveSAGAThreads(Common::OutSaveFile *outS);
void loadSAGAThreads(Common::InSaveFile *in, int32 chunkSize);
2021-05-17 18:47:39 +00:00
// Dispose of the active SAGA threads
2021-09-11 09:13:35 +00:00
void cleanupSAGAThreads();
2021-05-17 18:47:39 +00:00
// Dispose of an active SAGA thread
void deleteThread(Thread *p);
2021-05-17 18:47:39 +00:00
2021-08-14 13:29:19 +00:00
void newThread(Thread *p, ThreadID id);
void newThread(Thread *p);
2021-05-17 18:47:39 +00:00
// Return the ID of the specified SAGA thread
ThreadID getThreadID(Thread *thread);
// Return a pointer to a SAGA thread, given a thread ID
Thread *getThreadAddress(ThreadID id);
/* ===================================================================== *
Class Thread: An execution context of a script
* ===================================================================== */
// A script task is called a thread
scriptResult runMethod(
uint16 scriptClassID, // which script class
int16 bType, // builtin type
uint16 index, // object index
uint16 methodNum, // method number to call
scriptCallFrame &args);
class Thread {
friend char *STRING(int strNum);
2021-05-17 18:47:39 +00:00
friend scriptResult runScript(uint16 exportEntryNum, scriptCallFrame &args);
2021-05-17 18:47:39 +00:00
friend void wakeUpThread(ThreadID, int16);
public:
SegmentRef _programCounter; // current PC location
2021-05-17 18:47:39 +00:00
uint8 *_stackPtr; // current stack location
byte *_codeSeg; // base of current data segment
// *_stringBase; // base of string resource
2021-05-17 18:47:39 +00:00
uint8 *_stackBase; // base of module stack
2021-05-17 18:47:39 +00:00
enum threadFlags {
2022-10-29 12:15:21 +00:00
kTFWaiting = (1 << 0), // thread waiting for event
kTFFinished = (1 << 1), // thread finished normally
kTFAborted = (1 << 2), // thread is aborted
kTFExtended = (1 << 3), // this is an extended sequence
kTFExpectResult = (1 << 4), // script is expecting result on stack
kTFSynchronous = (1 << 5), // when this bit is set this thread will
2021-05-17 18:47:39 +00:00
// run until it is finished or this bit
// is cleared
2022-10-29 12:15:21 +00:00
kTFAsleep = (kTFWaiting | kTFFinished | kTFAborted)
2021-05-17 18:47:39 +00:00
};
int16 _stackSize, // allocated size of stack
_flags, // execution flags
_framePtr, // pointer to call frame
_returnVal; // return value from ccalls
2021-05-17 18:47:39 +00:00
bool _valid;
2021-05-17 18:47:39 +00:00
// Various signals that a script can wait upon
enum WaitTypes {
2022-10-29 12:15:21 +00:00
kWaitNone = 0, // waiting for nothing
kWaitDelay, // waiting for a timer
kWaitFrameDelay, // waiting for frame count
kWaitOther, // waiting for to be awoken
kWaitTagSemaphore // waiting for a tag semaphore
// kWaitSpeech, // waiting for speech to finish
// kWaitDialogEnd, // waiting for my dialog to finish
// kWaitDialogBegin, // waiting for other dialog to finish
// kWaitWalk, // waiting to finish walking
// kWaitRequest, // a request is up
2021-05-17 18:47:39 +00:00
};
WaitTypes _waitType; // what we're waiting for
2021-05-17 18:47:39 +00:00
union {
Alarm _waitAlarm; // for time-delay
FrameAlarm _waitFrameAlarm; // for frame count delay
ActiveItem *_waitParam; // for other waiting
2021-05-17 18:47:39 +00:00
};
scriptCallFrame _threadArgs; // arguments from C to thread
2021-05-17 18:47:39 +00:00
// For 'cfunc' member functions, the address of the object who's
// member function is being invoked.
void *_thisObject;
uint16 _argCount; // number of args to cfunc
2021-05-17 18:47:39 +00:00
// Constructor
Thread(uint16 segNum, uint16 segOff, scriptCallFrame &args);
// Constructor -- reconstruct from archive buffer
Thread(void **buf);
2021-08-14 13:29:19 +00:00
Thread(Common::SeekableReadStream *stream, ThreadID id);
2021-05-17 18:47:39 +00:00
// Destructor
~Thread();
// Return the number of bytes need to archive this thread in an
// arhive buffer
2021-09-11 09:13:35 +00:00
int32 archiveSize();
2021-05-17 18:47:39 +00:00
// Create an archive of this thread in an archive buffer
void *archive(void *buf);
void write(Common::MemoryWriteStreamDynamic *out);
2021-05-17 18:47:39 +00:00
// Dispatch all asynchronous threads
2021-09-11 09:13:35 +00:00
static void dispatch();
2021-05-17 18:47:39 +00:00
// Intepret a single thread
2021-09-11 09:13:35 +00:00
scriptResult run();
2021-05-17 18:47:39 +00:00
// Tells thread to wait for an event
2022-10-29 12:15:21 +00:00
void waitForEvent(WaitTypes wt, ActiveItem *param) {
_flags |= kTFWaiting;
_waitType = wt;
_waitParam = param;
2021-05-17 18:47:39 +00:00
}
// Convert to extended script, and back to synchonous script
2021-09-11 09:13:35 +00:00
void setExtended();
void clearExtended();
2021-05-17 18:47:39 +00:00
2021-09-11 09:13:35 +00:00
bool interpret();
2021-05-17 18:47:39 +00:00
private:
uint8 *strAddress(int strNum);
};
const int maxTimeSlice = 16, // max instructions per call
kStackSize = 512; // thread stack size
/* ============================================================================ *
C-Function dispatch table
* ============================================================================ */
typedef int16 C_Call(int16 *);
struct CallTable {
C_Call **table;
uint16 numEntries;
uint16 classID;
};
extern CallTable globalCFuncs,
actorCFuncs,
tagCFuncs,
missionCFuncs;
//extern C_Call *ccall_table[];
//extern int16 ccall_count;
/* ===================================================================== *
Externals
* ===================================================================== */
extern Thread *thisThread; // task queue
// Thread control
//void killThread( Thread *th );
/*
2022-10-29 12:15:21 +00:00
void wakeUpActorThread(WaitTypes wakeupType, void *obj);
void wakeUpThreads(WaitTypes wakeupType);
void wakeUpThreadsDelayed(WaitTypes wakeupType, int newdelay);
void abortObjectThreads(Thread *keep, uint16 id);
bool abortAllThreads(void);
2021-05-17 18:47:39 +00:00
*/
// Run a script function
scriptResult runScript(uint16 exportEntryNum, scriptCallFrame &args);
// Run a script class method.
scriptResult runObjectMethod(
ObjectID id, uint16 methodNum, scriptCallFrame &args);
scriptResult runTagMethod(
uint16 tagNum, uint16 methodNum, scriptCallFrame &args);
struct ResImportTable {
int16 deadActorProto,
2021-06-07 16:19:10 +00:00
reserved[2];
2021-05-17 18:47:39 +00:00
int16 EXP_spellEffect_CreateFireWisp,
EXP_spellEffect_CreateWindWisp,
EXP_spellEffect_CreateWraith,
EXP_spellEffect_TeleportToShrine,
EXP_spellEffect_Rejoin,
EXP_spellEffect_Timequake,
EXP_spellEffect_CreateFood;
2021-05-17 18:47:39 +00:00
};
extern ResImportTable *resImports;
} // end of namespace Saga2
#endif