-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy pathtlDeflate.h
253 lines (213 loc) · 5.69 KB
/
tlDeflate.h
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
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
/*
KLayout Layout Viewer
Copyright (C) 2013-2018 Matthias Koefferlein
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 St, Fifth Floor, Boston, MA 02110-1301 USA
*/
#ifndef HDR_tlDeflate
#define HDR_tlDeflate
#include "config.h"
#include "tlStream.h"
#include "tlException.h"
// forware definition of the zlib stream structure - we can omit the zlib header here
struct z_stream_s;
namespace tl
{
class HuffmannDecoder;
/**
* @brief A bit stream reader according to the DEFLATE specification
*
* This filter reads bytes from a tl::Stream and delivers bits, taken from
* these bytes. The bits are delivered in the order specified by the DEFLATE
* format specification (least significant bit first).
*/
class KLAYOUT_DLL BitStream
{
public:
/**
* @brief Constructor
*
* Constructs a bit stream filter attached to the given stream
*/
BitStream (tl::InputStream &input)
: mp_input (&input),
m_mask (0), m_byte (0)
{
// ...
}
/**
* @brief Get a byte
*
* This method simply passes the byte.
* The method expects the next byte to be available.
*/
unsigned char get_byte ()
{
m_mask = 0;
const char *c = mp_input->get (1, true /*bypass_deflate*/);
if (c == 0) {
throw tl::Exception (tl::translate ("Unexpected end of file (DEFLATE implementation)"));
}
return *c;
}
/**
* @brief Get a single bit
*
* This method gets the next bit available.
* The method expects the next byte to be available, if one needs to be read.
*/
bool get_bit ()
{
if (m_mask == 0) {
m_byte = get_byte ();
m_mask = 0x01;
}
bool b = ((m_byte & m_mask) != 0);
m_mask <<= 1;
return b;
}
/**
* @brief Get a sequence of bits
*
* This method gets the next n bits and delivers them as a single unsigned int,
* packing the first bit into the least signification bit. This is the specification
* for reading multiple bit values except Huffmann codes.
*/
unsigned int get_bits (unsigned int n)
{
// KLUDGE: take directly from the byte for performance.
unsigned int r = 0;
unsigned int m = 1;
while (n-- > 0) {
r |= get_bit () ? m : 0;
m <<= 1;
}
return r;
}
/**
* @brief Skip the next bits up to the next byte boundary
*/
void skip_to_byte ()
{
m_mask = 0;
}
private:
tl::InputStream *mp_input;
unsigned char m_mask;
unsigned char m_byte;
};
/**
* @brief A Deflating filter, similar to InflateFilter
*
* This filter can be used to produce a DEFLATE-compressed stream on an
* output stream. Similar to InflateFilter it is put between the source
* and an output stream.
*/
class KLAYOUT_DLL DeflateFilter
{
public:
/**
* @brief Constructor: creates a filter in front of the output stream
*/
DeflateFilter (tl::OutputStream &output);
/**
* @brief Destructor
*
* Note that this method will not flush the stream since that may
* throw an exception. flush() has to be called explcitly.
*/
~DeflateFilter ();
/**
* @brief Outputs a series of bytes into the deflated stream
*/
void put (const char *b, size_t n);
/**
* @brief Flushes the buffer and writes all remaining bytes
*
* Note: this method must be called always before the stream
* is closed and the filter is destroyed. Otherwise, the last bytes may be lost.
*/
void flush ();
/**
* @brief Get the uncompressed count collected so far
*/
size_t uncompressed () const
{
return m_uc;
}
/**
* @brief Get the compressed count collected so far
*/
size_t compressed () const
{
return m_cc;
}
private:
bool m_finished;
char m_buffer[65536];
tl::OutputStream *mp_output;
z_stream_s *mp_stream;
size_t m_uc, m_cc;
};
/**
* @brief The DEFLATE decompression (inflating) filter
*
* This class is the main DEFLATE decoder. It is called "filter", since it takes bytes from
* the input (from the "input" stream) and delivers bytes on the output ("get" method).
*/
class KLAYOUT_DLL InflateFilter
{
public:
/**
* @brief Constructor
*
* Constructs a filter attached to the given Stream object.
*/
InflateFilter (tl::InputStream &input);
/**
* @brief Destructor
*/
~InflateFilter ();
/**
* @brief Get the next byte(s)
*
* This method returns a contiguous block of decoded bytes with the given length.
* The maximum size of the block available is half the buffer size.
*/
const char *get (size_t n);
/**
* @brief Undo the last "get" operation
*
* This method ungets the last bytes obtained with the "get" method.
* the size to unget must match the last get's size.
*/
void unget (size_t n);
/**
* @brief Report true, if no more bytes can be delivered
*/
bool at_end ();
private:
BitStream m_input;
char m_buffer[65536];
unsigned int m_b_insert;
unsigned int m_b_read;
bool m_at_end;
// processor state
bool m_last_block;
int m_uncompressed_length;
HuffmannDecoder *mp_lit_decoder, *mp_dist_decoder;
void put_byte (char b);
void put_byte_dist (unsigned int d);
bool process ();
};
}
#endif