2012-01-10 19:41:51 +00:00
|
|
|
/* ResidualVM - A 3D game interpreter
|
2009-09-17 09:12:59 +00:00
|
|
|
*
|
2012-01-10 19:41:51 +00:00
|
|
|
* ResidualVM is the legal property of its developers, whose names
|
2009-09-17 09:12:59 +00:00
|
|
|
* are too numerous to list here. Please refer to the AUTHORS
|
|
|
|
* file distributed with this source distribution.
|
|
|
|
*
|
2012-01-05 09:07:48 +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.
|
2009-09-17 09:12:59 +00:00
|
|
|
|
2012-01-05 09:07:48 +00:00
|
|
|
* This program is distributed in the hope that it will be useful,
|
2009-09-17 09:12:59 +00:00
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
2012-01-05 09:07:48 +00:00
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
2009-09-17 09:12:59 +00:00
|
|
|
|
2012-01-05 09:07:48 +00:00
|
|
|
* 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.
|
2009-09-17 09:12:59 +00:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2009-09-14 14:37:24 +00:00
|
|
|
#include "engines/myst3/archive.h"
|
|
|
|
#include "common/debug.h"
|
2009-09-15 15:21:17 +00:00
|
|
|
#include "common/memstream.h"
|
2009-09-14 14:37:24 +00:00
|
|
|
|
2009-09-17 09:12:59 +00:00
|
|
|
namespace Myst3 {
|
|
|
|
|
2009-09-14 14:37:24 +00:00
|
|
|
void Archive::_decryptHeader(Common::SeekableReadStream &inStream, Common::WriteStream &outStream) {
|
|
|
|
static const uint32 addKey = 0x3C6EF35F;
|
|
|
|
static const uint32 multKey = 0x0019660D;
|
|
|
|
|
|
|
|
inStream.seek(0);
|
2012-01-14 16:51:36 +00:00
|
|
|
uint32 size = inStream.readUint32LE();
|
|
|
|
|
|
|
|
bool encrypted = size > 1000000;
|
2009-09-14 14:37:24 +00:00
|
|
|
|
|
|
|
inStream.seek(0);
|
|
|
|
|
2012-01-14 16:51:36 +00:00
|
|
|
if (encrypted) {
|
|
|
|
uint32 decryptedSize = size ^ addKey;
|
|
|
|
|
|
|
|
uint32 currentKey = 0;
|
|
|
|
for (uint i = 0; i < decryptedSize; i++) {
|
|
|
|
currentKey += addKey;
|
|
|
|
outStream.writeUint32LE(inStream.readUint32LE() ^ currentKey);
|
|
|
|
currentKey *= multKey;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (uint i = 0; i < size; i++) {
|
|
|
|
outStream.writeUint32LE(inStream.readUint32LE());
|
|
|
|
}
|
2009-09-14 14:37:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-15 16:45:55 +00:00
|
|
|
void Archive::_readDirectory() {
|
2009-09-15 15:21:17 +00:00
|
|
|
Common::MemoryWriteStreamDynamic buf(DisposeAfterUse::YES);
|
2009-09-15 16:45:55 +00:00
|
|
|
_decryptHeader(_file, buf);
|
2009-09-14 14:37:24 +00:00
|
|
|
|
|
|
|
Common::MemoryReadStream directory(buf.getData(), buf.size());
|
|
|
|
directory.skip(sizeof(uint32));
|
|
|
|
|
2012-01-07 20:31:52 +00:00
|
|
|
while (directory.pos() + 4 < directory.size()) {
|
2012-01-05 12:17:48 +00:00
|
|
|
DirectoryEntry entry(this);
|
2012-01-07 20:31:52 +00:00
|
|
|
|
|
|
|
if (_multipleRoom)
|
|
|
|
entry.readFromStream(directory, 0);
|
|
|
|
else
|
|
|
|
entry.readFromStream(directory, _roomName);
|
|
|
|
|
|
|
|
_directory.push_back(entry);
|
2009-09-14 14:37:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-15 16:45:55 +00:00
|
|
|
void Archive::dumpToFiles() {
|
2009-09-14 14:37:24 +00:00
|
|
|
for (uint i = 0; i < _directory.size(); i++) {
|
2009-09-15 16:45:55 +00:00
|
|
|
_directory[i].dumpToFiles(_file);
|
2009-09-14 14:37:24 +00:00
|
|
|
}
|
|
|
|
}
|
2009-09-15 16:45:55 +00:00
|
|
|
|
2012-01-05 12:17:48 +00:00
|
|
|
Common::MemoryReadStream *Archive::dumpToMemory(uint32 offset, uint32 size) {
|
|
|
|
_file.seek(offset);
|
|
|
|
return static_cast<Common::MemoryReadStream *>(_file.readStream(size));
|
|
|
|
}
|
|
|
|
|
2012-02-11 07:40:32 +00:00
|
|
|
const DirectorySubEntry *Archive::getDescription(const char *room, uint32 index, uint16 face, DirectorySubEntry::ResourceType type) {
|
2009-09-15 16:45:55 +00:00
|
|
|
for (uint i = 0; i < _directory.size(); i++) {
|
2012-01-07 21:17:21 +00:00
|
|
|
if (_directory[i].getIndex() == index
|
|
|
|
&& !strcmp(_directory[i].getRoom(), room)) {
|
2011-09-11 08:15:41 +00:00
|
|
|
return _directory[i].getItemDescription(face, type);
|
2009-09-15 16:45:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-01-07 20:31:52 +00:00
|
|
|
bool Archive::open(const char *fileName, const char *room) {
|
|
|
|
// Copy the room name if provided
|
|
|
|
// If the room name is not provided, it is assumed that
|
|
|
|
// we are opening a multi-room archive
|
|
|
|
_multipleRoom = room == 0;
|
|
|
|
if (!_multipleRoom)
|
|
|
|
Common::strlcpy(_roomName, room, sizeof(_roomName));
|
|
|
|
|
2009-09-15 16:45:55 +00:00
|
|
|
if (_file.open(fileName)) {
|
|
|
|
_readDirectory();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Archive::close() {
|
2011-09-03 13:39:16 +00:00
|
|
|
_directory.clear();
|
2009-09-15 16:45:55 +00:00
|
|
|
_file.close();
|
|
|
|
}
|
2009-09-17 09:12:59 +00:00
|
|
|
|
|
|
|
} // end of namespace Myst3
|