mirror of
https://github.com/ZDoom/Raze.git
synced 2025-01-10 02:50:49 +00:00
435 lines
11 KiB
C++
435 lines
11 KiB
C++
/*
|
|
** ancientzip.cpp
|
|
**
|
|
**---------------------------------------------------------------------------
|
|
** Copyright 2010-2011 Randy Heit
|
|
** All rights reserved.
|
|
**
|
|
** Redistribution and use in source and binary forms, with or without
|
|
** modification, are permitted provided that the following conditions
|
|
** are met:
|
|
**
|
|
** 1. Redistributions of source code must retain the above copyright
|
|
** notice, this list of conditions and the following disclaimer.
|
|
** 2. Redistributions in binary form must reproduce the above copyright
|
|
** notice, this list of conditions and the following disclaimer in the
|
|
** documentation and/or other materials provided with the distribution.
|
|
** 3. The name of the author may not be used to endorse or promote products
|
|
** derived from this software without specific prior written permission.
|
|
**
|
|
** THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
** IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
** OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
** IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
** INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
** NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
** THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
**---------------------------------------------------------------------------
|
|
**
|
|
** Based in information from
|
|
**
|
|
gunzip.c by Pasi Ojala, a1bert@iki.fi
|
|
http://www.iki.fi/a1bert/
|
|
|
|
A hopefully easier to understand guide to GZip
|
|
(deflate) decompression routine than the GZip
|
|
source code.
|
|
|
|
*/
|
|
|
|
/*----------------------------------------------------------------------*/
|
|
|
|
|
|
#include <stdlib.h>
|
|
#include "ancientzip.h"
|
|
|
|
/****************************************************************
|
|
Bit-I/O variables and routines/macros
|
|
|
|
These routines work in the bit level because the target
|
|
environment does not have a barrel shifter. Trying to
|
|
handle several bits at once would've only made the code
|
|
slower.
|
|
|
|
If the environment supports multi-bit shifts, you should
|
|
write these routines again (see e.g. the GZIP sources).
|
|
|
|
[RH] Since the target environment is not a C64, I did as
|
|
suggested and rewrote these using zlib as a reference.
|
|
|
|
****************************************************************/
|
|
|
|
#define READBYTE(c) \
|
|
do { \
|
|
c = 0; \
|
|
if (InLeft) { \
|
|
InLeft--; \
|
|
if (bs < be) \
|
|
c = ReadBuf[bs++]; \
|
|
else { \
|
|
be = (decltype(be))In->Read(&ReadBuf, sizeof(ReadBuf)); \
|
|
c = ReadBuf[0]; \
|
|
bs = 1; \
|
|
} \
|
|
} \
|
|
} while (0)
|
|
|
|
/* Get a byte of input into the bit accumulator. */
|
|
#define PULLBYTE() \
|
|
do { \
|
|
unsigned char next; \
|
|
READBYTE(next); \
|
|
Hold += (unsigned int)(next) << Bits; \
|
|
Bits += 8; \
|
|
} while (0)
|
|
|
|
/* Assure that there are at least n bits in the bit accumulator. */
|
|
#define NEEDBITS(n) \
|
|
do { \
|
|
while (Bits < (unsigned)(n)) \
|
|
PULLBYTE(); \
|
|
} while (0)
|
|
|
|
/* Return the low n bits of the bit accumulator (n < 16) */
|
|
#define BITS(n) \
|
|
((unsigned)Hold & ((1U << (n)) - 1))
|
|
|
|
/* Remove n bits from the bit accumulator */
|
|
#define DROPBITS(n) \
|
|
do { \
|
|
Hold >>= (n); \
|
|
Bits -= (unsigned)(n); \
|
|
} while (0)
|
|
|
|
#define READBITS(c, a) \
|
|
do { \
|
|
NEEDBITS(a); \
|
|
c = BITS(a); \
|
|
DROPBITS(a); \
|
|
} while (0)
|
|
|
|
/****************************************************************
|
|
Shannon-Fano tree routines
|
|
****************************************************************/
|
|
|
|
static const unsigned char BitReverse4[] = {
|
|
0x00, 0x08, 0x04, 0x0c, 0x02, 0x0a, 0x06, 0x0e,
|
|
0x01, 0x09, 0x05, 0x0d, 0x03, 0x0b, 0x07, 0x0f
|
|
};
|
|
|
|
#define FIRST_BIT_LEN 8
|
|
#define REST_BIT_LEN 4
|
|
|
|
void FZipExploder::InsertCode(TArray<HuffNode> &decoder, unsigned int pos, int bits, unsigned short code, int len, unsigned char value)
|
|
{
|
|
assert(len > 0);
|
|
unsigned int node = pos + (code & ((1 << bits) - 1));
|
|
|
|
if (len > bits)
|
|
{ // This code uses more bits than this level has room for. Store the bottom bits
|
|
// in this table, then proceed to the next one.
|
|
unsigned int child = decoder[node].ChildTable;
|
|
if (child == 0)
|
|
{ // Need to create child table.
|
|
child = InitTable(decoder, 1 << REST_BIT_LEN);
|
|
decoder[node].ChildTable = child;
|
|
decoder[node].Length = bits;
|
|
decoder[node].Value = 0;
|
|
}
|
|
else
|
|
{
|
|
assert(decoder[node].Length == bits);
|
|
assert(decoder[node].Value == 0);
|
|
}
|
|
InsertCode(decoder, child, REST_BIT_LEN, code >> bits, len - bits, value);
|
|
}
|
|
else
|
|
{ // If this code uses fewer bits than this level of the table, it is
|
|
// inserted repeatedly for each value that matches it at its lower
|
|
// bits.
|
|
for (int i = 1 << (bits - len); --i >= 0; node += 1 << len)
|
|
{
|
|
decoder[node].Length = len;
|
|
decoder[node].Value = value;
|
|
assert(decoder[node].ChildTable == 0);
|
|
}
|
|
}
|
|
}
|
|
|
|
unsigned int FZipExploder::InitTable(TArray<HuffNode> &decoder, int numspots)
|
|
{
|
|
unsigned int start = decoder.Size();
|
|
decoder.Reserve(numspots);
|
|
memset(&decoder[start], 0, sizeof(HuffNode)*numspots);
|
|
return start;
|
|
}
|
|
|
|
int FZipExploder::buildercmp(const void *a, const void *b)
|
|
{
|
|
const TableBuilder *v1 = (const TableBuilder *)a;
|
|
const TableBuilder *v2 = (const TableBuilder *)b;
|
|
int d = v1->Length - v2->Length;
|
|
if (d == 0) {
|
|
d = v1->Value - v2->Value;
|
|
}
|
|
return d;
|
|
}
|
|
|
|
int FZipExploder::BuildDecoder(TArray<HuffNode> &decoder, TableBuilder *values, int numvals)
|
|
{
|
|
int i;
|
|
|
|
qsort(values, numvals, sizeof(*values), buildercmp);
|
|
|
|
// Generate the Shannon-Fano tree:
|
|
unsigned short code = 0;
|
|
unsigned short code_increment = 0;
|
|
unsigned short last_bit_length = 0;
|
|
for (i = numvals - 1; i >= 0; --i)
|
|
{
|
|
code += code_increment;
|
|
if (values[i].Length != last_bit_length)
|
|
{
|
|
last_bit_length = values[i].Length;
|
|
code_increment = 1 << (16 - last_bit_length);
|
|
}
|
|
// Reverse the order of the bits in the code before storing it.
|
|
values[i].Code = BitReverse4[code >> 12] |
|
|
(BitReverse4[(code >> 8) & 0xf] << 4) |
|
|
(BitReverse4[(code >> 4) & 0xf] << 8) |
|
|
(BitReverse4[code & 0xf] << 12);
|
|
}
|
|
|
|
// Insert each code into the hierarchical table. The top level is FIRST_BIT_LEN bits,
|
|
// and the other levels are REST_BIT_LEN bits. If a code does not completely fill
|
|
// a level, every permutation for the remaining bits is filled in to
|
|
// match this one.
|
|
InitTable(decoder, 1 << FIRST_BIT_LEN); // Set up the top level.
|
|
for (i = 0; i < numvals; ++i)
|
|
{
|
|
InsertCode(decoder, 0, FIRST_BIT_LEN, values[i].Code, values[i].Length, values[i].Value);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
int FZipExploder::DecodeSFValue(const TArray<HuffNode> &decoder)
|
|
{
|
|
unsigned int bits = FIRST_BIT_LEN, table = 0, code;
|
|
const HuffNode *pos;
|
|
do
|
|
{
|
|
NEEDBITS(bits);
|
|
code = BITS(bits);
|
|
bits = REST_BIT_LEN;
|
|
pos = &decoder[table + code];
|
|
DROPBITS(pos->Length);
|
|
table = pos->ChildTable;
|
|
}
|
|
while (table != 0);
|
|
return pos->Value;
|
|
}
|
|
|
|
|
|
int FZipExploder::DecodeSF(TArray<HuffNode> &decoder, int numvals)
|
|
{
|
|
TableBuilder builder[256];
|
|
unsigned char a, c;
|
|
int i, n, v = 0;
|
|
|
|
READBYTE(c);
|
|
n = c + 1;
|
|
for (i = 0; i < n; i++) {
|
|
int nv, bl;
|
|
READBYTE(a);
|
|
nv = ((a >> 4) & 15) + 1;
|
|
bl = (a & 15) + 1;
|
|
while (nv--) {
|
|
builder[v].Length = bl;
|
|
builder[v].Value = v;
|
|
v++;
|
|
}
|
|
}
|
|
if (v != numvals)
|
|
return 1; /* bad table */
|
|
return BuildDecoder(decoder, builder, v);
|
|
}
|
|
|
|
int FZipExploder::Explode(unsigned char *out, unsigned int outsize,
|
|
FileReader &in, unsigned int insize,
|
|
int flags)
|
|
{
|
|
int c, i, minMatchLen = 3, len, dist;
|
|
int lowDistanceBits;
|
|
unsigned int bIdx = 0;
|
|
|
|
Hold = 0;
|
|
Bits = 0;
|
|
In = ∈
|
|
InLeft = insize;
|
|
bs = be = 0;
|
|
|
|
if ((flags & 4)) {
|
|
/* 3 trees: literals, lengths, distance top 6 */
|
|
minMatchLen = 3;
|
|
if (DecodeSF(LiteralDecoder, 256))
|
|
return 1;
|
|
} else {
|
|
/* 2 trees: lengths, distance top 6 */
|
|
minMatchLen = 2;
|
|
}
|
|
if (DecodeSF(LengthDecoder, 64))
|
|
return 1;
|
|
if (DecodeSF(DistanceDecoder, 64))
|
|
return 1;
|
|
|
|
lowDistanceBits = (flags & 2) ? /* 8k dictionary */ 7 : /* 4k dictionary */ 6;
|
|
while (bIdx < outsize)
|
|
{
|
|
READBITS(c, 1);
|
|
if (c) {
|
|
/* literal data */
|
|
if ((flags & 4)) {
|
|
c = DecodeSFValue(LiteralDecoder);
|
|
} else {
|
|
READBITS(c, 8);
|
|
}
|
|
out[bIdx++] = c;
|
|
} else {
|
|
READBITS(dist, lowDistanceBits);
|
|
c = DecodeSFValue(DistanceDecoder);
|
|
dist |= (c << lowDistanceBits);
|
|
len = DecodeSFValue(LengthDecoder);
|
|
if (len == 63) {
|
|
READBITS(c, 8);
|
|
len += c;
|
|
}
|
|
len += minMatchLen;
|
|
dist++;
|
|
if (bIdx + len > outsize) {
|
|
throw CExplosionError("Not enough output space");
|
|
}
|
|
if ((unsigned int)dist > bIdx) {
|
|
/* Anything before the first input byte is zero. */
|
|
int zeros = dist - bIdx;
|
|
if (len < zeros)
|
|
zeros = len;
|
|
for(i = zeros; i; i--)
|
|
out[bIdx++] = 0;
|
|
len -= zeros;
|
|
}
|
|
for(i = len; i; i--, bIdx++) {
|
|
out[bIdx] = out[bIdx - dist];
|
|
}
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* HSIZE is defined as 2^13 (8192) in unzip.h */
|
|
#define HSIZE 8192
|
|
#define BOGUSCODE 256
|
|
#define CODE_MASK (HSIZE - 1) /* 0x1fff (lower bits are parent's index) */
|
|
#define FREE_CODE HSIZE /* 0x2000 (code is unused or was cleared) */
|
|
#define HAS_CHILD (HSIZE << 1) /* 0x4000 (code has a child--do not clear) */
|
|
|
|
int ShrinkLoop(unsigned char *out, unsigned int outsize, FileReader &_In, unsigned int InLeft)
|
|
{
|
|
FileReader *In = &_In;
|
|
unsigned char ReadBuf[256];
|
|
unsigned short Parent[HSIZE];
|
|
unsigned char Value[HSIZE], Stack[HSIZE];
|
|
unsigned char *newstr;
|
|
int len;
|
|
int KwKwK, codesize = 9; /* start at 9 bits/code */
|
|
int code, oldcode, freecode, curcode;
|
|
unsigned int Bits = 0, Hold = 0;
|
|
unsigned int size = 0;
|
|
unsigned int bs = 0, be = 0;
|
|
|
|
freecode = BOGUSCODE;
|
|
for (code = 0; code < BOGUSCODE; code++)
|
|
{
|
|
Value[code] = code;
|
|
Parent[code] = BOGUSCODE;
|
|
}
|
|
for (code = BOGUSCODE+1; code < HSIZE; code++)
|
|
Parent[code] = FREE_CODE;
|
|
|
|
READBITS(oldcode, codesize);
|
|
if (size < outsize) {
|
|
out[size++] = oldcode;
|
|
}
|
|
|
|
while (size < outsize)
|
|
{
|
|
READBITS(code, codesize);
|
|
if (code == BOGUSCODE) { /* possible to have consecutive escapes? */
|
|
READBITS(code, codesize);
|
|
if (code == 1) {
|
|
codesize++;
|
|
} else if (code == 2) {
|
|
/* clear leafs (nodes with no children) */
|
|
/* first loop: mark each parent as such */
|
|
for (code = BOGUSCODE+1; code < HSIZE; ++code) {
|
|
curcode = (Parent[code] & CODE_MASK);
|
|
|
|
if (curcode > BOGUSCODE)
|
|
Parent[curcode] |= HAS_CHILD; /* set parent's child-bit */
|
|
}
|
|
|
|
/* second loop: clear all nodes *not* marked as parents; reset flag bits */
|
|
for (code = BOGUSCODE+1; code < HSIZE; ++code) {
|
|
if (Parent[code] & HAS_CHILD) { /* just clear child-bit */
|
|
Parent[code] &= ~HAS_CHILD;
|
|
} else { /* leaf: lose it */
|
|
Parent[code] = FREE_CODE;
|
|
}
|
|
}
|
|
freecode = BOGUSCODE;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
newstr = &Stack[HSIZE-1];
|
|
curcode = code;
|
|
|
|
if (Parent[curcode] == FREE_CODE) {
|
|
KwKwK = 1;
|
|
newstr--; /* last character will be same as first character */
|
|
curcode = oldcode;
|
|
len = 1;
|
|
} else {
|
|
KwKwK = 0;
|
|
len = 0;
|
|
}
|
|
|
|
do {
|
|
*newstr-- = Value[curcode];
|
|
len++;
|
|
curcode = (Parent[curcode] & CODE_MASK);
|
|
} while (curcode != BOGUSCODE);
|
|
|
|
newstr++;
|
|
if (KwKwK) {
|
|
Stack[HSIZE-1] = *newstr;
|
|
}
|
|
|
|
do {
|
|
freecode++;
|
|
} while (Parent[freecode] != FREE_CODE);
|
|
|
|
Parent[freecode] = oldcode;
|
|
Value[freecode] = *newstr;
|
|
oldcode = code;
|
|
|
|
while (len--) {
|
|
out[size++] = *newstr++;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|