mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-25 12:05:53 +00:00
378 lines
11 KiB
C++
378 lines
11 KiB
C++
/* 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 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* 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, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*
|
|
*
|
|
* 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/calender.h"
|
|
#include "saga2/savefile.h"
|
|
|
|
namespace Saga2 {
|
|
|
|
#ifdef _WIN32 // Set structure alignment packing value to 1 byte
|
|
#pragma pack( push, 1 )
|
|
#endif
|
|
|
|
typedef int16 ThreadID;
|
|
|
|
// Various result codes returned from runScript
|
|
|
|
enum scriptResult {
|
|
|
|
// Code returned when attempt to run a non-existant script
|
|
scriptResultNoScript = 0,
|
|
|
|
// Code returned when script was aborted before completion
|
|
scriptResultAborted,
|
|
|
|
// Code returned when script finished
|
|
scriptResultFinished,
|
|
|
|
// Script spun off as async thread; no answer available.
|
|
scriptResultAsync,
|
|
};
|
|
|
|
// 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
|
|
actionResultFailure = 0,
|
|
|
|
// Code returned by script when script completes the action
|
|
// successfully and C-code should not complete the action
|
|
actionResultSuccess,
|
|
|
|
// Code returned by script when requested action should complete
|
|
// the action
|
|
actionResultNotDone,
|
|
};
|
|
|
|
// 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 {
|
|
builtinTypeObject = -1,
|
|
builtinTypeTAG = -2,
|
|
builtinAbstract = -3,
|
|
builtinTypeMission = -4,
|
|
};
|
|
|
|
/* ===================================================================== *
|
|
SAGA management functions
|
|
* ===================================================================== */
|
|
|
|
// Load the SAGA data segment from the resource file
|
|
void initSAGADataSeg(void);
|
|
|
|
// Save the SAGA data segment to a save file
|
|
void saveSAGADataSeg(SaveFileConstructor &saveGame);
|
|
|
|
// Load the SAGA data segment from a save file
|
|
void loadSAGADataSeg(SaveFileReader &saveGame);
|
|
|
|
// Dispose of the SAGA data segment -- do nothing
|
|
inline void cleanupSAGADataSeg(void) {}
|
|
|
|
/* ===================================================================== *
|
|
Thread management functions
|
|
* ===================================================================== */
|
|
|
|
class Thread;
|
|
|
|
// Initialize the SAGA thread list
|
|
void initSAGAThreads(void);
|
|
|
|
// Save the active SAGA threads to a save file
|
|
void saveSAGAThreads(SaveFileConstructor &saveGame);
|
|
|
|
// Load the active SAGA threads from a save file
|
|
void loadSAGAThreads(SaveFileReader &saveGame);
|
|
|
|
// Dispose of the active SAGA threads
|
|
void cleanupSAGAThreads(void);
|
|
|
|
|
|
// Get a new SAGA thread from the global thread list
|
|
void *newThread(void);
|
|
|
|
// Get a specific SAGA thread from the global thread list
|
|
void *newThread(ThreadID id);
|
|
|
|
// Dispose of an active SAGA thread
|
|
void deleteThread(void *p);
|
|
|
|
|
|
// 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);
|
|
|
|
friend scriptResult runScript(
|
|
uint16 exportEntryNum, scriptCallFrame &args);
|
|
|
|
friend void wakeUpThread(ThreadID, int16);
|
|
|
|
public:
|
|
SegmentRef programCounter; // current PC location
|
|
|
|
uint8 *stackPtr; // current stack location
|
|
UByteHandle codeSeg; // base of current data segment
|
|
// *stringBase; // base of string resource
|
|
|
|
uint8 *stackBase; // base of module stack
|
|
|
|
enum threadFlags {
|
|
waiting = (1 << 0), // thread waiting for event
|
|
finished = (1 << 1), // thread finished normally
|
|
aborted = (1 << 2), // thread is aborted
|
|
extended = (1 << 3), // this is an extended sequence
|
|
expectResult = (1 << 4), // script is expecting result on stack
|
|
synchronous = (1 << 5), // when this bit is set this thread will
|
|
// run until it is finished or this bit
|
|
// is cleared
|
|
|
|
asleep = (waiting | finished | aborted),
|
|
};
|
|
|
|
int16 stackSize, // allocated size of stack
|
|
flags, // execution flags
|
|
framePtr, // pointer to call frame
|
|
returnVal; // return value from ccalls
|
|
|
|
// Various signals that a script can wait upon
|
|
enum WaitTypes {
|
|
waitNone = 0, // waiting for nothing
|
|
waitDelay, // waiting for a timer
|
|
waitFrameDelay, // waiting for frame count
|
|
waitOther, // waiting for to be awoken
|
|
waitTagSemaphore, // waiting for a tag semaphore
|
|
|
|
// waitSpeech, // waiting for speech to finish
|
|
// waitDialogEnd, // waiting for my dialog to finish
|
|
// waitDialogBegin, // waiting for other dialog to finish
|
|
// waitWalk, // waiting to finish walking
|
|
// waitRequest, // a request is up
|
|
};
|
|
|
|
enum WaitTypes waitType; // what we're waiting for
|
|
union {
|
|
Alarm waitAlarm; // for time-delay
|
|
FrameAlarm waitFrameAlarm; // for frame count delay
|
|
long waitParam; // for other waiting
|
|
};
|
|
|
|
scriptCallFrame threadArgs; // arguments from C to thread
|
|
|
|
// 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
|
|
|
|
// Constructor
|
|
Thread(uint16 segNum, uint16 segOff, scriptCallFrame &args);
|
|
|
|
// Constructor -- reconstruct from archive buffer
|
|
Thread(void **buf);
|
|
|
|
// Destructor
|
|
~Thread();
|
|
|
|
// Return the number of bytes need to archive this thread in an
|
|
// arhive buffer
|
|
int32 archiveSize(void);
|
|
|
|
// Create an archive of this thread in an archive buffer
|
|
void *archive(void *buf);
|
|
|
|
// Overloaded new operator functions
|
|
void *operator new (size_t) {
|
|
return newThread();
|
|
}
|
|
void *operator new (size_t, ThreadID id) {
|
|
return newThread(id);
|
|
}
|
|
|
|
// Overloaded delete operator function
|
|
void operator delete (void *p) {
|
|
deleteThread(p);
|
|
}
|
|
|
|
// Dispatch all asynchronous threads
|
|
static void dispatch(void);
|
|
|
|
// Intepret a single thread
|
|
scriptResult run(void);
|
|
|
|
// Tells thread to wait for an event
|
|
void waitForEvent(enum WaitTypes wt, long param) {
|
|
flags |= waiting;
|
|
waitType = wt;
|
|
waitParam = param;
|
|
}
|
|
|
|
// Convert to extended script, and back to synchonous script
|
|
void setExtended(void);
|
|
void clearExtended(void);
|
|
|
|
private:
|
|
uint8 *strAddress(int strNum);
|
|
|
|
bool interpret(void);
|
|
};
|
|
|
|
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 );
|
|
|
|
/*
|
|
void wakeUpActorThread( enum WaitTypes wakeupType, void *obj );
|
|
void wakeUpThreads( enum WaitTypes wakeupType );
|
|
void wakeUpThreadsDelayed( enum WaitTypes wakeupType, int newdelay );
|
|
void abortObjectThreads( Thread *keep, uint16 id );
|
|
bool abortAllThreads( void );
|
|
*/
|
|
|
|
// 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,
|
|
reserved[ 2 ];
|
|
int16 EXP_spellEffect_CreateFireWisp,
|
|
EXP_spellEffect_CreateWindWisp,
|
|
EXP_spellEffect_CreateWraith,
|
|
EXP_spellEffect_TeleportToShrine,
|
|
EXP_spellEffect_Rejoin,
|
|
EXP_spellEffect_Timequake,
|
|
EXP_spellEffect_CreateFood ;
|
|
};
|
|
|
|
extern ResImportTable *resImports;
|
|
|
|
#ifdef _WIN32 // Reset old structure alignment
|
|
#pragma pack( pop )
|
|
#endif
|
|
|
|
} // end of namespace Saga2
|
|
|
|
#endif
|