2019-06-16 03:25:48 +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.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef GLK_QUETZAL
|
|
|
|
#define GLK_QUETZAL
|
|
|
|
|
|
|
|
#include "common/array.h"
|
|
|
|
#include "common/endian.h"
|
|
|
|
#include "common/memstream.h"
|
|
|
|
#include "common/stream.h"
|
2019-06-16 16:32:56 +00:00
|
|
|
#include "engines/savestate.h"
|
2019-06-16 03:25:48 +00:00
|
|
|
#include "glk/blorb.h"
|
2019-06-17 00:55:16 +00:00
|
|
|
#include "glk/glk_types.h"
|
2019-06-16 03:25:48 +00:00
|
|
|
|
|
|
|
namespace Glk {
|
|
|
|
|
|
|
|
enum QueztalTag {
|
|
|
|
ID_IFSF = MKTAG('I', 'F', 'S', 'F'),
|
|
|
|
ID_IFZS = MKTAG('I', 'F', 'Z', 'S'),
|
|
|
|
ID_IFhd = MKTAG('I', 'F', 'h', 'd'),
|
|
|
|
ID_UMem = MKTAG('U', 'M', 'e', 'm'),
|
|
|
|
ID_CMem = MKTAG('C', 'M', 'e', 'm'),
|
|
|
|
ID_Stks = MKTAG('S', 't', 'k', 's'),
|
|
|
|
ID_SCVM = MKTAG('S', 'C', 'V', 'M')
|
|
|
|
};
|
|
|
|
|
2019-10-14 00:10:34 +00:00
|
|
|
class QuetzalBase {
|
|
|
|
public:
|
|
|
|
static uint32 getInterpreterTag(InterpreterType interpType);
|
|
|
|
};
|
2019-06-17 00:55:16 +00:00
|
|
|
|
2019-06-16 03:25:48 +00:00
|
|
|
/**
|
|
|
|
* Quetzal save file reader
|
|
|
|
*/
|
2019-10-14 00:10:34 +00:00
|
|
|
class QuetzalReader : public QuetzalBase {
|
2019-06-16 03:25:48 +00:00
|
|
|
struct Chunk {
|
|
|
|
uint32 _id;
|
|
|
|
size_t _offset, _size;
|
|
|
|
};
|
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* Iterator for the chunks list
|
|
|
|
*/
|
2019-06-16 05:28:55 +00:00
|
|
|
struct Iterator {
|
2019-06-16 03:25:48 +00:00
|
|
|
private:
|
|
|
|
Common::SeekableReadStream *_stream;
|
|
|
|
Common::Array<Chunk> &_chunks;
|
|
|
|
int _index;
|
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* Constructor
|
|
|
|
*/
|
2019-06-16 05:28:55 +00:00
|
|
|
Iterator(Common::SeekableReadStream *stream, Common::Array<Chunk> &chunks, int index) :
|
|
|
|
_stream(stream), _chunks(chunks), _index(index) {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Deference
|
|
|
|
*/
|
|
|
|
Chunk &operator*() const { return _chunks[_index]; }
|
2019-06-16 03:25:48 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Incrementer
|
|
|
|
*/
|
2019-06-16 05:28:55 +00:00
|
|
|
Iterator &operator++() {
|
|
|
|
++_index;
|
|
|
|
return *this;
|
|
|
|
}
|
2019-06-16 03:25:48 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Decrementer
|
|
|
|
*/
|
2019-06-16 05:28:55 +00:00
|
|
|
Iterator &operator--() {
|
|
|
|
--_index;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Equality test
|
|
|
|
*/
|
|
|
|
bool operator==(const Iterator &rhs) { return _index == rhs._index; }
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Inequality test
|
|
|
|
*/
|
|
|
|
bool operator!=(const Iterator &rhs) { return _index != rhs._index; }
|
2019-06-16 03:25:48 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a read stream for the contents of a chunk
|
|
|
|
*/
|
|
|
|
Common::SeekableReadStream *getStream() {
|
2019-06-16 05:28:55 +00:00
|
|
|
_stream->seek(_chunks[_index]._offset);
|
|
|
|
return _stream->readStream(_chunks[_index]._size);
|
2019-06-16 03:25:48 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
private:
|
|
|
|
Common::SeekableReadStream *_stream;
|
|
|
|
Common::Array<Chunk> _chunks;
|
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* Constructor
|
|
|
|
*/
|
|
|
|
QuetzalReader() : _stream(nullptr) {}
|
|
|
|
|
2019-06-16 04:50:01 +00:00
|
|
|
/**
|
|
|
|
* Clear
|
|
|
|
*/
|
|
|
|
void clear();
|
|
|
|
|
2019-06-16 03:25:48 +00:00
|
|
|
/**
|
|
|
|
* Opens a Quetzal file for access
|
|
|
|
*/
|
2019-06-16 16:32:56 +00:00
|
|
|
bool open(Common::SeekableReadStream *stream, uint32 formType = 0);
|
2019-06-16 03:25:48 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Return an iterator for the beginning of the chunks list
|
|
|
|
*/
|
2019-06-16 05:28:55 +00:00
|
|
|
Iterator begin() { return Iterator(_stream, _chunks, 0); }
|
2019-06-16 03:25:48 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Return an iterator for the beginning of the chunks list
|
|
|
|
*/
|
2019-06-16 05:28:55 +00:00
|
|
|
Iterator end() { return Iterator(_stream, _chunks, _chunks.size()); }
|
2019-06-16 16:32:56 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Loads a Quetzal save and extracts it's description from an ANNO chunk
|
|
|
|
*/
|
|
|
|
static bool getSavegameDescription(Common::SeekableReadStream *rs, Common::String &saveName);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Loads a Quetzal save and extract's it's description and meta info
|
|
|
|
*/
|
|
|
|
static bool getSavegameMetaInfo(Common::SeekableReadStream *rs, SaveStateDescriptor &ssd);
|
2019-06-16 20:29:15 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Support method for reading a string from a stream
|
|
|
|
*/
|
|
|
|
static Common::String readString(Common::ReadStream *src);
|
2019-06-16 03:25:48 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Quetzal save file writer
|
|
|
|
*/
|
2019-10-14 00:10:34 +00:00
|
|
|
class QuetzalWriter : public QuetzalBase {
|
2019-06-16 03:25:48 +00:00
|
|
|
/**
|
|
|
|
* Chunk entry
|
|
|
|
*/
|
|
|
|
struct Chunk {
|
|
|
|
uint32 _id;
|
|
|
|
Common::MemoryWriteStreamDynamic _stream;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Constructor
|
|
|
|
*/
|
|
|
|
Chunk() : _id(0), _stream(DisposeAfterUse::YES) {}
|
2019-06-16 04:50:01 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Constructor
|
|
|
|
*/
|
|
|
|
Chunk(uint32 id) : _id(id), _stream(DisposeAfterUse::YES) {}
|
2019-06-16 03:25:48 +00:00
|
|
|
};
|
|
|
|
private:
|
|
|
|
Common::Array<Chunk> _chunks;
|
2019-06-16 15:41:10 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Add chunks common to all Glk savegames
|
|
|
|
*/
|
|
|
|
void addCommonChunks(const Common::String &saveName);
|
2019-06-16 03:25:48 +00:00
|
|
|
public:
|
2019-06-16 04:50:01 +00:00
|
|
|
/**
|
|
|
|
* Clear
|
|
|
|
*/
|
|
|
|
void clear() { _chunks.clear(); }
|
|
|
|
|
2019-06-16 03:25:48 +00:00
|
|
|
/**
|
|
|
|
* Add a chunk
|
|
|
|
*/
|
|
|
|
Common::WriteStream &add(uint32 chunkId);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Save the added chunks to file
|
|
|
|
*/
|
2019-06-16 15:41:10 +00:00
|
|
|
void save(Common::WriteStream *out, const Common::String &saveName, uint32 formType = ID_IFSF);
|
2019-06-16 03:25:48 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
} // End of namespace Glk
|
|
|
|
|
|
|
|
#endif
|