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-02-18 00:36:45 +00:00
|
|
|
namespace Gob {
|
|
|
|
|
2005-05-06 17:04:33 +00:00
|
|
|
enum {
|
2006-08-15 21:19:09 +00:00
|
|
|
// We only compute MD5 of the first 5000 bytes of our data files.
|
|
|
|
kMD5FileSizeLimit = 5000
|
2005-05-06 17:04:33 +00:00
|
|
|
};
|
|
|
|
|
2006-04-08 11:21:04 +00:00
|
|
|
struct GameSettings {
|
2006-02-17 08:44:16 +00:00
|
|
|
const char *gameid;
|
|
|
|
const char *description;
|
|
|
|
uint32 features;
|
2006-04-18 12:39:02 +00:00
|
|
|
Common::Language lang;
|
2006-02-17 08:44:16 +00:00
|
|
|
const char *md5sum;
|
2006-08-12 12:35:18 +00:00
|
|
|
const char *startTotBase;
|
2006-02-17 08:44:16 +00:00
|
|
|
};
|
|
|
|
|
2006-04-08 11:21:04 +00:00
|
|
|
static const GameSettings gob_games[] = {
|
2005-05-13 21:24:30 +00:00
|
|
|
// Supplied by Florian Zeitz on scummvm-devel
|
2006-08-15 21:19:09 +00:00
|
|
|
{"gob1", "Gobliiins (DOS EGA)", GF_GOB1 | GF_EGA, Common::UNK_LANG, "c65e9cc8ba23a38456242e1f2b1caad4", "intro"},
|
|
|
|
{"gob1", "Gobliiins (DOS EGA Ru)", GF_GOB1 | GF_EGA, Common::RU_RUS, "f9233283a0be2464248d83e14b95f09c", "intro"},
|
|
|
|
//{"gob1", "Gobliiins (Windows)", GF_GOB1, Common::UNK_LANG, "8a5e850c49d7cacdba5f5eb1fcc77b89", "intro"},
|
2005-05-13 21:24:30 +00:00
|
|
|
|
|
|
|
// Supplied by Theruler76 in bug report #1201233
|
2006-08-15 21:19:09 +00:00
|
|
|
{"gob1", "Gobliiins (DOS VGA)", GF_GOB1, Common::UNK_LANG, "26a9118c0770fa5ac93a9626761600b2", "intro"},
|
2005-05-13 21:24:30 +00:00
|
|
|
|
|
|
|
// CD 1.000 version. Multilingual
|
2006-08-15 21:19:09 +00:00
|
|
|
{"gob1", "Gobliiins (CD)", GF_GOB1 | GF_CD, Common::UNK_LANG, "2fbf4b5b82bbaee87eb45d4404c28998", "intro"},
|
2005-05-13 21:24:30 +00:00
|
|
|
// CD 1.02 version. Multilingual
|
2006-08-15 21:19:09 +00:00
|
|
|
{"gob1", "Gobliiins (CD)", GF_GOB1 | GF_CD, Common::UNK_LANG, "8bd873137b6831c896ee8ad217a6a398", "intro"},
|
2005-05-13 21:24:30 +00:00
|
|
|
|
2006-08-15 21:19:09 +00:00
|
|
|
{"gob1", "Gobliiins (Amiga)", GF_GOB1, Common::UNK_LANG, "c65e9cc8ba23a38456242e1f2b1caad4", "intro"},
|
|
|
|
{"gob1", "Gobliiins (Amiga)", GF_GOB1, Common::UNK_LANG, "972f22c6ff8144a6636423f0354ca549", "intro"},
|
2005-05-13 21:24:30 +00:00
|
|
|
|
2006-08-15 21:19:09 +00:00
|
|
|
{"gob1", "Gobliiins (Interactive Demo)", GF_GOB1, Common::UNK_LANG, "e72bd1e3828c7dec4c8a3e58c48bdfdb", "intro"},
|
2006-01-08 20:03:20 +00:00
|
|
|
|
2006-08-15 21:19:09 +00:00
|
|
|
{"gob1", "Gobliiins (Mac)", GF_GOB1 | GF_MAC, Common::UNK_LANG, "00a42a7d2d22e6b6ab1b8c673c4ed267", "intro"},
|
2005-05-13 21:24:30 +00:00
|
|
|
|
2006-08-15 21:19:09 +00:00
|
|
|
{"gob2", "Gobliins 2 (DOS Fra)", GF_GOB2, Common::FR_FRA, "a13ecb4f6d8fd881ebbcc02e45cb5475", "intro"},
|
|
|
|
{"gob2", "Gobliins 2 (DOS Grb)", GF_GOB2, Common::EN_GRB, "b45b984ee8017efd6ea965b9becd4d66", "intro"},
|
|
|
|
{"gob2", "Gobliins 2 (DOS USA)", GF_GOB2, Common::EN_USA, "dedb5d31d8c8050a8cf77abedcc53dae", "intro"},
|
2006-08-12 12:35:18 +00:00
|
|
|
{"gob2", "Gobliins 2 (DOS)", GF_GOB2, Common::DE_DEU, "a13892cdf4badda85a6f6fb47603a128", "intro"},
|
2006-08-15 21:19:09 +00:00
|
|
|
{"gob2", "Gobliins 2 (DOS Ru)", GF_GOB2, Common::RU_RUS, "cd3e1df8b273636ee32e34b7064f50e8", "intro"},
|
2005-05-13 21:24:30 +00:00
|
|
|
// CD 1.000.
|
2006-08-15 21:19:09 +00:00
|
|
|
{"gob2", "Gobliins 2 (CD)", GF_GOB2, Common::EN_USA, "9de5fbb41cf97182109e5fecc9d90347", "intro"},
|
2005-05-13 21:24:30 +00:00
|
|
|
// CD 1.01
|
2006-08-15 21:19:09 +00:00
|
|
|
{"gob2", "Gobliins 2 (CD)", GF_GOB2, Common::UNK_LANG, "24a6b32757752ccb1917ce92fd7c2a04", "intro"},
|
|
|
|
{"gob2", "Gobliins 2 (Demo)", GF_GOB2, Common::UNK_LANG, "8b1c98ff2ab2e14f47a1b891e9b92217", "usa"},
|
|
|
|
|
|
|
|
{"gob2", "Ween: The Prohpecy", GF_GOB2, Common::UNK_LANG, "2bb8878a8042244dd2b96ff682381baa", "intro"},
|
|
|
|
{"gob2", "Ween: The Prophecy (Fr)", GF_GOB2, Common::UNK_LANG, "4b10525a3782aa7ecd9d833b5c1d308b", "intro"},
|
|
|
|
{"gob2", "Ween: The Prophecy (Demo)", GF_GOB2, Common::UNK_LANG, "2e9c2898f6bf206ede801e3b2e7ee428", "intro"},
|
|
|
|
|
|
|
|
{"gob2", "Bargon Attack", GF_GOB2, Common::UNK_LANG, "da3c54be18ab73fbdb32db24624a9c23", "intro"},
|
2005-05-13 21:24:30 +00:00
|
|
|
|
2006-02-18 01:04:58 +00:00
|
|
|
#if 0
|
2006-08-15 21:19:09 +00:00
|
|
|
{"gob3", "Goblins Quest 3", GF_GOB3, Common::UNK_LANG, "32b0f57f5ae79a9ae97e8011df38af42", "intro"},
|
|
|
|
//{"gob3", "Goblins Quest 3", GF_GOB3, Common::UNK_LANG, "d129f639f6ca8d6b5f0f4e15edb91058", "intro"},
|
|
|
|
{"gob3", "Goblins Quest 3", GF_GOB3, Common::UNK_LANG, "1e2f64ec8dfa89f42ee49936a27e66e7", "intro"},
|
|
|
|
{"gob3", "Goblins Quest 3 (Fr)", GF_GOB3, Common::FR_FRA, "e42a4f2337d6549487a80864d7826972", "intro"},
|
|
|
|
{"gob3", "Goblins Quest 3 (Ru)", GF_GOB3, Common::RU_RUS, "4e3af248a48a2321364736afab868527", "intro"},
|
|
|
|
{"gob3", "Goblins Quest 3 (Mac)", GF_GOB3, Common::UNK_LANG, "8d28ce1591b0e9cc79bf41cad0fc4c9c", "intro"},
|
2005-05-13 21:24:30 +00:00
|
|
|
// CD 1.000
|
2006-08-15 21:19:09 +00:00
|
|
|
{"gob3", "Goblins Quest 3 (CD)", GF_GOB3, Common::UNK_LANG, "6f2c226c62dd7ab0ab6f850e89d3fc47", "intro"},
|
2005-05-23 20:53:54 +00:00
|
|
|
// CD 1.02. Spanish "Computer Gaming World"* distribution in Spain
|
2006-08-15 21:19:09 +00:00
|
|
|
{"gob3", "Goblins Quest 3 (CD)", GF_GOB3, Common::UNK_LANG, "c3e9132ea9dc0fb866b6d60dcda10261", "intro"},
|
2005-05-13 21:24:30 +00:00
|
|
|
|
2006-08-15 21:19:09 +00:00
|
|
|
{"gob3", "Goblins Quest 3 (Interactive Demo)", GF_GOB3, Common::UNK_LANG, "7aebd94e49c2c5c518c9e7b74f25de9d", "intro"},
|
|
|
|
{"gob3", "Goblins Quest 3 (Demo)", GF_GOB3, "b9b898fccebe02b69c086052d5024a55", "intro"},
|
|
|
|
{"gob3", "Goblins Quest 3 (Interactive Demo)", GF_GOB3, Common::UNK_LANG, "e5dcbc9f6658ebb1e8fe26bc4da0806d", "intro"},
|
2005-05-13 21:24:30 +00:00
|
|
|
|
2005-05-15 13:39:27 +00:00
|
|
|
// CD 1.0
|
2006-08-15 21:19:09 +00:00
|
|
|
{"woodruff", "The Bizarre Adventures of Woodruff and the Schnibble", GF_WOODRUFF, Common::UNK_LANG, "dccf9d31cb720b34d75487408821b77e", "intro"},
|
2005-05-17 11:46:07 +00:00
|
|
|
|
|
|
|
// CD 1.00, German release (INTRO.STRK seems to be multilingual, though?)
|
2006-08-15 21:19:09 +00:00
|
|
|
{"woodruff", "The Bizarre Adventures of Woodruff and the Schnibble", GF_WOODRUFF, Common::UNK_LANG, "5f5f4e0a72c33391e67a47674b120cc6", "intro"},
|
2005-05-06 17:04:33 +00:00
|
|
|
#endif
|
2006-08-12 12:35:18 +00:00
|
|
|
{0, 0, 0, Common::UNK_LANG, 0, 0}
|
2005-04-05 15:07:40 +00:00
|
|
|
};
|
|
|
|
|
2005-05-13 21:24:30 +00:00
|
|
|
// Keep list of different supported games
|
2006-03-09 02:52:51 +00:00
|
|
|
static const PlainGameDescriptor gob_list[] = {
|
2006-02-18 00:12:36 +00:00
|
|
|
{"gob1", "Gobliiins"},
|
|
|
|
{"gob2", "Gobliins 2"},
|
|
|
|
{0, 0}
|
2005-05-13 21:24:30 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2006-01-05 16:06:55 +00:00
|
|
|
#define MAX_TIME_DELTA 100
|
|
|
|
|
2006-08-12 12:35:18 +00:00
|
|
|
GobEngine::GobEngine(OSystem * syst, uint32 features, Common::Language lang,
|
|
|
|
const char *startTotBase)
|
2006-01-05 16:06:55 +00:00
|
|
|
: Engine(syst) {
|
|
|
|
// 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"));
|
|
|
|
|
|
|
|
_features = features;
|
2006-04-18 12:39:02 +00:00
|
|
|
_language = lang;
|
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-08-12 12:35:18 +00:00
|
|
|
if (startTotBase == 0) {
|
|
|
|
_startTot = new char[10];
|
|
|
|
_startTot0 = new char[11];
|
|
|
|
strcpy(_startTot, "intro.tot");
|
|
|
|
strcpy(_startTot0, "intro0.tot");
|
|
|
|
} else {
|
|
|
|
_startTot = new char[strlen(startTotBase) + 5];
|
|
|
|
_startTot0 = new char[strlen(startTotBase) + 6];
|
|
|
|
strcpy(_startTot, startTotBase);
|
|
|
|
strcpy(_startTot0, startTotBase);
|
|
|
|
strcat(_startTot, ".tot");
|
|
|
|
strcat(_startTot0, "0.tot");
|
|
|
|
}
|
|
|
|
|
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-02-24 23:31:31 +00:00
|
|
|
Common::addSpecialDebugLevel(DEBUG_FUNCOP, "FuncOpcodes", "Script FuncOpcodes debug level");
|
|
|
|
Common::addSpecialDebugLevel(DEBUG_DRAWOP, "DrawOpcodes", "Script DrawOpcodes debug level");
|
|
|
|
Common::addSpecialDebugLevel(DEBUG_GOBOP, "GoblinOpcodes", "Script GoblinOpcodes debug level");
|
|
|
|
Common::addSpecialDebugLevel(DEBUG_MUSIC, "Music", "CD and adlib music debug level");
|
2006-02-25 10:16:46 +00:00
|
|
|
Common::addSpecialDebugLevel(DEBUG_PARSER, "Parser", "Parser debug level");
|
2006-02-24 23:31:31 +00:00
|
|
|
Common::addSpecialDebugLevel(DEBUG_GAMEFLOW, "Gameflow", "Gameflow debug level");
|
|
|
|
Common::addSpecialDebugLevel(DEBUG_FILEIO, "FileIO", "File Input/Output debug level");
|
|
|
|
Common::addSpecialDebugLevel(DEBUG_GRAPHICS, "Graphics", "Graphics debug level");
|
|
|
|
Common::addSpecialDebugLevel(DEBUG_COLLISIONS, "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;
|
2006-01-07 22:28:54 +00:00
|
|
|
delete _music;
|
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;
|
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-08-12 09:16:08 +00:00
|
|
|
void GobEngine::writeVarDebug(uint32 offs, uint32 v) {
|
2006-11-21 13:28:45 +00:00
|
|
|
warning("Setting var %u(%u) to %u", offs, offs >> 2, v);
|
2006-08-12 09:16:08 +00:00
|
|
|
(*(uint32 *)(_global->_inter_variables + (offs))) = v;
|
|
|
|
}
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
if ((in = _saveFileMan->openForLoading(_saveFiles[(int) sFile]))) {
|
|
|
|
size = getSaveSize(*in);
|
|
|
|
delete in;
|
|
|
|
} else
|
|
|
|
size = -1;
|
|
|
|
|
|
|
|
debugC(1, DEBUG_FILEIO, "Requested size of file \"%s\": %d", _saveFiles[(int) sFile], size);
|
|
|
|
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GobEngine::saveGame(enum SaveFiles sFile, int16 dataVar, int32 size, int32 offset) {
|
|
|
|
int32 retSize;
|
|
|
|
int16 index;
|
|
|
|
int16 top;
|
|
|
|
bool writePal;
|
|
|
|
char *sName;
|
|
|
|
char *buf;
|
|
|
|
char *oBuf;
|
|
|
|
int32 iSize;
|
|
|
|
int32 oSize;
|
|
|
|
int32 oOff;
|
|
|
|
Video::SurfaceDesc *destDesc;
|
|
|
|
Video::SurfaceDesc *srcDesc;
|
|
|
|
Common::InSaveFile *in;
|
|
|
|
Common::OutSaveFile *out;
|
|
|
|
|
|
|
|
index = 0;
|
|
|
|
oBuf = 0;
|
|
|
|
in = 0;
|
|
|
|
writePal = false;
|
|
|
|
sName = _saveFiles[(int) sFile];
|
|
|
|
|
|
|
|
// WRITE_VAR(1, 1)
|
|
|
|
*((uint32*)(_global->_inter_variables + 4)) = 1;
|
|
|
|
|
|
|
|
if (size < 0) {
|
|
|
|
if (size < -1000) {
|
|
|
|
writePal = true;
|
|
|
|
size += 1000;
|
|
|
|
}
|
|
|
|
index = -size - 1;
|
|
|
|
assert((index >= 0) && (index < 50)); // Just to be sure...
|
|
|
|
buf = (char *) _draw->_spritesArray[index]->vidPtr;
|
|
|
|
size = _draw->getSpriteRectSize(index);
|
|
|
|
if ((_draw->_spritesArray[index]->vidMode & 0x80) == 0)
|
|
|
|
size = -size;
|
|
|
|
} else if (size == 0) {
|
|
|
|
dataVar = 0;
|
|
|
|
size = READ_LE_UINT32(_game->_totFileData + 0x2C) * 4;
|
|
|
|
buf = _global->_inter_variables;
|
|
|
|
} else
|
|
|
|
buf = _global->_inter_variables + dataVar;
|
|
|
|
|
|
|
|
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;
|
2006-11-19 17:52:52 +00:00
|
|
|
oSize = MAX(iSize, oOff + ABS(size));
|
|
|
|
oBuf = new char[oSize];
|
|
|
|
memset(oBuf, 0, oSize);
|
|
|
|
|
|
|
|
if (in) {
|
|
|
|
in->read(oBuf, iSize);
|
|
|
|
delete in;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!(out = _saveFileMan->openForSaving(sName))) {
|
|
|
|
warning("Can't write file \"%s\"", sName);
|
|
|
|
delete[] oBuf;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (writePal) {
|
|
|
|
memcpy(oBuf + oOff, (char *) _global->_pPaletteDesc->vgaPal, 768);
|
|
|
|
oOff += 768;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (size < 0) {
|
|
|
|
srcDesc = _draw->_spritesArray[index];
|
|
|
|
destDesc = _video->initSurfDesc(_global->_videoMode, srcDesc->width, 25, 0);
|
|
|
|
for (top = 0, retSize = 0; top < srcDesc->height; top += 25) {
|
|
|
|
int16 height = MIN(25, srcDesc->height - top);
|
|
|
|
_video->drawSprite(srcDesc, destDesc, 0, top, srcDesc->width - 1,
|
|
|
|
top + height - 1, 0, 0, 0);
|
|
|
|
memcpy(oBuf + oOff, (char *) destDesc->vidPtr, srcDesc->width * 25);
|
|
|
|
oOff += srcDesc->width * 25;
|
|
|
|
}
|
|
|
|
_video->freeSurfDesc(destDesc);
|
|
|
|
} else
|
|
|
|
memcpy(oBuf + oOff, buf, size);
|
|
|
|
|
|
|
|
out->write(oBuf, oSize);
|
|
|
|
out->flush();
|
|
|
|
|
|
|
|
if (out->ioFailed())
|
|
|
|
warning("Can't write file \"%s\"", sName);
|
|
|
|
else {
|
|
|
|
debugC(1, DEBUG_FILEIO, "Saved file \"%s\" (%d, %d bytes at %d)",
|
|
|
|
sName, dataVar, size, offset);
|
|
|
|
// WRITE_VAR(1, 0)
|
|
|
|
*((uint32*)(_global->_inter_variables + 4)) = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
delete out;
|
|
|
|
delete[] oBuf;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GobEngine::loadGame(enum SaveFiles sFile, int16 dataVar, int32 size, int32 offset) {
|
|
|
|
int32 sSize;
|
|
|
|
int32 retSize;
|
2006-11-22 15:50:30 +00:00
|
|
|
int16 index = 0;
|
2006-11-19 17:52:52 +00:00
|
|
|
int16 y;
|
|
|
|
char *buf;
|
|
|
|
char *sName;
|
|
|
|
bool readPal;
|
|
|
|
Video::SurfaceDesc *destDesc;
|
|
|
|
Video::SurfaceDesc *srcDesc;
|
|
|
|
Common::InSaveFile *in;
|
|
|
|
|
|
|
|
readPal = false;
|
|
|
|
sName = _saveFiles[(int) sFile];
|
|
|
|
if (size < 0) {
|
|
|
|
if (size < -1000) {
|
|
|
|
readPal = true;
|
|
|
|
size += 1000;
|
|
|
|
}
|
|
|
|
index = -size - 1;
|
|
|
|
assert((index >= 0) && (index < 50)); // Just to be sure...
|
|
|
|
buf = (char *) _draw->_spritesArray[index]->vidPtr;
|
|
|
|
size = _draw->getSpriteRectSize(index);
|
|
|
|
if ((_draw->_spritesArray[index]->vidMode & 0x80) == 0)
|
|
|
|
size = -size;
|
|
|
|
} else if (size == 0) {
|
|
|
|
dataVar = 0;
|
|
|
|
size = READ_LE_UINT32(_game->_totFileData + 0x2C) * 4;
|
|
|
|
buf = _global->_inter_variables;
|
|
|
|
} else
|
|
|
|
buf = _global->_inter_variables + dataVar;
|
|
|
|
|
|
|
|
if (_global->_inter_resStr[0] == 0) {
|
|
|
|
if (readPal)
|
|
|
|
size += 768;
|
|
|
|
// WRITE_VAR(1, size)
|
|
|
|
*((uint32*)(_global->_inter_variables + 4)) = (uint32) size;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// WRITE_VAR(1, 1)
|
|
|
|
*((uint32*)(_global->_inter_variables + 4)) = 1;
|
|
|
|
if(!(in = _saveFileMan->openForLoading(sName)))
|
|
|
|
return;
|
|
|
|
|
|
|
|
debugC(1, DEBUG_FILEIO, "Loading file \"%s\" (%d, %d bytes at %d)",
|
|
|
|
sName, dataVar, size, offset);
|
|
|
|
|
|
|
|
sSize = getSaveSize(*in);
|
|
|
|
_draw->animateCursor(4);
|
|
|
|
if (offset < 0)
|
|
|
|
in->seek(sSize - (-offset - 1), 0);
|
|
|
|
else
|
|
|
|
in->seek(offset, 0);
|
|
|
|
|
|
|
|
if (readPal) {
|
|
|
|
retSize = in->read((char *) _global->_pPaletteDesc->vgaPal, 768);
|
|
|
|
_draw->_applyPal = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (size < 0) {
|
|
|
|
destDesc = _draw->_spritesArray[index];
|
|
|
|
srcDesc = _video->initSurfDesc(_global->_videoMode, destDesc->width, 25, 0);
|
|
|
|
for (y = 0, retSize = 0; y < destDesc->height; y += 25) {
|
|
|
|
int16 height = MIN(25, destDesc->height - y);
|
|
|
|
retSize += in->read((char *) srcDesc->vidPtr, destDesc->width * 25);
|
|
|
|
_video->drawSprite(srcDesc, destDesc, 0, 0, destDesc->width - 1, height - 1, 0, y, 0);
|
|
|
|
}
|
|
|
|
_video->freeSurfDesc(srcDesc);
|
|
|
|
} else
|
|
|
|
retSize = in->read(buf, size);
|
|
|
|
|
|
|
|
if (retSize == size)
|
|
|
|
// WRITE_VAR(1, 0)
|
|
|
|
*((uint32*)(_global->_inter_variables + 4)) = 0;
|
|
|
|
|
|
|
|
delete in;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2006-04-15 20:36:41 +00:00
|
|
|
int GobEngine::init() {
|
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);
|
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");
|
2006-05-29 18:24:52 +00:00
|
|
|
if ((_features & Gob::GF_MAC) || (_features & Gob::GF_GOB1) || (_features & Gob::GF_GOB2)) {
|
2006-07-23 15:08:48 +00:00
|
|
|
if (MidiDriver::parseMusicDriver(ConfMan.get("music_driver")) == MD_NULL)
|
2006-05-29 18:24:52 +00:00
|
|
|
_music = new Music_Dummy(this);
|
|
|
|
else
|
|
|
|
_music = new Music(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;
|
|
|
|
_global->_soundFlags = 0;
|
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;
|
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
|
2006-02-18 00:36:45 +00:00
|
|
|
|
|
|
|
using namespace Gob;
|
|
|
|
|
2006-02-18 11:15:37 +00:00
|
|
|
GameList Engine_GOB_gameIDList() {
|
2006-02-18 00:36:45 +00:00
|
|
|
GameList games;
|
2006-03-09 02:52:51 +00:00
|
|
|
const PlainGameDescriptor *g = gob_list;
|
2006-02-18 00:36:45 +00:00
|
|
|
|
|
|
|
while (g->gameid) {
|
|
|
|
games.push_back(*g);
|
|
|
|
g++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return games;
|
|
|
|
}
|
|
|
|
|
2006-03-09 02:52:51 +00:00
|
|
|
GameDescriptor Engine_GOB_findGameID(const char *gameid) {
|
|
|
|
const PlainGameDescriptor *g = gob_list;
|
2006-02-18 11:15:37 +00:00
|
|
|
while (g->gameid) {
|
2006-03-02 22:29:01 +00:00
|
|
|
if (0 == scumm_stricmp(gameid, g->gameid))
|
2006-02-18 11:15:37 +00:00
|
|
|
break;
|
|
|
|
g++;
|
|
|
|
}
|
|
|
|
return *g;
|
|
|
|
}
|
|
|
|
|
2006-02-18 00:36:45 +00:00
|
|
|
DetectedGameList Engine_GOB_detectGames(const FSList &fslist) {
|
|
|
|
DetectedGameList detectedGames;
|
2006-04-08 11:21:04 +00:00
|
|
|
const GameSettings *g;
|
2006-02-18 00:36:45 +00:00
|
|
|
FSList::const_iterator file;
|
|
|
|
|
|
|
|
// Iterate over all files in the given directory
|
|
|
|
for (file = fslist.begin(); file != fslist.end(); file++) {
|
|
|
|
if (file->isDirectory())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// All the supported games have an intro.stk file.
|
2006-07-22 14:36:09 +00:00
|
|
|
if (scumm_stricmp(file->name().c_str(), "intro.stk") == 0)
|
2006-02-18 00:36:45 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (file == fslist.end())
|
|
|
|
return detectedGames;
|
|
|
|
|
|
|
|
uint8 md5sum[16];
|
|
|
|
char md5str[32 + 1];
|
|
|
|
|
2006-07-22 14:59:44 +00:00
|
|
|
if (Common::md5_file(*file, md5sum, kMD5FileSizeLimit)) {
|
2006-02-18 00:36:45 +00:00
|
|
|
for (int i = 0; i < 16; i++) {
|
|
|
|
sprintf(md5str + i * 2, "%02x", (int)md5sum[i]);
|
|
|
|
}
|
|
|
|
for (g = gob_games; g->gameid; g++) {
|
|
|
|
if (strcmp(g->md5sum, (char *)md5str) == 0) {
|
2006-03-09 01:42:56 +00:00
|
|
|
detectedGames.push_back(DetectedGame(g->gameid, g->description));
|
2006-02-18 00:36:45 +00:00
|
|
|
}
|
|
|
|
}
|
2006-03-28 09:42:54 +00:00
|
|
|
if (detectedGames.empty()) {
|
2006-02-18 00:36:45 +00:00
|
|
|
printf("Unknown MD5 (%s)! Please report the details (language, platform, etc.) of this game to the ScummVM team\n", md5str);
|
|
|
|
|
2006-03-09 02:52:51 +00:00
|
|
|
const PlainGameDescriptor *g1 = gob_list;
|
2006-02-18 00:36:45 +00:00
|
|
|
while (g1->gameid) {
|
|
|
|
detectedGames.push_back(*g1);
|
|
|
|
g1++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return detectedGames;
|
|
|
|
}
|
|
|
|
|
2006-04-29 00:27:20 +00:00
|
|
|
PluginError Engine_GOB_create(OSystem *syst, Engine **engine) {
|
2006-02-18 00:36:45 +00:00
|
|
|
// Detect game features based on MD5
|
|
|
|
uint8 md5sum[16];
|
|
|
|
char md5str[32 + 1];
|
|
|
|
|
2006-04-16 12:50:39 +00:00
|
|
|
if (Common::md5_file("intro.stk", md5sum, kMD5FileSizeLimit)) {
|
2006-02-18 00:36:45 +00:00
|
|
|
for (int j = 0; j < 16; j++) {
|
|
|
|
sprintf(md5str + j*2, "%02x", (int)md5sum[j]);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
error("Engine_GOB_create(): Cannot find intro.stk");
|
|
|
|
}
|
|
|
|
|
2006-04-08 11:21:04 +00:00
|
|
|
const GameSettings *g;
|
2006-02-18 00:36:45 +00:00
|
|
|
bool found = false;
|
|
|
|
|
|
|
|
// TODO
|
|
|
|
// Fallback. Maybe we will be able to determine game type from game
|
|
|
|
// data contents
|
2006-04-15 20:36:41 +00:00
|
|
|
Common::String realGame(ConfMan.get("gameid"));
|
2006-02-18 00:36:45 +00:00
|
|
|
uint32 features;
|
2006-08-12 12:35:18 +00:00
|
|
|
const char *startTotBase=NULL;
|
|
|
|
|
2006-03-02 22:29:01 +00:00
|
|
|
if (!scumm_stricmp(realGame.c_str(), "gob2"))
|
2006-02-18 00:36:45 +00:00
|
|
|
features = GF_GOB2;
|
|
|
|
else
|
|
|
|
features = GF_GOB1;
|
|
|
|
|
|
|
|
for (g = gob_games; g->gameid; g++) {
|
|
|
|
if (strcmp(g->md5sum, (char *)md5str) == 0) {
|
|
|
|
features = g->features;
|
|
|
|
|
|
|
|
if (g->description)
|
|
|
|
g_system->setWindowCaption(g->description);
|
|
|
|
|
2006-08-12 12:35:18 +00:00
|
|
|
startTotBase = g->startTotBase;
|
2006-02-18 00:36:45 +00:00
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!found) {
|
|
|
|
printf("Unknown MD5 (%s)! Please report the details (language, platform, etc.) of this game to the ScummVM team\n", md5str);
|
|
|
|
}
|
|
|
|
|
2006-04-29 00:27:20 +00:00
|
|
|
assert(engine);
|
2006-08-12 12:35:18 +00:00
|
|
|
*engine = new GobEngine(syst, features, g->lang, startTotBase);
|
2006-04-29 00:27:20 +00:00
|
|
|
return kNoError;
|
2006-02-18 00:36:45 +00:00
|
|
|
}
|
|
|
|
|
2006-07-31 13:41:21 +00:00
|
|
|
REGISTER_PLUGIN(GOB, "Gob Engine", "Goblins Games (C) Coktel Vision");
|