2009-02-17 15:20:21 +00:00
|
|
|
/* ScummVM - Graphic Adventure Engine
|
|
|
|
*
|
|
|
|
* ScummVM is the legal property of its developers, whose names
|
|
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
|
|
* file distributed with this source distribution.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 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.
|
|
|
|
*
|
|
|
|
* $URL$
|
|
|
|
* $Id$
|
|
|
|
*
|
|
|
|
*/
|
2009-02-15 06:10:59 +00:00
|
|
|
|
|
|
|
#include "common/system.h"
|
|
|
|
#include "common/config-manager.h"
|
2010-05-04 11:59:22 +00:00
|
|
|
#include "common/debug-channels.h"
|
2010-07-30 22:47:01 +00:00
|
|
|
#include "common/EventRecorder.h"
|
2010-08-05 11:19:32 +00:00
|
|
|
#include "common/file.h" // for Common::File::exists()
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2009-02-15 08:20:53 +00:00
|
|
|
#include "engines/advancedDetector.h"
|
2010-05-04 11:58:12 +00:00
|
|
|
#include "engines/util.h"
|
|
|
|
|
2009-02-15 11:03:21 +00:00
|
|
|
#include "sci/sci.h"
|
2009-07-03 14:22:50 +00:00
|
|
|
#include "sci/debug.h"
|
2009-02-20 21:26:31 +00:00
|
|
|
#include "sci/console.h"
|
2009-12-04 17:38:24 +00:00
|
|
|
#include "sci/event.h"
|
2009-05-11 13:31:17 +00:00
|
|
|
|
2010-02-13 17:44:58 +00:00
|
|
|
#include "sci/engine/features.h"
|
2010-06-10 07:32:05 +00:00
|
|
|
#include "sci/engine/message.h"
|
2009-02-27 02:23:40 +00:00
|
|
|
#include "sci/engine/state.h"
|
2009-02-24 05:51:55 +00:00
|
|
|
#include "sci/engine/kernel.h"
|
2010-02-02 22:52:41 +00:00
|
|
|
#include "sci/engine/script.h" // for script_adjust_opcode_formats
|
2010-06-10 09:18:57 +00:00
|
|
|
#include "sci/engine/selector.h" // for SELECTOR
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2010-01-05 01:22:16 +00:00
|
|
|
#include "sci/sound/audio.h"
|
|
|
|
#include "sci/sound/soundcmd.h"
|
2010-06-15 15:44:24 +00:00
|
|
|
#include "sci/graphics/animate.h"
|
|
|
|
#include "sci/graphics/cache.h"
|
|
|
|
#include "sci/graphics/compare.h"
|
|
|
|
#include "sci/graphics/controls.h"
|
|
|
|
#include "sci/graphics/coordadjuster.h"
|
|
|
|
#include "sci/graphics/cursor.h"
|
2010-05-24 17:21:11 +00:00
|
|
|
#include "sci/graphics/maciconbar.h"
|
2010-06-09 21:41:20 +00:00
|
|
|
#include "sci/graphics/menu.h"
|
2010-06-15 13:34:40 +00:00
|
|
|
#include "sci/graphics/paint16.h"
|
2010-06-15 15:44:24 +00:00
|
|
|
#include "sci/graphics/paint32.h"
|
2010-07-24 16:47:12 +00:00
|
|
|
#include "sci/graphics/picture.h"
|
2010-01-31 12:35:15 +00:00
|
|
|
#include "sci/graphics/ports.h"
|
2010-01-05 01:37:57 +00:00
|
|
|
#include "sci/graphics/palette.h"
|
|
|
|
#include "sci/graphics/screen.h"
|
2010-06-15 15:44:24 +00:00
|
|
|
#include "sci/graphics/text16.h"
|
|
|
|
#include "sci/graphics/transitions.h"
|
2009-10-03 20:49:18 +00:00
|
|
|
|
2010-01-29 22:02:28 +00:00
|
|
|
#ifdef ENABLE_SCI32
|
2010-06-15 15:44:24 +00:00
|
|
|
#include "sci/graphics/frameout.h"
|
2010-01-29 22:02:28 +00:00
|
|
|
#endif
|
|
|
|
|
2009-02-20 14:45:28 +00:00
|
|
|
namespace Sci {
|
|
|
|
|
2010-02-13 17:42:49 +00:00
|
|
|
SciEngine *g_sci = 0;
|
|
|
|
|
|
|
|
|
2009-06-06 10:21:48 +00:00
|
|
|
class GfxDriver;
|
2009-02-21 10:23:36 +00:00
|
|
|
|
2010-06-25 16:16:29 +00:00
|
|
|
SciEngine::SciEngine(OSystem *syst, const ADGameDescription *desc, SciGameId gameId)
|
|
|
|
: Engine(syst), _gameDescription(desc), _gameId(gameId) {
|
2009-02-20 21:26:31 +00:00
|
|
|
|
2010-02-14 12:32:25 +00:00
|
|
|
assert(g_sci == 0);
|
2010-02-13 17:42:49 +00:00
|
|
|
g_sci = this;
|
2010-06-15 12:15:52 +00:00
|
|
|
|
|
|
|
_gfxMacIconBar = 0;
|
|
|
|
|
|
|
|
_audio = 0;
|
2010-02-13 17:44:58 +00:00
|
|
|
_features = 0;
|
2010-06-15 12:15:52 +00:00
|
|
|
_resMan = 0;
|
|
|
|
_gamestate = 0;
|
|
|
|
_kernel = 0;
|
|
|
|
_vocabulary = 0;
|
2010-07-19 15:32:26 +00:00
|
|
|
_vocabularyLanguage = 1; // we load english vocabulary on startup
|
2010-06-15 12:15:52 +00:00
|
|
|
_eventMan = 0;
|
|
|
|
_console = 0;
|
2010-02-13 17:42:49 +00:00
|
|
|
|
2009-02-17 18:16:48 +00:00
|
|
|
// Set up the engine specific debug levels
|
2010-04-27 21:40:52 +00:00
|
|
|
DebugMan.addDebugChannel(kDebugLevelError, "Error", "Script error debugging");
|
|
|
|
DebugMan.addDebugChannel(kDebugLevelNodes, "Lists", "Lists and nodes debugging");
|
|
|
|
DebugMan.addDebugChannel(kDebugLevelGraphics, "Graphics", "Graphics debugging");
|
|
|
|
DebugMan.addDebugChannel(kDebugLevelStrings, "Strings", "Strings debugging");
|
|
|
|
DebugMan.addDebugChannel(kDebugLevelMemory, "Memory", "Memory debugging");
|
|
|
|
DebugMan.addDebugChannel(kDebugLevelFuncCheck, "Func", "Function parameter debugging");
|
|
|
|
DebugMan.addDebugChannel(kDebugLevelBresen, "Bresenham", "Bresenham algorithms debugging");
|
|
|
|
DebugMan.addDebugChannel(kDebugLevelSound, "Sound", "Sound debugging");
|
|
|
|
DebugMan.addDebugChannel(kDebugLevelBaseSetter, "Base", "Base Setter debugging");
|
|
|
|
DebugMan.addDebugChannel(kDebugLevelParser, "Parser", "Parser debugging");
|
|
|
|
DebugMan.addDebugChannel(kDebugLevelSaid, "Said", "Said specs debugging");
|
|
|
|
DebugMan.addDebugChannel(kDebugLevelFile, "File", "File I/O debugging");
|
|
|
|
DebugMan.addDebugChannel(kDebugLevelTime, "Time", "Time debugging");
|
|
|
|
DebugMan.addDebugChannel(kDebugLevelRoom, "Room", "Room number debugging");
|
|
|
|
DebugMan.addDebugChannel(kDebugLevelAvoidPath, "Pathfinding", "Pathfinding debugging");
|
|
|
|
DebugMan.addDebugChannel(kDebugLevelDclInflate, "DCL", "DCL inflate debugging");
|
|
|
|
DebugMan.addDebugChannel(kDebugLevelVM, "VM", "VM debugging");
|
|
|
|
DebugMan.addDebugChannel(kDebugLevelScripts, "Scripts", "Notifies when scripts are unloaded");
|
|
|
|
DebugMan.addDebugChannel(kDebugLevelGC, "GC", "Garbage Collector debugging");
|
|
|
|
DebugMan.addDebugChannel(kDebugLevelResMan, "ResMan", "Resource manager debugging");
|
|
|
|
DebugMan.addDebugChannel(kDebugLevelOnStartup, "OnStartup", "Enter debugger at start of game");
|
2009-02-17 18:16:48 +00:00
|
|
|
|
2010-05-04 11:56:52 +00:00
|
|
|
const Common::FSNode gameDataDir(ConfMan.get("path"));
|
|
|
|
|
|
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "actors"); // KQ6 hi-res portraits
|
|
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "aud"); // resource.aud and audio files
|
2010-08-01 10:07:24 +00:00
|
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "audio");// resource.aud and audio files
|
2010-08-01 16:23:44 +00:00
|
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "audiosfx");// resource.aud and audio files
|
2010-05-04 11:56:52 +00:00
|
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "wav"); // speech files in WAV format
|
|
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "sfx"); // music/sound files in WAV format
|
2010-06-16 23:30:22 +00:00
|
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "avi"); // AVI movie files for Windows versions
|
|
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "seq"); // SEQ movie files for DOS versions
|
|
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "robot"); // robot movie files
|
2010-07-28 06:03:52 +00:00
|
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "robots"); // robot movie files
|
2010-08-04 15:17:09 +00:00
|
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "movie"); // vmd movie files
|
2010-06-17 00:07:03 +00:00
|
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "movies"); // vmd movie files
|
2010-06-16 23:30:22 +00:00
|
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "vmd"); // vmd movie files
|
2010-02-05 03:29:04 +00:00
|
|
|
|
2010-02-05 09:02:36 +00:00
|
|
|
// Add the patches directory, except for KQ6CD; The patches folder in some versions of KQ6CD
|
|
|
|
// is for the demo of Phantasmagoria, included in the disk
|
2010-06-25 16:16:29 +00:00
|
|
|
if (_gameId != GID_KQ6)
|
2010-05-04 11:56:52 +00:00
|
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "patches"); // resource patches
|
2009-02-17 18:16:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SciEngine::~SciEngine() {
|
|
|
|
// Remove all of our debug levels here
|
2010-04-27 21:40:52 +00:00
|
|
|
DebugMan.clearAllDebugChannels();
|
2009-02-20 21:26:31 +00:00
|
|
|
|
2010-06-28 08:18:55 +00:00
|
|
|
#ifdef ENABLE_SCI32
|
|
|
|
delete _gfxFrameout;
|
|
|
|
#endif
|
|
|
|
delete _gfxMenu;
|
|
|
|
delete _gfxControls;
|
|
|
|
delete _gfxText16;
|
|
|
|
delete _gfxAnimate;
|
|
|
|
delete _gfxPaint;
|
|
|
|
delete _gfxTransitions;
|
|
|
|
delete _gfxCompare;
|
|
|
|
delete _gfxCoordAdjuster;
|
|
|
|
delete _gfxPorts;
|
|
|
|
delete _gfxCache;
|
|
|
|
delete _gfxPalette;
|
|
|
|
delete _gfxCursor;
|
|
|
|
delete _gfxScreen;
|
|
|
|
|
2009-11-04 09:36:18 +00:00
|
|
|
delete _audio;
|
2010-06-29 09:00:08 +00:00
|
|
|
delete _soundCmd;
|
2009-07-11 23:45:54 +00:00
|
|
|
delete _kernel;
|
|
|
|
delete _vocabulary;
|
2009-02-20 21:26:31 +00:00
|
|
|
delete _console;
|
2010-02-13 17:44:58 +00:00
|
|
|
delete _features;
|
2010-05-24 21:51:45 +00:00
|
|
|
delete _gfxMacIconBar;
|
2010-02-13 17:42:49 +00:00
|
|
|
|
2010-06-28 08:18:55 +00:00
|
|
|
delete _eventMan;
|
|
|
|
delete _gamestate->_segMan;
|
|
|
|
delete _gamestate;
|
|
|
|
delete _resMan; // should be deleted last
|
2010-02-13 17:42:49 +00:00
|
|
|
g_sci = 0;
|
2009-02-17 18:16:48 +00:00
|
|
|
}
|
|
|
|
|
2010-08-17 20:36:28 +00:00
|
|
|
extern void showScummVMDialog(const Common::String &message);
|
|
|
|
|
2009-03-01 04:42:46 +00:00
|
|
|
Common::Error SciEngine::run() {
|
2010-07-30 22:47:01 +00:00
|
|
|
g_eventRec.registerRandomSource(_rng, "sci");
|
|
|
|
|
2010-01-26 19:25:33 +00:00
|
|
|
// Assign default values to the config manager, in case settings are missing
|
|
|
|
ConfMan.registerDefault("undither", "true");
|
|
|
|
ConfMan.registerDefault("enable_fb01", "false");
|
|
|
|
|
2009-09-02 12:02:37 +00:00
|
|
|
_resMan = new ResourceManager();
|
2010-06-15 12:15:52 +00:00
|
|
|
assert(_resMan);
|
|
|
|
_resMan->addAppropriateSources();
|
|
|
|
_resMan->init();
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2010-06-15 12:15:52 +00:00
|
|
|
// TODO: Add error handling. Check return values of addAppropriateSources
|
|
|
|
// and init. We first have to *add* sensible return values, though ;).
|
|
|
|
/*
|
2009-09-02 12:02:37 +00:00
|
|
|
if (!_resMan) {
|
2009-10-10 00:07:19 +00:00
|
|
|
warning("No resources found, aborting");
|
2009-02-17 18:16:48 +00:00
|
|
|
return Common::kNoGameDataFoundError;
|
2009-02-15 06:10:59 +00:00
|
|
|
}
|
2010-06-15 12:15:52 +00:00
|
|
|
*/
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2010-07-03 06:55:10 +00:00
|
|
|
// Reset, so that error()s before SoundCommandParser is initialized wont cause a crash
|
|
|
|
_soundCmd = NULL;
|
|
|
|
|
2010-06-09 18:42:21 +00:00
|
|
|
// Add the after market GM patches for the specified game, if they exist
|
2010-06-17 23:14:34 +00:00
|
|
|
_resMan->addNewGMPatch(_gameId);
|
2010-08-23 20:29:13 +00:00
|
|
|
_gameObjectAddress = _resMan->findGameObject();
|
|
|
|
_gameSuperClassAddress = NULL_REG;
|
2010-06-09 18:42:21 +00:00
|
|
|
|
2010-05-18 12:16:48 +00:00
|
|
|
SegManager *segMan = new SegManager(_resMan);
|
|
|
|
|
2010-06-28 08:18:55 +00:00
|
|
|
// Initialize the game screen
|
|
|
|
_gfxScreen = new GfxScreen(_resMan);
|
2010-06-09 18:42:21 +00:00
|
|
|
_gfxScreen->debugUnditherSetState(ConfMan.getBool("undither"));
|
|
|
|
|
2009-10-31 10:54:19 +00:00
|
|
|
// Create debugger console. It requires GFX to be initialized
|
|
|
|
_console = new Console(this);
|
2010-05-18 12:16:48 +00:00
|
|
|
_kernel = new Kernel(_resMan, segMan);
|
2010-08-23 20:29:13 +00:00
|
|
|
|
2010-06-09 18:42:21 +00:00
|
|
|
_features = new GameFeatures(segMan, _kernel);
|
2010-08-07 08:59:43 +00:00
|
|
|
// Only SCI0, SCI01 and SCI1 EGA games used a parser
|
2010-07-19 13:50:06 +00:00
|
|
|
_vocabulary = (getSciVersion() <= SCI_VERSION_1_EGA) ? new Vocabulary(_resMan, false) : NULL;
|
2010-08-07 08:59:43 +00:00
|
|
|
// Also, XMAS1990 apparently had a parser too. Refer to http://forums.scummvm.org/viewtopic.php?t=9135
|
|
|
|
if (getGameId() == GID_CHRISTMAS1990)
|
|
|
|
_vocabulary = new Vocabulary(_resMan, false);
|
2009-11-04 09:36:18 +00:00
|
|
|
_audio = new AudioPlayer(_resMan);
|
2010-06-01 15:11:20 +00:00
|
|
|
_gamestate = new EngineState(segMan);
|
2010-06-17 23:11:34 +00:00
|
|
|
_eventMan = new EventManager(_resMan->detectFontExtended());
|
2009-10-08 08:00:30 +00:00
|
|
|
|
2010-06-09 21:41:20 +00:00
|
|
|
// The game needs to be initialized before the graphics system is initialized, as
|
|
|
|
// the graphics code checks parts of the seg manager upon initialization (e.g. for
|
|
|
|
// the presence of the fastCast object)
|
2010-06-10 07:32:05 +00:00
|
|
|
if (!initGame()) { /* Initialize */
|
2010-06-09 21:41:20 +00:00
|
|
|
warning("Game initialization failed: Aborting...");
|
|
|
|
// TODO: Add an "init failed" error?
|
|
|
|
return Common::kUnknownError;
|
|
|
|
}
|
|
|
|
|
2010-08-23 20:29:13 +00:00
|
|
|
// we try to find the super class address of the game object, we can't do that earlier
|
|
|
|
const Object *gameObject = segMan->getObject(_gameObjectAddress);
|
|
|
|
if (!gameObject) {
|
|
|
|
warning("Could not get game object, aborting...");
|
|
|
|
return Common::kUnknownError;
|
|
|
|
}
|
|
|
|
_gameSuperClassAddress = gameObject->getSuperClassSelector();
|
|
|
|
|
2010-06-15 07:21:52 +00:00
|
|
|
script_adjust_opcode_formats();
|
2009-08-30 14:53:58 +00:00
|
|
|
|
2010-06-29 09:00:08 +00:00
|
|
|
// Must be called after game_init(), as they use _features
|
|
|
|
_kernel->loadKernelNames(_features);
|
|
|
|
_soundCmd = new SoundCommandParser(_resMan, segMan, _kernel, _audio, _features->detectDoSoundType());
|
2009-11-12 15:24:11 +00:00
|
|
|
|
2010-01-12 00:51:37 +00:00
|
|
|
syncSoundSettings();
|
|
|
|
|
2010-06-28 08:18:55 +00:00
|
|
|
// Initialize all graphics related subsystems
|
2010-06-15 13:34:40 +00:00
|
|
|
initGraphics();
|
2009-10-03 20:49:18 +00:00
|
|
|
|
2010-02-13 11:58:15 +00:00
|
|
|
debug("Emulating SCI version %s\n", getSciVersionDesc(getSciVersion()));
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2010-08-24 08:31:57 +00:00
|
|
|
// Patch in our save/restore code, so that dialogs are replaced
|
|
|
|
patchGameSaveRestore(segMan);
|
2010-08-23 23:04:07 +00:00
|
|
|
|
2010-08-01 22:41:06 +00:00
|
|
|
if (_gameDescription->flags & ADGF_ADDENGLISH) {
|
|
|
|
// if game is multilingual
|
2010-08-02 08:47:06 +00:00
|
|
|
Common::Language selectedLanguage = Common::parseLanguage(ConfMan.get("language"));
|
|
|
|
if (selectedLanguage == Common::EN_ANY) {
|
2010-08-01 22:41:06 +00:00
|
|
|
// and english was selected as language
|
|
|
|
if (SELECTOR(printLang) != -1) // set text language to english
|
2010-08-23 20:29:13 +00:00
|
|
|
writeSelectorValue(segMan, _gameObjectAddress, SELECTOR(printLang), 1);
|
2010-08-01 22:41:06 +00:00
|
|
|
if (SELECTOR(parseLang) != -1) // and set parser language to english as well
|
2010-08-23 20:29:13 +00:00
|
|
|
writeSelectorValue(segMan, _gameObjectAddress, SELECTOR(parseLang), 1);
|
2010-08-01 22:41:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-01-30 19:08:00 +00:00
|
|
|
// Check whether loading a savestate was requested
|
2010-06-28 20:58:32 +00:00
|
|
|
int saveSlot = ConfMan.getInt("save_slot");
|
|
|
|
if (saveSlot >= 0) {
|
|
|
|
reg_t restoreArgv[2] = { NULL_REG, make_reg(0, saveSlot) }; // special call (argv[0] is NULL)
|
2010-06-27 23:20:08 +00:00
|
|
|
kRestoreGame(_gamestate, 2, restoreArgv);
|
|
|
|
|
2010-08-05 11:19:32 +00:00
|
|
|
// TODO: The best way to do the following would be to invoke Game::init
|
|
|
|
// here and stop when the room is about to be changed, otherwise some
|
|
|
|
// game initialization won't take place
|
|
|
|
|
|
|
|
// Set audio language for KQ5CD (bug #3039477)
|
|
|
|
if (g_sci->getGameId() == GID_KQ5 && Common::File::exists("AUDIO001.002")) {
|
|
|
|
reg_t doAudioArgv[2] = { make_reg(0, 9), make_reg(0, 1) };
|
|
|
|
kDoAudio(_gamestate, 2, doAudioArgv);
|
|
|
|
}
|
|
|
|
|
2010-06-27 23:20:08 +00:00
|
|
|
// Initialize the game menu, if there is one.
|
|
|
|
// This is not done when loading, so we must do it manually.
|
|
|
|
reg_t menuBarObj = _gamestate->_segMan->findObjectByName("MenuBar");
|
2010-06-28 22:01:26 +00:00
|
|
|
if (menuBarObj.isNull())
|
|
|
|
menuBarObj = _gamestate->_segMan->findObjectByName("TheMenuBar"); // LSL2
|
2010-06-27 23:20:08 +00:00
|
|
|
if (menuBarObj.isNull())
|
|
|
|
menuBarObj = _gamestate->_segMan->findObjectByName("menuBar"); // LSL6
|
|
|
|
if (!menuBarObj.isNull()) {
|
2010-06-28 22:01:26 +00:00
|
|
|
// Reset abortScriptProcessing before initializing the game menu, so that the
|
|
|
|
// VM call performed by invokeSelector will actually run.
|
|
|
|
_gamestate->abortScriptProcessing = kAbortNone;
|
|
|
|
Object *menuBar = _gamestate->_segMan->getObject(menuBarObj);
|
|
|
|
// Invoke the first method (init) of the menuBar object
|
|
|
|
invokeSelector(_gamestate, menuBarObj, menuBar->getFuncSelector(0), 0, _gamestate->stack_base);
|
|
|
|
_gamestate->abortScriptProcessing = kAbortLoadGame;
|
2010-06-27 23:20:08 +00:00
|
|
|
}
|
2010-01-30 19:08:00 +00:00
|
|
|
}
|
|
|
|
|
2010-08-17 20:36:28 +00:00
|
|
|
// Show any special warnings for buggy scripts with severe game bugs,
|
|
|
|
// which have been patched by Sierra
|
|
|
|
if (getGameId() == GID_LONGBOW) {
|
|
|
|
// Longbow 1.0 has a buggy script which prevents the game
|
|
|
|
// from progressing during the Green Man riddle sequence.
|
|
|
|
// A patch for this buggy script has been released by Sierra,
|
|
|
|
// and is necessary to complete the game without issues.
|
|
|
|
// The patched script is included in Longbow 1.1.
|
|
|
|
// Refer to bug #3036609.
|
|
|
|
Resource *buggyScript = _resMan->findResource(ResourceId(kResourceTypeScript, 180), 0);
|
|
|
|
|
|
|
|
if (buggyScript->size == 12354 || buggyScript->size == 12362) {
|
|
|
|
showScummVMDialog("A known buggy game script has been detected, which could "
|
|
|
|
"prevent you from progressing later on in the game, during "
|
|
|
|
"the sequence with the Green Man's riddles. Please, apply "
|
|
|
|
"the latest patch for this game by Sierra to avoid possible "
|
|
|
|
"problems");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-06-10 07:32:05 +00:00
|
|
|
runGame();
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2010-01-12 00:51:37 +00:00
|
|
|
ConfMan.flushToDisk();
|
|
|
|
|
2009-02-15 08:20:53 +00:00
|
|
|
return Common::kNoError;
|
2009-02-15 06:10:59 +00:00
|
|
|
}
|
|
|
|
|
2010-08-23 23:04:07 +00:00
|
|
|
static byte patchGameRestore[] = {
|
|
|
|
0x39, 0x02, // pushi 02
|
|
|
|
0x76, // push0
|
|
|
|
0x38, 0xff, 0xff, // pushi -1
|
|
|
|
0x43, 0xff, 0x04, // call kRestoreGame (will get fixed directly)
|
|
|
|
0x48, // ret
|
|
|
|
};
|
|
|
|
|
|
|
|
void SciEngine::patchGameSaveRestore(SegManager *segMan) {
|
|
|
|
const Object *gameSuperObject = segMan->getObject(_gameSuperClassAddress);
|
|
|
|
const uint16 gameSuperMethodCount = gameSuperObject->getMethodCount();
|
|
|
|
reg_t methodAddress;
|
|
|
|
Script *script = NULL;
|
|
|
|
const byte *scriptRestorePtr = NULL;
|
|
|
|
const uint16 kernelCount = _kernel->getKernelNamesSize();
|
|
|
|
byte kernelIdRestore = 0;
|
|
|
|
|
2010-08-24 08:31:57 +00:00
|
|
|
// DISABLE NEXT LINE FOR REPLACING SIERRA GAME RESTORE DIALOG WITH SCUMMVM RESTORE
|
|
|
|
return;
|
|
|
|
|
2010-08-23 23:04:07 +00:00
|
|
|
for (uint16 kernelNr = 0; kernelNr < kernelCount; kernelNr++) {
|
|
|
|
Common::String kernelName = _kernel->getKernelName(kernelNr);
|
|
|
|
if (kernelName == "RestoreGame")
|
|
|
|
kernelIdRestore = kernelNr;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (uint16 methodNr = 0; methodNr < gameSuperMethodCount; methodNr++) {
|
|
|
|
uint16 selectorId = gameSuperObject->getFuncSelector(methodNr);
|
|
|
|
Common::String methodName = _kernel->getSelectorName(selectorId);
|
|
|
|
if (methodName == "restore") {
|
|
|
|
methodAddress = gameSuperObject->getFunction(methodNr);
|
|
|
|
script = segMan->getScript(methodAddress.segment);
|
|
|
|
scriptRestorePtr = script->getBuf(methodAddress.offset);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (scriptRestorePtr) {
|
|
|
|
// Now patch in our code
|
|
|
|
byte *patchPtr = (byte *)scriptRestorePtr;
|
|
|
|
memcpy(patchPtr, patchGameRestore, sizeof(patchGameRestore));
|
|
|
|
patchPtr[7] = kernelIdRestore;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-06-10 07:32:05 +00:00
|
|
|
bool SciEngine::initGame() {
|
|
|
|
// Script 0 needs to be allocated here before anything else!
|
|
|
|
int script0Segment = _gamestate->_segMan->getScriptSegment(0, SCRIPT_GET_LOCK);
|
|
|
|
DataStack *stack = _gamestate->_segMan->allocateStack(VM_STACK_SIZE, NULL);
|
|
|
|
|
|
|
|
_gamestate->_msgState = new MessageState(_gamestate->_segMan);
|
2010-06-10 11:43:20 +00:00
|
|
|
_gamestate->gcCountDown = GC_INTERVAL - 1;
|
2010-06-10 07:32:05 +00:00
|
|
|
|
|
|
|
// Script 0 should always be at segment 1
|
|
|
|
if (script0Segment != 1) {
|
|
|
|
debug(2, "Failed to instantiate script.000");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
_gamestate->initGlobals();
|
|
|
|
_gamestate->_segMan->initSysStrings();
|
|
|
|
|
|
|
|
_gamestate->r_acc = _gamestate->r_prev = NULL_REG;
|
|
|
|
|
|
|
|
_gamestate->_executionStack.clear(); // Start without any execution stack
|
2010-06-10 11:18:10 +00:00
|
|
|
_gamestate->executionStackBase = -1; // No vm is running yet
|
2010-06-10 07:32:05 +00:00
|
|
|
_gamestate->_executionStackPosChanged = false;
|
|
|
|
|
|
|
|
_gamestate->abortScriptProcessing = kAbortNone;
|
2010-07-31 14:09:42 +00:00
|
|
|
_gamestate->gameIsRestarting = GAMEISRESTARTING_NONE;
|
2010-06-10 07:32:05 +00:00
|
|
|
|
|
|
|
_gamestate->stack_base = stack->_entries;
|
|
|
|
_gamestate->stack_top = stack->_entries + stack->_capacity;
|
|
|
|
|
2010-06-27 21:18:19 +00:00
|
|
|
if (!_gamestate->_segMan->instantiateScript(0)) {
|
2010-06-17 23:45:38 +00:00
|
|
|
error("initGame(): Could not instantiate script 0");
|
2010-06-10 07:32:05 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reset parser
|
|
|
|
if (_vocabulary) {
|
2010-07-19 15:30:27 +00:00
|
|
|
_vocabulary->reset();
|
2010-06-10 07:32:05 +00:00
|
|
|
}
|
|
|
|
|
2010-06-14 08:36:52 +00:00
|
|
|
_gamestate->gameStartTime = _gamestate->lastWaitTime = _gamestate->_screenUpdateTime = g_system->getMillis();
|
2010-06-10 07:32:05 +00:00
|
|
|
|
|
|
|
// Load game language into printLang property of game object
|
2010-06-10 11:18:10 +00:00
|
|
|
setSciLanguage();
|
2010-06-10 07:32:05 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-06-15 13:34:40 +00:00
|
|
|
void SciEngine::initGraphics() {
|
2010-06-28 08:18:55 +00:00
|
|
|
|
|
|
|
// Reset all graphics objects
|
|
|
|
_gfxAnimate = 0;
|
|
|
|
_gfxCache = 0;
|
|
|
|
_gfxCompare = 0;
|
|
|
|
_gfxControls = 0;
|
|
|
|
_gfxCoordAdjuster = 0;
|
|
|
|
_gfxCursor = 0;
|
|
|
|
_gfxMacIconBar = 0;
|
|
|
|
_gfxMenu = 0;
|
|
|
|
_gfxPaint = 0;
|
|
|
|
_gfxPaint16 = 0;
|
|
|
|
_gfxPalette = 0;
|
|
|
|
_gfxPorts = 0;
|
|
|
|
_gfxText16 = 0;
|
|
|
|
_gfxTransitions = 0;
|
|
|
|
#ifdef ENABLE_SCI32
|
|
|
|
_gfxFrameout = 0;
|
|
|
|
_gfxPaint32 = 0;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (_resMan->isSci11Mac() && getSciVersion() == SCI_VERSION_1_1)
|
|
|
|
_gfxMacIconBar = new GfxMacIconBar();
|
|
|
|
|
|
|
|
bool paletteMerging = true;
|
|
|
|
if (getSciVersion() >= SCI_VERSION_1_1) {
|
|
|
|
// there are some games that use inbetween SCI1.1 interpreter, so we have to detect if it's merging or copying
|
|
|
|
if (getSciVersion() == SCI_VERSION_1_1)
|
|
|
|
paletteMerging = _resMan->detectForPaletteMergingForSci11();
|
|
|
|
else
|
|
|
|
paletteMerging = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
_gfxPalette = new GfxPalette(_resMan, _gfxScreen, paletteMerging);
|
|
|
|
_gfxCache = new GfxCache(_resMan, _gfxScreen, _gfxPalette);
|
|
|
|
_gfxCursor = new GfxCursor(_resMan, _gfxPalette, _gfxScreen);
|
|
|
|
|
|
|
|
#ifdef ENABLE_SCI32
|
|
|
|
if (getSciVersion() >= SCI_VERSION_2) {
|
|
|
|
// SCI32 graphic objects creation
|
|
|
|
_gfxCoordAdjuster = new GfxCoordAdjuster32(_gamestate->_segMan);
|
|
|
|
_gfxCursor->init(_gfxCoordAdjuster, _eventMan);
|
|
|
|
_gfxCompare = new GfxCompare(_gamestate->_segMan, g_sci->getKernel(), _gfxCache, _gfxScreen, _gfxCoordAdjuster);
|
|
|
|
_gfxPaint32 = new GfxPaint32(g_sci->getResMan(), _gamestate->_segMan, g_sci->getKernel(), _gfxCoordAdjuster, _gfxCache, _gfxScreen, _gfxPalette);
|
|
|
|
_gfxPaint = _gfxPaint32;
|
|
|
|
_gfxFrameout = new GfxFrameout(_gamestate->_segMan, g_sci->getResMan(), _gfxCoordAdjuster, _gfxCache, _gfxScreen, _gfxPalette, _gfxPaint32);
|
|
|
|
} else {
|
|
|
|
#endif
|
|
|
|
// SCI0-SCI1.1 graphic objects creation
|
|
|
|
_gfxPorts = new GfxPorts(_gamestate->_segMan, _gfxScreen);
|
|
|
|
_gfxCoordAdjuster = new GfxCoordAdjuster16(_gfxPorts);
|
|
|
|
_gfxCursor->init(_gfxCoordAdjuster, g_sci->getEventManager());
|
|
|
|
_gfxCompare = new GfxCompare(_gamestate->_segMan, g_sci->getKernel(), _gfxCache, _gfxScreen, _gfxCoordAdjuster);
|
|
|
|
_gfxTransitions = new GfxTransitions(_gfxScreen, _gfxPalette, g_sci->getResMan()->isVGA());
|
|
|
|
_gfxPaint16 = new GfxPaint16(g_sci->getResMan(), _gamestate->_segMan, g_sci->getKernel(), _gfxCache, _gfxPorts, _gfxCoordAdjuster, _gfxScreen, _gfxPalette, _gfxTransitions, _audio);
|
|
|
|
_gfxPaint = _gfxPaint16;
|
|
|
|
_gfxAnimate = new GfxAnimate(_gamestate, _gfxCache, _gfxPorts, _gfxPaint16, _gfxScreen, _gfxPalette, _gfxCursor, _gfxTransitions);
|
|
|
|
_gfxText16 = new GfxText16(g_sci->getResMan(), _gfxCache, _gfxPorts, _gfxPaint16, _gfxScreen);
|
|
|
|
_gfxControls = new GfxControls(_gamestate->_segMan, _gfxPorts, _gfxPaint16, _gfxText16, _gfxScreen);
|
|
|
|
_gfxMenu = new GfxMenu(g_sci->getEventManager(), _gamestate->_segMan, _gfxPorts, _gfxPaint16, _gfxText16, _gfxScreen, _gfxCursor);
|
|
|
|
|
|
|
|
_gfxMenu->reset();
|
|
|
|
#ifdef ENABLE_SCI32
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2010-06-15 13:34:40 +00:00
|
|
|
if (_gfxPorts) {
|
|
|
|
_gfxPorts->init(_features->usesOldGfxFunctions(), _gfxPaint16, _gfxText16);
|
|
|
|
_gfxPaint16->init(_gfxAnimate, _gfxText16);
|
|
|
|
}
|
|
|
|
// Set default (EGA, amiga or resource 999) palette
|
|
|
|
_gfxPalette->setDefault();
|
|
|
|
}
|
|
|
|
|
2010-06-10 07:32:05 +00:00
|
|
|
void SciEngine::initStackBaseWithSelector(Selector selector) {
|
|
|
|
_gamestate->stack_base[0] = make_reg(0, (uint16)selector);
|
|
|
|
_gamestate->stack_base[1] = NULL_REG;
|
|
|
|
|
|
|
|
// Register the first element on the execution stack
|
2010-08-23 20:29:13 +00:00
|
|
|
if (!send_selector(_gamestate, _gameObjectAddress, _gameObjectAddress, _gamestate->stack_base, 2, _gamestate->stack_base)) {
|
|
|
|
_console->printObject(_gameObjectAddress);
|
2010-06-10 07:32:05 +00:00
|
|
|
error("initStackBaseWithSelector: error while registering the first selector in the call stack");
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void SciEngine::runGame() {
|
2010-06-10 09:18:57 +00:00
|
|
|
initStackBaseWithSelector(SELECTOR(play)); // Call the play selector
|
2010-06-10 07:32:05 +00:00
|
|
|
|
|
|
|
// Attach the debug console on game startup, if requested
|
|
|
|
if (DebugMan.isDebugChannelEnabled(kDebugLevelOnStartup))
|
|
|
|
_console->attach();
|
|
|
|
|
|
|
|
do {
|
|
|
|
_gamestate->_executionStackPosChanged = false;
|
2010-07-20 23:15:07 +00:00
|
|
|
run_vm(_gamestate);
|
2010-06-10 07:32:05 +00:00
|
|
|
exitGame();
|
|
|
|
|
|
|
|
if (_gamestate->abortScriptProcessing == kAbortRestartGame) {
|
|
|
|
_gamestate->_segMan->resetSegMan();
|
|
|
|
initGame();
|
2010-06-10 09:18:57 +00:00
|
|
|
initStackBaseWithSelector(SELECTOR(play));
|
2010-08-24 08:31:57 +00:00
|
|
|
patchGameSaveRestore(_gamestate->_segMan);
|
2010-07-31 14:09:42 +00:00
|
|
|
_gamestate->gameIsRestarting = GAMEISRESTARTING_RESTART;
|
2010-07-26 13:40:07 +00:00
|
|
|
if (_gfxMenu)
|
|
|
|
_gfxMenu->reset();
|
|
|
|
_gamestate->abortScriptProcessing = kAbortNone;
|
2010-06-10 07:32:05 +00:00
|
|
|
} else if (_gamestate->abortScriptProcessing == kAbortLoadGame) {
|
|
|
|
_gamestate->abortScriptProcessing = kAbortNone;
|
2010-07-22 12:38:48 +00:00
|
|
|
_gamestate->_executionStack.clear();
|
2010-06-10 09:18:57 +00:00
|
|
|
initStackBaseWithSelector(SELECTOR(replay));
|
2010-08-24 08:31:57 +00:00
|
|
|
patchGameSaveRestore(_gamestate->_segMan);
|
2010-07-22 12:38:48 +00:00
|
|
|
_gamestate->shrinkStackToBase();
|
2010-07-26 13:40:07 +00:00
|
|
|
_gamestate->abortScriptProcessing = kAbortNone;
|
2010-06-10 07:32:05 +00:00
|
|
|
} else {
|
|
|
|
break; // exit loop
|
|
|
|
}
|
|
|
|
} while (true);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SciEngine::exitGame() {
|
|
|
|
if (_gamestate->abortScriptProcessing != kAbortLoadGame) {
|
|
|
|
_gamestate->_executionStack.clear();
|
|
|
|
_audio->stopAllAudio();
|
2010-06-29 09:00:08 +00:00
|
|
|
g_sci->_soundCmd->clearPlayList();
|
2010-06-10 07:32:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// TODO Free parser segment here
|
|
|
|
|
|
|
|
// TODO Free scripts here
|
|
|
|
|
|
|
|
// Close all opened file handles
|
|
|
|
_gamestate->_fileHandles.clear();
|
|
|
|
_gamestate->_fileHandles.resize(5);
|
|
|
|
}
|
|
|
|
|
2009-07-03 14:22:50 +00:00
|
|
|
// Invoked by error() when a severe error occurs
|
2009-05-11 13:31:17 +00:00
|
|
|
GUI::Debugger *SciEngine::getDebugger() {
|
2009-07-07 06:53:53 +00:00
|
|
|
if (_gamestate) {
|
|
|
|
ExecStack *xs = &(_gamestate->_executionStack.back());
|
2010-07-12 23:20:33 +00:00
|
|
|
xs->addr.pc.offset = _debugState.old_pc_offset;
|
|
|
|
xs->sp = _debugState.old_sp;
|
2009-07-07 06:53:53 +00:00
|
|
|
}
|
|
|
|
|
2010-07-12 23:20:33 +00:00
|
|
|
_debugState.runningStep = 0; // Stop multiple execution
|
|
|
|
_debugState.seeking = kDebugSeekNothing; // Stop special seeks
|
2009-07-03 14:22:50 +00:00
|
|
|
|
|
|
|
return _console;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Used to obtain the engine's console in order to print messages to it
|
|
|
|
Console *SciEngine::getSciDebugger() {
|
2009-05-11 13:31:17 +00:00
|
|
|
return _console;
|
|
|
|
}
|
|
|
|
|
2010-06-25 16:16:29 +00:00
|
|
|
const char *SciEngine::getGameIdStr() const {
|
|
|
|
return _gameDescription->gameid;
|
|
|
|
}
|
|
|
|
|
2009-02-20 14:45:28 +00:00
|
|
|
Common::Language SciEngine::getLanguage() const {
|
2009-10-09 23:19:53 +00:00
|
|
|
return _gameDescription->language;
|
2009-02-20 14:45:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Common::Platform SciEngine::getPlatform() const {
|
2009-10-09 23:19:53 +00:00
|
|
|
return _gameDescription->platform;
|
2009-02-20 14:45:28 +00:00
|
|
|
}
|
|
|
|
|
2009-12-30 16:00:56 +00:00
|
|
|
bool SciEngine::isDemo() const {
|
2010-06-25 16:12:38 +00:00
|
|
|
return _gameDescription->flags & ADGF_DEMO;
|
2009-12-30 16:00:56 +00:00
|
|
|
}
|
|
|
|
|
2009-02-20 23:41:15 +00:00
|
|
|
Common::String SciEngine::getSavegameName(int nr) const {
|
2009-07-25 10:26:17 +00:00
|
|
|
return _targetName + Common::String::printf(".%03d", nr);
|
2009-02-20 23:41:15 +00:00
|
|
|
}
|
|
|
|
|
2009-02-27 01:17:24 +00:00
|
|
|
Common::String SciEngine::getSavegamePattern() const {
|
|
|
|
return _targetName + ".???";
|
|
|
|
}
|
|
|
|
|
2010-01-01 09:40:28 +00:00
|
|
|
Common::String SciEngine::getFilePrefix() const {
|
2010-06-25 16:16:29 +00:00
|
|
|
if (_gameId == GID_QFG2) {
|
2010-01-01 09:40:28 +00:00
|
|
|
// Quest for Glory 2 wants to read files from Quest for Glory 1 (EGA/VGA) to import character data
|
|
|
|
if (_gamestate->currentRoomNumber() == 805)
|
|
|
|
return "qfg1";
|
|
|
|
// TODO: Include import-room for qfg1vga
|
2010-06-25 16:16:29 +00:00
|
|
|
} else if (_gameId == GID_QFG3) {
|
2010-01-01 09:40:28 +00:00
|
|
|
// Quest for Glory 3 wants to read files from Quest for Glory 2 to import character data
|
|
|
|
if (_gamestate->currentRoomNumber() == 54)
|
|
|
|
return "qfg2";
|
2010-07-15 06:04:52 +00:00
|
|
|
} else if (_gameId == GID_QFG4) {
|
|
|
|
// Quest for Glory 4 wants to read files from Quest for Glory 3 to import character data
|
|
|
|
if (_gamestate->currentRoomNumber() == 54)
|
|
|
|
return "qfg3";
|
2010-01-01 09:40:28 +00:00
|
|
|
}
|
|
|
|
return _targetName;
|
|
|
|
}
|
|
|
|
|
2009-02-27 01:17:24 +00:00
|
|
|
Common::String SciEngine::wrapFilename(const Common::String &name) const {
|
2010-01-01 09:40:28 +00:00
|
|
|
return getFilePrefix() + "-" + name;
|
2009-02-27 01:17:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Common::String SciEngine::unwrapFilename(const Common::String &name) const {
|
2010-01-01 09:40:28 +00:00
|
|
|
Common::String prefix = getFilePrefix() + "-";
|
2009-02-27 01:17:24 +00:00
|
|
|
if (name.hasPrefix(prefix.c_str()))
|
|
|
|
return Common::String(name.c_str() + prefix.size());
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
2009-06-02 19:03:43 +00:00
|
|
|
void SciEngine::pauseEngineIntern(bool pause) {
|
|
|
|
_mixer->pauseAll(pause);
|
|
|
|
}
|
|
|
|
|
2010-01-12 00:51:37 +00:00
|
|
|
void SciEngine::syncSoundSettings() {
|
|
|
|
Engine::syncSoundSettings();
|
|
|
|
|
|
|
|
bool mute = false;
|
|
|
|
if (ConfMan.hasKey("mute"))
|
|
|
|
mute = ConfMan.getBool("mute");
|
|
|
|
|
|
|
|
int soundVolumeMusic = (mute ? 0 : ConfMan.getInt("music_volume"));
|
|
|
|
|
2010-06-29 09:00:08 +00:00
|
|
|
if (_gamestate && g_sci->_soundCmd) {
|
2010-01-12 00:51:37 +00:00
|
|
|
int vol = (soundVolumeMusic + 1) * SoundCommandParser::kMaxSciVolume / Audio::Mixer::kMaxMixerVolume;
|
2010-06-29 09:00:08 +00:00
|
|
|
g_sci->_soundCmd->setMasterVolume(vol);
|
2010-01-12 00:51:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-02-20 14:45:28 +00:00
|
|
|
} // End of namespace Sci
|