mirror of
https://github.com/libretro/scummvm.git
synced 2025-04-03 07:11:49 +00:00
262 lines
7.7 KiB
C++
262 lines
7.7 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 "immortal/immortal.h"
|
|
|
|
/* Decompression:
|
|
* This decompression algorithm follows the source assembly very closely,
|
|
* which is itself a modification to LZW (a derivitive of LZ78).
|
|
* In: Source data as File, size of data
|
|
* Out: Pointer to uncompressed data as SeekableReadStream
|
|
*/
|
|
namespace Immortal {
|
|
|
|
Common::SeekableReadStream *ImmortalEngine::unCompress(Common::File *src, int srcLen) {
|
|
/* Note: this function does not seek() in the file, which means
|
|
* that if there is a header on the data, the expectation is that
|
|
* seek() was already used to move past the header before this function.
|
|
*/
|
|
|
|
// If the source data has no length, we certainly do not want to decompress it
|
|
if (srcLen == 0) {
|
|
return nullptr;
|
|
}
|
|
|
|
/* This will be the dynamically re-allocated writeable memory stream.
|
|
* We do not want it to be deleted from scope, as this location is where
|
|
* the readstream being returned will point to.
|
|
*/
|
|
Common::MemoryWriteStreamDynamic dstW(DisposeAfterUse::NO);
|
|
|
|
// The 20k bytes of memory that compression gets allocated to work with for the dictionary and the stack of chars
|
|
uint16 start[0x4000]; // Really needs a better name, remember to do this future me
|
|
uint16 ptk[0x4000]; // Pointer To Keys? Also needs a better name
|
|
byte stack[0x4000]; // Stack of chars to be stored
|
|
|
|
// These are the main variables we'll need for this
|
|
uint16 findEmpty;
|
|
uint16 code; // Needs to be ASL to index with
|
|
uint16 inputCode;
|
|
uint16 finalChar;
|
|
uint16 myCode; // Silly name is silly
|
|
uint16 oldCode;
|
|
uint16 index; // The Y register was used to index the byte array's, this will sort of take its place
|
|
uint16 evenOdd = 0;
|
|
uint16 topStack = 0;
|
|
|
|
byte outByte; // If only we could SEP #$20 like the 65816
|
|
|
|
setupDictionary(start, ptk, findEmpty); // Clear the dictionary and also set findEmpty to 8k
|
|
bool carry = true; // This will represent the carry flag so we can make this a clean loop
|
|
|
|
code = getInputCode(carry, src, srcLen, evenOdd); // Get the first code
|
|
if (carry == false) {
|
|
return nullptr; // This is essentially the same as the first error check, but the source returns an error code and didn't even check it here so we might as well
|
|
}
|
|
|
|
finalChar = code;
|
|
oldCode = code;
|
|
myCode = code;
|
|
|
|
outByte = code & kMaskLow;
|
|
dstW.writeByte(outByte); // Take just the lower byte and write it the output
|
|
|
|
// :nextcode
|
|
while (carry == true) {
|
|
|
|
code = getInputCode(carry, src, srcLen, evenOdd); // Get the next code
|
|
if (carry == true) {
|
|
|
|
index = code << 1;
|
|
inputCode = code;
|
|
myCode = code;
|
|
|
|
// Split up the conditional statement to be easier to follow
|
|
uint16 cond;
|
|
cond = start[index] & kMaskLast;
|
|
cond |= ptk[index];
|
|
|
|
if ((cond & kMaskHigh) == 0) { // Empty code
|
|
index = topStack;
|
|
outByte = finalChar & kMaskLow;
|
|
stack[index] = outByte;
|
|
topStack++;
|
|
myCode = oldCode;
|
|
}
|
|
|
|
// :nextsymbol
|
|
index = myCode << 1;
|
|
while (index >= 0x200) {
|
|
myCode = start[index] & kMask12Bit;
|
|
outByte = ptk[index] & kMaskLow;
|
|
index = topStack;
|
|
stack[index] = outByte;
|
|
topStack++;
|
|
index = myCode << 1;
|
|
}
|
|
|
|
// :singlechar
|
|
finalChar = (myCode >> 1);
|
|
outByte = finalChar & kMaskLow;
|
|
dstW.writeByte(outByte);
|
|
|
|
// :dump
|
|
while (topStack != 0xFFFF) { // Dump the chars on the stack into the output file
|
|
outByte = stack[topStack] & kMaskLow;
|
|
dstW.writeByte(outByte);
|
|
topStack--;
|
|
}
|
|
|
|
topStack = 0;
|
|
code = getMember(oldCode, finalChar, findEmpty, start, ptk);
|
|
oldCode = inputCode;
|
|
}
|
|
|
|
}
|
|
|
|
/* Return a readstream with a pointer to the data in the write stream.
|
|
* This one we do want to dispose after using, because it will be in the scope of the engine itself
|
|
*/
|
|
return new Common::MemoryReadStream(dstW.getData(), dstW.size(), DisposeAfterUse::YES);
|
|
}
|
|
|
|
void ImmortalEngine::setupDictionary(uint16 start[], uint16 ptk[], uint16 &findEmpty) {
|
|
// Clear the whole dictionary
|
|
for (int i = 0x3FFF; i >= 0; i--) {
|
|
start[i] = 0;
|
|
ptk[i] = 0;
|
|
}
|
|
|
|
// Set the initial 256 bytes to be value 256, these are the characters without extensions
|
|
for (int i = 255; i >= 0; i--) {
|
|
ptk[i] = 256;
|
|
}
|
|
|
|
// This shouldn't really be done inside the function, but for the sake of consistency with the source, we will
|
|
findEmpty = 0x8000;
|
|
}
|
|
|
|
int ImmortalEngine::getInputCode(bool &carry, Common::File *src, int &srcLen, uint16 &evenOdd) {
|
|
// Check if we're at the end of the file
|
|
if (srcLen == 0) {
|
|
carry = false;
|
|
return 0;
|
|
}
|
|
|
|
uint16 c;
|
|
if (evenOdd != 0) { // Odd
|
|
srcLen--;
|
|
evenOdd--;
|
|
c = (src->readUint16BE() >> 3) & 0x00FE; // & #-1-1
|
|
} else { // Even
|
|
srcLen -= 2;
|
|
evenOdd++;
|
|
c = (src->readUint16BE() & kMask12Bit) << 1;
|
|
src->seek(-1, SEEK_CUR);
|
|
}
|
|
return c;
|
|
}
|
|
|
|
uint16 ImmortalEngine::getMember(uint16 codeW, uint16 k, uint16 &findEmpty, uint16 start[], uint16 ptk[]) {
|
|
// This function is effectively void, as the return value is only used in compression
|
|
|
|
// k and codeW are local variables with the value of oldCode and finalChar
|
|
|
|
uint16 hash;
|
|
uint16 tmp;
|
|
bool ag = true;
|
|
|
|
hash = (k << 3) ^ k;
|
|
hash = (hash << 1) ^ codeW;
|
|
hash <<= 1;
|
|
|
|
hash = (hash >= 0x200) ? hash : hash + 0x200;
|
|
|
|
uint16 a = start[hash] & 0x0F00;
|
|
uint16 b = ptk[hash] & kMaskHigh;
|
|
if (a | b) {
|
|
start[hash] = codeW;
|
|
ptk[hash] = k | 0x100;
|
|
return ptk[hash];
|
|
}
|
|
|
|
// This loop is a bit wacky, due to the way the jumps were stuctured in the source
|
|
while (ag == true) {
|
|
if ((start[hash] & kMask12Bit) == codeW) {
|
|
if ((ptk[hash] & kMaskLow) == k) {
|
|
return hash >> 1;
|
|
}
|
|
}
|
|
|
|
tmp = start[hash] & kMaskLast;
|
|
if (tmp == 0) {
|
|
// I've separated this into it's own function for the sake of this loop being readable
|
|
appendList(codeW, k, hash, findEmpty, start, ptk, tmp);
|
|
ag = false;
|
|
|
|
} else {
|
|
hash = xba(ptk[hash]);
|
|
hash = (hash & kMaskLow) | (tmp >> 4);
|
|
hash <<= 1;
|
|
}
|
|
}
|
|
return hash;
|
|
}
|
|
|
|
void ImmortalEngine::appendList(uint16 codeW, uint16 k, uint16 &hash, uint16 &findEmpty, uint16 start[], uint16 ptk[], uint16 &tmp) {
|
|
uint16 prev;
|
|
uint16 link;
|
|
|
|
prev = hash;
|
|
if (hash >= 0x200) {
|
|
setupDictionary(start, ptk, findEmpty);
|
|
|
|
} else {
|
|
bool found = false;
|
|
while (found == false) {
|
|
hash -= 2;
|
|
if (hash >= 0x200) {
|
|
setupDictionary(start, ptk, findEmpty);
|
|
found = true;
|
|
}
|
|
|
|
// Split up the conditional statement to be easier to follow
|
|
uint16 cond;
|
|
cond = start[hash] & kMaskLast;
|
|
cond |= ptk[hash];
|
|
|
|
if ((cond & kMaskHigh) == 0) {
|
|
findEmpty = hash;
|
|
start[hash] = codeW;
|
|
ptk[hash] = k | 0x100;
|
|
|
|
link = hash >> 1;
|
|
|
|
ptk[prev] = (link << 8) | (ptk[prev] & kMaskLow);
|
|
start[prev] |= (link >> 4) & kMaskLast;
|
|
found = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
} // namespace immortal
|