2005-04-05 15:07:40 +00:00
|
|
|
/* ScummVM - Scumm Interpreter
|
2006-01-18 17:39:49 +00:00
|
|
|
* Copyright (C) 2004-2006 The ScummVM project
|
2005-04-05 15:07:40 +00:00
|
|
|
*
|
|
|
|
* 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
|
2005-04-09 19:19:54 +00:00
|
|
|
* along with this program; if not, write to the Free Software
|
2005-10-18 01:30:26 +00:00
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2005-04-05 15:07:40 +00:00
|
|
|
*
|
2006-02-11 10:11:37 +00:00
|
|
|
* $URL$
|
|
|
|
* $Id$
|
2005-04-05 15:07:40 +00:00
|
|
|
*
|
|
|
|
*/
|
2005-06-24 15:23:51 +00:00
|
|
|
#include "common/stdafx.h"
|
2006-11-19 17:52:52 +00:00
|
|
|
#include "common/endian.h"
|
2005-04-05 15:07:40 +00:00
|
|
|
|
|
|
|
#include "base/plugins.h"
|
2006-02-17 08:44:16 +00:00
|
|
|
#include "common/config-manager.h"
|
2006-06-24 08:07:48 +00:00
|
|
|
#include "common/fs.h"
|
2005-05-06 17:04:33 +00:00
|
|
|
#include "common/md5.h"
|
2005-04-05 15:07:40 +00:00
|
|
|
|
|
|
|
#include "gob/gob.h"
|
|
|
|
|
|
|
|
#include "gob/global.h"
|
|
|
|
#include "gob/game.h"
|
|
|
|
#include "gob/sound.h"
|
|
|
|
#include "gob/init.h"
|
2006-01-03 23:14:39 +00:00
|
|
|
#include "gob/inter.h"
|
|
|
|
#include "gob/draw.h"
|
|
|
|
#include "gob/anim.h"
|
|
|
|
#include "gob/cdrom.h"
|
|
|
|
#include "gob/goblin.h"
|
|
|
|
#include "gob/map.h"
|
|
|
|
#include "gob/mult.h"
|
|
|
|
#include "gob/pack.h"
|
|
|
|
#include "gob/palanim.h"
|
|
|
|
#include "gob/parse.h"
|
|
|
|
#include "gob/scenery.h"
|
|
|
|
#include "gob/timer.h"
|
|
|
|
#include "gob/util.h"
|
2006-01-07 22:28:54 +00:00
|
|
|
#include "gob/music.h"
|
2005-04-05 15:07:40 +00:00
|
|
|
|
2006-07-23 15:08:48 +00:00
|
|
|
#include "sound/mididrv.h"
|
|
|
|
|
2006-11-27 14:19:30 +00:00
|
|
|
// Use the original saves. Just for testing purposes, will be removed later
|
|
|
|
// The new method is more convenient, and, more importantly, endian-safe
|
|
|
|
//#define GOB_ORIGSAVES
|
|
|
|
|
2006-02-18 00:36:45 +00:00
|
|
|
namespace Gob {
|
|
|
|
|
2005-05-13 21:24:30 +00:00
|
|
|
|
2006-01-05 16:06:55 +00:00
|
|
|
#define MAX_TIME_DELTA 100
|
|
|
|
|
2007-01-30 02:17:46 +00:00
|
|
|
GobEngine::GobEngine(OSystem *syst) : Engine(syst) {
|
2006-01-05 16:06:55 +00:00
|
|
|
// Setup mixer
|
|
|
|
if (!_mixer->isReady()) {
|
|
|
|
warning("Sound initialization failed.");
|
|
|
|
}
|
|
|
|
|
|
|
|
_mixer->setVolumeForSoundType(Audio::Mixer::kSFXSoundType, ConfMan.getInt("sfx_volume"));
|
|
|
|
_mixer->setVolumeForSoundType(Audio::Mixer::kMusicSoundType, ConfMan.getInt("music_volume"));
|
|
|
|
|
2006-01-14 09:57:20 +00:00
|
|
|
_copyProtection = ConfMan.getBool("copy_protection");
|
2006-05-31 08:44:14 +00:00
|
|
|
_quitRequested = false;
|
2006-02-24 23:31:31 +00:00
|
|
|
|
2006-11-19 17:52:52 +00:00
|
|
|
int i;
|
|
|
|
_saveFiles = new char*[3];
|
|
|
|
for (i = 0; i < 3; i++)
|
|
|
|
_saveFiles[i] = new char[_targetName.size() + 5];
|
|
|
|
sprintf(_saveFiles[0], "%s.cat", _targetName.c_str());
|
|
|
|
sprintf(_saveFiles[1], "%s.sav", _targetName.c_str());
|
|
|
|
sprintf(_saveFiles[2], "%s.blo", _targetName.c_str());
|
2006-11-27 14:19:30 +00:00
|
|
|
_saveSlotFile = new char[_targetName.size() + 5];
|
|
|
|
sprintf(_saveSlotFile, "%s.s00", _targetName.c_str());
|
|
|
|
memset(_saveIndex, 0, 600);
|
|
|
|
memset(_saveIndexSizes, 0, 600);
|
2006-11-19 17:52:52 +00:00
|
|
|
|
2007-01-13 15:35:02 +00:00
|
|
|
Common::addSpecialDebugLevel(kDebugFuncOp, "FuncOpcodes", "Script FuncOpcodes debug level");
|
|
|
|
Common::addSpecialDebugLevel(kDebugDrawOp, "DrawOpcodes", "Script DrawOpcodes debug level");
|
|
|
|
Common::addSpecialDebugLevel(kDebugGobOp, "GoblinOpcodes", "Script GoblinOpcodes debug level");
|
2007-01-23 19:40:28 +00:00
|
|
|
Common::addSpecialDebugLevel(kDebugMusic, "Music", "CD, Adlib and Infogrames music debug level");
|
2007-01-13 15:35:02 +00:00
|
|
|
Common::addSpecialDebugLevel(kDebugParser, "Parser", "Parser debug level");
|
|
|
|
Common::addSpecialDebugLevel(kDebugGameFlow, "Gameflow", "Gameflow debug level");
|
|
|
|
Common::addSpecialDebugLevel(kDebugFileIO, "FileIO", "File Input/Output debug level");
|
|
|
|
Common::addSpecialDebugLevel(kDebugGraphics, "Graphics", "Graphics debug level");
|
|
|
|
Common::addSpecialDebugLevel(kDebugCollisions, "Collisions", "Collisions debug level");
|
2005-04-05 15:07:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
GobEngine::~GobEngine() {
|
2006-11-20 13:03:30 +00:00
|
|
|
delete _mult;
|
2006-01-03 23:14:39 +00:00
|
|
|
delete _game;
|
|
|
|
delete _snd;
|
|
|
|
delete _global;
|
|
|
|
delete _draw;
|
|
|
|
delete _anim;
|
|
|
|
delete _cdrom;
|
|
|
|
delete _dataio;
|
|
|
|
delete _goblin;
|
|
|
|
delete _init;
|
|
|
|
delete _inter;
|
|
|
|
delete _map;
|
|
|
|
delete _pack;
|
|
|
|
delete _palanim;
|
|
|
|
delete _parse;
|
|
|
|
delete _scenery;
|
|
|
|
delete _gtimer;
|
|
|
|
delete _util;
|
2007-01-23 19:40:28 +00:00
|
|
|
if (_adlib)
|
|
|
|
delete _adlib;
|
2006-11-20 13:03:30 +00:00
|
|
|
delete _video;
|
2006-08-12 13:40:16 +00:00
|
|
|
delete[] _startTot;
|
|
|
|
delete[] _startTot0;
|
2006-11-19 17:52:52 +00:00
|
|
|
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < 3; i++)
|
|
|
|
delete[] _saveFiles[i];
|
|
|
|
delete[] _saveFiles;
|
2006-11-27 14:19:30 +00:00
|
|
|
delete[] _saveSlotFile;
|
2005-04-05 15:07:40 +00:00
|
|
|
}
|
|
|
|
|
2006-01-05 16:06:55 +00:00
|
|
|
int GobEngine::go() {
|
|
|
|
_init->initGame(0);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GobEngine::shutdown() {
|
2006-05-31 08:44:14 +00:00
|
|
|
_quitRequested = true;
|
2006-01-05 16:06:55 +00:00
|
|
|
}
|
|
|
|
|
2006-11-19 17:52:52 +00:00
|
|
|
// Seeking with SEEK_END (and therefore also pos()) doesn't work with
|
|
|
|
// gzip'd save files, so reading the whole thing in is necessary
|
|
|
|
uint32 GobEngine::getSaveSize(Common::InSaveFile &in) {
|
|
|
|
char buf[1024];
|
|
|
|
uint32 size;
|
|
|
|
uint32 i;
|
|
|
|
uint32 pos;
|
|
|
|
|
|
|
|
size = 0;
|
|
|
|
pos = in.pos();
|
|
|
|
in.seek(0, SEEK_SET);
|
|
|
|
while ((i = in.read(buf, 1024)) > 0)
|
|
|
|
size += i;
|
|
|
|
in.seek(0, pos);
|
|
|
|
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
int32 GobEngine::getSaveSize(enum SaveFiles sFile) {
|
|
|
|
int32 size;
|
|
|
|
Common::InSaveFile *in;
|
|
|
|
|
2006-11-27 14:19:30 +00:00
|
|
|
size = -1;
|
|
|
|
#ifndef GOB_ORIGSAVES
|
|
|
|
int i;
|
|
|
|
if (sFile == SAVE_CAT) {
|
|
|
|
for (i = 14; i >= 0; i--)
|
|
|
|
if ((in = _saveFileMan->openForLoading(getSaveSlotFile(i)))) {
|
|
|
|
size = (i + 1) * READ_LE_UINT32(_game->_totFileData + 0x2C) * 4 + 600;
|
|
|
|
delete in;
|
|
|
|
break;
|
|
|
|
}
|
2007-01-13 15:35:02 +00:00
|
|
|
debugC(1, kDebugFileIO, "Requested save games size: %d", size);
|
2006-11-27 14:19:30 +00:00
|
|
|
return size;
|
|
|
|
}
|
|
|
|
#endif // GOB_ORIGSAVES
|
|
|
|
|
2007-01-28 13:19:17 +00:00
|
|
|
if (sFile == SAVE_SAV)
|
|
|
|
size = _global->_savedBack == 0 ? -1 : _global->_savedBackSize;
|
|
|
|
else if ((in = _saveFileMan->openForLoading(_saveFiles[(int) sFile]))) {
|
2006-11-19 17:52:52 +00:00
|
|
|
size = getSaveSize(*in);
|
|
|
|
delete in;
|
2006-11-27 14:19:30 +00:00
|
|
|
}
|
2006-11-19 17:52:52 +00:00
|
|
|
|
2007-01-13 15:35:02 +00:00
|
|
|
debugC(1, kDebugFileIO, "Requested size of file \"%s\": %d", _saveFiles[(int) sFile], size);
|
2006-11-19 17:52:52 +00:00
|
|
|
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
2006-11-27 14:19:30 +00:00
|
|
|
const char *GobEngine::getSaveSlotFile(int slot) {
|
|
|
|
static char *slotBase = _saveSlotFile + strlen(_targetName.c_str()) + 2;
|
|
|
|
|
|
|
|
snprintf(slotBase, 3, "%02d", slot);
|
|
|
|
return _saveSlotFile;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GobEngine::saveGameData(enum SaveFiles sFile, int16 dataVar, int32 size, int32 offset) {
|
2006-11-19 17:52:52 +00:00
|
|
|
int16 index;
|
|
|
|
bool writePal;
|
|
|
|
char *sName;
|
|
|
|
char *buf;
|
|
|
|
char *oBuf;
|
2007-01-28 13:19:17 +00:00
|
|
|
int32 retSize;
|
2006-11-19 17:52:52 +00:00
|
|
|
int32 iSize;
|
|
|
|
int32 oSize;
|
|
|
|
int32 oOff;
|
|
|
|
Video::SurfaceDesc *srcDesc;
|
|
|
|
Common::InSaveFile *in;
|
|
|
|
Common::OutSaveFile *out;
|
|
|
|
|
2007-01-28 13:19:17 +00:00
|
|
|
retSize = 0;
|
2006-11-19 17:52:52 +00:00
|
|
|
index = 0;
|
|
|
|
oBuf = 0;
|
|
|
|
in = 0;
|
|
|
|
writePal = false;
|
|
|
|
sName = _saveFiles[(int) sFile];
|
|
|
|
|
2006-11-27 14:19:30 +00:00
|
|
|
WRITE_VAR(1, 1);
|
2006-11-19 17:52:52 +00:00
|
|
|
|
2007-01-28 13:19:17 +00:00
|
|
|
if (sFile == SAVE_SAV) {
|
|
|
|
_global->_savedBackSize = -1;
|
|
|
|
if (size >= 0) {
|
|
|
|
warning("Invalid attempt at saving a sprite");
|
|
|
|
return;
|
|
|
|
}
|
2006-11-19 17:52:52 +00:00
|
|
|
if (size < -1000) {
|
|
|
|
size += 1000;
|
2007-01-28 13:19:17 +00:00
|
|
|
writePal = true;
|
|
|
|
memcpy((char *) _global->_savedPal, (char *) _global->_pPaletteDesc->vgaPal, 768);
|
2006-11-19 17:52:52 +00:00
|
|
|
}
|
|
|
|
index = -size - 1;
|
2007-01-28 13:19:17 +00:00
|
|
|
if ((index < 0) || (index >= 50)) {
|
|
|
|
warning("Invalid attempt at saving a sprite");
|
2006-11-27 14:19:30 +00:00
|
|
|
return;
|
|
|
|
}
|
2007-01-28 13:19:17 +00:00
|
|
|
srcDesc = _draw->_spritesArray[index];
|
|
|
|
|
|
|
|
if (_global->_savedBack)
|
|
|
|
_video->freeSurfDesc(_global->_savedBack);
|
|
|
|
|
|
|
|
_global->_savedBack =
|
|
|
|
_video->initSurfDesc(_vm->_global->_videoMode, srcDesc->width, srcDesc->height, 0);
|
|
|
|
_vm->_video->drawSprite(srcDesc, _global->_savedBack, 0, 0,
|
|
|
|
srcDesc->width - 1, srcDesc->height - 1, 0, 0, 0);
|
|
|
|
|
|
|
|
_global->_savedBackSize = _draw->getSpriteRectSize(index);
|
|
|
|
if (writePal)
|
|
|
|
_global->_savedBackSize += 768;
|
|
|
|
|
|
|
|
WRITE_VAR(1, 0);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (size < 0) {
|
|
|
|
warning("Invalid saving procedure");
|
|
|
|
return;
|
2006-11-27 14:19:30 +00:00
|
|
|
}
|
2006-11-19 17:52:52 +00:00
|
|
|
|
2007-01-28 13:19:17 +00:00
|
|
|
int32 varSize = READ_LE_UINT32(_game->_totFileData + 0x2C) * 4;
|
|
|
|
|
|
|
|
if (size == 0) {
|
|
|
|
dataVar = 0;
|
|
|
|
size = varSize;
|
|
|
|
}
|
|
|
|
buf = _global->_inter_variables + dataVar;
|
|
|
|
#ifndef GOB_ORIGSAVES
|
|
|
|
if (sFile == SAVE_CAT) {
|
|
|
|
if(saveGame((offset - 600) / varSize, dataVar, size, offset))
|
|
|
|
WRITE_VAR(1, 0);
|
|
|
|
return;
|
|
|
|
} else if (offset != 0) {
|
|
|
|
warning("Can't write file \"%s\": Can't correctly enfore endianness with offset", sName);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif // GOB_ORIGSAVES
|
|
|
|
|
2006-11-19 17:52:52 +00:00
|
|
|
if ((in = _saveFileMan->openForLoading(sName)))
|
|
|
|
iSize = getSaveSize(*in);
|
|
|
|
else
|
|
|
|
iSize = 0;
|
|
|
|
|
2006-11-21 13:28:45 +00:00
|
|
|
oOff = offset < 0 ? MAX((int32) 0, iSize - (-offset - 1)) : offset;
|
2007-01-28 13:19:17 +00:00
|
|
|
oSize = MAX(iSize, oOff + size);
|
2006-11-19 17:52:52 +00:00
|
|
|
oBuf = new char[oSize];
|
|
|
|
memset(oBuf, 0, oSize);
|
|
|
|
|
|
|
|
if (in) {
|
|
|
|
in->read(oBuf, iSize);
|
|
|
|
delete in;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!(out = _saveFileMan->openForSaving(sName))) {
|
2006-11-27 14:19:30 +00:00
|
|
|
warning("Can't open file \"%s\" for writing", sName);
|
2006-11-19 17:52:52 +00:00
|
|
|
delete[] oBuf;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2007-01-28 13:19:17 +00:00
|
|
|
retSize = writeDataEndian(*out, buf, _global->_inter_variablesSizes + dataVar, size);
|
2006-11-19 17:52:52 +00:00
|
|
|
|
|
|
|
out->flush();
|
|
|
|
|
2007-01-28 13:19:17 +00:00
|
|
|
if (out->ioFailed() || (retSize != size))
|
2006-11-19 17:52:52 +00:00
|
|
|
warning("Can't write file \"%s\"", sName);
|
|
|
|
else {
|
2007-01-13 15:35:02 +00:00
|
|
|
debugC(1, kDebugFileIO, "Saved file \"%s\" (%d, %d bytes at %d)",
|
2006-11-19 17:52:52 +00:00
|
|
|
sName, dataVar, size, offset);
|
2006-11-27 14:19:30 +00:00
|
|
|
WRITE_VAR(1, 0);
|
2006-11-19 17:52:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
delete out;
|
|
|
|
delete[] oBuf;
|
|
|
|
}
|
|
|
|
|
2006-11-27 14:19:30 +00:00
|
|
|
bool GobEngine::saveGame(int saveSlot, int16 dataVar, int32 size, int32 offset) {
|
|
|
|
int32 varSize;
|
|
|
|
char *varBuf;
|
|
|
|
byte *sizeBuf;
|
|
|
|
Common::OutSaveFile *out;
|
|
|
|
|
|
|
|
varBuf = _global->_inter_variables + dataVar;
|
|
|
|
sizeBuf = _global->_inter_variablesSizes + dataVar;
|
|
|
|
varSize = READ_LE_UINT32(_game->_totFileData + 0x2C) * 4;
|
|
|
|
if ((offset == 0) && (size == 600)) {
|
|
|
|
memcpy(_saveIndex, varBuf, size);
|
|
|
|
memcpy(_saveIndexSizes, sizeBuf, size);
|
|
|
|
return true;
|
|
|
|
} else if((((offset - 600) % varSize) == 0) && (size == varSize)) {
|
|
|
|
if (!(out = _saveFileMan->openForSaving(getSaveSlotFile(saveSlot)))) {
|
|
|
|
warning("Can't open file \"%s\" for writing", getSaveSlotFile(saveSlot));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
writeDataEndian(*out, _saveIndex + saveSlot * 40, _saveIndexSizes + saveSlot * 40, 40);
|
|
|
|
writeDataEndian(*out, varBuf, sizeBuf, size);
|
|
|
|
out->flush();
|
|
|
|
if (out->ioFailed()) {
|
|
|
|
warning("Can't save to slot %d", saveSlot);
|
|
|
|
return false;
|
|
|
|
}
|
2007-01-13 15:35:02 +00:00
|
|
|
debugC(1, kDebugFileIO, "Saved to slot %d", saveSlot);
|
2006-11-27 14:19:30 +00:00
|
|
|
delete out;
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
warning("Invalid saving procedure");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32 GobEngine::writeDataEndian(Common::OutSaveFile &out, char *varBuf, byte *sizeBuf,
|
|
|
|
int32 size) {
|
|
|
|
|
|
|
|
#ifndef GOB_ORIGSAVES
|
|
|
|
|
|
|
|
int i;
|
|
|
|
char tmp[4];
|
2006-11-27 20:56:41 +00:00
|
|
|
uint32 written;
|
2006-11-27 14:19:30 +00:00
|
|
|
|
|
|
|
written = 0;
|
|
|
|
for (i = 0; i < size; i++, varBuf++) {
|
|
|
|
if (sizeBuf[i] == 3)
|
|
|
|
WRITE_LE_UINT32(tmp, *((uint32 *) varBuf));
|
|
|
|
else if (sizeBuf[i] == 1)
|
|
|
|
WRITE_LE_UINT16(tmp, *((uint16 *) varBuf));
|
|
|
|
else if (sizeBuf[i] == 0)
|
|
|
|
*tmp = *varBuf;
|
|
|
|
else {
|
|
|
|
warning("Can't write data, corrupted variables sizes");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
written += out.write(tmp, sizeBuf[i] + 1);
|
|
|
|
varBuf += sizeBuf[i];
|
|
|
|
i += sizeBuf[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
out.write(sizeBuf, size);
|
|
|
|
|
|
|
|
return written;
|
|
|
|
|
|
|
|
#else // GOB_ORIGSAVES
|
|
|
|
|
|
|
|
return out.write(varBuf, size);
|
|
|
|
|
|
|
|
#endif // GOB_ORIGSAVES
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void GobEngine::loadGameData(enum SaveFiles sFile, int16 dataVar, int32 size, int32 offset) {
|
2006-11-19 17:52:52 +00:00
|
|
|
int32 sSize;
|
|
|
|
int32 retSize;
|
2006-11-27 14:19:30 +00:00
|
|
|
int16 index;
|
2006-11-19 17:52:52 +00:00
|
|
|
char *buf;
|
|
|
|
char *sName;
|
|
|
|
bool readPal;
|
|
|
|
Video::SurfaceDesc *destDesc;
|
|
|
|
Common::InSaveFile *in;
|
|
|
|
|
2006-11-27 14:19:30 +00:00
|
|
|
index = 0;
|
2006-11-19 17:52:52 +00:00
|
|
|
readPal = false;
|
|
|
|
sName = _saveFiles[(int) sFile];
|
2006-11-27 14:19:30 +00:00
|
|
|
|
|
|
|
WRITE_VAR(1, 1);
|
2007-01-28 13:19:17 +00:00
|
|
|
|
|
|
|
if (sFile == SAVE_SAV) {
|
|
|
|
if (size >= 0) {
|
|
|
|
warning("Invalid attempt at loading a sprite");
|
|
|
|
return;
|
|
|
|
}
|
2006-11-19 17:52:52 +00:00
|
|
|
if (size < -1000) {
|
|
|
|
size += 1000;
|
2007-01-28 13:19:17 +00:00
|
|
|
readPal = true;
|
|
|
|
memcpy((char *) _global->_pPaletteDesc->vgaPal, (char *) _global->_savedPal, 768);
|
2006-11-19 17:52:52 +00:00
|
|
|
}
|
|
|
|
index = -size - 1;
|
2007-01-28 13:19:17 +00:00
|
|
|
if ((index < 0) || (index >= 50)) {
|
|
|
|
warning("Invalid attempt at loading a sprite");
|
2006-11-27 14:19:30 +00:00
|
|
|
return;
|
2007-01-28 13:19:17 +00:00
|
|
|
}
|
|
|
|
destDesc = _draw->_spritesArray[index];
|
|
|
|
|
|
|
|
if ((destDesc->width != _global->_savedBack->width) ||
|
|
|
|
(destDesc->height != _global->_savedBack->height)) {
|
|
|
|
warning("Resolution doesn't match while loading a sprite");
|
2006-11-27 14:19:30 +00:00
|
|
|
return;
|
|
|
|
}
|
2007-01-28 13:19:17 +00:00
|
|
|
|
|
|
|
_vm->_video->drawSprite(_global->_savedBack, destDesc, 0, 0,
|
|
|
|
destDesc->width - 1, destDesc->height - 1, 0, 0, 0);
|
|
|
|
if (index == 21) {
|
|
|
|
_video->drawSprite(_draw->_backSurface, _draw->_frontSurface, 0, 0,
|
|
|
|
_draw->_frontSurface->width - 1, _draw->_frontSurface->height - 1, 0, 0, 0);
|
|
|
|
_video->waitRetrace(_global->_videoMode);
|
|
|
|
}
|
|
|
|
|
|
|
|
WRITE_VAR(1, 0);
|
|
|
|
return;
|
2006-11-27 14:19:30 +00:00
|
|
|
}
|
2006-11-19 17:52:52 +00:00
|
|
|
|
2007-01-28 13:19:17 +00:00
|
|
|
if (size < 0) {
|
|
|
|
warning("Invalid loading procedure");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int32 varSize;
|
|
|
|
varSize = READ_LE_UINT32(_game->_totFileData + 0x2C) * 4;
|
|
|
|
if (size == 0) {
|
|
|
|
dataVar = 0;
|
|
|
|
size = varSize;
|
|
|
|
}
|
|
|
|
buf = _global->_inter_variables + dataVar;
|
|
|
|
#ifndef GOB_ORIGSAVES
|
|
|
|
if (sFile == SAVE_CAT) {
|
|
|
|
if(loadGame((offset - 600) / varSize, dataVar, size, offset))
|
|
|
|
WRITE_VAR(1, 0);
|
|
|
|
return;
|
|
|
|
} else if (offset != 0) {
|
|
|
|
warning("Can't read file \"%s\": Can't correctly enfore endianness with offset", sName);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif // GOB_ORIGSAVES
|
|
|
|
|
2006-11-19 17:52:52 +00:00
|
|
|
if (_global->_inter_resStr[0] == 0) {
|
2006-11-27 14:19:30 +00:00
|
|
|
WRITE_VAR(1, size);
|
2006-11-19 17:52:52 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2006-11-27 14:19:30 +00:00
|
|
|
if(!(in = _saveFileMan->openForLoading(sName))) {
|
|
|
|
warning("Can't open file \"%s\" for reading", sName);
|
2006-11-19 17:52:52 +00:00
|
|
|
return;
|
2006-11-27 14:19:30 +00:00
|
|
|
}
|
2006-11-19 17:52:52 +00:00
|
|
|
|
2007-01-13 15:35:02 +00:00
|
|
|
debugC(1, kDebugFileIO, "Loading file \"%s\" (%d, %d bytes at %d)",
|
2006-11-19 17:52:52 +00:00
|
|
|
sName, dataVar, size, offset);
|
|
|
|
|
|
|
|
sSize = getSaveSize(*in);
|
|
|
|
_draw->animateCursor(4);
|
|
|
|
if (offset < 0)
|
|
|
|
in->seek(sSize - (-offset - 1), 0);
|
|
|
|
else
|
|
|
|
in->seek(offset, 0);
|
|
|
|
|
2007-01-28 13:19:17 +00:00
|
|
|
retSize = readDataEndian(*in, buf, _global->_inter_variablesSizes + dataVar, size);
|
2006-11-19 17:52:52 +00:00
|
|
|
|
|
|
|
if (retSize == size)
|
2006-11-27 14:19:30 +00:00
|
|
|
WRITE_VAR(1, 0);
|
2006-11-19 17:52:52 +00:00
|
|
|
|
|
|
|
delete in;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2006-11-27 14:19:30 +00:00
|
|
|
bool GobEngine::loadGame(int saveSlot, int16 dataVar, int32 size, int32 offset) {
|
|
|
|
int i;
|
|
|
|
int32 varSize;
|
|
|
|
char *varBuf;
|
|
|
|
byte *sizeBuf;
|
|
|
|
Common::InSaveFile *in;
|
|
|
|
|
|
|
|
varBuf = _global->_inter_variables + dataVar;
|
|
|
|
sizeBuf = _global->_inter_variablesSizes + dataVar;
|
|
|
|
varSize = READ_LE_UINT32(_game->_totFileData + 0x2C) * 4;
|
|
|
|
if ((offset == 0) && (size == 600)) {
|
|
|
|
for (i = 0; i < 15; i++, varBuf += 40) {
|
|
|
|
if ((in = _saveFileMan->openForLoading(getSaveSlotFile(i)))) {
|
|
|
|
in->read(varBuf, 40);
|
|
|
|
delete in;
|
|
|
|
} else
|
|
|
|
memset(varBuf, 0, 40);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
} else if((((offset - 600) % varSize) == 0) && (size == varSize)) {
|
|
|
|
if (!(in = _saveFileMan->openForLoading(getSaveSlotFile(saveSlot)))) {
|
|
|
|
warning("Can't load from slot %d", saveSlot);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (((getSaveSize(*in) / 2) - 40) != (uint32) varSize) {
|
|
|
|
warning("Can't load from slot %d: Wrong size", saveSlot);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
in->seek(80);
|
|
|
|
readDataEndian(*in, varBuf, sizeBuf, size);
|
|
|
|
delete in;
|
2007-01-13 15:35:02 +00:00
|
|
|
debugC(1, kDebugFileIO, "Loading from slot %d", saveSlot);
|
2006-11-27 14:19:30 +00:00
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
warning("Invalid loading procedure");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32 GobEngine::readDataEndian(Common::InSaveFile &in, char *varBuf, byte *sizeBuf,
|
|
|
|
int32 size) {
|
|
|
|
|
|
|
|
#ifndef GOB_ORIGSAVES
|
|
|
|
|
|
|
|
uint32 read;
|
|
|
|
char *vars;
|
|
|
|
char *sizes;
|
2006-11-27 20:56:41 +00:00
|
|
|
int i;
|
2006-11-27 14:19:30 +00:00
|
|
|
|
|
|
|
vars = new char[size];
|
|
|
|
sizes = new char[size];
|
|
|
|
|
|
|
|
read = in.read(vars, size);
|
|
|
|
if (in.read(sizes, size) != read) {
|
|
|
|
warning("Can't read data: Corrupted variables sizes");
|
|
|
|
delete[] vars;
|
|
|
|
delete[] sizes;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < size; i++) {
|
|
|
|
if (sizes[i] == 3)
|
|
|
|
*((uint32 *) (vars + i)) = READ_LE_UINT32(vars + i);
|
|
|
|
else if (sizes[i] == 1)
|
|
|
|
*((uint16 *) (vars + i)) = READ_LE_UINT16(vars + i);
|
|
|
|
else if (sizes[i] != 0) {
|
|
|
|
warning("Can't read data: Corrupted variables sizes");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
i += sizes[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(varBuf, vars, size);
|
|
|
|
memcpy(sizeBuf, sizes, size);
|
|
|
|
delete[] vars;
|
|
|
|
delete[] sizes;
|
|
|
|
|
|
|
|
return read;
|
|
|
|
|
|
|
|
#else // GOB_ORIGSAVES
|
|
|
|
|
|
|
|
return in.read(varBuf, size);
|
|
|
|
|
|
|
|
#endif // GOB_ORIGSAVES
|
|
|
|
}
|
|
|
|
|
2006-04-15 20:36:41 +00:00
|
|
|
int GobEngine::init() {
|
2007-01-30 02:17:46 +00:00
|
|
|
// Detect game
|
|
|
|
if (!detectGame()) {
|
|
|
|
GUIErrorMessage("No valid games were found in the specified directory.");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2007-01-23 19:40:28 +00:00
|
|
|
_adlib = 0;
|
2006-01-03 23:14:39 +00:00
|
|
|
_snd = new Snd(this);
|
|
|
|
_global = new Global(this);
|
|
|
|
_anim = new Anim();
|
|
|
|
_cdrom = new CDROM(this);
|
|
|
|
_dataio = new DataIO(this);
|
|
|
|
_pack = new Pack();
|
|
|
|
_palanim = new PalAnim(this);
|
|
|
|
_gtimer = new GTimer();
|
|
|
|
_util = new Util(this);
|
2006-02-02 21:12:00 +00:00
|
|
|
if (_features & Gob::GF_GOB1) {
|
2006-01-05 16:06:55 +00:00
|
|
|
_inter = new Inter_v1(this);
|
2006-02-02 21:12:00 +00:00
|
|
|
_parse = new Parse_v1(this);
|
2006-02-21 14:22:25 +00:00
|
|
|
_mult = new Mult_v1(this);
|
2006-04-13 16:25:07 +00:00
|
|
|
_draw = new Draw_v1(this);
|
2006-04-18 09:59:18 +00:00
|
|
|
_game = new Game_v1(this);
|
|
|
|
_video = new Video_v1(this);
|
2006-05-01 12:43:50 +00:00
|
|
|
_init = new Init_v1(this);
|
2006-05-11 19:43:30 +00:00
|
|
|
_map = new Map_v1(this);
|
|
|
|
_goblin = new Goblin_v1(this);
|
2006-05-29 18:24:52 +00:00
|
|
|
_scenery = new Scenery_v1(this);
|
2006-02-02 21:12:00 +00:00
|
|
|
}
|
|
|
|
else if (_features & Gob::GF_GOB2) {
|
2006-01-05 16:06:55 +00:00
|
|
|
_inter = new Inter_v2(this);
|
2006-02-02 21:12:00 +00:00
|
|
|
_parse = new Parse_v2(this);
|
2006-02-21 14:22:25 +00:00
|
|
|
_mult = new Mult_v2(this);
|
2006-04-13 16:25:07 +00:00
|
|
|
_draw = new Draw_v2(this);
|
2006-04-18 09:59:18 +00:00
|
|
|
_game = new Game_v2(this);
|
|
|
|
_video = new Video_v2(this);
|
2007-02-04 15:45:15 +00:00
|
|
|
_init = new Init_v2(this);
|
|
|
|
_map = new Map_v2(this);
|
|
|
|
_goblin = new Goblin_v2(this);
|
|
|
|
_scenery = new Scenery_v2(this);
|
|
|
|
}
|
|
|
|
else if (_features & Gob::GF_BARGON) {
|
|
|
|
_inter = new Inter_Bargon(this);
|
|
|
|
_parse = new Parse_v2(this);
|
|
|
|
_mult = new Mult_v2(this);
|
|
|
|
_draw = new Draw_Bargon(this);
|
|
|
|
_game = new Game_v2(this);
|
|
|
|
_video = new Video_v2(this);
|
2006-05-01 12:43:50 +00:00
|
|
|
_init = new Init_v2(this);
|
2006-05-11 19:43:30 +00:00
|
|
|
_map = new Map_v2(this);
|
|
|
|
_goblin = new Goblin_v2(this);
|
2006-05-29 18:24:52 +00:00
|
|
|
_scenery = new Scenery_v2(this);
|
2006-02-02 21:12:00 +00:00
|
|
|
}
|
2006-01-05 16:06:55 +00:00
|
|
|
else
|
|
|
|
error("GobEngine::init(): Unknown version of game engine");
|
2007-01-25 14:18:12 +00:00
|
|
|
_noMusic = MidiDriver::parseMusicDriver(ConfMan.get("music_driver")) == MD_NULL;
|
2007-01-30 22:32:51 +00:00
|
|
|
if (!_noMusic && !(_platform == Common::kPlatformAmiga) &&
|
|
|
|
(((_platform == Common::kPlatformMacintosh) && (_features & Gob::GF_GOB1)) ||
|
2007-01-30 22:19:55 +00:00
|
|
|
(_features & Gob::GF_GOB2)))
|
|
|
|
_adlib = new Adlib(this);
|
2006-11-27 14:19:30 +00:00
|
|
|
_vm = this;
|
2006-01-03 23:14:39 +00:00
|
|
|
|
2005-04-19 08:29:43 +00:00
|
|
|
_system->beginGFXTransaction();
|
2006-04-15 20:36:41 +00:00
|
|
|
initCommonGFX(false);
|
2005-04-19 08:29:43 +00:00
|
|
|
_system->initSize(320, 200);
|
|
|
|
_system->endGFXTransaction();
|
|
|
|
|
2005-11-01 11:18:50 +00:00
|
|
|
// On some systems it's not safe to run CD audio games from the CD.
|
|
|
|
if (_features & GF_CD)
|
|
|
|
checkCD();
|
|
|
|
|
2005-04-22 08:39:51 +00:00
|
|
|
int cd_num = ConfMan.getInt("cdrom");
|
|
|
|
if (cd_num >= 0)
|
|
|
|
_system->openCD(cd_num);
|
|
|
|
|
2006-01-04 01:48:15 +00:00
|
|
|
_global->_debugFlag = 1;
|
|
|
|
_global->_doRangeClamp = 1;
|
2005-04-05 15:07:40 +00:00
|
|
|
|
2006-01-04 01:48:15 +00:00
|
|
|
_global->_videoMode = 0x13;
|
|
|
|
_global->_useMouse = 1;
|
2007-01-30 22:19:55 +00:00
|
|
|
_global->_soundFlags = MIDI_FLAG | SPEAKER_FLAG | BLASTER_FLAG | ADLIB_FLAG;
|
2005-04-26 06:29:53 +00:00
|
|
|
|
2006-04-18 12:39:02 +00:00
|
|
|
if (ConfMan.hasKey("language"))
|
|
|
|
_language = Common::parseLanguage(ConfMan.get("language"));
|
|
|
|
|
|
|
|
switch (_language) {
|
2005-04-26 06:29:53 +00:00
|
|
|
case Common::FR_FRA:
|
2006-04-18 12:39:02 +00:00
|
|
|
case Common::RU_RUS:
|
2006-01-04 01:48:15 +00:00
|
|
|
_global->_language = 0;
|
2005-04-26 06:29:53 +00:00
|
|
|
break;
|
|
|
|
case Common::DE_DEU:
|
2006-01-04 01:48:15 +00:00
|
|
|
_global->_language = 1;
|
2005-04-26 06:29:53 +00:00
|
|
|
break;
|
2007-02-01 21:59:51 +00:00
|
|
|
case Common::EN_ANY:
|
2006-04-18 12:39:02 +00:00
|
|
|
case Common::EN_GRB:
|
|
|
|
_global->_language = 2;
|
|
|
|
break;
|
2005-04-26 06:29:53 +00:00
|
|
|
case Common::ES_ESP:
|
2006-01-04 01:48:15 +00:00
|
|
|
_global->_language = 3;
|
2005-04-26 06:29:53 +00:00
|
|
|
break;
|
|
|
|
case Common::IT_ITA:
|
2006-01-04 01:48:15 +00:00
|
|
|
_global->_language = 4;
|
2005-04-26 06:29:53 +00:00
|
|
|
break;
|
2006-04-18 12:39:02 +00:00
|
|
|
case Common::EN_USA:
|
|
|
|
_global->_language = 5;
|
|
|
|
break;
|
|
|
|
case Common::NL_NLD:
|
|
|
|
_global->_language = 6;
|
|
|
|
break;
|
|
|
|
case Common::KO_KOR:
|
|
|
|
_global->_language = 7;
|
|
|
|
break;
|
|
|
|
case Common::HB_ISR:
|
|
|
|
_global->_language = 8;
|
|
|
|
break;
|
2005-04-26 06:29:53 +00:00
|
|
|
default:
|
|
|
|
// Default to English
|
2006-01-04 01:48:15 +00:00
|
|
|
_global->_language = 2;
|
2005-04-26 06:29:53 +00:00
|
|
|
break;
|
|
|
|
}
|
2005-04-05 15:07:40 +00:00
|
|
|
|
2005-08-10 15:48:50 +00:00
|
|
|
// FIXME: This is the ugly way of reducing redraw overhead. It works
|
|
|
|
// well for 320x200 but it's unclear how well it will work for
|
|
|
|
// 640x480.
|
|
|
|
|
|
|
|
g_system->setFeatureState(OSystem::kFeatureAutoComputeDirtyRects, true);
|
2005-04-05 15:07:40 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // End of namespace Gob
|