2012-09-04 20:17:23 +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.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This file is based on WME Lite.
|
|
|
|
* http://dead-code.org/redir.php?target=wmelite
|
|
|
|
* Copyright (c) 2011 Jan Nedoma
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "engines/wintermute/dcgf.h"
|
|
|
|
#include "engines/wintermute/base/base_file_manager.h"
|
|
|
|
#include "engines/wintermute/base/base_game.h"
|
|
|
|
#include "engines/wintermute/base/base_engine.h"
|
|
|
|
#include "engines/wintermute/base/base_persistence_manager.h"
|
|
|
|
#include "engines/wintermute/base/base_save_thumb_helper.h"
|
|
|
|
#include "engines/wintermute/platform_osystem.h"
|
|
|
|
#include "engines/wintermute/math/vector2.h"
|
|
|
|
#include "engines/wintermute/base/gfx/base_image.h"
|
|
|
|
#include "engines/wintermute/base/sound/base_sound.h"
|
2012-12-13 20:35:08 +00:00
|
|
|
#include "engines/wintermute/graphics/transparent_surface.h"
|
2012-09-04 20:17:23 +00:00
|
|
|
#include "engines/wintermute/wintermute.h"
|
|
|
|
#include "graphics/decoders/bmp.h"
|
2012-12-13 20:35:08 +00:00
|
|
|
#include "graphics/scaler.h"
|
2012-09-04 20:17:23 +00:00
|
|
|
#include "common/memstream.h"
|
|
|
|
#include "common/str.h"
|
|
|
|
#include "common/system.h"
|
|
|
|
#include "common/savefile.h"
|
|
|
|
|
|
|
|
namespace Wintermute {
|
|
|
|
|
2013-01-31 22:59:02 +00:00
|
|
|
// The original WME-Lite savegames had the following:
|
|
|
|
//#define SAVE_MAGIC 0x45564153
|
|
|
|
//#define SAVE_MAGIC_2 0x32564153
|
|
|
|
// In case anyone tries to load original savegames, or for that matter
|
|
|
|
// in case we ever want to attempt to support original savegames, we
|
|
|
|
// avoid those numbers, and use this instead:
|
|
|
|
#define SAVE_MAGIC_3 0x12564154
|
2012-09-04 20:17:23 +00:00
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
BasePersistenceManager::BasePersistenceManager(const char *savePrefix, bool deleteSingleton) {
|
|
|
|
_saving = false;
|
|
|
|
_offset = 0;
|
2013-01-26 17:06:42 +00:00
|
|
|
_saveStream = nullptr;
|
|
|
|
_loadStream = nullptr;
|
2012-09-04 20:17:23 +00:00
|
|
|
_deleteSingleton = deleteSingleton;
|
|
|
|
if (BaseEngine::instance().getGameRef()) {
|
|
|
|
_gameRef = BaseEngine::instance().getGameRef();
|
|
|
|
} else {
|
2013-01-26 17:06:42 +00:00
|
|
|
_gameRef = nullptr;
|
2012-09-04 20:17:23 +00:00
|
|
|
}
|
|
|
|
|
2013-01-26 17:06:42 +00:00
|
|
|
_richBuffer = nullptr;
|
2012-09-04 20:17:23 +00:00
|
|
|
_richBufferSize = 0;
|
|
|
|
|
2013-01-26 17:06:42 +00:00
|
|
|
_scummVMThumbnailData = nullptr;
|
2013-01-22 19:58:12 +00:00
|
|
|
_scummVMThumbSize = 0;
|
|
|
|
|
2013-01-26 17:06:42 +00:00
|
|
|
_savedDescription = nullptr;
|
2012-09-04 20:17:23 +00:00
|
|
|
// _savedTimestamp = 0;
|
|
|
|
_savedVerMajor = _savedVerMinor = _savedVerBuild = 0;
|
|
|
|
_savedExtMajor = _savedExtMinor = 0;
|
|
|
|
|
|
|
|
_thumbnailDataSize = 0;
|
2013-01-26 17:06:42 +00:00
|
|
|
_thumbnailData = nullptr;
|
2012-09-04 20:17:23 +00:00
|
|
|
if (savePrefix) {
|
|
|
|
_savePrefix = savePrefix;
|
|
|
|
} else if (_gameRef) {
|
|
|
|
_savePrefix = _gameRef->getGameId();
|
|
|
|
} else {
|
|
|
|
_savePrefix = "wmesav";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
BasePersistenceManager::~BasePersistenceManager() {
|
|
|
|
cleanup();
|
2013-01-26 17:06:42 +00:00
|
|
|
if (_deleteSingleton && BaseEngine::instance().getGameRef() == nullptr)
|
2012-09-04 20:17:23 +00:00
|
|
|
BaseEngine::destroy();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
void BasePersistenceManager::cleanup() {
|
|
|
|
_offset = 0;
|
|
|
|
|
|
|
|
delete[] _richBuffer;
|
2013-01-26 17:06:42 +00:00
|
|
|
_richBuffer = nullptr;
|
2012-09-04 20:17:23 +00:00
|
|
|
_richBufferSize = 0;
|
|
|
|
|
|
|
|
delete[] _savedDescription;
|
2013-01-26 17:06:42 +00:00
|
|
|
_savedDescription = nullptr; // ref to buffer
|
2012-09-04 20:17:23 +00:00
|
|
|
// _savedTimestamp = 0;
|
|
|
|
_savedVerMajor = _savedVerMinor = _savedVerBuild = 0;
|
|
|
|
_savedExtMajor = _savedExtMinor = 0;
|
|
|
|
|
|
|
|
_thumbnailDataSize = 0;
|
|
|
|
if (_thumbnailData) {
|
|
|
|
delete[] _thumbnailData;
|
2013-01-26 17:06:42 +00:00
|
|
|
_thumbnailData = nullptr;
|
2012-09-04 20:17:23 +00:00
|
|
|
}
|
|
|
|
|
2013-01-22 19:58:12 +00:00
|
|
|
_scummVMThumbSize = 0;
|
|
|
|
if (_scummVMThumbnailData) {
|
|
|
|
delete[] _scummVMThumbnailData;
|
2013-01-26 17:06:42 +00:00
|
|
|
_scummVMThumbnailData = nullptr;
|
2013-01-22 19:58:12 +00:00
|
|
|
}
|
|
|
|
|
2012-09-04 20:17:23 +00:00
|
|
|
delete _loadStream;
|
|
|
|
delete _saveStream;
|
2013-01-26 17:06:42 +00:00
|
|
|
_loadStream = nullptr;
|
|
|
|
_saveStream = nullptr;
|
2012-09-04 20:17:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Common::String BasePersistenceManager::getFilenameForSlot(int slot) const {
|
|
|
|
// 3 Digits, to allow for one save-slot for autosave + slot 1 - 100 (which will be numbered 0-99 filename-wise)
|
2013-01-22 19:58:12 +00:00
|
|
|
return Common::String::format("%s.%03d", _savePrefix.c_str(), slot);
|
2012-09-04 20:17:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void BasePersistenceManager::getSaveStateDesc(int slot, SaveStateDescriptor &desc) {
|
|
|
|
Common::String filename = getFilenameForSlot(slot);
|
|
|
|
debugC(kWintermuteDebugSaveGame, "Trying to list savegame %s in slot %d", filename.c_str(), slot);
|
|
|
|
if (DID_FAIL(readHeader(filename))) {
|
2013-01-31 22:59:02 +00:00
|
|
|
debugC(kWintermuteDebugSaveGame, "getSavedDesc(%d) - Failed for %s", slot, filename.c_str());
|
2012-09-04 20:17:23 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
desc.setSaveSlot(slot);
|
|
|
|
desc.setDescription(_savedDescription);
|
|
|
|
desc.setDeletableFlag(true);
|
|
|
|
desc.setWriteProtectedFlag(false);
|
|
|
|
|
2013-01-22 19:58:12 +00:00
|
|
|
int thumbSize = 0;
|
2013-01-26 17:06:42 +00:00
|
|
|
byte *thumbData = nullptr;
|
2013-01-22 19:58:12 +00:00
|
|
|
if (_scummVMThumbSize > 0) {
|
|
|
|
thumbSize = _scummVMThumbSize;
|
|
|
|
thumbData = _scummVMThumbnailData;
|
|
|
|
} else if (_thumbnailDataSize > 0) {
|
|
|
|
thumbSize = _thumbnailDataSize;
|
|
|
|
thumbData = _thumbnailData;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (thumbSize > 0) {
|
2013-01-24 16:14:51 +00:00
|
|
|
Common::MemoryReadStream thumbStream(thumbData, thumbSize, DisposeAfterUse::NO);
|
2012-09-04 20:17:23 +00:00
|
|
|
Graphics::BitmapDecoder bmpDecoder;
|
|
|
|
if (bmpDecoder.loadStream(thumbStream)) {
|
2013-01-24 16:14:51 +00:00
|
|
|
const Graphics::Surface *bmpSurface = bmpDecoder.getSurface();
|
|
|
|
TransparentSurface *scaleableSurface = new TransparentSurface(*bmpSurface, false);
|
2012-12-13 20:35:08 +00:00
|
|
|
Graphics::Surface *scaled = scaleableSurface->scale(kThumbnailWidth, kThumbnailHeight2);
|
2013-01-24 16:14:51 +00:00
|
|
|
Graphics::Surface *thumb = scaled->convertTo(g_system->getOverlayFormat());
|
|
|
|
desc.setThumbnail(thumb);
|
2012-12-13 20:35:08 +00:00
|
|
|
delete scaleableSurface;
|
2013-01-24 16:14:51 +00:00
|
|
|
scaled->free();
|
|
|
|
delete scaled;
|
2012-09-04 20:17:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
desc.setSaveDate(_savedTimestamp.tm_year, _savedTimestamp.tm_mon, _savedTimestamp.tm_mday);
|
|
|
|
desc.setSaveTime(_savedTimestamp.tm_hour, _savedTimestamp.tm_min);
|
|
|
|
desc.setPlayTime(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void BasePersistenceManager::deleteSaveSlot(int slot) {
|
|
|
|
Common::String filename = getFilenameForSlot(slot);
|
|
|
|
g_system->getSavefileManager()->removeSavefile(filename);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32 BasePersistenceManager::getMaxUsedSlot() {
|
2013-01-22 19:58:12 +00:00
|
|
|
Common::String saveMask = Common::String::format("%s.???", _savePrefix.c_str());
|
2012-09-04 20:17:23 +00:00
|
|
|
Common::StringArray saves = g_system->getSavefileManager()->listSavefiles(saveMask);
|
|
|
|
Common::StringArray::iterator it = saves.begin();
|
|
|
|
int ret = -1;
|
|
|
|
for (; it != saves.end(); ++it) {
|
|
|
|
int num = -1;
|
2013-01-22 19:58:12 +00:00
|
|
|
sscanf(it->c_str(), ".%d", &num);
|
2012-09-04 20:17:23 +00:00
|
|
|
ret = MAX(ret, num);
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool BasePersistenceManager::getSaveExists(int slot) {
|
|
|
|
Common::String filename = getFilenameForSlot(slot);
|
|
|
|
if (DID_FAIL(readHeader(filename))) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
bool BasePersistenceManager::initSave(const char *desc) {
|
|
|
|
if (!desc) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup();
|
|
|
|
_saving = true;
|
|
|
|
|
|
|
|
_saveStream = new Common::MemoryWriteStreamDynamic(DisposeAfterUse::YES);
|
|
|
|
|
|
|
|
if (_saveStream) {
|
|
|
|
// get thumbnails
|
|
|
|
if (!_gameRef->_cachedThumbnail) {
|
|
|
|
_gameRef->_cachedThumbnail = new BaseSaveThumbHelper(_gameRef);
|
|
|
|
if (DID_FAIL(_gameRef->_cachedThumbnail->storeThumbnail(true))) {
|
|
|
|
delete _gameRef->_cachedThumbnail;
|
2013-01-26 17:06:42 +00:00
|
|
|
_gameRef->_cachedThumbnail = nullptr;
|
2012-09-04 20:17:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32 magic = DCGF_MAGIC;
|
|
|
|
putDWORD(magic);
|
|
|
|
|
2013-01-31 22:59:02 +00:00
|
|
|
magic = SAVE_MAGIC_3;
|
2012-09-04 20:17:23 +00:00
|
|
|
putDWORD(magic);
|
|
|
|
|
|
|
|
byte verMajor, verMinor, extMajor, extMinor;
|
|
|
|
_gameRef->getVersion(&verMajor, &verMinor, &extMajor, &extMinor);
|
|
|
|
_saveStream->writeByte(verMajor);
|
|
|
|
_saveStream->writeByte(verMinor);
|
|
|
|
_saveStream->writeByte(extMajor);
|
|
|
|
_saveStream->writeByte(extMinor);
|
|
|
|
|
|
|
|
// new in ver 2
|
|
|
|
putDWORD((uint32)DCGF_VER_BUILD);
|
|
|
|
putString(_gameRef->getName());
|
|
|
|
|
|
|
|
// thumbnail data size
|
|
|
|
bool thumbnailOK = false;
|
|
|
|
|
|
|
|
if (_gameRef->_cachedThumbnail) {
|
|
|
|
if (_gameRef->_cachedThumbnail->_thumbnail) {
|
|
|
|
Common::MemoryWriteStreamDynamic thumbStream(DisposeAfterUse::YES);
|
|
|
|
if (_gameRef->_cachedThumbnail->_thumbnail->writeBMPToStream(&thumbStream)) {
|
|
|
|
_saveStream->writeUint32LE(thumbStream.size());
|
|
|
|
_saveStream->write(thumbStream.getData(), thumbStream.size());
|
|
|
|
} else {
|
|
|
|
_saveStream->writeUint32LE(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
thumbnailOK = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!thumbnailOK) {
|
|
|
|
putDWORD(0);
|
|
|
|
}
|
2013-01-22 19:58:12 +00:00
|
|
|
thumbnailOK = false;
|
|
|
|
// Again for the ScummVM-thumb:
|
|
|
|
if (_gameRef->_cachedThumbnail) {
|
|
|
|
if (_gameRef->_cachedThumbnail->_scummVMThumb) {
|
|
|
|
Common::MemoryWriteStreamDynamic scummVMthumbStream(DisposeAfterUse::YES);
|
|
|
|
if (_gameRef->_cachedThumbnail->_scummVMThumb->writeBMPToStream(&scummVMthumbStream)) {
|
|
|
|
_saveStream->writeUint32LE(scummVMthumbStream.size());
|
|
|
|
_saveStream->write(scummVMthumbStream.getData(), scummVMthumbStream.size());
|
|
|
|
} else {
|
|
|
|
_saveStream->writeUint32LE(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
thumbnailOK = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!thumbnailOK) {
|
|
|
|
putDWORD(0);
|
|
|
|
}
|
|
|
|
|
2012-09-04 20:17:23 +00:00
|
|
|
|
|
|
|
// in any case, destroy the cached thumbnail once used
|
|
|
|
delete _gameRef->_cachedThumbnail;
|
2013-01-26 17:06:42 +00:00
|
|
|
_gameRef->_cachedThumbnail = nullptr;
|
2012-09-04 20:17:23 +00:00
|
|
|
|
|
|
|
uint32 dataOffset = _offset +
|
|
|
|
sizeof(uint32) + // data offset
|
|
|
|
sizeof(uint32) + strlen(desc) + 1 + // description
|
|
|
|
sizeof(uint32); // timestamp
|
|
|
|
|
|
|
|
putDWORD(dataOffset);
|
|
|
|
putString(desc);
|
|
|
|
|
|
|
|
g_system->getTimeAndDate(_savedTimestamp);
|
|
|
|
putTimeDate(_savedTimestamp);
|
|
|
|
_savedPlayTime = g_system->getMillis();
|
|
|
|
_saveStream->writeUint32LE(_savedPlayTime);
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool BasePersistenceManager::readHeader(const Common::String &filename) {
|
|
|
|
cleanup();
|
|
|
|
|
|
|
|
_saving = false;
|
|
|
|
|
|
|
|
_loadStream = g_system->getSavefileManager()->openForLoading(filename);
|
2013-01-31 22:59:02 +00:00
|
|
|
|
2012-09-04 20:17:23 +00:00
|
|
|
if (_loadStream) {
|
|
|
|
uint32 magic;
|
|
|
|
magic = getDWORD();
|
|
|
|
|
|
|
|
if (magic != DCGF_MAGIC) {
|
|
|
|
cleanup();
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
magic = getDWORD();
|
|
|
|
|
2013-01-31 22:59:02 +00:00
|
|
|
if (magic == SAVE_MAGIC_3) {
|
2012-09-04 20:17:23 +00:00
|
|
|
_savedVerMajor = _loadStream->readByte();
|
|
|
|
_savedVerMinor = _loadStream->readByte();
|
|
|
|
_savedExtMajor = _loadStream->readByte();
|
|
|
|
_savedExtMinor = _loadStream->readByte();
|
|
|
|
|
2013-01-31 22:59:02 +00:00
|
|
|
_savedVerBuild = (byte)getDWORD();
|
|
|
|
_savedName = getStringObj();
|
|
|
|
|
|
|
|
// load thumbnail
|
|
|
|
_thumbnailDataSize = getDWORD();
|
|
|
|
if (_thumbnailDataSize > 0) {
|
|
|
|
_thumbnailData = new byte[_thumbnailDataSize];
|
|
|
|
if (_thumbnailData) {
|
|
|
|
getBytes(_thumbnailData, _thumbnailDataSize);
|
|
|
|
} else {
|
|
|
|
_thumbnailDataSize = 0;
|
2013-01-22 19:58:12 +00:00
|
|
|
}
|
2013-01-31 22:59:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
_scummVMThumbSize = getDWORD();
|
|
|
|
_scummVMThumbnailData = new byte[_scummVMThumbSize];
|
|
|
|
if (_scummVMThumbnailData) {
|
|
|
|
getBytes(_scummVMThumbnailData, _scummVMThumbSize);
|
2012-09-04 20:17:23 +00:00
|
|
|
} else {
|
2013-01-31 22:59:02 +00:00
|
|
|
_scummVMThumbSize = 0;
|
2012-09-04 20:17:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
uint32 dataOffset = getDWORD();
|
|
|
|
|
|
|
|
_savedDescription = getString();
|
|
|
|
_savedTimestamp = getTimeDate();
|
|
|
|
_savedPlayTime = _loadStream->readUint32LE();
|
|
|
|
|
|
|
|
_offset = dataOffset;
|
|
|
|
|
|
|
|
return STATUS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup();
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
bool BasePersistenceManager::initLoad(const Common::String &filename) {
|
|
|
|
if (DID_FAIL(readHeader(filename))) {
|
|
|
|
cleanup();
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
_saving = false;
|
|
|
|
|
|
|
|
if (_savedName == "" || scumm_stricmp(_savedName.c_str(), _gameRef->getName()) != 0) {
|
|
|
|
debugC(kWintermuteDebugSaveGame, "ERROR: Saved game name doesn't match current game");
|
|
|
|
cleanup();
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
// if save is newer version than we are, fail
|
|
|
|
if (_savedVerMajor > DCGF_VER_MAJOR ||
|
|
|
|
(_savedVerMajor == DCGF_VER_MAJOR && _savedVerMinor > DCGF_VER_MINOR) ||
|
|
|
|
(_savedVerMajor == DCGF_VER_MAJOR && _savedVerMinor == DCGF_VER_MINOR && _savedVerBuild > DCGF_VER_BUILD)
|
|
|
|
) {
|
|
|
|
|
|
|
|
debugC(kWintermuteDebugSaveGame, "ERROR: Saved game version is newer than current game");
|
|
|
|
debugC(kWintermuteDebugSaveGame, "ERROR: Expected %d.%d.%d got %d.%d.%d", DCGF_VER_MAJOR, DCGF_VER_MINOR, DCGF_VER_BUILD, _savedVerMajor, _savedVerMinor, _savedVerBuild);
|
|
|
|
cleanup();
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
// if save is older than the minimal version we support
|
|
|
|
if (_savedVerMajor < SAVEGAME_VER_MAJOR ||
|
|
|
|
(_savedVerMajor == SAVEGAME_VER_MAJOR && _savedVerMinor < SAVEGAME_VER_MINOR) ||
|
|
|
|
(_savedVerMajor == SAVEGAME_VER_MAJOR && _savedVerMinor == SAVEGAME_VER_MINOR && _savedVerBuild < SAVEGAME_VER_BUILD)
|
|
|
|
) {
|
|
|
|
debugC(kWintermuteDebugSaveGame, "ERROR: Saved game is too old and cannot be used by this version of game engine");
|
|
|
|
debugC(kWintermuteDebugSaveGame, "ERROR: Expected %d.%d.%d got %d.%d.%d", DCGF_VER_MAJOR, DCGF_VER_MINOR, DCGF_VER_BUILD, _savedVerMajor, _savedVerMinor, _savedVerBuild);
|
|
|
|
cleanup();
|
|
|
|
return STATUS_FAILED;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return STATUS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
bool BasePersistenceManager::saveFile(const Common::String &filename) {
|
|
|
|
byte *prefixBuffer = _richBuffer;
|
|
|
|
uint32 prefixSize = _richBufferSize;
|
|
|
|
byte *buffer = ((Common::MemoryWriteStreamDynamic *)_saveStream)->getData();
|
|
|
|
uint32 bufferSize = ((Common::MemoryWriteStreamDynamic *)_saveStream)->size();
|
|
|
|
|
|
|
|
Common::SaveFileManager *saveMan = ((WintermuteEngine *)g_engine)->getSaveFileMan();
|
|
|
|
Common::OutSaveFile *file = saveMan->openForSaving(filename);
|
|
|
|
file->write(prefixBuffer, prefixSize);
|
|
|
|
file->write(buffer, bufferSize);
|
|
|
|
bool retVal = !file->err();
|
|
|
|
file->finalize();
|
|
|
|
delete file;
|
|
|
|
return retVal;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
bool BasePersistenceManager::putBytes(byte *buffer, uint32 size) {
|
|
|
|
_saveStream->write(buffer, size);
|
|
|
|
if (_saveStream->err()) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
bool BasePersistenceManager::getBytes(byte *buffer, uint32 size) {
|
|
|
|
_loadStream->read(buffer, size);
|
|
|
|
if (_loadStream->err()) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
void BasePersistenceManager::putDWORD(uint32 val) {
|
|
|
|
_saveStream->writeUint32LE(val);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
uint32 BasePersistenceManager::getDWORD() {
|
|
|
|
uint32 ret = _loadStream->readUint32LE();
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
void BasePersistenceManager::putString(const Common::String &val) {
|
|
|
|
if (!val.size()) {
|
|
|
|
putString("(null)");
|
|
|
|
} else {
|
|
|
|
_saveStream->writeUint32LE(val.size());
|
|
|
|
_saveStream->writeString(val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Common::String BasePersistenceManager::getStringObj() {
|
|
|
|
uint32 len = _loadStream->readUint32LE();
|
|
|
|
char *ret = new char[len + 1];
|
|
|
|
_loadStream->read(ret, len);
|
|
|
|
ret[len] = '\0';
|
|
|
|
|
|
|
|
Common::String retString = ret;
|
|
|
|
delete[] ret;
|
|
|
|
|
|
|
|
if (retString == "(null)") {
|
|
|
|
retString = "";
|
|
|
|
}
|
|
|
|
|
|
|
|
return retString;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
char *BasePersistenceManager::getString() {
|
|
|
|
uint32 len = _loadStream->readUint32LE();
|
|
|
|
char *ret = new char[len + 1];
|
|
|
|
_loadStream->read(ret, len);
|
|
|
|
ret[len] = '\0';
|
|
|
|
|
|
|
|
if (!strcmp(ret, "(null)")) {
|
|
|
|
delete[] ret;
|
2013-01-26 17:06:42 +00:00
|
|
|
return nullptr;
|
2012-09-04 20:17:23 +00:00
|
|
|
} else {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool BasePersistenceManager::putTimeDate(const TimeDate &t) {
|
|
|
|
_saveStream->writeSint32LE(t.tm_sec);
|
|
|
|
_saveStream->writeSint32LE(t.tm_min);
|
|
|
|
_saveStream->writeSint32LE(t.tm_hour);
|
|
|
|
_saveStream->writeSint32LE(t.tm_mday);
|
|
|
|
_saveStream->writeSint32LE(t.tm_mon);
|
|
|
|
_saveStream->writeSint32LE(t.tm_year);
|
2013-01-31 23:48:47 +00:00
|
|
|
_saveStream->writeSint32LE(t.tm_wday);
|
2012-09-04 20:17:23 +00:00
|
|
|
|
|
|
|
if (_saveStream->err()) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
TimeDate BasePersistenceManager::getTimeDate() {
|
|
|
|
TimeDate t;
|
|
|
|
t.tm_sec = _loadStream->readSint32LE();
|
|
|
|
t.tm_min = _loadStream->readSint32LE();
|
|
|
|
t.tm_hour = _loadStream->readSint32LE();
|
|
|
|
t.tm_mday = _loadStream->readSint32LE();
|
|
|
|
t.tm_mon = _loadStream->readSint32LE();
|
|
|
|
t.tm_year = _loadStream->readSint32LE();
|
2013-01-31 23:48:47 +00:00
|
|
|
t.tm_wday = _loadStream->readSint32LE();
|
2012-09-04 20:17:23 +00:00
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
|
|
|
void BasePersistenceManager::putFloat(float val) {
|
2013-01-31 22:59:02 +00:00
|
|
|
int32 exponent = 0;
|
2013-01-31 23:48:47 +00:00
|
|
|
float significand = frexp(val, &exponent);
|
2013-01-31 22:59:02 +00:00
|
|
|
Common::String str = Common::String::format("FS%f", significand);
|
2012-09-04 20:17:23 +00:00
|
|
|
_saveStream->writeUint32LE(str.size());
|
|
|
|
_saveStream->writeString(str);
|
2013-01-31 22:59:02 +00:00
|
|
|
_saveStream->writeSint32LE(exponent);
|
2012-09-04 20:17:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
float BasePersistenceManager::getFloat() {
|
|
|
|
char *str = getString();
|
|
|
|
float value = 0.0f;
|
2013-01-31 22:59:02 +00:00
|
|
|
float significand = 0.0f;
|
|
|
|
int32 exponent = _loadStream->readSint32LE();
|
|
|
|
int ret = sscanf(str, "FS%f", &significand);
|
2013-01-31 23:48:47 +00:00
|
|
|
value = ldexp(significand, exponent);
|
2012-09-04 20:17:23 +00:00
|
|
|
if (ret != 1) {
|
|
|
|
warning("%s not parsed as float", str);
|
|
|
|
}
|
|
|
|
delete[] str;
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
void BasePersistenceManager::putDouble(double val) {
|
2013-01-31 22:59:02 +00:00
|
|
|
int32 exponent = 0;
|
|
|
|
double significand = frexp(val, &exponent);
|
|
|
|
Common::String str = Common::String::format("DS%f", significand);
|
2012-09-04 20:17:23 +00:00
|
|
|
_saveStream->writeUint32LE(str.size());
|
|
|
|
_saveStream->writeString(str);
|
2013-01-31 22:59:02 +00:00
|
|
|
_saveStream->writeSint32LE(exponent);
|
2012-09-04 20:17:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
double BasePersistenceManager::getDouble() {
|
|
|
|
char *str = getString();
|
2013-01-31 22:59:02 +00:00
|
|
|
double value = 0.0f;
|
|
|
|
float significand = 0.0f;
|
|
|
|
int32 exponent = _loadStream->readSint32LE();
|
|
|
|
int ret = sscanf(str, "DS%f", &significand);
|
|
|
|
value = ldexp(significand, exponent);
|
2012-09-04 20:17:23 +00:00
|
|
|
if (ret != 1) {
|
|
|
|
warning("%s not parsed as double", str);
|
|
|
|
}
|
|
|
|
delete[] str;
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
// bool
|
|
|
|
bool BasePersistenceManager::transfer(const char *name, bool *val) {
|
|
|
|
if (_saving) {
|
|
|
|
_saveStream->writeByte(*val);
|
|
|
|
if (_saveStream->err()) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
} else {
|
|
|
|
*val = _loadStream->readByte();
|
|
|
|
if (_loadStream->err()) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
// int
|
|
|
|
bool BasePersistenceManager::transfer(const char *name, int *val) {
|
|
|
|
if (_saving) {
|
|
|
|
_saveStream->writeSint32LE(*val);
|
|
|
|
if (_saveStream->err()) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
} else {
|
|
|
|
*val = _loadStream->readSint32LE();
|
|
|
|
if (_loadStream->err()) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
// DWORD
|
|
|
|
bool BasePersistenceManager::transfer(const char *name, uint32 *val) {
|
|
|
|
if (_saving) {
|
|
|
|
_saveStream->writeUint32LE(*val);
|
|
|
|
if (_saveStream->err()) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
} else {
|
|
|
|
*val = _loadStream->readUint32LE();
|
|
|
|
if (_loadStream->err()) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
// float
|
|
|
|
bool BasePersistenceManager::transfer(const char *name, float *val) {
|
|
|
|
if (_saving) {
|
|
|
|
putFloat(*val);
|
|
|
|
if (_saveStream->err()) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
} else {
|
|
|
|
*val = getFloat();
|
|
|
|
if (_loadStream->err()) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
// double
|
|
|
|
bool BasePersistenceManager::transfer(const char *name, double *val) {
|
|
|
|
if (_saving) {
|
|
|
|
putDouble(*val);
|
|
|
|
if (_saveStream->err()) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
} else {
|
|
|
|
*val = getDouble();
|
|
|
|
if (_loadStream->err()) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
// char*
|
|
|
|
bool BasePersistenceManager::transfer(const char *name, char **val) {
|
|
|
|
if (_saving) {
|
|
|
|
putString(*val);
|
|
|
|
return STATUS_OK;
|
|
|
|
} else {
|
|
|
|
char *str = getString();
|
|
|
|
if (_loadStream->err()) {
|
|
|
|
delete[] str;
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
*val = str;
|
|
|
|
return STATUS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
// const char*
|
|
|
|
bool BasePersistenceManager::transfer(const char *name, const char **val) {
|
|
|
|
if (_saving) {
|
|
|
|
putString(*val);
|
|
|
|
return STATUS_OK;
|
|
|
|
} else {
|
|
|
|
char *str = getString();
|
|
|
|
if (_loadStream->err()) {
|
|
|
|
delete[] str;
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
*val = str;
|
|
|
|
return STATUS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
// Common::String
|
|
|
|
bool BasePersistenceManager::transfer(const char *name, Common::String *val) {
|
|
|
|
if (_saving) {
|
|
|
|
putString(*val);
|
|
|
|
return STATUS_OK;
|
|
|
|
} else {
|
|
|
|
char *str = getString();
|
|
|
|
if (_loadStream->err()) {
|
|
|
|
delete[] str;
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
if (str) {
|
|
|
|
*val = str;
|
|
|
|
delete[] str;
|
|
|
|
} else {
|
|
|
|
*val = "";
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
bool BasePersistenceManager::transfer(const char *name, AnsiStringArray &val) {
|
|
|
|
size_t size;
|
|
|
|
|
|
|
|
if (_saving) {
|
|
|
|
size = val.size();
|
|
|
|
_saveStream->writeUint32LE(size);
|
|
|
|
|
|
|
|
for (AnsiStringArray::iterator it = val.begin(); it != val.end(); ++it) {
|
|
|
|
putString((*it).c_str());
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
val.clear();
|
|
|
|
size = _loadStream->readUint32LE();
|
|
|
|
|
|
|
|
for (size_t i = 0; i < size; i++) {
|
|
|
|
char *str = getString();
|
|
|
|
if (_loadStream->err()) {
|
|
|
|
delete[] str;
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
if (str) {
|
|
|
|
val.push_back(str);
|
|
|
|
}
|
|
|
|
delete[] str;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return STATUS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
// BYTE
|
|
|
|
bool BasePersistenceManager::transfer(const char *name, byte *val) {
|
|
|
|
if (_saving) {
|
|
|
|
_saveStream->writeByte(*val);
|
|
|
|
if (_saveStream->err()) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
} else {
|
|
|
|
*val = _loadStream->readByte();
|
|
|
|
if (_loadStream->err()) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
// RECT
|
|
|
|
bool BasePersistenceManager::transfer(const char *name, Rect32 *val) {
|
|
|
|
if (_saving) {
|
|
|
|
_saveStream->writeSint32LE(val->left);
|
|
|
|
_saveStream->writeSint32LE(val->top);
|
|
|
|
_saveStream->writeSint32LE(val->right);
|
|
|
|
_saveStream->writeSint32LE(val->bottom);
|
|
|
|
if (_saveStream->err()) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
} else {
|
|
|
|
val->left = _loadStream->readSint32LE();
|
|
|
|
val->top = _loadStream->readSint32LE();
|
|
|
|
val->right = _loadStream->readSint32LE();
|
|
|
|
val->bottom = _loadStream->readSint32LE();
|
|
|
|
if (_loadStream->err()) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
// POINT
|
|
|
|
bool BasePersistenceManager::transfer(const char *name, Point32 *val) {
|
|
|
|
if (_saving) {
|
|
|
|
_saveStream->writeSint32LE(val->x);
|
|
|
|
_saveStream->writeSint32LE(val->y);
|
|
|
|
if (_saveStream->err()) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
} else {
|
|
|
|
val->x = _loadStream->readSint32LE();
|
|
|
|
val->y = _loadStream->readSint32LE();
|
|
|
|
if (_loadStream->err()) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
// Vector2
|
|
|
|
bool BasePersistenceManager::transfer(const char *name, Vector2 *val) {
|
|
|
|
if (_saving) {
|
|
|
|
putFloat(val->x);
|
|
|
|
putFloat(val->y);
|
|
|
|
if (_saveStream->err()) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
} else {
|
|
|
|
val->x = getFloat();
|
|
|
|
val->y = getFloat();
|
|
|
|
if (_loadStream->err()) {
|
|
|
|
return STATUS_FAILED;
|
|
|
|
}
|
|
|
|
return STATUS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
// generic pointer
|
|
|
|
bool BasePersistenceManager::transfer(const char *name, void *val) {
|
|
|
|
int classID = -1, instanceID = -1;
|
|
|
|
|
|
|
|
if (_saving) {
|
|
|
|
SystemClassRegistry::getInstance()->getPointerID(*(void **)val, &classID, &instanceID);
|
2013-01-26 17:06:42 +00:00
|
|
|
if (*(void **)val != nullptr && (classID == -1 || instanceID == -1)) {
|
2012-09-04 20:17:23 +00:00
|
|
|
debugC(kWintermuteDebugSaveGame, "Warning: invalid instance '%s'", name);
|
|
|
|
}
|
|
|
|
|
|
|
|
_saveStream->writeUint32LE(classID);
|
|
|
|
_saveStream->writeUint32LE(instanceID);
|
|
|
|
} else {
|
|
|
|
classID = _loadStream->readUint32LE();
|
|
|
|
instanceID = _loadStream->readUint32LE();
|
|
|
|
|
|
|
|
*(void **)val = SystemClassRegistry::getInstance()->idToPointer(classID, instanceID);
|
|
|
|
}
|
|
|
|
|
|
|
|
return STATUS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
bool BasePersistenceManager::checkVersion(byte verMajor, byte verMinor, byte verBuild) {
|
|
|
|
if (_saving) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// it's ok if we are same or newer than the saved game
|
|
|
|
if (verMajor > _savedVerMajor ||
|
|
|
|
(verMajor == _savedVerMajor && verMinor > _savedVerMinor) ||
|
|
|
|
(verMajor == _savedVerMajor && verMinor == _savedVerMinor && verBuild > _savedVerBuild)
|
|
|
|
) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // end of namespace Wintermute
|