2014-06-22 17:56:56 +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.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
// John_Doe's implementation
|
|
|
|
|
|
|
|
#include "prince/decompress.h"
|
|
|
|
|
|
|
|
namespace Prince {
|
|
|
|
|
|
|
|
static const uint16 table1[] = {
|
|
|
|
0x8000, 0x0002,
|
|
|
|
0x4000, 0x0004,
|
|
|
|
0x2000, 0x0008,
|
|
|
|
0x1000, 0x0010,
|
|
|
|
0x0800, 0x0020,
|
|
|
|
0x0400, 0x0040,
|
|
|
|
0x0200, 0x0080,
|
|
|
|
0x0100, 0x0100,
|
|
|
|
0x0080, 0x0200,
|
|
|
|
0x0040, 0x0400
|
|
|
|
};
|
|
|
|
|
|
|
|
static const uint32 table2[] = {
|
|
|
|
0x0000F000,
|
|
|
|
0x0020FC00,
|
|
|
|
0x00A0FF00,
|
|
|
|
0x02A0FF80,
|
|
|
|
0x06A0FFC0,
|
|
|
|
0x0EA0FFE0,
|
|
|
|
0x1EA0FFF0,
|
|
|
|
0x3EA0FFF8
|
|
|
|
};
|
|
|
|
|
|
|
|
static const uint16 table3[] = {
|
|
|
|
0x8000, 0x0000,
|
|
|
|
0x4000, 0x0002,
|
|
|
|
0x2000, 0x0006,
|
|
|
|
0x1000, 0x000E,
|
|
|
|
0x0800, 0x001E,
|
|
|
|
0x0400, 0x003E,
|
|
|
|
0x0200, 0x007E,
|
|
|
|
0x0100, 0x00FE,
|
|
|
|
0x0080, 0x01FE,
|
|
|
|
0x0040, 0x03FE,
|
|
|
|
0x0020, 0x07FE,
|
|
|
|
0x0010, 0x0FFE,
|
|
|
|
0x0008, 0x1FFE,
|
|
|
|
0x0004, 0x3FFE,
|
|
|
|
0x0002, 0x7FFE,
|
|
|
|
0x0001, 0xFFFE
|
|
|
|
};
|
|
|
|
|
|
|
|
void Decompressor::decompress(byte *source, byte *dest, uint32 destSize) {
|
|
|
|
byte *destEnd = dest + destSize;
|
|
|
|
int more;
|
|
|
|
_src = source;
|
|
|
|
_dst = dest;
|
|
|
|
_bitBuffer = 0x80;
|
|
|
|
while (_dst < destEnd) {
|
|
|
|
uint32 ebp;
|
|
|
|
uint16 offset, length;
|
|
|
|
if (getBit()) {
|
|
|
|
if (getBit()) {
|
|
|
|
if (getBit()) {
|
|
|
|
if (getBit()) {
|
|
|
|
if (getBit()) {
|
|
|
|
if (getBit()) {
|
|
|
|
uint32 tableIndex = 0;
|
|
|
|
while (getBit())
|
|
|
|
tableIndex++;
|
|
|
|
length = table3[tableIndex * 2 + 0];
|
|
|
|
do {
|
|
|
|
more = !(length & 0x8000);
|
|
|
|
length = (length << 1) | getBit();
|
|
|
|
} while (more);
|
|
|
|
length += table3[tableIndex * 2 + 1];
|
|
|
|
length++;
|
|
|
|
memcpy(_dst, _src, length);
|
|
|
|
_src += length;
|
|
|
|
_dst += length;
|
|
|
|
}
|
|
|
|
*_dst++ = *_src++;
|
|
|
|
}
|
|
|
|
*_dst++ = *_src++;
|
|
|
|
}
|
|
|
|
*_dst++ = *_src++;
|
|
|
|
}
|
|
|
|
*_dst++ = *_src++;
|
|
|
|
}
|
|
|
|
*_dst++ = *_src++;
|
|
|
|
}
|
|
|
|
if (!getBit()) {
|
|
|
|
if (getBit()) {
|
|
|
|
uint32 tableIndex = getBit();
|
|
|
|
tableIndex = (tableIndex << 1) | getBit();
|
|
|
|
tableIndex = (tableIndex << 1) | getBit();
|
|
|
|
ebp = table2[tableIndex];
|
|
|
|
length = 1;
|
|
|
|
} else {
|
|
|
|
ebp = 0x0000FF00;
|
|
|
|
length = 0;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
uint32 tableIndex = 0;
|
|
|
|
while (getBit())
|
|
|
|
tableIndex++;
|
|
|
|
length = table1[tableIndex * 2 + 0];
|
|
|
|
do {
|
|
|
|
more = !(length & 0x8000);
|
|
|
|
length = (length << 1) | getBit();
|
|
|
|
} while (more);
|
|
|
|
length += table1[tableIndex * 2 + 1];
|
|
|
|
tableIndex = getBit();
|
|
|
|
tableIndex = (tableIndex << 1) | getBit();
|
|
|
|
tableIndex = (tableIndex << 1) | getBit();
|
|
|
|
ebp = table2[tableIndex];
|
|
|
|
}
|
|
|
|
offset = ebp & 0xFFFF;
|
|
|
|
do {
|
|
|
|
if (_bitBuffer == 0x80) {
|
|
|
|
if (offset >= 0xFF00) {
|
|
|
|
offset = (offset << 8) | *_src++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
more = offset & 0x8000;
|
|
|
|
offset = (offset << 1) | getBit();
|
|
|
|
} while (more);
|
|
|
|
offset += (ebp >> 16);
|
|
|
|
length += 2;
|
|
|
|
while (length--) {
|
|
|
|
if (_dst >= destEnd) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
*_dst = *(_dst - offset);
|
|
|
|
_dst++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int Decompressor::getBit() {
|
|
|
|
int bit = (_bitBuffer & 0x80) >> 7;
|
|
|
|
_bitBuffer <<= 1;
|
|
|
|
if (_bitBuffer == 0) {
|
|
|
|
_bitBuffer = *_src++;
|
|
|
|
bit = (_bitBuffer & 0x80) >> 7;
|
|
|
|
_bitBuffer <<= 1;
|
|
|
|
_bitBuffer |= 1;
|
|
|
|
}
|
|
|
|
return bit;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // End of namespace Prince
|