scummvm/engines/gargoyle/detection.cpp

276 lines
8.8 KiB
C++
Raw Normal View History

2018-10-14 13:02:24 -07: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.
*
*/
#include "gargoyle/gargoyle.h"
#include "base/plugins.h"
#include "common/md5.h"
#include "common/memstream.h"
2018-10-14 13:02:24 -07:00
#include "common/savefile.h"
#include "common/str-array.h"
#include "common/system.h"
#include "engines/advancedDetector.h"
2018-10-14 13:02:24 -07:00
#include "graphics/colormasks.h"
#include "graphics/surface.h"
#define MAX_SAVES 99
namespace Gargoyle {
struct GargoyleGameDescription {
ADGameDescription _desc;
Common::String _filename;
InterpreterType _interpType;
Common::String _md5;
2018-10-14 13:02:24 -07:00
};
const Common::String &GargoyleEngine::getFilename() const {
return _gameDescription->_filename;
}
2018-10-14 13:02:24 -07:00
uint32 GargoyleEngine::getFeatures() const {
return _gameDescription->_desc.flags;
2018-10-14 13:02:24 -07:00
}
bool GargoyleEngine::isDemo() const {
return (bool)(_gameDescription->_desc.flags & ADGF_DEMO);
2018-10-14 13:02:24 -07:00
}
Common::Language GargoyleEngine::getLanguage() const {
return _gameDescription->_desc.language;
2018-10-14 13:02:24 -07:00
}
InterpreterType GargoyleEngine::getInterpreterType() const {
return _gameDescription->_interpType;
}
const Common::String &GargoyleEngine::getGameMD5() const {
return _gameDescription->_md5;
}
2018-10-14 13:02:24 -07:00
} // End of namespace Gargoyle
static const PlainGameDescriptor gargoyleGames[] = {
{"zcode", "Zcode Games" },
{"scottadams", "Scott Adams Games"},
// Scott Adams games
{ "adventureland", "Adventureland" },
{ "pirateadventure", "Pirate Adventure" },
{ "missionimpossible", "Mission Impossible" },
{ "voodoocastle", "Voodoo Castle" },
{ "thecount", "The Count" },
{ "strangeodyssey", "Strange Odyssey" },
{ "mysteryfunhouse", "Mystery Fun House" },
{ "pyramidofdoom", "Pyramid Of Doom" },
{ "ghosttown", "Ghost Town" },
{ "savageisland1", "Savage Island, Part 1" },
{ "savageisland2", "Savage Island, Part 2" },
{ "goldenvoyage", "The Golden Voyage" },
{ "adventure13", "Adventure 13" },
{ "adventure14", "Adventure 14" },
{ "buckaroobonzai", "Buckaroo Banzai" },
2018-10-14 13:02:24 -07:00
{0, 0}
};
#include "common/config-manager.h"
#include "common/file.h"
2018-10-14 13:02:24 -07:00
#include "gargoyle/detection_tables.h"
#include "gargoyle/frotz/detection.h"
#include "gargoyle/frotz/frotz.h"
2018-10-17 21:14:04 -07:00
#include "gargoyle/scott/detection.h"
#include "gargoyle/scott/scott.h"
2018-10-14 13:02:24 -07:00
class GargoyleMetaEngine : public AdvancedMetaEngine {
public:
GargoyleMetaEngine() : AdvancedMetaEngine(Gargoyle::gameDescriptions, sizeof(Gargoyle::GargoyleGameDescription), gargoyleGames) {
2018-10-14 13:02:24 -07:00
_maxScanDepth = 3;
}
virtual const char *getName() const {
return "Gargoyle Engine";
}
virtual const char *getOriginalCopyright() const {
return "Gargoyle Engine (c) 2018";
2018-10-14 13:02:24 -07:00
}
2018-10-17 21:14:04 -07:00
virtual bool hasFeature(MetaEngineFeature f) const override;
virtual bool createInstance(OSystem *syst, Engine **engine, const ADGameDescription *desc) const override;
2018-10-14 13:02:24 -07:00
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;
2018-10-17 21:14:04 -07:00
virtual DetectedGames detectGames(const Common::FSList &fslist) const override;
virtual ADDetectedGames detectGame(const Common::FSNode &parent, const FileMap &allFiles, Common::Language language, Common::Platform platform, const Common::String &extra) const override;
2018-10-14 13:02:24 -07:00
};
bool GargoyleMetaEngine::hasFeature(MetaEngineFeature f) const {
return
2018-11-10 05:42:22 +00:00
(f == kSupportsListSaves) ||
(f == kSupportsLoadingDuringStartup) ||
(f == kSupportsDeleteSave) ||
(f == kSavesSupportMetaInfo) ||
(f == kSavesSupportCreationDate) ||
(f == kSavesSupportPlayTime) ||
(f == kSimpleSavesNames);
2018-10-14 13:02:24 -07:00
}
bool Gargoyle::GargoyleEngine::hasFeature(EngineFeature f) const {
return
2018-11-10 05:42:22 +00:00
(f == kSupportsRTL) ||
(f == kSupportsLoadingDuringRuntime) ||
(f == kSupportsSavingDuringRuntime);
2018-10-14 13:02:24 -07:00
}
bool GargoyleMetaEngine::createInstance(OSystem *syst, Engine **engine, const ADGameDescription *desc) const {
2018-11-10 18:25:20 +00:00
const Gargoyle::GargoyleGameDescription *gd = (const Gargoyle::GargoyleGameDescription *)desc;
switch (gd->_interpType) {
case Gargoyle::INTERPRETER_FROTZ:
*engine = new Gargoyle::Frotz::Frotz(syst, gd);
break;
case Gargoyle::INTERPRETER_SCOTT:
*engine = new Gargoyle::Scott::Scott(syst, gd);
break;
default:
error("Unknown interpreter");
}
2018-10-14 13:02:24 -07:00
return gd != 0;
}
SaveStateList GargoyleMetaEngine::listSaves(const char *target) const {
2018-11-04 15:11:11 -08:00
Common::SaveFileManager *saveFileMan = g_system->getSavefileManager();
Common::StringArray filenames;
Common::String saveDesc;
Common::String pattern = Common::String::format("%s.0##", target);
Gargoyle::SavegameHeader header;
filenames = saveFileMan->listSavefiles(pattern);
2018-10-14 13:02:24 -07:00
SaveStateList saveList;
2018-11-04 15:11:11 -08:00
for (Common::StringArray::const_iterator file = filenames.begin(); file != filenames.end(); ++file) {
const char *ext = strrchr(file->c_str(), '.');
int slot = ext ? atoi(ext + 1) : -1;
if (slot >= 0 && slot <= MAX_SAVES) {
Common::InSaveFile *in = g_system->getSavefileManager()->openForLoading(*file);
if (in) {
if (Gargoyle::FileStream::readSavegameHeader(in, header))
saveList.push_back(SaveStateDescriptor(slot, header._saveName));
delete in;
}
}
}
// Sort saves based on slot number.
Common::sort(saveList.begin(), saveList.end(), SaveStateDescriptorSlotComparator());
2018-10-14 13:02:24 -07:00
return saveList;
}
int GargoyleMetaEngine::getMaximumSaveSlot() const {
return MAX_SAVES;
}
void GargoyleMetaEngine::removeSaveState(const char *target, int slot) const {
}
SaveStateDescriptor GargoyleMetaEngine::querySaveMetaInfos(const char *target, int slot) const {
Common::String filename = Common::String::format("%s.%03d", target, slot);
Common::InSaveFile *in = g_system->getSavefileManager()->openForLoading(filename);
if (in) {
Gargoyle::SavegameHeader header;
if (Gargoyle::FileStream::readSavegameHeader(in, header)) {
// Create the return descriptor
SaveStateDescriptor desc(slot, header._saveName);
desc.setSaveDate(header._year, header._month, header._day);
desc.setSaveTime(header._hour, header._minute);
desc.setPlayTime(header._totalFrames * GAME_FRAME_TIME);
delete in;
return desc;
}
}
2018-10-14 13:02:24 -07:00
return SaveStateDescriptor();
}
2018-10-17 21:14:04 -07:00
DetectedGames GargoyleMetaEngine::detectGames(const Common::FSList &fslist) const {
DetectedGames detectedGames;
Gargoyle::Frotz::FrotzMetaEngine::detectGames(fslist, detectedGames);
2018-10-17 21:14:04 -07:00
Gargoyle::Scott::ScottMetaEngine::detectGames(fslist, detectedGames);
return detectedGames;
}
2018-10-14 13:02:24 -07:00
static Gargoyle::GargoyleGameDescription gameDescription;
ADDetectedGames GargoyleMetaEngine::detectGame(const Common::FSNode &parent, const FileMap &allFiles, Common::Language language, Common::Platform platform, const Common::String &extra) const {
static char gameId[100];
strcpy(gameId, ConfMan.get("gameid").c_str());
Common::String filename = ConfMan.get("filename");
2018-11-03 21:05:59 -07:00
Common::FSList fslist;
DetectedGames detectedGames;
fslist.push_back(parent.getChild(filename));
ADDetectedGames results;
Common::File f;
// Check each sub-engine for any detected games
if (Gargoyle::Frotz::FrotzMetaEngine::detectGames(fslist, detectedGames))
gameDescription._interpType = Gargoyle::INTERPRETER_FROTZ;
else if (Gargoyle::Scott::ScottMetaEngine::detectGames(fslist, detectedGames))
gameDescription._interpType = Gargoyle::INTERPRETER_SCOTT;
else
// No match found, so return no results
return results;
// Set up the game description and return it
if (f.open(parent.getChild(filename))) {
2018-11-03 21:05:59 -07:00
DetectedGame gd = detectedGames.front();
gameDescription._desc.gameId = gameId;
2018-11-03 21:05:59 -07:00
gameDescription._desc.language = gd.language;
gameDescription._desc.platform = gd.platform;
gameDescription._desc.guiOptions = GUIO4(GUIO_NOSPEECH, GUIO_NOSFX, GUIO_NOMUSIC, GUIO_NOSUBTITLES);
gameDescription._filename = filename;
gameDescription._md5 = Common::computeStreamMD5AsString(f, 5000);
ADDetectedGame dg((ADGameDescription *)&gameDescription);
2018-11-03 21:05:59 -07:00
results.push_back(dg);
}
2018-11-03 21:05:59 -07:00
return results;
}
2018-10-14 13:02:24 -07:00
#if PLUGIN_ENABLED_DYNAMIC(GARGOYLE)
2018-11-10 05:42:22 +00:00
REGISTER_PLUGIN_DYNAMIC(Gargoyle, PLUGIN_TYPE_ENGINE, GargoyleMetaEngine);
2018-10-14 13:02:24 -07:00
#else
2018-11-10 05:42:22 +00:00
REGISTER_PLUGIN_STATIC(GARGOYLE, PLUGIN_TYPE_ENGINE, GargoyleMetaEngine);
2018-10-14 13:02:24 -07:00
#endif