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.
|
2009-02-15 06:10:59 +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
|
|
|
|
* 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 08:20:53 +00:00
|
|
|
#include "engines/advancedDetector.h"
|
2009-02-15 06:10:59 +00:00
|
|
|
#include "base/plugins.h"
|
2010-02-17 23:38:43 +00:00
|
|
|
#include "common/file.h"
|
2009-10-11 15:51:43 +00:00
|
|
|
#include "common/savefile.h"
|
2010-02-17 23:37:32 +00:00
|
|
|
#include "common/system.h"
|
2009-10-11 15:51:43 +00:00
|
|
|
#include "graphics/thumbnail.h"
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2009-02-15 11:03:21 +00:00
|
|
|
#include "sci/sci.h"
|
2009-08-18 10:01:18 +00:00
|
|
|
#include "sci/engine/kernel.h"
|
2009-10-11 15:51:43 +00:00
|
|
|
#include "sci/engine/savegame.h"
|
2010-05-19 15:57:58 +00:00
|
|
|
#include "sci/engine/script.h"
|
2009-08-17 05:55:21 +00:00
|
|
|
#include "sci/engine/seg_manager.h"
|
2009-10-11 15:51:43 +00:00
|
|
|
#include "sci/engine/state.h"
|
2009-09-01 17:09:59 +00:00
|
|
|
#include "sci/engine/vm.h" // for convertSierraGameId
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2009-02-20 14:45:28 +00:00
|
|
|
namespace Sci {
|
|
|
|
|
2009-02-15 06:10:59 +00:00
|
|
|
// Titles of the games
|
|
|
|
static const PlainGameDescriptor SciGameTitles[] = {
|
2009-02-22 14:40:56 +00:00
|
|
|
{"sci", "Sierra SCI Game"},
|
|
|
|
{"sci-fanmade", "Fanmade SCI Game"},
|
2010-01-31 08:28:10 +00:00
|
|
|
// === SCI0 games =========================================================
|
2009-02-15 06:10:59 +00:00
|
|
|
{"astrochicken", "Astro Chicken"},
|
2009-02-20 23:00:27 +00:00
|
|
|
{"christmas1988", "Christmas Card 1988"},
|
2009-02-15 06:10:59 +00:00
|
|
|
{"iceman", "Codename: Iceman"},
|
|
|
|
{"camelot", "Conquests of Camelot: King Arthur, Quest for the Grail"},
|
2010-01-31 08:28:10 +00:00
|
|
|
{"funseeker", "Fun Seeker's Guide"},
|
|
|
|
{"hoyle1", "Hoyle Official Book of Games: Volume 1"},
|
|
|
|
{"hoyle2", "Hoyle Official Book of Games: Volume 2"},
|
|
|
|
{"kq4sci", "King's Quest IV: The Perils of Rosella, SCI Remake"},
|
|
|
|
{"laurabow", "Laura Bow: The Colonel's Bequest"},
|
|
|
|
{"lsl2", "Leisure Suit Larry 2: Goes Looking for Love (in Several Wrong Places)"},
|
|
|
|
{"lsl3", "Leisure Suit Larry 3: Passionate Patti in Pursuit of the Pulsating Pectorals"},
|
|
|
|
{"pq2", "Police Quest II: The Vengeance"},
|
|
|
|
{"qfg1", "Quest for Glory I: So You Want to Be a Hero"}, // EGA is SCI0, VGA SCI1.1
|
|
|
|
{"sq3", "Space Quest III: The Pirates of Pestulon"},
|
|
|
|
// === SCI01 games ========================================================
|
|
|
|
{"qfg2", "Quest for Glory II: Trial by Fire"},
|
|
|
|
{"kq1sci", "King's Quest I: Quest for the Crown, SCI Remake"},
|
|
|
|
// === SCI1 games =========================================================
|
|
|
|
{"castlebrain", "Castle of Dr. Brain"},
|
|
|
|
{"christmas1990", "Christmas Card 1990: The Seasoned Professional"},
|
2009-09-01 19:05:21 +00:00
|
|
|
{"cnick-lsl", "Crazy Nick's Software Picks: Leisure Suit Larry's Casino"},
|
|
|
|
{"cnick-kq", "Crazy Nick's Software Picks: King Graham's Board Game Challenge"},
|
|
|
|
{"cnick-laurabow", "Crazy Nick's Software Picks: Parlor Games with Laura Bow"},
|
|
|
|
{"cnick-longbow", "Crazy Nick's Software Picks: Robin Hood's Game of Skill and Chance"},
|
|
|
|
{"cnick-sq", "Crazy Nick's Software Picks: Roger Wilco's Spaced Out Game Pack"},
|
2010-01-31 08:28:10 +00:00
|
|
|
{"ecoquest", "EcoQuest: The Search for Cetus"}, // floppy is SCI1, CD SCI1.1
|
|
|
|
{"fairytales", "Mixed-up Fairy Tales"},
|
2009-05-02 15:35:57 +00:00
|
|
|
{"hoyle3", "Hoyle Official Book of Games: Volume 3"},
|
2009-02-15 06:10:59 +00:00
|
|
|
{"jones", "Jones in the Fast Lane"},
|
|
|
|
{"kq5", "King's Quest V: Absence Makes the Heart Go Yonder"},
|
2010-01-31 08:28:10 +00:00
|
|
|
{"longbow", "Conquests of the Longbow: The Adventures of Robin Hood"},
|
2009-09-01 19:40:29 +00:00
|
|
|
{"lsl1sci", "Leisure Suit Larry in the Land of the Lounge Lizards"},
|
2009-02-15 06:10:59 +00:00
|
|
|
{"lsl5", "Leisure Suit Larry 5: Passionate Patti Does a Little Undercover Work"},
|
2009-02-22 23:32:28 +00:00
|
|
|
{"msastrochicken", "Ms. Astro Chicken"},
|
2009-09-01 19:40:29 +00:00
|
|
|
{"pq1sci", "Police Quest: In Pursuit of the Death Angel"},
|
2009-02-15 06:10:59 +00:00
|
|
|
{"pq3", "Police Quest III: The Kindred"},
|
2009-09-01 19:40:29 +00:00
|
|
|
{"sq1sci", "Space Quest I: The Sarien Encounter"},
|
2010-01-31 08:28:10 +00:00
|
|
|
{"sq4", "Space Quest IV: Roger Wilco and the Time Rippers"}, // floppy is SCI1, CD SCI1.1
|
|
|
|
// === SCI1.1 games =======================================================
|
|
|
|
{"christmas1992", "Christmas Card 1992"},
|
|
|
|
{"ecoquest2", "EcoQuest II: Lost Secret of the Rainforest"},
|
|
|
|
{"freddypharkas", "Freddy Pharkas: Frontier Pharmacist"},
|
|
|
|
{"hoyle4", "Hoyle Classic Card Games"},
|
|
|
|
{"kq6", "King's Quest VI: Heir Today, Gone Tomorrow"},
|
|
|
|
{"laurabow2", "Laura Bow 2: The Dagger of Amon Ra"},
|
|
|
|
{"qfg3", "Quest for Glory III: Wages of War"},
|
2009-02-15 06:10:59 +00:00
|
|
|
{"sq5", "Space Quest V: The Next Mutation"},
|
|
|
|
{"islandbrain", "The Island of Dr. Brain"},
|
2010-01-31 08:28:10 +00:00
|
|
|
{"lsl6", "Leisure Suit Larry 6: Shape Up or Slip Out!"},
|
|
|
|
{"mothergoose", "Mixed-Up Mother Goose"}, // floppy is SCI1.1, CD SCI2.1
|
|
|
|
{"pepper", "Pepper's Adventure in Time"},
|
|
|
|
{"slater", "Slater & Charlie Go Camping"},
|
|
|
|
// === SCI2 games =========================================================
|
2009-08-26 10:01:05 +00:00
|
|
|
{"gk1", "Gabriel Knight: Sins of the Fathers"}, // demo is SCI11, full version SCI32
|
2010-01-31 08:28:10 +00:00
|
|
|
{"pq4", "Police Quest IV: Open Season"}, // floppy is SCI2, CD SCI2.1
|
|
|
|
{"qfg4", "Quest for Glory IV: Shadows of Darkness"}, // floppy is SCI2, CD SCI2.1
|
|
|
|
// === SCI2.1 games ========================================================
|
2009-03-30 07:53:32 +00:00
|
|
|
{"gk2", "The Beast Within: A Gabriel Knight Mystery"},
|
2010-01-31 08:28:10 +00:00
|
|
|
// TODO: Inside The Chest/Behind the Developer's Shield
|
2009-03-30 07:53:32 +00:00
|
|
|
{"kq7", "King's Quest VII: The Princeless Bride"},
|
2010-01-31 08:28:10 +00:00
|
|
|
// TODO: King's Questions
|
2009-03-30 07:53:32 +00:00
|
|
|
{"phantasmagoria", "Phantasmagoria"},
|
2009-05-02 15:35:57 +00:00
|
|
|
{"pqswat", "Police Quest: SWAT"},
|
2009-03-30 07:53:32 +00:00
|
|
|
{"shivers", "Shivers"},
|
|
|
|
{"sq6", "Space Quest 6: The Spinal Frontier"},
|
2009-02-15 06:10:59 +00:00
|
|
|
{"torin", "Torin's Passage"},
|
2010-01-31 08:28:10 +00:00
|
|
|
// === SCI3 games =========================================================
|
|
|
|
{"lsl7", "Leisure Suit Larry 7: Love for Sail!"},
|
|
|
|
{"lighthouse", "Lighthouse: The Dark Being"},
|
|
|
|
{"phantasmagoria2", "Phantasmagoria II: A Puzzle of Flesh"},
|
|
|
|
{"shivers2", "Shivers II: Harvest of Souls"},
|
|
|
|
{"rama", "RAMA"},
|
2009-02-15 06:10:59 +00:00
|
|
|
{0, 0}
|
|
|
|
};
|
|
|
|
|
2009-08-24 07:57:04 +00:00
|
|
|
#include "sci/detection_tables.h"
|
2009-02-15 06:10:59 +00:00
|
|
|
|
2009-02-19 02:04:31 +00:00
|
|
|
/**
|
|
|
|
* The fallback game descriptor used by the SCI engine's fallbackDetector.
|
2010-02-17 23:36:50 +00:00
|
|
|
* Contents of this struct are overwritten by the fallbackDetector.
|
2009-02-19 02:04:31 +00:00
|
|
|
*/
|
2009-10-09 23:19:53 +00:00
|
|
|
static ADGameDescription s_fallbackDesc = {
|
|
|
|
"",
|
|
|
|
"",
|
|
|
|
AD_ENTRY1(0, 0), // This should always be AD_ENTRY1(0, 0) in the fallback descriptor
|
|
|
|
Common::UNK_LANG,
|
|
|
|
Common::kPlatformPC,
|
|
|
|
ADGF_NO_FLAGS,
|
|
|
|
Common::GUIO_NONE
|
2009-02-19 02:04:31 +00:00
|
|
|
};
|
|
|
|
|
2010-02-17 23:36:50 +00:00
|
|
|
static char s_fallbackGameIdBuf[256];
|
|
|
|
|
2009-02-19 02:04:31 +00:00
|
|
|
|
2009-02-15 08:20:53 +00:00
|
|
|
static const ADParams detectionParams = {
|
2009-02-15 06:10:59 +00:00
|
|
|
// Pointer to ADGameDescription or its superset structure
|
2009-08-24 07:57:04 +00:00
|
|
|
(const byte *)Sci::SciGameDescriptions,
|
2009-02-15 06:10:59 +00:00
|
|
|
// Size of that superset structure
|
2009-10-09 23:19:53 +00:00
|
|
|
sizeof(ADGameDescription),
|
2009-02-15 06:10:59 +00:00
|
|
|
// Number of bytes to compute MD5 sum for
|
|
|
|
5000,
|
|
|
|
// List of all engine targets
|
|
|
|
SciGameTitles,
|
|
|
|
// Structure for autoupgrading obsolete targets
|
|
|
|
0,
|
|
|
|
// Name of single gameid (optional)
|
|
|
|
"sci",
|
|
|
|
// List of files for file-based fallback detection (optional)
|
2009-02-18 21:05:05 +00:00
|
|
|
0,
|
2009-02-15 06:10:59 +00:00
|
|
|
// Flags
|
2009-06-06 17:56:41 +00:00
|
|
|
0,
|
|
|
|
// Additional GUI options (for every game}
|
|
|
|
Common::GUIO_NONE
|
2009-02-15 06:10:59 +00:00
|
|
|
};
|
|
|
|
|
2009-02-15 08:20:53 +00:00
|
|
|
class SciMetaEngine : public AdvancedMetaEngine {
|
2009-02-15 06:10:59 +00:00
|
|
|
public:
|
2009-02-15 08:20:53 +00:00
|
|
|
SciMetaEngine() : AdvancedMetaEngine(detectionParams) {}
|
2009-02-15 06:10:59 +00:00
|
|
|
|
|
|
|
virtual const char *getName() const {
|
2009-03-30 07:53:32 +00:00
|
|
|
return "SCI Engine [SCI0, SCI01, SCI10, SCI11"
|
|
|
|
#ifdef ENABLE_SCI32
|
|
|
|
", SCI32"
|
|
|
|
#endif
|
|
|
|
"]";
|
2009-02-15 06:10:59 +00:00
|
|
|
}
|
|
|
|
|
2009-03-05 12:04:58 +00:00
|
|
|
virtual const char *getOriginalCopyright() const {
|
2009-02-15 06:10:59 +00:00
|
|
|
return "Sierra's Creative Interpreter (C) Sierra Online";
|
|
|
|
}
|
|
|
|
|
2009-02-15 08:20:53 +00:00
|
|
|
virtual bool createInstance(OSystem *syst, Engine **engine, const ADGameDescription *gd) const;
|
2009-02-18 21:05:05 +00:00
|
|
|
const ADGameDescription *fallbackDetect(const Common::FSList &fslist) const;
|
2009-10-11 15:51:43 +00:00
|
|
|
virtual bool hasFeature(MetaEngineFeature f) const;
|
|
|
|
virtual SaveStateList listSaves(const char *target) const;
|
|
|
|
virtual int getMaximumSaveSlot() const;
|
|
|
|
virtual void removeSaveState(const char *target, int slot) const;
|
|
|
|
SaveStateDescriptor querySaveMetaInfos(const char *target, int slot) const;
|
2009-02-15 06:10:59 +00:00
|
|
|
};
|
|
|
|
|
2009-08-26 00:27:14 +00:00
|
|
|
Common::Language charToScummVMLanguage(const char c) {
|
|
|
|
switch (c) {
|
|
|
|
case 'F':
|
|
|
|
return Common::FR_FRA;
|
|
|
|
case 'S':
|
|
|
|
return Common::ES_ESP;
|
|
|
|
case 'I':
|
|
|
|
return Common::IT_ITA;
|
|
|
|
case 'G':
|
|
|
|
return Common::DE_DEU;
|
|
|
|
case 'J':
|
|
|
|
case 'j':
|
|
|
|
return Common::JA_JPN;
|
|
|
|
case 'P':
|
|
|
|
return Common::PT_BRA;
|
|
|
|
default:
|
|
|
|
return Common::UNK_LANG;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-05-19 15:57:58 +00:00
|
|
|
#define READ_UINT16(buf) (!resMan->isSci11Mac() ? READ_LE_UINT16(buf) : READ_BE_UINT16(buf))
|
|
|
|
|
|
|
|
// Finds the internal ID of the current game from script 0
|
|
|
|
Common::String getSierraGameId(ResourceManager *resMan) {
|
|
|
|
Resource *script = resMan->findResource(ResourceId(kResourceTypeScript, 0), false);
|
|
|
|
Script *script000 = new Script();
|
|
|
|
script000->init(0, resMan);
|
|
|
|
script000->mcpyInOut(0, script->data, script->size);
|
|
|
|
uint16 curOffset = (getSciVersion() == SCI_VERSION_0_EARLY) ? 2 : 0;
|
|
|
|
uint16 objLength = 0;
|
|
|
|
int objType = 0;
|
|
|
|
int16 exportsOffset = 0;
|
|
|
|
Common::String sierraId;
|
|
|
|
|
|
|
|
// Now find the export table of the script
|
|
|
|
do {
|
|
|
|
objType = READ_UINT16(script000->_buf + curOffset);
|
|
|
|
if (!objType)
|
|
|
|
break;
|
|
|
|
|
|
|
|
objLength = READ_UINT16(script000->_buf + curOffset + 2);
|
|
|
|
curOffset += 4; // skip header
|
|
|
|
|
|
|
|
if (objType == SCI_OBJ_EXPORTS) {
|
|
|
|
exportsOffset = READ_UINT16(script000->_buf + curOffset + 2);
|
|
|
|
break;
|
|
|
|
}
|
2010-05-19 23:29:27 +00:00
|
|
|
curOffset += objLength - 4;
|
|
|
|
} while (objType != 0 && curOffset < script->size - 2);
|
2010-05-19 15:57:58 +00:00
|
|
|
|
|
|
|
// The game object is the first export. Script 0 is always at segment 1
|
|
|
|
reg_t gameObj = make_reg(1, exportsOffset);
|
|
|
|
|
|
|
|
// TODO: stop using the segment manager and read the object name here
|
|
|
|
SegManager *segMan = new SegManager(resMan);
|
|
|
|
script_instantiate(resMan, segMan, 0);
|
|
|
|
sierraId = segMan->getObjectName(gameObj);
|
|
|
|
delete segMan;
|
|
|
|
|
|
|
|
delete script000;
|
|
|
|
|
|
|
|
return sierraId;
|
|
|
|
}
|
|
|
|
|
2009-02-18 21:05:05 +00:00
|
|
|
const ADGameDescription *SciMetaEngine::fallbackDetect(const Common::FSList &fslist) const {
|
2009-02-19 02:04:31 +00:00
|
|
|
bool foundResMap = false;
|
|
|
|
bool foundRes000 = false;
|
2009-02-18 22:20:28 +00:00
|
|
|
|
2009-08-24 13:47:38 +00:00
|
|
|
// Set some defaults
|
2009-10-09 23:19:53 +00:00
|
|
|
s_fallbackDesc.extra = "";
|
|
|
|
s_fallbackDesc.language = Common::EN_ANY;
|
|
|
|
s_fallbackDesc.flags = ADGF_NO_FLAGS;
|
|
|
|
s_fallbackDesc.platform = Common::kPlatformPC; // default to PC platform
|
|
|
|
s_fallbackDesc.gameid = "sci";
|
2009-08-24 13:47:38 +00:00
|
|
|
|
2009-02-18 22:20:28 +00:00
|
|
|
// First grab all filenames
|
|
|
|
for (Common::FSList::const_iterator file = fslist.begin(); file != fslist.end(); ++file) {
|
2009-02-20 14:45:28 +00:00
|
|
|
if (file->isDirectory())
|
|
|
|
continue;
|
|
|
|
|
2009-02-18 22:20:28 +00:00
|
|
|
Common::String filename = file->getName();
|
|
|
|
filename.toLowercase();
|
2009-02-20 14:45:28 +00:00
|
|
|
|
2010-05-10 18:29:13 +00:00
|
|
|
if (filename.contains("resource.map") || filename.contains("resmap.00") || filename.contains("Data1")) {
|
2009-08-18 12:49:34 +00:00
|
|
|
// HACK: resource.map is located in the same directory as the other resource files,
|
2009-08-18 10:01:18 +00:00
|
|
|
// therefore add the directory here, so that the game files can be opened later on
|
2009-08-21 22:25:55 +00:00
|
|
|
// We now add the parent directory temporary to our SearchMan so the engine code
|
|
|
|
// used in the detection can access all files via Common::File without any problems.
|
|
|
|
// In all branches returning from this function, we need to have a call to
|
|
|
|
// SearchMan.remove to remove it from the default directory pool again.
|
|
|
|
//
|
|
|
|
// A proper solution to remove this hack would be to have the code, which is needed
|
|
|
|
// for detection, to operate on Stream objects, so they can be easily called from
|
|
|
|
// the detection code. This might be easily to achieve through refactoring the
|
|
|
|
// code needed for detection.
|
|
|
|
assert(!SearchMan.hasArchive("SCI_detection"));
|
|
|
|
SearchMan.addDirectory("SCI_detection", file->getParent());
|
2009-02-19 02:04:31 +00:00
|
|
|
foundResMap = true;
|
2009-08-18 10:01:18 +00:00
|
|
|
}
|
2009-02-20 14:45:28 +00:00
|
|
|
|
2009-08-24 13:47:38 +00:00
|
|
|
// Determine if we got a CD version and set the CD flag accordingly, by checking for
|
2009-08-26 10:01:05 +00:00
|
|
|
// resource.aud for SCI1.1 CD games, or audio001.002 for SCI1 CD games. We assume that
|
|
|
|
// the file should be over 10MB, as it contains all the game speech and is usually
|
|
|
|
// around 450MB+. The size check is for some floppy game versions like KQ6 floppy, which
|
|
|
|
// also have a small resource.aud file
|
|
|
|
if (filename.contains("resource.aud") || filename.contains("audio001.002")) {
|
2009-08-24 13:47:38 +00:00
|
|
|
Common::SeekableReadStream *tmpStream = file->createReadStream();
|
|
|
|
if (tmpStream->size() > 10 * 1024 * 1024) {
|
|
|
|
// We got a CD version, so set the CD flag accordingly
|
2009-10-09 23:19:53 +00:00
|
|
|
s_fallbackDesc.flags |= ADGF_CD;
|
|
|
|
s_fallbackDesc.extra = "CD";
|
2009-08-24 13:47:38 +00:00
|
|
|
}
|
|
|
|
delete tmpStream;
|
|
|
|
}
|
|
|
|
|
2009-02-19 02:04:31 +00:00
|
|
|
if (filename.contains("resource.000") || filename.contains("resource.001")
|
|
|
|
|| filename.contains("ressci.000") || filename.contains("ressci.001"))
|
|
|
|
foundRes000 = true;
|
2009-08-26 00:27:14 +00:00
|
|
|
|
2010-05-10 18:29:13 +00:00
|
|
|
// Data1 contains both map and volume for SCI1.1+ Mac games
|
|
|
|
if (filename.contains("Data1")) {
|
|
|
|
foundResMap = foundRes000 = true;
|
|
|
|
s_fallbackDesc.platform = Common::kPlatformMacintosh;
|
|
|
|
}
|
|
|
|
|
2009-08-26 00:27:14 +00:00
|
|
|
// Determine the game platform
|
|
|
|
// The existence of any of these files indicates an Amiga game
|
|
|
|
if (filename.contains("9.pat") || filename.contains("spal") ||
|
|
|
|
filename.contains("patch.005") || filename.contains("bank.001"))
|
2009-10-09 23:19:53 +00:00
|
|
|
s_fallbackDesc.platform = Common::kPlatformAmiga;
|
2009-08-26 00:27:14 +00:00
|
|
|
|
|
|
|
// The existence of 7.pat indicates a Mac game
|
|
|
|
if (filename.contains("7.pat"))
|
2009-10-09 23:19:53 +00:00
|
|
|
s_fallbackDesc.platform = Common::kPlatformMacintosh;
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2009-08-26 00:27:14 +00:00
|
|
|
// The data files for Atari ST versions are the same as their DOS counterparts
|
2009-02-18 22:20:28 +00:00
|
|
|
}
|
2009-02-20 14:45:28 +00:00
|
|
|
|
2009-02-19 02:04:31 +00:00
|
|
|
// If these files aren't found, it can't be SCI
|
2009-08-21 22:25:55 +00:00
|
|
|
if (!foundResMap && !foundRes000) {
|
|
|
|
SearchMan.remove("SCI_detection");
|
2009-02-19 02:04:31 +00:00
|
|
|
return 0;
|
2009-08-21 22:25:55 +00:00
|
|
|
}
|
2009-02-20 14:45:28 +00:00
|
|
|
|
2009-09-02 12:02:37 +00:00
|
|
|
ResourceManager *resMan = new ResourceManager(fslist);
|
|
|
|
ViewType gameViews = resMan->getViewType();
|
2009-08-24 13:47:38 +00:00
|
|
|
|
|
|
|
// Have we identified the game views? If not, stop here
|
|
|
|
if (gameViews == kViewUnknown) {
|
|
|
|
SearchMan.remove("SCI_detection");
|
2009-09-02 12:02:37 +00:00
|
|
|
delete resMan;
|
2010-01-17 22:32:53 +00:00
|
|
|
// Can't be SCI (or unsupported SCI views). Pinball Creep by sierra also uses resource.map/resource.000 files
|
|
|
|
// but doesnt share sci format at all, if we dont return 0 here we will detect this game as SCI
|
|
|
|
return 0;
|
2009-08-24 13:47:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef ENABLE_SCI32
|
|
|
|
// Is SCI32 compiled in? If not, and this is a SCI32 game,
|
|
|
|
// stop here
|
2009-09-23 10:55:35 +00:00
|
|
|
if (getSciVersion() >= SCI_VERSION_2) {
|
2009-08-24 13:47:38 +00:00
|
|
|
SearchMan.remove("SCI_detection");
|
2009-09-02 12:02:37 +00:00
|
|
|
delete resMan;
|
2009-08-24 13:47:38 +00:00
|
|
|
return (const ADGameDescription *)&s_fallbackDesc;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2009-08-24 14:00:29 +00:00
|
|
|
// EGA views
|
2009-10-09 23:19:53 +00:00
|
|
|
if (gameViews == kViewEga && s_fallbackDesc.platform != Common::kPlatformAmiga)
|
|
|
|
s_fallbackDesc.extra = "EGA";
|
2009-08-24 14:00:29 +00:00
|
|
|
|
2009-08-26 00:27:14 +00:00
|
|
|
// Set the platform to Amiga if the game is using Amiga views
|
|
|
|
if (gameViews == kViewAmiga)
|
2009-10-09 23:19:53 +00:00
|
|
|
s_fallbackDesc.platform = Common::kPlatformAmiga;
|
2009-02-18 22:20:28 +00:00
|
|
|
|
2009-08-17 05:55:21 +00:00
|
|
|
// Determine the game id
|
2010-05-19 15:57:58 +00:00
|
|
|
Common::String gameId = convertSierraGameId(getSierraGameId(resMan).c_str(), &s_fallbackDesc.flags, resMan);
|
2010-02-17 23:36:50 +00:00
|
|
|
strncpy(s_fallbackGameIdBuf, gameId.c_str(), sizeof(s_fallbackGameIdBuf) - 1);
|
|
|
|
s_fallbackGameIdBuf[sizeof(s_fallbackGameIdBuf) - 1] = 0; // Make sure string is NULL terminated
|
|
|
|
s_fallbackDesc.gameid = s_fallbackGameIdBuf;
|
2009-08-26 00:27:14 +00:00
|
|
|
|
|
|
|
// Try to determine the game language
|
|
|
|
// Load up text 0 and start looking for "#" characters
|
|
|
|
// Non-English versions contain strings like XXXX#YZZZZ
|
|
|
|
// Where XXXX is the English string, #Y a separator indicating the language
|
2009-08-26 00:37:04 +00:00
|
|
|
// (e.g. #G for German) and ZZZZ is the translated text
|
2009-08-26 01:26:56 +00:00
|
|
|
// NOTE: This doesn't work for games which use message instead of text resources
|
2010-01-25 01:39:44 +00:00
|
|
|
// (like, for example, Eco Quest 1 and all SCI1.1 games and newer, e.g. Freddy Pharkas).
|
2009-08-26 10:01:05 +00:00
|
|
|
// As far as we know, these games store the messages of each language in separate
|
|
|
|
// resources, and it's not possible to detect that easily
|
2010-01-06 18:25:43 +00:00
|
|
|
// Also look for "%J" which is used in japanese games
|
2009-09-02 12:02:37 +00:00
|
|
|
Resource *text = resMan->findResource(ResourceId(kResourceTypeText, 0), 0);
|
2009-08-26 00:27:14 +00:00
|
|
|
uint seeker = 0;
|
|
|
|
if (text) {
|
|
|
|
while (seeker < text->size) {
|
2010-01-06 18:25:43 +00:00
|
|
|
if (text->data[seeker] == '#') {
|
|
|
|
if (seeker + 1 < text->size)
|
|
|
|
s_fallbackDesc.language = charToScummVMLanguage(text->data[seeker + 1]);
|
2009-08-26 00:27:14 +00:00
|
|
|
break;
|
|
|
|
}
|
2010-01-06 18:25:43 +00:00
|
|
|
if (text->data[seeker] == '%') {
|
|
|
|
if ((seeker + 1 < text->size) && (text->data[seeker + 1] == 'J')) {
|
|
|
|
s_fallbackDesc.language = charToScummVMLanguage(text->data[seeker + 1]);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2009-08-26 00:27:14 +00:00
|
|
|
seeker++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-02 12:02:37 +00:00
|
|
|
delete resMan;
|
2009-08-17 05:55:21 +00:00
|
|
|
|
2009-09-01 19:40:29 +00:00
|
|
|
// Fill in extras field
|
2009-10-09 23:19:53 +00:00
|
|
|
if (!strcmp(s_fallbackDesc.gameid, "lsl1sci") ||
|
|
|
|
!strcmp(s_fallbackDesc.gameid, "pq1sci") ||
|
|
|
|
!strcmp(s_fallbackDesc.gameid, "sq1sci"))
|
|
|
|
s_fallbackDesc.extra = "VGA Remake";
|
2009-09-01 19:40:29 +00:00
|
|
|
|
2009-10-09 23:19:53 +00:00
|
|
|
if (!strcmp(s_fallbackDesc.gameid, "qfg1") && !Common::File::exists("resource.001"))
|
|
|
|
s_fallbackDesc.extra = "VGA Remake";
|
2009-09-01 19:40:29 +00:00
|
|
|
|
2009-09-24 07:18:38 +00:00
|
|
|
// Add "demo" to the description for demos
|
2009-10-09 23:19:53 +00:00
|
|
|
if (s_fallbackDesc.flags & ADGF_DEMO)
|
|
|
|
s_fallbackDesc.extra = "demo";
|
2009-09-24 07:18:38 +00:00
|
|
|
|
2009-08-21 22:25:55 +00:00
|
|
|
SearchMan.remove("SCI_detection");
|
|
|
|
|
2009-08-15 12:09:47 +00:00
|
|
|
return (const ADGameDescription *)&s_fallbackDesc;
|
2009-02-18 21:05:05 +00:00
|
|
|
}
|
|
|
|
|
2009-02-15 08:20:53 +00:00
|
|
|
bool SciMetaEngine::createInstance(OSystem *syst, Engine **engine, const ADGameDescription *gd) const {
|
2009-10-09 23:19:53 +00:00
|
|
|
const ADGameDescription *desc = (const ADGameDescription *)gd;
|
2009-02-15 06:10:59 +00:00
|
|
|
|
|
|
|
*engine = new SciEngine(syst, desc);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2009-10-11 15:51:43 +00:00
|
|
|
bool SciMetaEngine::hasFeature(MetaEngineFeature f) const {
|
|
|
|
return
|
|
|
|
(f == kSupportsListSaves) ||
|
2010-01-21 22:20:16 +00:00
|
|
|
(f == kSupportsLoadingDuringStartup) ||
|
2009-10-11 15:51:43 +00:00
|
|
|
(f == kSupportsDeleteSave) ||
|
|
|
|
(f == kSavesSupportMetaInfo) ||
|
|
|
|
(f == kSavesSupportThumbnail) ||
|
|
|
|
(f == kSavesSupportCreationDate);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SciEngine::hasFeature(EngineFeature f) const {
|
|
|
|
return
|
|
|
|
//(f == kSupportsRTL) ||
|
|
|
|
(f == kSupportsLoadingDuringRuntime);
|
|
|
|
//(f == kSupportsSavingDuringRuntime);
|
|
|
|
}
|
|
|
|
|
|
|
|
SaveStateList SciMetaEngine::listSaves(const char *target) const {
|
|
|
|
Common::SaveFileManager *saveFileMan = g_system->getSavefileManager();
|
2010-03-18 15:54:40 +00:00
|
|
|
Common::StringArray filenames;
|
2009-10-11 15:51:43 +00:00
|
|
|
Common::String pattern = target;
|
|
|
|
pattern += ".???";
|
|
|
|
|
|
|
|
filenames = saveFileMan->listSavefiles(pattern);
|
|
|
|
sort(filenames.begin(), filenames.end()); // Sort (hopefully ensuring we are sorted numerically..)
|
|
|
|
|
|
|
|
SaveStateList saveList;
|
|
|
|
int slotNum = 0;
|
2010-03-18 15:54:40 +00:00
|
|
|
for (Common::StringArray::const_iterator file = filenames.begin(); file != filenames.end(); ++file) {
|
2009-10-11 15:51:43 +00:00
|
|
|
// Obtain the last 3 digits of the filename, since they correspond to the save slot
|
|
|
|
slotNum = atoi(file->c_str() + file->size() - 3);
|
|
|
|
|
|
|
|
if (slotNum >= 0 && slotNum < 999) {
|
|
|
|
Common::InSaveFile *in = saveFileMan->openForLoading(*file);
|
|
|
|
if (in) {
|
|
|
|
SavegameMetadata meta;
|
|
|
|
if (!get_savegame_metadata(in, &meta)) {
|
|
|
|
// invalid
|
|
|
|
delete in;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
saveList.push_back(SaveStateDescriptor(slotNum, meta.savegame_name));
|
|
|
|
delete in;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return saveList;
|
|
|
|
}
|
|
|
|
|
|
|
|
SaveStateDescriptor SciMetaEngine::querySaveMetaInfos(const char *target, int slot) const {
|
|
|
|
Common::String fileName = Common::String::printf("%s.%03d", target, slot);
|
|
|
|
Common::InSaveFile *in = g_system->getSavefileManager()->openForLoading(fileName);
|
|
|
|
|
|
|
|
if (in) {
|
|
|
|
SavegameMetadata meta;
|
|
|
|
if (!get_savegame_metadata(in, &meta)) {
|
|
|
|
// invalid
|
|
|
|
delete in;
|
|
|
|
|
|
|
|
SaveStateDescriptor desc(slot, "Invalid");
|
|
|
|
return desc;
|
|
|
|
}
|
|
|
|
|
|
|
|
SaveStateDescriptor desc(slot, meta.savegame_name);
|
|
|
|
|
|
|
|
Graphics::Surface *thumbnail = new Graphics::Surface();
|
|
|
|
assert(thumbnail);
|
|
|
|
if (!Graphics::loadThumbnail(*in, *thumbnail)) {
|
|
|
|
delete thumbnail;
|
|
|
|
thumbnail = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
desc.setThumbnail(thumbnail);
|
|
|
|
|
|
|
|
desc.setDeletableFlag(true);
|
|
|
|
desc.setWriteProtectedFlag(false);
|
|
|
|
|
|
|
|
int day = (meta.savegame_date >> 24) & 0xFF;
|
|
|
|
int month = (meta.savegame_date >> 16) & 0xFF;
|
|
|
|
int year = meta.savegame_date & 0xFFFF;
|
|
|
|
|
|
|
|
desc.setSaveDate(year, month, day);
|
|
|
|
|
|
|
|
int hour = (meta.savegame_time >> 16) & 0xFF;
|
|
|
|
int minutes = (meta.savegame_time >> 8) & 0xFF;
|
|
|
|
|
|
|
|
desc.setSaveTime(hour, minutes);
|
|
|
|
|
|
|
|
// TODO: played time
|
|
|
|
|
|
|
|
delete in;
|
|
|
|
|
|
|
|
return desc;
|
|
|
|
}
|
|
|
|
|
|
|
|
return SaveStateDescriptor();
|
|
|
|
}
|
|
|
|
|
|
|
|
int SciMetaEngine::getMaximumSaveSlot() const { return 999; }
|
|
|
|
|
|
|
|
void SciMetaEngine::removeSaveState(const char *target, int slot) const {
|
|
|
|
Common::String fileName = Common::String::printf("%s.%03d", target, slot);
|
|
|
|
g_system->getSavefileManager()->removeSavefile(fileName);
|
|
|
|
}
|
|
|
|
|
|
|
|
Common::Error SciEngine::loadGameState(int slot) {
|
|
|
|
Common::String fileName = Common::String::printf("%s.%03d", _targetName.c_str(), slot);
|
|
|
|
Common::SaveFileManager *saveFileMan = g_engine->getSaveFileManager();
|
|
|
|
Common::SeekableReadStream *in = saveFileMan->openForLoading(fileName);
|
|
|
|
|
|
|
|
if (in) {
|
|
|
|
// found a savegame file
|
2010-01-31 01:26:06 +00:00
|
|
|
gamestate_restore(_gamestate, in);
|
2009-10-11 15:51:43 +00:00
|
|
|
delete in;
|
|
|
|
}
|
|
|
|
|
2010-01-31 01:26:06 +00:00
|
|
|
if (_gamestate->r_acc != make_reg(0, 1)) {
|
2009-10-11 15:51:43 +00:00
|
|
|
return Common::kNoError;
|
|
|
|
} else {
|
2009-10-13 20:52:05 +00:00
|
|
|
warning("Restoring gamestate '%s' failed", fileName.c_str());
|
2009-10-11 15:51:43 +00:00
|
|
|
return Common::kUnknownError;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Common::Error SciEngine::saveGameState(int slot, const char *desc) {
|
|
|
|
Common::String fileName = Common::String::printf("%s.%03d", _targetName.c_str(), slot);
|
|
|
|
Common::SaveFileManager *saveFileMan = g_engine->getSaveFileManager();
|
|
|
|
Common::OutSaveFile *out = saveFileMan->openForSaving(fileName);
|
|
|
|
const char *version = "";
|
|
|
|
if (!out) {
|
2010-01-31 10:02:38 +00:00
|
|
|
warning("Opening savegame \"%s\" for writing failed", fileName.c_str());
|
2009-10-11 15:51:43 +00:00
|
|
|
return Common::kWritingFailed;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gamestate_save(_gamestate, out, desc, version)) {
|
2010-01-31 10:02:38 +00:00
|
|
|
warning("Saving the game state to '%s' failed", fileName.c_str());
|
2010-01-31 08:28:10 +00:00
|
|
|
return Common::kWritingFailed;
|
|
|
|
} else {
|
|
|
|
out->finalize();
|
|
|
|
if (out->err()) {
|
|
|
|
warning("Writing the savegame failed.");
|
|
|
|
return Common::kWritingFailed;
|
|
|
|
}
|
|
|
|
delete out;
|
2009-10-11 15:51:43 +00:00
|
|
|
}
|
2010-01-31 08:28:10 +00:00
|
|
|
|
|
|
|
return Common::kNoError;
|
2009-10-11 15:51:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool SciEngine::canLoadGameStateCurrently() {
|
|
|
|
return !_gamestate->execution_stack_base;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SciEngine::canSaveGameStateCurrently() {
|
|
|
|
return !_gamestate->execution_stack_base;
|
|
|
|
}
|
|
|
|
|
2009-02-20 14:45:28 +00:00
|
|
|
} // End of namespace Sci
|
|
|
|
|
2009-02-15 06:10:59 +00:00
|
|
|
#if PLUGIN_ENABLED_DYNAMIC(SCI)
|
2009-02-20 14:45:28 +00:00
|
|
|
REGISTER_PLUGIN_DYNAMIC(SCI, PLUGIN_TYPE_ENGINE, Sci::SciMetaEngine);
|
2009-02-15 06:10:59 +00:00
|
|
|
#else
|
2009-02-20 14:45:28 +00:00
|
|
|
REGISTER_PLUGIN_STATIC(SCI, PLUGIN_TYPE_ENGINE, Sci::SciMetaEngine);
|
2009-02-15 06:10:59 +00:00
|
|
|
#endif
|