mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-25 04:01:03 +00:00
975 lines
30 KiB
C++
975 lines
30 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
|
|
* aint32 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.
|
|
*/
|
|
|
|
#define FORBIDDEN_SYMBOL_ALLOW_ALL // FIXME: Remove
|
|
|
|
#include "common/debug.h"
|
|
#include "common/events.h"
|
|
#include "common/memstream.h"
|
|
|
|
#include "saga2/std.h"
|
|
#include "saga2/setup.h"
|
|
#include "saga2/transit.h"
|
|
#include "saga2/player.h"
|
|
#include "saga2/tile.h"
|
|
#include "saga2/messager.h"
|
|
#include "saga2/intrface.h"
|
|
#include "saga2/script.h"
|
|
#include "saga2/localize.h"
|
|
#include "saga2/mainmap.h"
|
|
#include "saga2/display.h"
|
|
#include "saga2/tower.h"
|
|
#include "saga2/tromode.h"
|
|
#include "saga2/loadsave.h"
|
|
#include "saga2/gamerate.h"
|
|
#include "saga2/msgbox.h"
|
|
#include "saga2/savefile.h"
|
|
|
|
namespace Saga2 {
|
|
|
|
/* ===================================================================== *
|
|
Optional Debugging Code
|
|
* ===================================================================== */
|
|
|
|
// enable the following to display event loop processing
|
|
#define DEBUG_LOOP 0
|
|
|
|
extern WindowDecoration autoMapDecorations[];
|
|
extern gToolBase G_BASE;
|
|
extern char *gameTimeStr;
|
|
extern bool underground;
|
|
extern char commandLineHelp[];
|
|
extern hResContext *tileRes; // tile resource handle
|
|
extern hResContext *listRes;
|
|
extern vDisplayPage protoPage;
|
|
|
|
/* ===================================================================== *
|
|
Globals
|
|
* ===================================================================== */
|
|
|
|
// command line options
|
|
bool cliWriteStatusF = false;
|
|
bool cliScriptDebug = false;
|
|
bool cliSpeechText = false;
|
|
bool cliDrawInv = false;
|
|
uint32 cliMemory = 0;
|
|
|
|
// User-interface variables
|
|
gMouseState mouseState;
|
|
|
|
// Display variables
|
|
gDisplayPort mainPort; // default rendering port
|
|
gMousePointer pointer(mainPort); // the actual pointer
|
|
BackWindow *mainWindow; // main window...
|
|
|
|
// Memory allocation heap
|
|
long memorySize = 8000000L;
|
|
|
|
// Global game state
|
|
bool gameRunning = true; // true while game running
|
|
bool allPlayerActorsDead = false;
|
|
//bool graphicsInit = false; // true if graphics init OK
|
|
bool checkExit = false; // true while game running
|
|
int gameKiller = 0; // will contain the exception that ends the game
|
|
|
|
// Resource files
|
|
hResource *resFile, // main resource file
|
|
*objResFile, // object resource file
|
|
*auxResFile, // auxillary data resource file
|
|
*scriptResFile, // script resources
|
|
*soundResFile,
|
|
*voiceResFile; // sound resources
|
|
|
|
// Import list from resource file.
|
|
ResImportTable *resImports;
|
|
|
|
// x location of status lines
|
|
uint16 writeStatusFX = 468;
|
|
uint16 writeStatusFY = 354;
|
|
|
|
/* ===================================================================== *
|
|
Locals
|
|
* ===================================================================== */
|
|
|
|
// game states
|
|
static bool cleanExit = true;
|
|
bool gameInitialized = false; // true when game initialized
|
|
bool fullInitialized = false;
|
|
bool delayReDraw = false;
|
|
|
|
// main heap
|
|
static uint8 *heapMemory;
|
|
|
|
|
|
/* ===================================================================== *
|
|
Debug
|
|
* ===================================================================== */
|
|
// frame counting
|
|
uint32 frames = 0;
|
|
static uint32 loops = 0;
|
|
static uint32 elapsed = 0;
|
|
static uint32 lastGameTime = 0;
|
|
|
|
// message handlers
|
|
static pMessager Status[10];
|
|
static pMessager Status2[10];
|
|
|
|
pMessager ratemess[3];
|
|
|
|
#if 1
|
|
frameSmoother frate(frameRate, TICKSPERSECOND, gameTime);
|
|
//frameSmoother lrate(frameRate,TICKSPERSECOND,gameTime);
|
|
frameCounter lrate(TICKSPERSECOND, gameTime);
|
|
frameCounter irate(TICKSPERSECOND, gameTime);
|
|
#else
|
|
frameCounter frate(TICKSPERSECOND, gameTime);
|
|
frameCounter lrate(TICKSPERSECOND, gameTime);
|
|
frameCounter irate(TICKSPERSECOND, gameTime);
|
|
#endif
|
|
|
|
/* ===================================================================== *
|
|
Prototypes
|
|
* ===================================================================== */
|
|
|
|
bool readCommandLine(int argc, char *argv[]);
|
|
void findProgramDir(char *argv); // save program home directory
|
|
|
|
APPFUNC(cmdWindowFunc); // main window event handler
|
|
|
|
// Exportable prototypes
|
|
void EventLoop(bool &running, bool modal); // handles input and distributes
|
|
void SystemEventLoop(void);
|
|
|
|
void runPathFinder(void);
|
|
|
|
bool setupGame(void);
|
|
|
|
void mainEnable(void);
|
|
void mainDisable(void);
|
|
void lightsOut(void);
|
|
|
|
void cleanupGame(void); // auto-cleanup function
|
|
void RShowMem(void);
|
|
void parseCommandLine(int argc, char *argv[]);
|
|
const char *getExeFromCommandLine(int argc, char *argv[]);
|
|
void WriteStatusF2(int16 line, const char *msg, ...);
|
|
bool initUserDialog(void);
|
|
void cleanupUserDialog(void);
|
|
int16 OptionsDialog(bool disableSaveResume = false);
|
|
|
|
static void mainLoop(bool &cleanExit, int argc, char *argv[]);
|
|
void displayUpdate(void);
|
|
|
|
void testTiles();
|
|
|
|
bool initResourceHandles();
|
|
bool initDisplayPort();
|
|
bool initPanelSystem();
|
|
bool initDisplay();
|
|
bool initGameMaps();
|
|
|
|
/********************************************************************/
|
|
/* */
|
|
/* MAIN FUNCTION */
|
|
/* */
|
|
/********************************************************************/
|
|
void termFaultHandler(void);
|
|
|
|
void main_saga2() {
|
|
gameInitialized = false;
|
|
|
|
mainDisable();
|
|
initCleanup();
|
|
|
|
// parse command-line arguments and store results
|
|
// if (!readCommandLine(argc, argv))
|
|
// abortMain;
|
|
|
|
// call the initialization code
|
|
gameInitialized = initializeGame();
|
|
cleanExit = gameInitialized;
|
|
|
|
if (gameInitialized) {
|
|
mainLoop(cleanExit, 0, NULL);
|
|
}
|
|
|
|
shutdownGame();
|
|
gameInitialized = false;
|
|
}
|
|
|
|
// ------------------------------------------------------------------------
|
|
// Inner chunk of main - this bizzare nesting is required because VC++
|
|
// doesn't like try{} catch(){ } blocks in the same routine as its
|
|
// __try{} __except(){} blocks
|
|
void updateActiveRegions(void);
|
|
|
|
static void mainLoop(bool &cleanExit_, int argc, char *argv[]) {
|
|
const char *exeFile = getExeFromCommandLine(argc, argv);
|
|
if (displayEnabled())
|
|
displayUpdate();
|
|
checkRestartGame(exeFile);
|
|
fullInitialized = true;
|
|
EventLoop(gameRunning, false);
|
|
}
|
|
|
|
/********************************************************************/
|
|
/* */
|
|
/* INITIALIZATION and CLEANUP CODE */
|
|
/* */
|
|
/********************************************************************/
|
|
|
|
//
|
|
// Note: the bulk of the Initialization & cleanup routines have
|
|
// been moved to TOWERFTA.CPP. This file together with
|
|
// TOWER.CPP automate initialization & cleanup. This is needed
|
|
// to accomodate differences in system startup between
|
|
// the windows & DOS versions
|
|
//
|
|
//
|
|
|
|
// ------------------------------------------------------------------------
|
|
// Game setup function
|
|
|
|
bool setupGame(void) {
|
|
return programInit();
|
|
}
|
|
|
|
// ------------------------------------------------------------------------
|
|
// Game cleanup function
|
|
|
|
void cleanupGame(void) {
|
|
programTerm();
|
|
}
|
|
|
|
|
|
/********************************************************************/
|
|
/* */
|
|
/* EVENT LOOP HANDLING */
|
|
/* */
|
|
/********************************************************************/
|
|
|
|
void processEventLoop(bool updateScreen = true);
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Main loop
|
|
|
|
void EventLoop(bool &running, bool) {
|
|
// Our typical main loop
|
|
while (running && gameRunning)
|
|
processEventLoop(displayEnabled());
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Main event which does everything (including handle user input)
|
|
|
|
void dumpGBASE(char *msg);
|
|
|
|
void processEventLoop(bool updateScreen) {
|
|
|
|
debugC(1, kDebugEventLoop, "EventLoop: starting event loop");
|
|
irate.updateFrameCount();
|
|
|
|
if (checkExit && verifyUserExit()) {
|
|
//gameRunning=false;
|
|
endGame();
|
|
return;
|
|
}
|
|
|
|
debugC(1, kDebugEventLoop, "EventLoop: audio event loop");
|
|
//FIXME: Disabled for debug purposes. Enable and implement later.
|
|
//audioEventLoop();
|
|
|
|
debugC(1, kDebugEventLoop, "EventLoop: game mode update");
|
|
if (GameMode::newmodeFlag)
|
|
GameMode::update();
|
|
|
|
Common::Event event;
|
|
while (g_vm->getEventManager()->pollEvent(event)) {
|
|
switch (event.type) {
|
|
case Common::EVENT_LBUTTONDOWN:
|
|
case Common::EVENT_RBUTTONDOWN:
|
|
case Common::EVENT_LBUTTONUP:
|
|
case Common::EVENT_RBUTTONUP:
|
|
case Common::EVENT_MOUSEMOVE:
|
|
G_BASE.handleMouse(event, g_system->getMillis());
|
|
break;
|
|
case Common::EVENT_KEYDOWN:
|
|
G_BASE.handleKeyStroke(event);
|
|
break;
|
|
case Common::EVENT_QUIT:
|
|
if (verifyUserExit())
|
|
endGame();
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
//if(!running) return; // This Is No Tasks Are Done After Saving Game
|
|
|
|
debugC(1, kDebugEventLoop, "EventLoop: timer update");
|
|
// Handle the timer events
|
|
// REM: Causing code corruption in windows for some reason...
|
|
G_BASE.handleTimerTick(gameTime >> 2);
|
|
|
|
// Handle updating of the display.
|
|
debugC(1, kDebugEventLoop, "EventLoop: display update");
|
|
if (!g_vm->checkVideo()) {
|
|
displayUpdate();
|
|
}
|
|
|
|
if (allPlayerActorsDead) {
|
|
allPlayerActorsDead = false;
|
|
setLostroMode();
|
|
}
|
|
}
|
|
|
|
void displayUpdate(void) {
|
|
if (displayEnabled()) { //updateScreen)
|
|
//debugC(1, kDebugEventLoop, "EventLoop: daytime transition update loop");
|
|
dayNightUpdate();
|
|
//debugC(1, kDebugEventLoop, "EventLoop: Game mode handle task");
|
|
GameMode::modeStackPtr[GameMode::modeStackCtr - 1]->handleTask();
|
|
lrate.updateFrameCount();
|
|
loops++;
|
|
elapsed += (g_system->getMillis() - lastGameTime);
|
|
lastGameTime = g_system->getMillis();
|
|
|
|
|
|
debugC(1, kDebugEventLoop, "EventLoop: Interface indicator updates");
|
|
updateIndicators();
|
|
|
|
g_system->updateScreen();
|
|
|
|
if (delayReDraw)
|
|
reDrawScreen();
|
|
// Call asynchronous resource loader
|
|
debugC(1, kDebugEventLoop, "EventLoop: resource update");
|
|
loadAsyncResources();
|
|
|
|
//FIXME: Disabled for debug purposes. Enable and implement later.
|
|
//audioEventLoop();
|
|
|
|
// Call the asynchronous path finder
|
|
debugC(1, kDebugEventLoop, "EventLoop: pathfinder update");
|
|
runPathFinder();
|
|
}
|
|
}
|
|
|
|
/* ===================================================================== *
|
|
Abbreviated event loop
|
|
* ===================================================================== */
|
|
|
|
void SystemEventLoop(void) {
|
|
if (
|
|
#ifdef DO_OUTRO_IN_CLEANUP
|
|
whichOutro == -1 &&
|
|
#endif
|
|
!gameRunning)
|
|
TroModeExternEvent();
|
|
|
|
Common::Event event;
|
|
while (g_vm->getEventManager()->pollEvent(event)) {
|
|
switch (event.type) {
|
|
case Common::EVENT_LBUTTONUP:
|
|
case Common::EVENT_RBUTTONUP:
|
|
case Common::EVENT_KEYDOWN:
|
|
case Common::EVENT_QUIT:
|
|
TroModeExternEvent();
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
g_system->updateScreen();
|
|
}
|
|
|
|
/********************************************************************/
|
|
/* */
|
|
/* COMMAND LINE PARSING WITHOUT CRASHES */
|
|
/* */
|
|
/********************************************************************/
|
|
|
|
// ------------------------------------------------------------------------
|
|
// Determines the EXE file executed from command line info
|
|
|
|
const char *getExeFromCommandLine(int argc, char *argv[]) {
|
|
if (argv == nullptr)
|
|
return "scummvm";
|
|
return argv[0];
|
|
}
|
|
|
|
// ------------------------------------------------------------------------
|
|
// Adds error handling to command line parsing
|
|
|
|
bool readCommandLine(int argc, char *argv[]) {
|
|
parseCommandLine(argc, argv);
|
|
|
|
return true;
|
|
}
|
|
|
|
/********************************************************************/
|
|
/* */
|
|
/* MOUSE EVENT QUEUE */
|
|
/* */
|
|
/********************************************************************/
|
|
|
|
// ------------------------------------------------------------------------
|
|
// Mouse handling
|
|
|
|
gMouseState prevState;
|
|
MouseExtState mouseQueue[64];
|
|
|
|
int16 queueIn = 0,
|
|
queueOut = 0;
|
|
|
|
inline int BUMP(int x) {
|
|
return (x + 1) & 63;
|
|
}
|
|
|
|
// ------------------------------------------------------------------------
|
|
// clears any queued input (mouse AND keyboard)
|
|
void resetInputDevices(void) {
|
|
Common::Event event;
|
|
while (g_vm->getEventManager()->pollEvent(event));
|
|
}
|
|
|
|
/********************************************************************/
|
|
/* */
|
|
/* RESOURCE MANAGEMENT CODE */
|
|
/* */
|
|
/********************************************************************/
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Opens a file using simple DOS i/o, allocates a buffer the same size
|
|
// as the file, and loads the file into the buffer
|
|
|
|
void *LoadFile(char *filename, const char desc[]) {
|
|
#if 0
|
|
int fHandle; // file handle
|
|
struct stat fileStat; // stat structure
|
|
uint8 *buffer; // allocated buffer
|
|
|
|
// Open the file
|
|
if ((fHandle = open(filename, O_RDONLY)) < 0)
|
|
error("Error opening %s", filename);
|
|
|
|
// Determine size of file
|
|
if (fstat(fHandle, &fileStat) < 0)
|
|
error("Error accessing %s", filename);
|
|
|
|
// Allocate the buffer
|
|
buffer = (uint8 *)mustAlloc(fileStat.st_size, desc);
|
|
|
|
// Read file into buffer
|
|
if (read(fHandle, buffer, fileStat.st_size) < 0)
|
|
error("Error reading %s", filename);
|
|
|
|
// Close file and return
|
|
close(fHandle);
|
|
return buffer;
|
|
#endif
|
|
warning("STUB: LoadFile(%s)", filename);
|
|
return nullptr;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Loads a resource into a buffer and returns a pointer
|
|
|
|
void *LoadResource(hResContext *con, uint32 id, const char desc[]) {
|
|
int32 size;
|
|
uint8 *buffer; // allocated buffer
|
|
|
|
debugC(3, kDebugResources, "LoadResource(): Loading resource %d (%s, %s)", id, tag2str(id), desc);
|
|
|
|
size = con->size(id);
|
|
if (size <= 0 || !con->seek(id)) {
|
|
error("LoadResource(): Error reading resource ID '%s'.", tag2str(id));
|
|
}
|
|
|
|
// Allocate the buffer
|
|
buffer = (uint8 *)malloc(size);
|
|
con->read(buffer, size);
|
|
con->rest();
|
|
|
|
return buffer;
|
|
}
|
|
|
|
Common::SeekableReadStream *loadResourceToStream(hResContext *con, uint32 id, const char desc[]) {
|
|
int32 size;
|
|
uint8 *buffer; // allocated buffer
|
|
|
|
debugC(3, kDebugResources, "loadResourceToStream(): Loading resource %d (%s, %s)", id, tag2str(id), desc);
|
|
|
|
size = con->size(id);
|
|
if (size <= 0 || !con->seek(id)) {
|
|
warning("loadResourceToStream(): Error reading resource ID '%s'.", tag2str(id));
|
|
return nullptr;
|
|
}
|
|
|
|
// Allocate the buffer
|
|
buffer = (uint8 *)malloc(size);
|
|
con->read(buffer, size);
|
|
con->rest();
|
|
|
|
return new Common::MemoryReadStream(buffer, size, DisposeAfterUse::YES);
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Loads a resource into a relocatable buffer and returns a handle
|
|
|
|
RHANDLE LoadResourceToHandle(hResContext *con, uint32 id, const char desc[]) {
|
|
int32 size;
|
|
RHANDLE buffer; // allocated buffer
|
|
|
|
debugC(3, kDebugResources, "LoadResourceToHandle(): Loading resource %d (%s, %s)", id, tag2str(id), desc);
|
|
|
|
size = con->size(id);
|
|
if (size <= 0 || !con->seek(id)) {
|
|
error("LoadResourceToHandle(): Error reading resource ID '%s'.", tag2str(id));
|
|
}
|
|
|
|
// Allocate the buffer
|
|
buffer = mustAllocHandle(size, desc);
|
|
con->read(*buffer, size);
|
|
con->rest();
|
|
|
|
return buffer;
|
|
}
|
|
|
|
typedef hResource *pHResource;
|
|
|
|
inline char drive(char *path) {
|
|
return (path[0] % 32);
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Routine to initialize an arbitrary resource file
|
|
|
|
static bool openResource(
|
|
pHResource &hr, // resource to initialize
|
|
const char *defaultPath, // backup path
|
|
const char *fileName, // file name & extension
|
|
const char *description) {
|
|
if (hr) delete hr;
|
|
hr = NULL;
|
|
|
|
hr = new hResource(fileName, defaultPath, description);
|
|
|
|
while (hr == NULL || !hr->_valid) {
|
|
if (hr) delete hr;
|
|
hr = NULL;
|
|
hr = new hResource(fileName, defaultPath, description);
|
|
}
|
|
|
|
if (hr == NULL || !hr->_valid) {
|
|
error("openResource: Cannot open resource: %s, %s", fileName, description);
|
|
// return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Routine to initialize all the resource files
|
|
|
|
bool openResources(void) {
|
|
|
|
if (
|
|
openResource(resFile, "..\\resfile\\", IMAGE_RESFILE,
|
|
"Imagery resource file") &&
|
|
|
|
openResource(objResFile, "..\\resfile\\", OBJECT_RESFILE,
|
|
"Object resource file") &&
|
|
|
|
openResource(auxResFile, "..\\resfile\\", AUX_RESFILE,
|
|
"Data resource file") &&
|
|
|
|
openResource(scriptResFile, "..\\scripts\\", SCRIPT_RESFILE,
|
|
"Script resource file") &&
|
|
openResource(voiceResFile, "..\\sound\\", VOICE_RESFILE,
|
|
"Voice resource file") &&
|
|
openResource(soundResFile, "..\\sound\\", SOUND_RESFILE,
|
|
"Sound resource file")) {
|
|
return true;
|
|
}
|
|
return false;
|
|
|
|
}
|
|
|
|
void testOpenImage() {
|
|
hResContext *decRes;
|
|
|
|
decRes = resFile->newContext(MKTAG('A', 'M', 'A', 'P'), "Automap Resources");
|
|
//checkAlloc(summaryData = LoadResource(decRes,
|
|
// MKTAG('S', 'U', 'M', currentMapNum),
|
|
// "summary data"));
|
|
|
|
WindowDecoration *dec = &autoMapDecorations[0];
|
|
dec->image = LoadResource(decRes, MKTAG('M', 'A', 'P', 0), "MAP0");
|
|
//dec->image = ImageCache.requestImage(decRes, MKTAG('M', 'A', 'P', 0) | MKTAG('B', 'R', 'D', dec->imageNumber));
|
|
Point16 pos(0, 0);
|
|
drawCompressedImage(mainPort, pos, dec->image);
|
|
}
|
|
|
|
void testScripts() {
|
|
scriptCallFrame scf;
|
|
//for (int i = 1; i < 100; ++i)
|
|
// runScript(i, scf);
|
|
runScript(1, scf);
|
|
}
|
|
|
|
void testTileRendering() {
|
|
tileRes = resFile->newContext(MKTAG('T', 'I', 'L', 'E'), "tile resources");
|
|
listRes = objResFile->newContext(MKTAG('L', 'I', 'S', 'T'), "list resources");
|
|
resImports = (ResImportTable *)LoadResource(listRes, MKTAG('I', 'M', 'P', 'O'), "res imports");
|
|
|
|
initResourceHandles();
|
|
mainPort.setDisplayPage(&protoPage);
|
|
initPanelSystem();
|
|
initDisplayPort();
|
|
initDisplay();
|
|
initGameMaps();
|
|
testTiles();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Routine to cleanup all the resource files
|
|
|
|
void closeResources(void) {
|
|
if (soundResFile) delete soundResFile;
|
|
soundResFile = NULL;
|
|
if (voiceResFile) delete voiceResFile;
|
|
voiceResFile = NULL;
|
|
if (scriptResFile) delete scriptResFile;
|
|
scriptResFile = NULL;
|
|
if (auxResFile) delete auxResFile;
|
|
auxResFile = NULL;
|
|
if (objResFile) delete objResFile;
|
|
objResFile = NULL;
|
|
if (resFile) delete resFile;
|
|
resFile = NULL;
|
|
}
|
|
|
|
/********************************************************************/
|
|
/* */
|
|
/* GLOBAL DATA SAVE / RESTORE */
|
|
/* */
|
|
/********************************************************************/
|
|
|
|
extern int32 objectIndex,
|
|
actorIndex;
|
|
extern bool brotherBandingEnabled,
|
|
centerActorIndicatorEnabled,
|
|
interruptableMotionsPaused,
|
|
objectStatesPaused,
|
|
actorStatesPaused,
|
|
actorTasksPaused,
|
|
combatBehaviorEnabled,
|
|
backgroundSimulationPaused;
|
|
|
|
// This structure is used archiving any globals which will need to be saved
|
|
// in a save game file.
|
|
|
|
struct GlobalsArchive {
|
|
int32 objectIndex,
|
|
actorIndex;
|
|
bool brotherBandingEnabled,
|
|
centerActorIndicatorEnabled,
|
|
interruptableMotionsPaused,
|
|
objectStatesPaused,
|
|
actorStatesPaused,
|
|
actorTasksPaused,
|
|
combatBehaviorEnabled,
|
|
backgroundSimulationPaused;
|
|
};
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Assign initial values to miscellaneous globals
|
|
|
|
void initGlobals(void) {
|
|
objectIndex = 0;
|
|
actorIndex = 0;
|
|
brotherBandingEnabled = true;
|
|
centerActorIndicatorEnabled = false;
|
|
interruptableMotionsPaused = false;
|
|
objectStatesPaused = false;
|
|
actorStatesPaused = false;
|
|
actorTasksPaused = false;
|
|
combatBehaviorEnabled = false;
|
|
backgroundSimulationPaused = false;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Store miscellaneous globals in a save file
|
|
|
|
void saveGlobals(SaveFileConstructor &saveGame) {
|
|
GlobalsArchive archive;
|
|
|
|
archive.objectIndex = objectIndex;
|
|
archive.actorIndex = actorIndex;
|
|
archive.brotherBandingEnabled = brotherBandingEnabled;
|
|
archive.centerActorIndicatorEnabled = centerActorIndicatorEnabled;
|
|
archive.interruptableMotionsPaused = interruptableMotionsPaused;
|
|
archive.objectStatesPaused = objectStatesPaused;
|
|
archive.actorStatesPaused = actorStatesPaused;
|
|
archive.actorTasksPaused = actorTasksPaused;
|
|
archive.combatBehaviorEnabled = combatBehaviorEnabled;
|
|
archive.backgroundSimulationPaused = backgroundSimulationPaused;
|
|
|
|
saveGame.writeChunk(
|
|
MakeID('G', 'L', 'O', 'B'),
|
|
&archive,
|
|
sizeof(archive));
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Restore miscellaneouse globals from a save file
|
|
|
|
void loadGlobals(SaveFileReader &saveGame) {
|
|
GlobalsArchive archive;
|
|
|
|
saveGame.read(&archive, sizeof(archive));
|
|
|
|
objectIndex = archive.objectIndex;
|
|
actorIndex = archive.actorIndex;
|
|
brotherBandingEnabled = archive.brotherBandingEnabled;
|
|
centerActorIndicatorEnabled = archive.centerActorIndicatorEnabled;
|
|
interruptableMotionsPaused = archive.interruptableMotionsPaused;
|
|
objectStatesPaused = archive.objectStatesPaused;
|
|
actorStatesPaused = archive.actorStatesPaused;
|
|
actorTasksPaused = archive.actorTasksPaused;
|
|
combatBehaviorEnabled = archive.combatBehaviorEnabled;
|
|
backgroundSimulationPaused = archive.backgroundSimulationPaused;
|
|
}
|
|
|
|
/********************************************************************/
|
|
/* */
|
|
/* ERROR / MESSAGE HANDLING */
|
|
/* */
|
|
/********************************************************************/
|
|
|
|
// ------------------------------------------------------------------------
|
|
// pops up a window to see if the user really wants to exit
|
|
|
|
bool verifyUserExit(void) {
|
|
if (!gameRunning)
|
|
return true;
|
|
if (FTAMessageBox("Are you sure you want to exit", ERROR_YE_BUTTON, ERROR_NO_BUTTON) != 0)
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Allocate visual messagers
|
|
|
|
bool initGUIMessagers(void) {
|
|
initUserDialog();
|
|
for (int i = 0; i < 10; i++) {
|
|
char debItem[16];
|
|
sprintf(debItem, "Status%1.1d", i);
|
|
Status[i] = new StatusLineMessager(debItem, i, &mainPort);
|
|
if (Status[i] == NULL)
|
|
return false;
|
|
sprintf(debItem, "Status%2.2d", i + 10);
|
|
Status2[i] = new StatusLineMessager(debItem, i, &mainPort, 468, 21 + (11 * i));
|
|
}
|
|
for (int j = 0; j < 3; j++)
|
|
ratemess[j] = new StatusLineMessager("FrameRates", j, &mainPort, 5, 450 + (11 * j), 500);
|
|
return true;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// cleanup visual messagers
|
|
|
|
void cleanupGUIMessagers(void) {
|
|
for (int i = 0; i < 10; i++) {
|
|
if (Status[i]) delete Status[i];
|
|
Status[i] = NULL;
|
|
if (Status2[i]) delete Status2[i];
|
|
Status2[i] = NULL;
|
|
}
|
|
cleanupUserDialog();
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Debugging status functions
|
|
|
|
#ifdef WriteStatus
|
|
void WriteStatusF(int16 line, const char *msg, ...) {
|
|
va_list argptr;
|
|
int cnt;
|
|
if (displayEnabled()) {
|
|
va_start(argptr, msg);
|
|
if (line > 9) {
|
|
if (Status2[line - 10])
|
|
cnt = Status2[line - 10]->va(msg, argptr);
|
|
} else {
|
|
if (Status[line])
|
|
cnt = Status[line]->va(msg, argptr);
|
|
}
|
|
va_end(argptr);
|
|
}
|
|
}
|
|
|
|
void WriteStatusF2(int16 line, const char *msg, ...) {
|
|
va_list argptr;
|
|
int cnt;
|
|
if (displayEnabled()) {
|
|
va_start(argptr, msg);
|
|
if (Status2[line])
|
|
cnt = Status2[line]->va(msg, argptr);
|
|
va_end(argptr);
|
|
}
|
|
}
|
|
#else
|
|
void WriteStatusF(int16, const char *, ...) {}
|
|
void WriteStatusF2(int16, const char *, ...) {}
|
|
#endif
|
|
|
|
//---------------------------------------------------------
|
|
// Game performance can be used as a gauge of how much
|
|
// CPU time is available. We'd like to keep the retu
|
|
|
|
int32 currentGamePerformance(void) {
|
|
int32 framePer = 100;
|
|
int32 lval = int(lrate.frameStat());
|
|
int32 fval = int(lrate.frameStat(grFramesPerSecond));
|
|
if (fval >= frameRate && lval > fval) {
|
|
framePer += (50 * ((lval - fval) / fval));
|
|
} else {
|
|
framePer = (100 * frate.frameStat(grFramesPerSecond)) / frameRate;
|
|
}
|
|
framePer = clamp(10, framePer, 240);
|
|
return framePer;
|
|
}
|
|
|
|
|
|
void updateFrameCount(void) {
|
|
frate.updateFrameCount();
|
|
}
|
|
|
|
int32 eloopsPerSecond = 0;
|
|
int32 framesPerSecond = 0;
|
|
|
|
int32 gamePerformance(void) {
|
|
if (framesPerSecond < frameRate) {
|
|
return (100 * framesPerSecond) / frameRate;
|
|
}
|
|
if (framesPerSecond == frameRate)
|
|
return 100;
|
|
return 100 + 50 * (eloopsPerSecond - frameRate) / frameRate;
|
|
|
|
}
|
|
|
|
|
|
/********************************************************************/
|
|
/* */
|
|
/* APPFUNC FOR MAIN WINDOW */
|
|
/* */
|
|
/********************************************************************/
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Function to handle miscellanous events to the window.
|
|
// Any panel events which are not handled by individual panels
|
|
// are sent to this function.
|
|
|
|
APPFUNC(cmdWindowFunc) {
|
|
int16 key, qual;
|
|
|
|
switch (ev.eventType) {
|
|
case gEventKeyDown:
|
|
key = ev.value & 0xffff;
|
|
qual = ev.value >> 16;
|
|
|
|
GameMode::modeStackPtr[GameMode::modeStackCtr - 1]->handleKey(key, qual);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
/********************************************************************/
|
|
/* */
|
|
/* MEMORY MANAGEMENT CODE */
|
|
/* */
|
|
/********************************************************************/
|
|
|
|
/* ===================================================================== *
|
|
Functions to initialize the memory manager.
|
|
* ===================================================================== */
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Initialize memory manager
|
|
|
|
bool initMemPool(void) {
|
|
uint32 take = pickHeapSize(memorySize);
|
|
memorySize = take;
|
|
if (NULL == (heapMemory = (uint8 *)malloc(take)))
|
|
return false;
|
|
//initMemHandler();
|
|
return true;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// De-initialize memory manager
|
|
|
|
void cleanupMemPool(void) {
|
|
//clearMemHandler();
|
|
if (heapMemory) {
|
|
free(heapMemory);
|
|
heapMemory = nullptr;
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Allocates memory, or throws exception if allocation fails.
|
|
|
|
void *mustAlloc(uint32 size, const char desc[]) {
|
|
void *ptr;
|
|
|
|
ptr = malloc(size);
|
|
// REM: Before we give up completely, try unloading some things...
|
|
if (ptr == NULL)
|
|
error("Local heap allocation size %d bytes failed.", size);
|
|
return ptr;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------
|
|
// Allocates relocatable memory, or throws exception if allocation fails.
|
|
|
|
RHANDLE mustAllocHandle(uint32 size, const char desc[]) {
|
|
void **ptr;
|
|
|
|
ptr = (void **)malloc(size);
|
|
// REM: Before we give up completely, try unloading some things...
|
|
if (ptr == NULL)
|
|
error("Local handle allocation size %d bytes failed.", size);
|
|
return ptr;
|
|
}
|
|
|
|
} // end of namespace Saga2
|