scummvm/engines/gob/save/savehandler.cpp
2021-12-26 18:48:43 +01:00

532 lines
12 KiB
C++

/* 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 3 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, see <http://www.gnu.org/licenses/>.
*
*/
#include "common/endian.h"
#include "gob/gob.h"
#include "gob/save/savehandler.h"
#include "gob/save/savefile.h"
#include "gob/save/saveconverter.h"
#include "gob/global.h"
#include "gob/video.h"
#include "gob/draw.h"
#include "gob/variables.h"
#include "gob/inter.h"
namespace Gob {
SlotFile::SlotFile(GobEngine *vm, uint32 slotCount, const Common::String &base) : _vm(vm) {
_base = base;
_slotCount = slotCount;
}
SlotFile::~SlotFile() {
}
uint32 SlotFileIndexed::getSlotMax() const {
Common::SaveFileManager *saveMan = g_system->getSavefileManager();
Common::InSaveFile *in;
// Find the last filled save slot and base the save file size calculate on that
for (int i = (_slotCount - 1); i >= 0; i--) {
Common::String slotFile = build(i);
if (slotFile.empty())
continue;
in = saveMan->openForLoading(slotFile);
if (in) {
delete in;
return i + 1;
}
}
return 0;
}
int32 SlotFileIndexed::tallyUpFiles(uint32 slotSize, uint32 indexSize) const {
uint32 maxSlot = getSlotMax();
if (maxSlot == 0)
return -1;
return ((maxSlot * slotSize) + indexSize);
}
void SlotFileIndexed::buildIndex(byte *buffer, SavePartInfo &info,
SaveConverter *converter, bool setLongest) const {
uint32 descLength = info.getDescMaxLength();
uint32 longest = 0;
byte *bufferStart = buffer;
// Iterate over all files
for (uint32 i = 0; i < _slotCount; i++, buffer += descLength) {
Common::String slotFile = build(i);
if (!slotFile.empty()) {
char *desc = nullptr;
if (converter && (desc = converter->getDescription(slotFile)))
// Old style save
memcpy(buffer, desc, descLength);
else if (SaveReader::getInfo(slotFile, info))
// New style save
memcpy(buffer, info.getDesc(), descLength);
else
// No known format, fill with 0
memset(buffer, 0, descLength);
delete[] desc;
longest = MAX<uint32>(longest, strlen((const char *) buffer));
} else
// No valid slot, fill with 0
memset(buffer, 0, descLength);
}
if (setLongest) {
uint32 slot0Len;
for (slot0Len = strlen((const char *) bufferStart); slot0Len < longest; slot0Len++)
bufferStart[slot0Len] = ' ';
bufferStart[slot0Len] = '\0';
}
}
bool SlotFileIndexed::exists(int slot) const {
Common::InSaveFile *in = openRead(slot);
bool result = (in != nullptr);
delete in;
return result;
}
Common::InSaveFile *SlotFileIndexed::openRead(int slot) const {
Common::String name = build(slot);
if (name.empty())
return nullptr;
Common::SaveFileManager *saveMan = g_system->getSavefileManager();
Common::InSaveFile *result = saveMan->openForLoading(name);
return result;
}
Common::OutSaveFile *SlotFileIndexed::openWrite(int slot) const {
Common::String name = build(slot);
if (name.empty())
return nullptr;
Common::SaveFileManager *saveMan = g_system->getSavefileManager();
Common::OutSaveFile *result = saveMan->openForSaving(name);
return result;
}
SlotFileIndexed::SlotFileIndexed(GobEngine *vm, uint32 slotCount,
const Common::String &base, const Common::String &extStub) : SlotFile(vm, slotCount, base) {
_ext = extStub;
}
SlotFileIndexed::~SlotFileIndexed() {
}
Common::String SlotFileIndexed::build(int slot) const {
if ((slot < 0) || (((uint32) slot) >= _slotCount))
return Common::String();
Common::String buf = Common::String::format("%02d", slot);
return _base + "." + _ext + buf;
}
SlotFileStatic::SlotFileStatic(GobEngine *vm, const Common::String &base,
const Common::String &ext) : SlotFile(vm, 1, base) {
_ext = "." + ext;
}
SlotFileStatic::~SlotFileStatic() {
}
int SlotFileStatic::getSlot(int32 offset) const {
return -1;
}
int SlotFileStatic::getSlotRemainder(int32 offset) const {
return -1;
}
Common::String SlotFileStatic::build() const {
return _base + _ext;
}
bool SlotFileStatic::exists() const {
Common::InSaveFile *in = openRead();
bool result = (in != nullptr);
delete in;
return result;
}
Common::InSaveFile *SlotFileStatic::openRead() const {
Common::String name = build();
if (name.empty())
return nullptr;
Common::SaveFileManager *saveMan = g_system->getSavefileManager();
Common::InSaveFile *result = saveMan->openForLoading(name);
return result;
}
Common::OutSaveFile *SlotFileStatic::openWrite() const {
Common::String name = build();
if (name.empty())
return nullptr;
Common::SaveFileManager *saveMan = g_system->getSavefileManager();
Common::OutSaveFile *result = saveMan->openForSaving(name);
return result;
}
SaveHandler::SaveHandler(GobEngine *vm) : _vm(vm) {
}
SaveHandler::~SaveHandler() {
}
uint32 SaveHandler::getVarSize(GobEngine *vm) {
// Sanity checks
if (!vm || !vm->_inter || !vm->_inter->_variables)
return 0;
return vm->_inter->_variables->getSize();
}
bool SaveHandler::deleteFile() {
return true;
}
TempSpriteHandler::TempSpriteHandler(GobEngine *vm) : SaveHandler(vm) {
_sprite = nullptr;
}
TempSpriteHandler::~TempSpriteHandler() {
delete _sprite;
}
int32 TempSpriteHandler::getSize() {
if (!_sprite)
return -1;
return _sprite->getSize();
}
bool TempSpriteHandler::load(int16 dataVar, int32 size, int32 offset) {
if (isDummy(size))
return true;
// Sprite available?
if (!_sprite)
return false;
// Sprite requested?
if (!isSprite(size))
return false;
// Index sane?
int index = getIndex(size);
if ((index < 0) || (index >= Draw::kSpriteCount))
return false;
SurfacePtr sprite = _vm->_draw->_spritesArray[index];
// Target sprite exists?
if (!sprite)
return false;
// Load the sprite
if (!_sprite->writeSprite(*sprite))
return false;
// Handle palette
if (usesPalette(size)) {
if (!_sprite->writePalette((byte *)_vm->_global->_pPaletteDesc->vgaPal))
return false;
_vm->_video->setFullPalette(_vm->_global->_pPaletteDesc);
}
if (index == 21) {
// We wrote into the backbuffer, blit
_vm->_draw->forceBlit();
_vm->_video->retrace();
} else if (index == 20)
// We wrote into the frontbuffer, retrace
_vm->_video->retrace();
return true;
}
bool TempSpriteHandler::save(int16 dataVar, int32 size, int32 offset) {
if (isDummy(size))
return true;
SurfacePtr sprite = createSprite(dataVar, size, offset);
if (!sprite)
return false;
// Save the sprite
if (!_sprite->readSprite(*sprite))
return false;
// Handle palette
if (usesPalette(size))
if (!_sprite->readPalette((const byte *)_vm->_global->_pPaletteDesc->vgaPal))
return false;
return true;
}
bool TempSpriteHandler::create(uint32 width, uint32 height, bool trueColor) {
delete _sprite;
_sprite = nullptr;
// Create a new temporary sprite
_sprite = new SavePartSprite(width, height, trueColor);
return true;
}
bool TempSpriteHandler::createFromSprite(int16 dataVar, int32 size, int32 offset) {
return createSprite(dataVar, size, offset) != nullptr;
}
SurfacePtr TempSpriteHandler::createSprite(int16 dataVar, int32 size, int32 offset) {
SurfacePtr sprt;
// Sprite requested?
if (!isSprite(size))
return sprt;
// Index sane?
int index = getIndex(size);
if ((index < 0) || (index >= Draw::kSpriteCount))
return sprt;
// Sprite exists?
if (!(sprt = _vm->_draw->_spritesArray[index]))
return sprt;
if (!create(sprt->getWidth(), sprt->getHeight(), sprt->getBPP() > 1))
sprt.reset();
return sprt;
}
// A size of 0 means no proper sprite should be saved/loaded,
// but no error should be thrown either.
bool TempSpriteHandler::isDummy(int32 size) {
return (size == 0);
}
// A negative size is the flag for using a sprite
bool TempSpriteHandler::isSprite(int32 size) {
return (size < 0);
}
// Contruct the index
int TempSpriteHandler::getIndex(int32 size) {
// Palette flag
if (size < -3000)
size += 3000;
if (size < -1000)
size += 1000;
return (-size - 1);
}
// A size smaller than -1000 indicates palette usage
bool TempSpriteHandler::usesPalette(int32 size) {
return (size < -1000);
}
NotesHandler::File::File(GobEngine *vm, const Common::String &base) :
SlotFileStatic(vm, base, "blo") {
}
NotesHandler::File::~File() {
}
NotesHandler::NotesHandler(uint32 notesSize, GobEngine *vm, const Common::String &target) :
SaveHandler(vm) {
_notesSize = notesSize;
_file = new File(vm, target);
_notes = new SavePartVars(vm, _notesSize);
}
NotesHandler::~NotesHandler() {
delete _file;
delete _notes;
}
int32 NotesHandler::getSize() {
Common::String fileName = _file->build();
if (fileName.empty())
return -1;
Common::InSaveFile *saveFile;
SaveConverter_Notes converter(_vm, _notesSize, fileName);
if (converter.isOldSave(&saveFile)) {
// Old save, get the size olden-style
int32 size = saveFile->size();
delete saveFile;
return size;
}
SaveReader reader(1, 0, fileName);
SaveHeader header;
if (!reader.load())
return -1;
if (!reader.readPartHeader(0, &header))
return -1;
// Return the part's size
return header.getSize();
}
bool NotesHandler::load(int16 dataVar, int32 size, int32 offset) {
if ((dataVar < 0) || (size < 0) || (offset < 0))
return false;
Common::String fileName = _file->build();
if (fileName.empty())
return false;
SaveReader *reader;
SaveConverter_Notes converter(_vm, _notesSize, fileName);
if (converter.isOldSave()) {
// Old save, plug the converter in
if (!converter.load())
return false;
reader = new SaveReader(1, 0, converter);
} else
// New save, load directly
reader = new SaveReader(1, 0, fileName);
SavePartVars vars(_vm, _notesSize);
if (!reader->load()) {
delete reader;
return false;
}
if (!reader->readPart(0, &vars)) {
delete reader;
return false;
}
if (!vars.writeInto(dataVar, offset, size)) {
delete reader;
return false;
}
delete reader;
return true;
}
bool NotesHandler::save(int16 dataVar, int32 size, int32 offset) {
if ((dataVar < 0) || (size < 0) || (offset < 0))
return false;
Common::String fileName = _file->build();
if (fileName.empty())
return false;
SaveWriter writer(1, 0, fileName);
SavePartVars vars(_vm, _notesSize);
if (!vars.readFrom(dataVar, offset, size))
return false;
return writer.writePart(0, &vars);
}
FakeFileHandler::FakeFileHandler(GobEngine *vm) : SaveHandler(vm) {
}
FakeFileHandler::~FakeFileHandler() {
}
int32 FakeFileHandler::getSize() {
if (_data.empty())
return -1;
return _data.size();
}
bool FakeFileHandler::load(int16 dataVar, int32 size, int32 offset) {
if (size <= 0)
return false;
if ((uint32)(offset + size) > _data.size())
return false;
_vm->_inter->_variables->copyFrom(dataVar, &_data[0] + offset, size);
return true;
}
bool FakeFileHandler::save(int16 dataVar, int32 size, int32 offset) {
if (size <= 0)
return false;
if ((uint32)(offset + size) > _data.size())
_data.resize(offset + size);
_vm->_inter->_variables->copyTo(dataVar, &_data[0] + offset, size);
return true;
}
bool FakeFileHandler::deleteFile() {
_data.clear();
return true;
}
} // End of namespace Gob