-
Notifications
You must be signed in to change notification settings - Fork 20
/
lzw.cpp
109 lines (100 loc) · 2.57 KB
/
lzw.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
/*
* Heart of Darkness engine rewrite
* Copyright (C) 2009-2011 Gregory Montoir (cyx@users.sourceforge.net)
*/
#include "intern.h"
enum {
kCodeWidth = 9,
kClearCode = 1 << (kCodeWidth - 1),
kEndCode = kClearCode + 1,
kNewCodes = kEndCode + 1,
kStackSize = 8192,
kMaxBits = 12
};
struct LzwDecoder {
uint16_t _prefix[1 << kMaxBits];
uint8_t _stack[kStackSize];
const uint8_t *_buf;
uint32_t _currentBits;
uint8_t _bitsLeft;
uint32_t nextCode(int codeSize);
int decode(uint8_t *dst);
};
static struct LzwDecoder _lzw;
uint32_t LzwDecoder::nextCode(int codeSize) { // 9 to 12bits
_currentBits |= (*_buf++) << _bitsLeft;
_bitsLeft += 8;
if (_bitsLeft < codeSize) {
_currentBits |= (*_buf++) << _bitsLeft;
_bitsLeft += 8;
}
const uint32_t code = _currentBits & ((1 << codeSize) - 1);
_currentBits >>= codeSize;
_bitsLeft -= codeSize;
return code;
}
int LzwDecoder::decode(uint8_t *dst) {
uint8_t *p = dst;
uint8_t *stackPtr = &_stack[kStackSize - 1];
uint32_t previousCode = 0;
uint32_t lastCode = 0;
uint32_t currentCode;
uint32_t currentSlot = kNewCodes;
uint32_t topSlot = 1 << kCodeWidth;
int codeSize = kCodeWidth;
while ((currentCode = nextCode(codeSize)) != kEndCode) {
if (currentCode == kClearCode) {
currentSlot = kNewCodes;
topSlot = 1 << kCodeWidth;
codeSize = kCodeWidth;
while ((currentCode = nextCode(codeSize)) == kClearCode) {
}
if (currentCode == kEndCode) {
break;
} else if (currentCode >= kNewCodes) {
currentCode = 0;
}
previousCode = lastCode = currentCode;
*p++ = (uint8_t)currentCode;
} else {
uint8_t *currentStackPtr = stackPtr;
uint32_t slot = currentSlot;
uint32_t code = currentCode;
if (currentCode >= slot) {
code = lastCode;
currentStackPtr = &_stack[kStackSize - 2];
*currentStackPtr = (uint8_t)previousCode;
}
while (code >= kNewCodes) {
--currentStackPtr;
assert(currentStackPtr >= &_stack[0]);
*currentStackPtr = _stack[code];
code = _prefix[code];
}
--currentStackPtr;
*currentStackPtr = (uint8_t)code;
if (slot < topSlot) {
_stack[slot] = (uint8_t)code;
previousCode = code;
_prefix[slot] = (uint16_t)lastCode;
lastCode = currentCode;
++slot;
currentSlot = slot;
}
if (slot >= topSlot && codeSize < kMaxBits) {
topSlot <<= 1;
++codeSize;
}
while (currentStackPtr < stackPtr) {
*p++ = *currentStackPtr++;
}
assert(currentStackPtr == stackPtr);
}
}
return p - dst;
}
int decodeLZW(const uint8_t *src, uint8_t *dst) {
memset(&_lzw, 0, sizeof(_lzw));
_lzw._buf = src;
return _lzw.decode(dst);
}