/* 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. * * $URL$ * $Id$ * */ #include "kyra/resource_intern.h" #include "kyra/resource.h" #include "common/stream.h" #include "common/endian.h" namespace Kyra { // Implementation of various Archive subclasses // -> PlainArchive implementation PlainArchive::PlainArchive(Common::SharedPtr file, const FileInputList &files) : _file(file), _files() { for (FileInputList::const_iterator i = files.begin(); i != files.end(); ++i) { Entry entry; entry.offset = i->offset; entry.size = i->size; _files[i->name] = entry; } } bool PlainArchive::hasFile(const Common::String &name) { return (_files.find(name) != _files.end()); } int PlainArchive::listMembers(Common::ArchiveMemberList &list) { int count = 0; for (FileMap::const_iterator i = _files.begin(); i != _files.end(); ++i) { list.push_back(Common::ArchiveMemberList::value_type(new Common::GenericArchiveMember(i->_key, this))); ++count; } return count; } Common::ArchiveMemberPtr PlainArchive::getMember(const Common::String &name) { if (!hasFile(name)) return Common::ArchiveMemberPtr(); return Common::ArchiveMemberPtr(new Common::GenericArchiveMember(name, this)); } Common::SeekableReadStream *PlainArchive::createReadStreamForMember(const Common::String &name) const { FileMap::const_iterator fDesc = _files.find(name); if (fDesc == _files.end()) return 0; Common::SeekableReadStream *parent = _file->createReadStream(); if (!parent) return 0; return new Common::SeekableSubReadStream(parent, fDesc->_value.offset, fDesc->_value.offset + fDesc->_value.size, true); } // -> CachedArchive implementation CachedArchive::CachedArchive(const FileInputList &files) : _files() { for (FileInputList::const_iterator i = files.begin(); i != files.end(); ++i) { Entry entry; entry.data = i->data; entry.size = i->size; Common::String name = i->name; name.toLowercase(); _files[name] = entry; } } CachedArchive::~CachedArchive() { for (FileMap::iterator i = _files.begin(); i != _files.end(); ++i) delete[] i->_value.data; _files.clear(); } bool CachedArchive::hasFile(const Common::String &name) { return (_files.find(name) != _files.end()); } int CachedArchive::listMembers(Common::ArchiveMemberList &list) { int count = 0; for (FileMap::const_iterator i = _files.begin(); i != _files.end(); ++i) { list.push_back(Common::ArchiveMemberList::value_type(new Common::GenericArchiveMember(i->_key, this))); ++count; } return count; } Common::ArchiveMemberPtr CachedArchive::getMember(const Common::String &name) { if (!hasFile(name)) return Common::ArchiveMemberPtr(); return Common::ArchiveMemberPtr(new Common::GenericArchiveMember(name, this)); } Common::SeekableReadStream *CachedArchive::createReadStreamForMember(const Common::String &name) const { FileMap::const_iterator fDesc = _files.find(name); if (fDesc == _files.end()) return 0; return new Common::MemoryReadStream(fDesc->_value.data, fDesc->_value.size, false); } // ResFileLoader implementations // -> ResLoaderPak implementation bool ResLoaderPak::checkFilename(Common::String filename) const { filename.toUppercase(); return (filename.hasSuffix(".PAK") || filename.hasSuffix(".APK") || filename.hasSuffix(".VRM") || filename.hasSuffix(".CMP") || filename.hasSuffix(".TLK") || filename.equalsIgnoreCase(StaticResource::staticDataFilename())); } bool ResLoaderPak::isLoadable(const Common::String &filename, Common::SeekableReadStream &stream) const { int32 filesize = stream.size(); int32 offset = 0; bool switchEndian = false; bool firstFile = true; offset = stream.readUint32LE(); if (offset > filesize) { switchEndian = true; offset = SWAP_BYTES_32(offset); } Common::String file; while (!stream.eos()) { // The start offset of a file should never be in the filelist if (offset < stream.pos() || offset > filesize) return false; byte c = 0; file.clear(); while (!stream.eos() && (c = stream.readByte()) != 0) file += c; if (stream.eos()) return false; // Quit now if we encounter an empty string if (file.empty()) { if (firstFile) return false; else break; } firstFile = false; offset = switchEndian ? stream.readUint32BE() : stream.readUint32LE(); if (!offset || offset == filesize) break; } return true; } namespace { Common::String readString(Common::SeekableReadStream &stream) { Common::String result; char c = 0; while ((c = stream.readByte()) != 0) result += c; return result; } struct PlainArchiveListSearch { PlainArchiveListSearch(const Common::String &search) : _search(search) {} bool operator()(const PlainArchive::InputEntry &entry) { return _search.equalsIgnoreCase(entry.name); } Common::String _search; }; } // end of anonymous namespace Common::Archive *ResLoaderPak::load(Common::SharedPtr memberFile, Common::SeekableReadStream &stream) const { int32 filesize = stream.size(); int32 startoffset = 0, endoffset = 0; bool switchEndian = false; bool firstFile = true; startoffset = stream.readUint32LE(); if (startoffset > filesize) { switchEndian = true; startoffset = SWAP_BYTES_32(startoffset); } PlainArchive::FileInputList files; Common::String file; while (!stream.eos()) { // The start offset of a file should never be in the filelist if (startoffset < stream.pos() || startoffset > filesize) { warning("PAK file '%s' is corrupted", memberFile->getDisplayName().c_str()); return false; } file.clear(); byte c = 0; while (!stream.eos() && (c = stream.readByte()) != 0) file += c; if (stream.eos()) { warning("PAK file '%s' is corrupted", memberFile->getDisplayName().c_str()); return false; } // Quit now if we encounter an empty string if (file.empty()) { if (firstFile) { warning("PAK file '%s' is corrupted", memberFile->getDisplayName().c_str()); return false; } else { break; } } firstFile = false; endoffset = switchEndian ? stream.readUint32BE() : stream.readUint32LE(); if (!endoffset) endoffset = filesize; if (startoffset != endoffset) { PlainArchive::InputEntry entry; entry.size = endoffset - startoffset; entry.offset = startoffset; entry.name = file; files.push_back(entry); } if (endoffset == filesize) break; startoffset = endoffset; } PlainArchive::FileInputList::const_iterator iter = Common::find_if(files.begin(), files.end(), PlainArchiveListSearch("LINKLIST")); if (iter != files.end()) { stream.seek(iter->offset, SEEK_SET); uint32 magic = stream.readUint32BE(); if (magic != MKID_BE('SCVM')) error("LINKLIST file does not contain 'SCVM' header"); uint32 links = stream.readUint32BE(); for (uint i = 0; i < links; ++i) { Common::String linksTo = readString(stream); uint32 sources = stream.readUint32BE(); iter = Common::find_if(files.begin(), files.end(), PlainArchiveListSearch(linksTo)); if (iter == files.end()) error("PAK file link destination '%s' not found", linksTo.c_str()); for (uint j = 0; j < sources; ++j) { Common::String dest = readString(stream); PlainArchive::InputEntry link = *iter; link.name = dest; files.push_back(link); // Better safe than sorry, we update the 'iter' value, in case push_back invalidated it iter = Common::find_if(files.begin(), files.end(), PlainArchiveListSearch(linksTo)); } } } return new PlainArchive(memberFile, files); } // -> ResLoaderInsMalcolm implementation bool ResLoaderInsMalcolm::checkFilename(Common::String filename) const { filename.toUppercase(); if (!filename.hasSuffix(".001")) return false; return true; } bool ResLoaderInsMalcolm::isLoadable(const Common::String &filename, Common::SeekableReadStream &stream) const { stream.seek(3, SEEK_SET); int32 size = stream.readUint32LE(); if (size+7 > stream.size()) return false; stream.seek(size+5, SEEK_SET); uint8 buffer[2]; stream.read(&buffer, 2); return (buffer[0] == 0x0D && buffer[1] == 0x0A); } Common::Archive *ResLoaderInsMalcolm::load(Common::SharedPtr memberFile, Common::SeekableReadStream &stream) const { Common::List filenames; PlainArchive::FileInputList files; // thanks to eriktorbjorn for this code (a bit modified though) stream.seek(3, SEEK_SET); // first file is the index table uint32 size = stream.readUint32LE(); Common::String temp; for (uint32 i = 0; i < size; ++i) { byte c = stream.readByte(); if (c == '\\') { temp.clear(); } else if (c == 0x0D) { // line endings are CRLF c = stream.readByte(); assert(c == 0x0A); ++i; filenames.push_back(temp); } else { temp += (char)c; } } stream.seek(3, SEEK_SET); for (Common::List::iterator file = filenames.begin(); file != filenames.end(); ++file) { PlainArchive::InputEntry entry; entry.size = stream.readUint32LE(); entry.offset = stream.pos(); entry.name = *file; entry.name.toLowercase(); stream.seek(entry.size, SEEK_CUR); files.push_back(entry); } return new PlainArchive(memberFile, files); } bool ResLoaderTlk::checkFilename(Common::String filename) const { filename.toUppercase(); return (filename.hasSuffix(".TLK")); } bool ResLoaderTlk::isLoadable(const Common::String &filename, Common::SeekableReadStream &stream) const { uint16 entries = stream.readUint16LE(); int32 entryTableSize = (entries * 8); if (entryTableSize + 2 > stream.size()) return false; int32 offset = 0; for (uint i = 0; i < entries; ++i) { stream.readUint32LE(); offset = stream.readUint32LE(); if (offset > stream.size()) return false; } return true; } Common::Archive *ResLoaderTlk::load(Common::SharedPtr file, Common::SeekableReadStream &stream) const { uint16 entries = stream.readUint16LE(); PlainArchive::FileInputList files; for (uint i = 0; i < entries; ++i) { PlainArchive::InputEntry entry; uint32 resFilename = stream.readUint32LE(); uint32 resOffset = stream.readUint32LE(); entry.offset = resOffset+4; char realFilename[20]; snprintf(realFilename, 20, "%.08u.AUD", resFilename); entry.name = realFilename; uint32 curOffset = stream.pos(); stream.seek(resOffset, SEEK_SET); entry.size = stream.readUint32LE(); stream.seek(curOffset, SEEK_SET); files.push_back(entry); } return new PlainArchive(file, files); } // InstallerLoader implementation class FileExpanderSource { public: FileExpanderSource(const uint8 *data, int dataSize) : _dataPtr(data), _endofBuffer(data + dataSize), _bitsLeft(8), _key(0), _index(0) {} ~FileExpanderSource() {} void advSrcRefresh(); void advSrcBitsBy1(); void advSrcBitsByIndex(uint8 newIndex); uint8 getKeyLower() { return _key & 0xff; } void setIndex(uint8 index) { _index = index; } uint16 getKeyMasked(uint8 newIndex); uint16 keyMaskedAlign(uint16 val); void copyBytes(uint8 *& dst); private: const uint8 *_dataPtr; const uint8 *_endofBuffer; uint16 _key; int8 _bitsLeft; uint8 _index; }; void FileExpanderSource::advSrcBitsBy1() { _key >>= 1; if (!--_bitsLeft) { if (_dataPtr < _endofBuffer) _key = ((*_dataPtr++) << 8 ) | (_key & 0xff); _bitsLeft = 8; } } void FileExpanderSource::advSrcBitsByIndex(uint8 newIndex) { _index = newIndex; _bitsLeft -= _index; if (_bitsLeft <= 0) { _key >>= (_index + _bitsLeft); _index = -_bitsLeft; _bitsLeft = 8 - _index; if (_dataPtr < _endofBuffer) _key = (*_dataPtr++ << 8) | (_key & 0xff); } _key >>= _index; } uint16 FileExpanderSource::getKeyMasked(uint8 newIndex) { static const uint8 mskTable[] = { 0x0F, 0x01, 0x03, 0x07, 0x0F, 0x1F, 0x3F, 0x7F, 0xFF }; _index = newIndex; uint16 res = 0; if (_index > 8) { newIndex = _index - 8; res = (_key & 0xff) & mskTable[8]; advSrcBitsByIndex(8); _index = newIndex; res |= (((_key & 0xff) & mskTable[_index]) << 8); advSrcBitsByIndex(_index); } else { res = (_key & 0xff) & mskTable[_index]; advSrcBitsByIndex(_index); } return res; } void FileExpanderSource::copyBytes(uint8 *& dst) { advSrcBitsByIndex(_bitsLeft); uint16 r = (READ_LE_UINT16(_dataPtr) ^ _key) + 1; _dataPtr += 2; if (r) error("decompression failure"); memcpy(dst, _dataPtr, _key); _dataPtr += _key; dst += _key; } uint16 FileExpanderSource::keyMaskedAlign(uint16 val) { val -= 0x101; _index = (val & 0xff) >> 2; int16 b = ((_bitsLeft << 8) | _index) - 1; _bitsLeft = b >> 8; _index = b & 0xff; uint16 res = (((val & 3) + 4) << _index) + 0x101; return res + getKeyMasked(_index); } void FileExpanderSource::advSrcRefresh() { _key = READ_LE_UINT16(_dataPtr); if (_dataPtr < _endofBuffer - 1) _dataPtr += 2; _bitsLeft = 8; } class FileExpander { public: FileExpander(); ~FileExpander(); bool process(uint8 *dst, const uint8 *src, uint32 outsize, uint32 insize); private: void generateTables(uint8 srcIndex, uint8 dstIndex, uint8 dstIndex2, int cnt); uint8 calcCmdAndIndex(const uint8 *tbl, int16 ¶); FileExpanderSource *_src; uint8 *_tables[9]; uint16 *_tables16[3]; }; FileExpander::FileExpander() : _src(0) { _tables[0] = new uint8[3914]; assert(_tables[0]); _tables[1] = _tables[0] + 320; _tables[2] = _tables[0] + 352; _tables[3] = _tables[0] + 864; _tables[4] = _tables[0] + 2016; _tables[5] = _tables[0] + 2528; _tables[6] = _tables[0] + 2656; _tables[7] = _tables[0] + 2736; _tables[8] = _tables[0] + 2756; _tables16[0] = (uint16 *)(_tables[0] + 3268); _tables16[1] = (uint16 *)(_tables[0] + 3302); _tables16[2] = (uint16 *)(_tables[0] + 3338); } FileExpander::~FileExpander() { delete _src; delete[] _tables[0]; } bool FileExpander::process(uint8 *dst, const uint8 *src, uint32 outsize, uint32 compressedSize) { static const uint8 indexTable[] = { 0x10, 0x11, 0x12, 0x00, 0x08, 0x07, 0x09, 0x06, 0x0A, 0x05, 0x0B, 0x04, 0x0C, 0x03, 0x0D, 0x02, 0x0E, 0x01, 0x0F }; memset(_tables[0], 0, 3914); uint8 *d = dst; uint16 tableSize0 = 0; uint16 tableSize1 = 0; bool needrefresh = true; bool postprocess = false; _src = new FileExpanderSource(src, compressedSize); while (d < dst + outsize) { if (needrefresh) { needrefresh = false; _src->advSrcRefresh(); } _src->advSrcBitsBy1(); int mode = _src->getKeyMasked(2) - 1; if (mode == 1) { tableSize0 = _src->getKeyMasked(5) + 257; tableSize1 = _src->getKeyMasked(5) + 1; memset(_tables[7], 0, 19); const uint8 *itbl = indexTable; int numbytes = _src->getKeyMasked(4) + 4; while (numbytes--) _tables[7][*itbl++] = _src->getKeyMasked(3); generateTables(7, 8, 255, 19); int cnt = tableSize0 + tableSize1; uint8 *tmp = _tables[0]; while (cnt) { uint16 cmd = _src->getKeyLower(); cmd = READ_LE_UINT16(&_tables[8][cmd << 1]); _src->advSrcBitsByIndex(_tables[7][cmd]); if (cmd < 16) { *tmp++ = cmd; cnt--; } else { uint8 tmpI = 0; if (cmd == 16) { cmd = _src->getKeyMasked(2) + 3; tmpI = *(tmp - 1); } else if (cmd == 17) { cmd = _src->getKeyMasked(3) + 3; } else { cmd = _src->getKeyMasked(7) + 11; } _src->setIndex(tmpI); memset(tmp, tmpI, cmd); tmp += cmd; cnt -= cmd; if (cnt < 0) error("decompression failure"); } } memcpy(_tables[1], _tables[0] + tableSize0, tableSize1); generateTables(0, 2, 3, tableSize0); generateTables(1, 4, 5, tableSize1); postprocess = true; } else if (mode < 0) { _src->copyBytes(d); postprocess = false; needrefresh = true; } else if (mode == 0){ uint8 *d2 = _tables[0]; memset(d2, 8, 144); memset(d2 + 144, 9, 112); memset(d2 + 256, 7, 24); memset(d2 + 280, 8, 8); d2 = _tables[1]; memset(d2, 5, 32); tableSize0 = 288; tableSize1 = 32; generateTables(0, 2, 3, tableSize0); generateTables(1, 4, 5, tableSize1); postprocess = true; } else { error("decompression failure"); } if (!postprocess) continue; int16 cmd = 0; do { cmd = ((int16*) _tables[2])[_src->getKeyLower()]; _src->advSrcBitsByIndex(cmd < 0 ? calcCmdAndIndex(_tables[3], cmd) : _tables[0][cmd]); if (cmd == 0x11d) { cmd = 0x200; } else if (cmd > 0x108) { cmd = _src->keyMaskedAlign(cmd); } if (!(cmd >> 8)) { *d++ = cmd & 0xff; } else if (cmd != 0x100) { cmd -= 0xfe; int16 offset = ((int16*) _tables[4])[_src->getKeyLower()]; _src->advSrcBitsByIndex(offset < 0 ? calcCmdAndIndex(_tables[5], offset) : _tables[1][offset]); if ((offset & 0xff) >= 4) { uint8 newIndex = ((offset & 0xff) >> 1) - 1; offset = (((offset & 1) + 2) << newIndex); offset += _src->getKeyMasked(newIndex); } uint8 *s2 = d - 1 - offset; if (s2 >= dst) { while (cmd--) *d++ = *s2++; } else { uint32 pos = dst - s2; s2 += (d - dst); if (pos < (uint32) cmd) { cmd -= pos; while (pos--) *d++ = *s2++; s2 = dst; } while (cmd--) *d++ = *s2++; } } } while (cmd != 0x100); } delete _src; _src = 0; return true; } void FileExpander::generateTables(uint8 srcIndex, uint8 dstIndex, uint8 dstIndex2, int cnt) { uint8 *tbl1 = _tables[srcIndex]; uint8 *tbl2 = _tables[dstIndex]; uint8 *tbl3 = dstIndex2 == 0xff ? 0 : _tables[dstIndex2]; if (!cnt) return; const uint8 *s = tbl1; memset(_tables16[0], 0, 32); for (int i = 0; i < cnt; i++) _tables16[0][(*s++)]++; _tables16[1][1] = 0; for (uint16 i = 1, r = 0; i < 16; i++) { r = (r + _tables16[0][i]) << 1; _tables16[1][i + 1] = r; } if (_tables16[1][16]) { uint16 r = 0; for (uint16 i = 1; i < 16; i++) r += _tables16[0][i]; if (r > 1) error("decompression failure"); } s = tbl1; uint16 *d = _tables16[2]; for (int i = 0; i < cnt; i++) { uint16 t = *s++; if (t) { _tables16[1][t]++; t = _tables16[1][t] - 1; } *d++ = t; } s = tbl1; d = _tables16[2]; for (int i = 0; i < cnt; i++) { int8 t = ((int8)(*s++)) - 1; if (t > 0) { uint16 v1 = *d; uint16 v2 = 0; do { v2 = (v2 << 1) | (v1 & 1); v1 >>= 1; } while (--t && v1); t++; uint8 c1 = (v1 & 1); while (t--) { uint8 c2 = v2 >> 15; v2 = (v2 << 1) | c1; c1 = c2; }; *d++ = v2; } else { d++; } } memset(tbl2, 0, 512); cnt--; s = tbl1 + cnt; d = &_tables16[2][cnt]; uint16 * bt = (uint16*) tbl3; uint16 inc = 0; uint16 cnt2 = 0; do { uint8 t = *s--; uint16 *s2 = (uint16*) tbl2; if (t && t < 9) { inc = 1 << t; uint16 o = *d; do { s2[o] = cnt; o += inc; } while (!(o & 0xf00)); } else if (t > 8) { if (!bt) error("decompression failure"); t -= 8; uint8 shiftCnt = 1; uint8 v = (*d) >> 8; s2 = &((uint16*) tbl2)[*d & 0xff]; do { if (!*s2) { *s2 = (uint16)(~cnt2); *(uint32*)&bt[cnt2] = 0; cnt2 += 2; } s2 = &bt[(uint16)(~*s2)]; if (v & shiftCnt) s2++; shiftCnt <<= 1; } while (--t); *s2 = cnt; } d--; } while (--cnt >= 0); } uint8 FileExpander::calcCmdAndIndex(const uint8 *tbl, int16 ¶) { const uint16 *t = (const uint16*)tbl; _src->advSrcBitsByIndex(8); uint8 newIndex = 0; uint16 v = _src->getKeyLower(); do { newIndex++; para = t[((~para) & 0xfffe) | (v & 1)]; v >>= 1; } while (para < 0); return newIndex; } namespace { struct InsArchive { Common::String filename; uint32 firstFile; uint32 startOffset; uint32 lastFile; uint32 endOffset; uint32 totalSize; }; } // end of anonymouse namespace Common::Archive *InstallerLoader::load(Resource *owner, const Common::String &filename, const Common::String &extension, const uint8 containerOffset) { uint32 pos = 0; uint32 bytesleft = 0; bool startFile = true; Common::String filenameBase = filename; Common::String filenameTemp; char filenameExt[4]; if (filenameBase.lastChar() != '.') filenameBase += '.'; InsArchive newArchive; Common::List archives; Common::SeekableReadStream *tmpFile = 0; for (int8 currentFile = 1; currentFile; currentFile++) { sprintf(filenameExt, extension.c_str(), currentFile); filenameTemp = filenameBase + Common::String(filenameExt); if (!(tmpFile = owner->createReadStream(filenameTemp))) { debug(3, "couldn't open file '%s'\n", filenameTemp.c_str()); break; } tmpFile->seek(pos, SEEK_SET); uint8 fileId = tmpFile->readByte(); pos++; uint32 size = tmpFile->size() - 1; if (startFile) { size -= 4; if (fileId == currentFile) { size -= containerOffset; pos += containerOffset; tmpFile->seek(containerOffset, SEEK_CUR); } else { size = size + 1 - pos; } newArchive.filename = filenameBase; bytesleft = newArchive.totalSize = tmpFile->readUint32LE(); pos += 4; newArchive.firstFile = currentFile; newArchive.startOffset = pos; startFile = false; } uint32 cs = MIN(size, bytesleft); bytesleft -= cs; delete tmpFile; tmpFile = 0; pos += cs; if (cs == size) { if (!bytesleft) { newArchive.lastFile = currentFile; newArchive.endOffset = --pos; archives.push_back(newArchive); currentFile = -1; } else { pos = 0; } } else { startFile = true; bytesleft = size - cs; newArchive.lastFile = currentFile--; newArchive.endOffset = --pos; archives.push_back(newArchive); } } FileExpander exp; CachedArchive::InputEntry newEntry; uint32 insize = 0; uint32 outsize = 0; uint8 *inbuffer = 0; uint8 *outbuffer = 0; uint32 inPart1 = 0; uint32 inPart2 = 0; uint8 compressionType = 0; Common::String entryStr; CachedArchive::FileInputList fileList; pos = 0; const uint32 kExecSize = 0x0bba; const uint32 kHeaderSize = 30; const uint32 kHeaderSize2 = 46; for (Common::List::iterator a = archives.begin(); a != archives.end(); ++a) { startFile = true; for (uint32 i = a->firstFile; i != (a->lastFile + 1); i++) { sprintf(filenameExt, extension.c_str(), i); filenameTemp = a->filename + Common::String(filenameExt); if (!(tmpFile = owner->createReadStream(filenameTemp))) { debug(3, "couldn't open file '%s'\n", filenameTemp.c_str()); break; } uint32 size = (i == a->lastFile) ? a->endOffset : tmpFile->size(); if (startFile) { startFile = false; pos = a->startOffset + kExecSize; if (pos > size) { pos -= size; delete tmpFile; tmpFile = 0; continue; } } else { if (inPart2) { tmpFile->seek(1, SEEK_SET); tmpFile->read(inbuffer + inPart1, inPart2); inPart2 = 0; if (compressionType > 0) exp.process(outbuffer, inbuffer, outsize, insize); else memcpy(outbuffer, inbuffer, outsize); delete[] inbuffer; inbuffer = 0; newEntry.data = outbuffer; newEntry.size = outsize; newEntry.name = entryStr; fileList.push_back(newEntry); } pos++; } while (pos < size) { uint8 hdr[43]; uint32 m = 0; tmpFile->seek(pos, SEEK_SET); if (pos + 42 > size) { m = size - pos; uint32 b = 42 - m; if (m >= 4) { uint32 id = tmpFile->readUint32LE(); if (id == 0x06054B50) { startFile = true; break; } else { tmpFile->seek(pos, SEEK_SET); } } sprintf(filenameExt, extension.c_str(), i + 1); filenameTemp = a->filename + Common::String(filenameExt); Common::SeekableReadStream *tmpFile2 = owner->createReadStream(filenameTemp); tmpFile->read(hdr, m); tmpFile2->read(hdr + m, b); delete tmpFile2; } else { tmpFile->read(hdr, 42); } uint32 id = READ_LE_UINT32(hdr); if (id == 0x04034B50) { compressionType = hdr[8]; insize = READ_LE_UINT32(hdr + 18); outsize = READ_LE_UINT32(hdr + 22); uint16 filestrlen = READ_LE_UINT16(hdr + 26); *(hdr + 30 + filestrlen) = 0; entryStr = Common::String((const char *)(hdr + 30)); pos += (kHeaderSize + filestrlen - m); tmpFile->seek(pos, SEEK_SET); outbuffer = new uint8[outsize]; if (!outbuffer) error("Out of memory: Can't uncompress installer files"); if (!inbuffer) { inbuffer = new uint8[insize]; if (!inbuffer) error("Out of memory: Can't uncompress installer files"); } if ((pos + insize) > size) { // this is for files that are split between two archive files inPart1 = size - pos; inPart2 = insize - inPart1; tmpFile->read(inbuffer, inPart1); } else { tmpFile->read(inbuffer, insize); inPart2 = 0; if (compressionType > 0) exp.process(outbuffer, inbuffer, outsize, insize); else memcpy(outbuffer, inbuffer, outsize); delete[] inbuffer; inbuffer = 0; newEntry.data = outbuffer; newEntry.size = outsize; newEntry.name = entryStr; fileList.push_back(newEntry); } pos += insize; if (pos > size) { pos -= size; break; } } else { uint32 filestrlen = READ_LE_UINT32(hdr + 28); pos += (kHeaderSize2 + filestrlen - m); } } delete tmpFile; tmpFile = 0; } } archives.clear(); return new CachedArchive(fileList); } } // end of namespace Kyra