1*593dc095SDavid du Colombier /* inflate.c -- zlib decompression
2*593dc095SDavid du Colombier * Copyright (C) 1995-2003 Mark Adler
37dd7cddfSDavid du Colombier * For conditions of distribution and use, see copyright notice in zlib.h
47dd7cddfSDavid du Colombier */
57dd7cddfSDavid du Colombier
6*593dc095SDavid du Colombier /*
7*593dc095SDavid du Colombier * Change history:
8*593dc095SDavid du Colombier *
9*593dc095SDavid du Colombier * 1.2.beta0 24 Nov 2002
10*593dc095SDavid du Colombier * - First version -- complete rewrite of inflate to simplify code, avoid
11*593dc095SDavid du Colombier * creation of window when not needed, minimize use of window when it is
12*593dc095SDavid du Colombier * needed, make inffast.c even faster, implement gzip decoding, and to
13*593dc095SDavid du Colombier * improve code readability and style over the previous zlib inflate code
14*593dc095SDavid du Colombier *
15*593dc095SDavid du Colombier * 1.2.beta1 25 Nov 2002
16*593dc095SDavid du Colombier * - Use pointers for available input and output checking in inffast.c
17*593dc095SDavid du Colombier * - Remove input and output counters in inffast.c
18*593dc095SDavid du Colombier * - Change inffast.c entry and loop from avail_in >= 7 to >= 6
19*593dc095SDavid du Colombier * - Remove unnecessary second byte pull from length extra in inffast.c
20*593dc095SDavid du Colombier * - Unroll direct copy to three copies per loop in inffast.c
21*593dc095SDavid du Colombier *
22*593dc095SDavid du Colombier * 1.2.beta2 4 Dec 2002
23*593dc095SDavid du Colombier * - Change external routine names to reduce potential conflicts
24*593dc095SDavid du Colombier * - Correct filename to inffixed.h for fixed tables in inflate.c
25*593dc095SDavid du Colombier * - Make hbuf[] unsigned char to match parameter type in inflate.c
26*593dc095SDavid du Colombier * - Change strm->next_out[-state->offset] to *(strm->next_out - state->offset)
27*593dc095SDavid du Colombier * to avoid negation problem on Alphas (64 bit) in inflate.c
28*593dc095SDavid du Colombier *
29*593dc095SDavid du Colombier * 1.2.beta3 22 Dec 2002
30*593dc095SDavid du Colombier * - Add comments on state->bits assertion in inffast.c
31*593dc095SDavid du Colombier * - Add comments on op field in inftrees.h
32*593dc095SDavid du Colombier * - Fix bug in reuse of allocated window after inflateReset()
33*593dc095SDavid du Colombier * - Remove bit fields--back to byte structure for speed
34*593dc095SDavid du Colombier * - Remove distance extra == 0 check in inflate_fast()--only helps for lengths
35*593dc095SDavid du Colombier * - Change post-increments to pre-increments in inflate_fast(), PPC biased?
36*593dc095SDavid du Colombier * - Add compile time option, POSTINC, to use post-increments instead (Intel?)
37*593dc095SDavid du Colombier * - Make MATCH copy in inflate() much faster for when inflate_fast() not used
38*593dc095SDavid du Colombier * - Use local copies of stream next and avail values, as well as local bit
39*593dc095SDavid du Colombier * buffer and bit count in inflate()--for speed when inflate_fast() not used
40*593dc095SDavid du Colombier *
41*593dc095SDavid du Colombier * 1.2.beta4 1 Jan 2003
42*593dc095SDavid du Colombier * - Split ptr - 257 statements in inflate_table() to avoid compiler warnings
43*593dc095SDavid du Colombier * - Move a comment on output buffer sizes from inffast.c to inflate.c
44*593dc095SDavid du Colombier * - Add comments in inffast.c to introduce the inflate_fast() routine
45*593dc095SDavid du Colombier * - Rearrange window copies in inflate_fast() for speed and simplification
46*593dc095SDavid du Colombier * - Unroll last copy for window match in inflate_fast()
47*593dc095SDavid du Colombier * - Use local copies of window variables in inflate_fast() for speed
48*593dc095SDavid du Colombier * - Pull out common write == 0 case for speed in inflate_fast()
49*593dc095SDavid du Colombier * - Make op and len in inflate_fast() unsigned for consistency
50*593dc095SDavid du Colombier * - Add FAR to lcode and dcode declarations in inflate_fast()
51*593dc095SDavid du Colombier * - Simplified bad distance check in inflate_fast()
52*593dc095SDavid du Colombier * - Added inflateBackInit(), inflateBack(), and inflateBackEnd() in new
53*593dc095SDavid du Colombier * source file infback.c to provide a call-back interface to inflate for
54*593dc095SDavid du Colombier * programs like gzip and unzip -- uses window as output buffer to avoid
55*593dc095SDavid du Colombier * window copying
56*593dc095SDavid du Colombier *
57*593dc095SDavid du Colombier * 1.2.beta5 1 Jan 2003
58*593dc095SDavid du Colombier * - Improved inflateBack() interface to allow the caller to provide initial
59*593dc095SDavid du Colombier * input in strm.
60*593dc095SDavid du Colombier * - Fixed stored blocks bug in inflateBack()
61*593dc095SDavid du Colombier *
62*593dc095SDavid du Colombier * 1.2.beta6 4 Jan 2003
63*593dc095SDavid du Colombier * - Added comments in inffast.c on effectiveness of POSTINC
64*593dc095SDavid du Colombier * - Typecasting all around to reduce compiler warnings
65*593dc095SDavid du Colombier * - Changed loops from while (1) or do {} while (1) to for (;;), again to
66*593dc095SDavid du Colombier * make compilers happy
67*593dc095SDavid du Colombier * - Changed type of window in inflateBackInit() to unsigned char *
68*593dc095SDavid du Colombier *
69*593dc095SDavid du Colombier * 1.2.beta7 27 Jan 2003
70*593dc095SDavid du Colombier * - Changed many types to unsigned or unsigned short to avoid warnings
71*593dc095SDavid du Colombier * - Added inflateCopy() function
72*593dc095SDavid du Colombier *
73*593dc095SDavid du Colombier * 1.2.0 9 Mar 2003
74*593dc095SDavid du Colombier * - Changed inflateBack() interface to provide separate opaque descriptors
75*593dc095SDavid du Colombier * for the in() and out() functions
76*593dc095SDavid du Colombier * - Changed inflateBack() argument and in_func typedef to swap the length
77*593dc095SDavid du Colombier * and buffer address return values for the input function
78*593dc095SDavid du Colombier * - Check next_in and next_out for Z_NULL on entry to inflate()
79*593dc095SDavid du Colombier *
80*593dc095SDavid du Colombier * The history for versions after 1.2.0 are in ChangeLog in zlib distribution.
81*593dc095SDavid du Colombier */
82*593dc095SDavid du Colombier
837dd7cddfSDavid du Colombier #include "zutil.h"
84*593dc095SDavid du Colombier #include "inftrees.h"
85*593dc095SDavid du Colombier #include "inflate.h"
86*593dc095SDavid du Colombier #include "inffast.h"
877dd7cddfSDavid du Colombier
88*593dc095SDavid du Colombier #ifdef MAKEFIXED
89*593dc095SDavid du Colombier # ifndef BUILDFIXED
90*593dc095SDavid du Colombier # define BUILDFIXED
91*593dc095SDavid du Colombier # endif
92*593dc095SDavid du Colombier #endif
937dd7cddfSDavid du Colombier
94*593dc095SDavid du Colombier /* function prototypes */
95*593dc095SDavid du Colombier local void fixedtables OF((struct inflate_state FAR *state));
96*593dc095SDavid du Colombier local int updatewindow OF((z_streamp strm, unsigned out));
97*593dc095SDavid du Colombier #ifdef BUILDFIXED
98*593dc095SDavid du Colombier void makefixed OF((void));
99*593dc095SDavid du Colombier #endif
100*593dc095SDavid du Colombier local unsigned syncsearch OF((unsigned FAR *have, unsigned char FAR *buf,
101*593dc095SDavid du Colombier unsigned len));
1027dd7cddfSDavid du Colombier
inflateReset(strm)103*593dc095SDavid du Colombier int ZEXPORT inflateReset(strm)
104*593dc095SDavid du Colombier z_streamp strm;
1057dd7cddfSDavid du Colombier {
106*593dc095SDavid du Colombier struct inflate_state FAR *state;
1077dd7cddfSDavid du Colombier
108*593dc095SDavid du Colombier if (strm == Z_NULL || strm->state == Z_NULL) return Z_STREAM_ERROR;
109*593dc095SDavid du Colombier state = (struct inflate_state FAR *)strm->state;
110*593dc095SDavid du Colombier strm->total_in = strm->total_out = state->total = 0;
111*593dc095SDavid du Colombier strm->msg = Z_NULL;
112*593dc095SDavid du Colombier strm->adler = 1; /* to support ill-conceived Java test suite */
113*593dc095SDavid du Colombier state->mode = HEAD;
114*593dc095SDavid du Colombier state->last = 0;
115*593dc095SDavid du Colombier state->havedict = 0;
116*593dc095SDavid du Colombier state->wsize = 0;
117*593dc095SDavid du Colombier state->whave = 0;
118*593dc095SDavid du Colombier state->hold = 0;
119*593dc095SDavid du Colombier state->bits = 0;
120*593dc095SDavid du Colombier state->lencode = state->distcode = state->next = state->codes;
121*593dc095SDavid du Colombier Tracev((stderr, "inflate: reset\n"));
1227dd7cddfSDavid du Colombier return Z_OK;
1237dd7cddfSDavid du Colombier }
1247dd7cddfSDavid du Colombier
inflateInit2_(strm,windowBits,version,stream_size)125*593dc095SDavid du Colombier int ZEXPORT inflateInit2_(strm, windowBits, version, stream_size)
126*593dc095SDavid du Colombier z_streamp strm;
127*593dc095SDavid du Colombier int windowBits;
1287dd7cddfSDavid du Colombier const char *version;
1297dd7cddfSDavid du Colombier int stream_size;
1307dd7cddfSDavid du Colombier {
131*593dc095SDavid du Colombier struct inflate_state FAR *state;
132*593dc095SDavid du Colombier
1337dd7cddfSDavid du Colombier if (version == Z_NULL || version[0] != ZLIB_VERSION[0] ||
134*593dc095SDavid du Colombier stream_size != (int)(sizeof(z_stream)))
1357dd7cddfSDavid du Colombier return Z_VERSION_ERROR;
136*593dc095SDavid du Colombier if (strm == Z_NULL) return Z_STREAM_ERROR;
137*593dc095SDavid du Colombier strm->msg = Z_NULL; /* in case we return an error */
138*593dc095SDavid du Colombier if (strm->zalloc == (alloc_func)0) {
139*593dc095SDavid du Colombier strm->zalloc = zcalloc;
140*593dc095SDavid du Colombier strm->opaque = (voidpf)0;
1417dd7cddfSDavid du Colombier }
142*593dc095SDavid du Colombier if (strm->zfree == (free_func)0) strm->zfree = zcfree;
143*593dc095SDavid du Colombier state = (struct inflate_state FAR *)
144*593dc095SDavid du Colombier ZALLOC(strm, 1, sizeof(struct inflate_state));
145*593dc095SDavid du Colombier if (state == Z_NULL) return Z_MEM_ERROR;
146*593dc095SDavid du Colombier Tracev((stderr, "inflate: allocated\n"));
147*593dc095SDavid du Colombier strm->state = (voidpf)state;
148*593dc095SDavid du Colombier if (windowBits < 0) {
149*593dc095SDavid du Colombier state->wrap = 0;
150*593dc095SDavid du Colombier windowBits = -windowBits;
1517dd7cddfSDavid du Colombier }
152*593dc095SDavid du Colombier else {
153*593dc095SDavid du Colombier state->wrap = (windowBits >> 4) + 1;
154*593dc095SDavid du Colombier #ifdef GUNZIP
155*593dc095SDavid du Colombier if (windowBits < 48) windowBits &= 15;
156*593dc095SDavid du Colombier #endif
157*593dc095SDavid du Colombier }
158*593dc095SDavid du Colombier if (windowBits < 8 || windowBits > 15) {
159*593dc095SDavid du Colombier ZFREE(strm, state);
160*593dc095SDavid du Colombier strm->state = Z_NULL;
1617dd7cddfSDavid du Colombier return Z_STREAM_ERROR;
1627dd7cddfSDavid du Colombier }
163*593dc095SDavid du Colombier state->wbits = (unsigned)windowBits;
164*593dc095SDavid du Colombier state->window = Z_NULL;
165*593dc095SDavid du Colombier return inflateReset(strm);
1667dd7cddfSDavid du Colombier }
1677dd7cddfSDavid du Colombier
inflateInit_(strm,version,stream_size)168*593dc095SDavid du Colombier int ZEXPORT inflateInit_(strm, version, stream_size)
169*593dc095SDavid du Colombier z_streamp strm;
1707dd7cddfSDavid du Colombier const char *version;
1717dd7cddfSDavid du Colombier int stream_size;
1727dd7cddfSDavid du Colombier {
173*593dc095SDavid du Colombier return inflateInit2_(strm, DEF_WBITS, version, stream_size);
1747dd7cddfSDavid du Colombier }
1757dd7cddfSDavid du Colombier
176*593dc095SDavid du Colombier /*
177*593dc095SDavid du Colombier Return state with length and distance decoding tables and index sizes set to
178*593dc095SDavid du Colombier fixed code decoding. Normally this returns fixed tables from inffixed.h.
179*593dc095SDavid du Colombier If BUILDFIXED is defined, then instead this routine builds the tables the
180*593dc095SDavid du Colombier first time it's called, and returns those tables the first time and
181*593dc095SDavid du Colombier thereafter. This reduces the size of the code by about 2K bytes, in
182*593dc095SDavid du Colombier exchange for a little execution time. However, BUILDFIXED should not be
183*593dc095SDavid du Colombier used for threaded applications, since the rewriting of the tables and virgin
184*593dc095SDavid du Colombier may not be thread-safe.
185*593dc095SDavid du Colombier */
fixedtables(state)186*593dc095SDavid du Colombier local void fixedtables(state)
187*593dc095SDavid du Colombier struct inflate_state FAR *state;
1887dd7cddfSDavid du Colombier {
189*593dc095SDavid du Colombier #ifdef BUILDFIXED
190*593dc095SDavid du Colombier static int virgin = 1;
191*593dc095SDavid du Colombier static code *lenfix, *distfix;
192*593dc095SDavid du Colombier static code fixed[544];
1937dd7cddfSDavid du Colombier
194*593dc095SDavid du Colombier /* build fixed huffman tables if first call (may not be thread safe) */
195*593dc095SDavid du Colombier if (virgin) {
196*593dc095SDavid du Colombier unsigned sym, bits;
197*593dc095SDavid du Colombier static code *next;
198*593dc095SDavid du Colombier
199*593dc095SDavid du Colombier /* literal/length table */
200*593dc095SDavid du Colombier sym = 0;
201*593dc095SDavid du Colombier while (sym < 144) state->lens[sym++] = 8;
202*593dc095SDavid du Colombier while (sym < 256) state->lens[sym++] = 9;
203*593dc095SDavid du Colombier while (sym < 280) state->lens[sym++] = 7;
204*593dc095SDavid du Colombier while (sym < 288) state->lens[sym++] = 8;
205*593dc095SDavid du Colombier next = fixed;
206*593dc095SDavid du Colombier lenfix = next;
207*593dc095SDavid du Colombier bits = 9;
208*593dc095SDavid du Colombier inflate_table(LENS, state->lens, 288, &(next), &(bits), state->work);
209*593dc095SDavid du Colombier
210*593dc095SDavid du Colombier /* distance table */
211*593dc095SDavid du Colombier sym = 0;
212*593dc095SDavid du Colombier while (sym < 32) state->lens[sym++] = 5;
213*593dc095SDavid du Colombier distfix = next;
214*593dc095SDavid du Colombier bits = 5;
215*593dc095SDavid du Colombier inflate_table(DISTS, state->lens, 32, &(next), &(bits), state->work);
216*593dc095SDavid du Colombier
217*593dc095SDavid du Colombier /* do this just once */
218*593dc095SDavid du Colombier virgin = 0;
219*593dc095SDavid du Colombier }
220*593dc095SDavid du Colombier #else /* !BUILDFIXED */
221*593dc095SDavid du Colombier # include "inffixed.h"
222*593dc095SDavid du Colombier #endif /* BUILDFIXED */
223*593dc095SDavid du Colombier state->lencode = lenfix;
224*593dc095SDavid du Colombier state->lenbits = 9;
225*593dc095SDavid du Colombier state->distcode = distfix;
226*593dc095SDavid du Colombier state->distbits = 5;
227*593dc095SDavid du Colombier }
228*593dc095SDavid du Colombier
229*593dc095SDavid du Colombier #ifdef MAKEFIXED
230*593dc095SDavid du Colombier #include <stdio.h>
231*593dc095SDavid du Colombier
232*593dc095SDavid du Colombier /*
233*593dc095SDavid du Colombier Write out the inffixed.h that is #include'd above. Defining MAKEFIXED also
234*593dc095SDavid du Colombier defines BUILDFIXED, so the tables are built on the fly. makefixed() writes
235*593dc095SDavid du Colombier those tables to stdout, which would be piped to inffixed.h. A small program
236*593dc095SDavid du Colombier can simply call makefixed to do this:
237*593dc095SDavid du Colombier
238*593dc095SDavid du Colombier void makefixed(void);
239*593dc095SDavid du Colombier
240*593dc095SDavid du Colombier int main(void)
241*593dc095SDavid du Colombier {
242*593dc095SDavid du Colombier makefixed();
243*593dc095SDavid du Colombier return 0;
244*593dc095SDavid du Colombier }
245*593dc095SDavid du Colombier
246*593dc095SDavid du Colombier Then that can be linked with zlib built with MAKEFIXED defined and run:
247*593dc095SDavid du Colombier
248*593dc095SDavid du Colombier a.out > inffixed.h
249*593dc095SDavid du Colombier */
makefixed()250*593dc095SDavid du Colombier void makefixed()
251*593dc095SDavid du Colombier {
252*593dc095SDavid du Colombier unsigned low, size;
253*593dc095SDavid du Colombier struct inflate_state state;
254*593dc095SDavid du Colombier
255*593dc095SDavid du Colombier fixedtables(&state);
256*593dc095SDavid du Colombier puts(" /* inffixed.h -- table for decoding fixed codes");
257*593dc095SDavid du Colombier puts(" * Generated automatically by makefixed().");
258*593dc095SDavid du Colombier puts(" */");
259*593dc095SDavid du Colombier puts("");
260*593dc095SDavid du Colombier puts(" /* WARNING: this file should *not* be used by applications.");
261*593dc095SDavid du Colombier puts(" It is part of the implementation of this library and is");
262*593dc095SDavid du Colombier puts(" subject to change. Applications should only use zlib.h.");
263*593dc095SDavid du Colombier puts(" */");
264*593dc095SDavid du Colombier puts("");
265*593dc095SDavid du Colombier size = 1U << 9;
266*593dc095SDavid du Colombier printf(" static const code lenfix[%u] = {", size);
267*593dc095SDavid du Colombier low = 0;
268*593dc095SDavid du Colombier for (;;) {
269*593dc095SDavid du Colombier if ((low % 7) == 0) printf("\n ");
270*593dc095SDavid du Colombier printf("{%u,%u,%d}", state.lencode[low].op, state.lencode[low].bits,
271*593dc095SDavid du Colombier state.lencode[low].val);
272*593dc095SDavid du Colombier if (++low == size) break;
273*593dc095SDavid du Colombier putchar(',');
274*593dc095SDavid du Colombier }
275*593dc095SDavid du Colombier puts("\n };");
276*593dc095SDavid du Colombier size = 1U << 5;
277*593dc095SDavid du Colombier printf("\n static const code distfix[%u] = {", size);
278*593dc095SDavid du Colombier low = 0;
279*593dc095SDavid du Colombier for (;;) {
280*593dc095SDavid du Colombier if ((low % 6) == 0) printf("\n ");
281*593dc095SDavid du Colombier printf("{%u,%u,%d}", state.distcode[low].op, state.distcode[low].bits,
282*593dc095SDavid du Colombier state.distcode[low].val);
283*593dc095SDavid du Colombier if (++low == size) break;
284*593dc095SDavid du Colombier putchar(',');
285*593dc095SDavid du Colombier }
286*593dc095SDavid du Colombier puts("\n };");
287*593dc095SDavid du Colombier }
288*593dc095SDavid du Colombier #endif /* MAKEFIXED */
289*593dc095SDavid du Colombier
290*593dc095SDavid du Colombier /*
291*593dc095SDavid du Colombier Update the window with the last wsize (normally 32K) bytes written before
292*593dc095SDavid du Colombier returning. If window does not exist yet, create it. This is only called
293*593dc095SDavid du Colombier when a window is already in use, or when output has been written during this
294*593dc095SDavid du Colombier inflate call, but the end of the deflate stream has not been reached yet.
295*593dc095SDavid du Colombier It is also called to create a window for dictionary data when a dictionary
296*593dc095SDavid du Colombier is loaded.
297*593dc095SDavid du Colombier
298*593dc095SDavid du Colombier Providing output buffers larger than 32K to inflate() should provide a speed
299*593dc095SDavid du Colombier advantage, since only the last 32K of output is copied to the sliding window
300*593dc095SDavid du Colombier upon return from inflate(), and since all distances after the first 32K of
301*593dc095SDavid du Colombier output will fall in the output data, making match copies simpler and faster.
302*593dc095SDavid du Colombier The advantage may be dependent on the size of the processor's data caches.
303*593dc095SDavid du Colombier */
updatewindow(strm,out)304*593dc095SDavid du Colombier local int updatewindow(strm, out)
305*593dc095SDavid du Colombier z_streamp strm;
306*593dc095SDavid du Colombier unsigned out;
307*593dc095SDavid du Colombier {
308*593dc095SDavid du Colombier struct inflate_state FAR *state;
309*593dc095SDavid du Colombier unsigned copy, dist;
310*593dc095SDavid du Colombier
311*593dc095SDavid du Colombier state = (struct inflate_state FAR *)strm->state;
312*593dc095SDavid du Colombier
313*593dc095SDavid du Colombier /* if it hasn't been done already, allocate space for the window */
314*593dc095SDavid du Colombier if (state->window == Z_NULL) {
315*593dc095SDavid du Colombier state->window = (unsigned char FAR *)
316*593dc095SDavid du Colombier ZALLOC(strm, 1U << state->wbits,
317*593dc095SDavid du Colombier sizeof(unsigned char));
318*593dc095SDavid du Colombier if (state->window == Z_NULL) return 1;
319*593dc095SDavid du Colombier }
320*593dc095SDavid du Colombier
321*593dc095SDavid du Colombier /* if window not in use yet, initialize */
322*593dc095SDavid du Colombier if (state->wsize == 0) {
323*593dc095SDavid du Colombier state->wsize = 1U << state->wbits;
324*593dc095SDavid du Colombier state->write = 0;
325*593dc095SDavid du Colombier state->whave = 0;
326*593dc095SDavid du Colombier }
327*593dc095SDavid du Colombier
328*593dc095SDavid du Colombier /* copy state->wsize or less output bytes into the circular window */
329*593dc095SDavid du Colombier copy = out - strm->avail_out;
330*593dc095SDavid du Colombier if (copy >= state->wsize) {
331*593dc095SDavid du Colombier zmemcpy(state->window, strm->next_out - state->wsize, state->wsize);
332*593dc095SDavid du Colombier state->write = 0;
333*593dc095SDavid du Colombier state->whave = state->wsize;
334*593dc095SDavid du Colombier }
335*593dc095SDavid du Colombier else {
336*593dc095SDavid du Colombier dist = state->wsize - state->write;
337*593dc095SDavid du Colombier if (dist > copy) dist = copy;
338*593dc095SDavid du Colombier zmemcpy(state->window + state->write, strm->next_out - copy, dist);
339*593dc095SDavid du Colombier copy -= dist;
340*593dc095SDavid du Colombier if (copy) {
341*593dc095SDavid du Colombier zmemcpy(state->window, strm->next_out - copy, copy);
342*593dc095SDavid du Colombier state->write = copy;
343*593dc095SDavid du Colombier state->whave = state->wsize;
344*593dc095SDavid du Colombier }
345*593dc095SDavid du Colombier else {
346*593dc095SDavid du Colombier state->write += dist;
347*593dc095SDavid du Colombier if (state->write == state->wsize) state->write = 0;
348*593dc095SDavid du Colombier if (state->whave < state->wsize) state->whave += dist;
349*593dc095SDavid du Colombier }
350*593dc095SDavid du Colombier }
351*593dc095SDavid du Colombier return 0;
352*593dc095SDavid du Colombier }
353*593dc095SDavid du Colombier
354*593dc095SDavid du Colombier /* Macros for inflate(): */
355*593dc095SDavid du Colombier
356*593dc095SDavid du Colombier /* check function to use adler32() for zlib or crc32() for gzip */
357*593dc095SDavid du Colombier #ifdef GUNZIP
358*593dc095SDavid du Colombier # define UPDATE(check, buf, len) \
359*593dc095SDavid du Colombier (state->flags ? crc32(check, buf, len) : adler32(check, buf, len))
360*593dc095SDavid du Colombier #else
361*593dc095SDavid du Colombier # define UPDATE(check, buf, len) adler32(check, buf, len)
362*593dc095SDavid du Colombier #endif
363*593dc095SDavid du Colombier
364*593dc095SDavid du Colombier /* check macros for header crc */
365*593dc095SDavid du Colombier #ifdef GUNZIP
366*593dc095SDavid du Colombier # define CRC2(check, word) \
367*593dc095SDavid du Colombier do { \
368*593dc095SDavid du Colombier hbuf[0] = (unsigned char)(word); \
369*593dc095SDavid du Colombier hbuf[1] = (unsigned char)((word) >> 8); \
370*593dc095SDavid du Colombier check = crc32(check, hbuf, 2); \
371*593dc095SDavid du Colombier } while (0)
372*593dc095SDavid du Colombier
373*593dc095SDavid du Colombier # define CRC4(check, word) \
374*593dc095SDavid du Colombier do { \
375*593dc095SDavid du Colombier hbuf[0] = (unsigned char)(word); \
376*593dc095SDavid du Colombier hbuf[1] = (unsigned char)((word) >> 8); \
377*593dc095SDavid du Colombier hbuf[2] = (unsigned char)((word) >> 16); \
378*593dc095SDavid du Colombier hbuf[3] = (unsigned char)((word) >> 24); \
379*593dc095SDavid du Colombier check = crc32(check, hbuf, 4); \
380*593dc095SDavid du Colombier } while (0)
381*593dc095SDavid du Colombier #endif
382*593dc095SDavid du Colombier
383*593dc095SDavid du Colombier /* Load registers with state in inflate() for speed */
384*593dc095SDavid du Colombier #define LOAD() \
385*593dc095SDavid du Colombier do { \
386*593dc095SDavid du Colombier put = strm->next_out; \
387*593dc095SDavid du Colombier left = strm->avail_out; \
388*593dc095SDavid du Colombier next = strm->next_in; \
389*593dc095SDavid du Colombier have = strm->avail_in; \
390*593dc095SDavid du Colombier hold = state->hold; \
391*593dc095SDavid du Colombier bits = state->bits; \
392*593dc095SDavid du Colombier } while (0)
393*593dc095SDavid du Colombier
394*593dc095SDavid du Colombier /* Restore state from registers in inflate() */
395*593dc095SDavid du Colombier #define RESTORE() \
396*593dc095SDavid du Colombier do { \
397*593dc095SDavid du Colombier strm->next_out = put; \
398*593dc095SDavid du Colombier strm->avail_out = left; \
399*593dc095SDavid du Colombier strm->next_in = next; \
400*593dc095SDavid du Colombier strm->avail_in = have; \
401*593dc095SDavid du Colombier state->hold = hold; \
402*593dc095SDavid du Colombier state->bits = bits; \
403*593dc095SDavid du Colombier } while (0)
404*593dc095SDavid du Colombier
405*593dc095SDavid du Colombier /* Clear the input bit accumulator */
406*593dc095SDavid du Colombier #define INITBITS() \
407*593dc095SDavid du Colombier do { \
408*593dc095SDavid du Colombier hold = 0; \
409*593dc095SDavid du Colombier bits = 0; \
410*593dc095SDavid du Colombier } while (0)
411*593dc095SDavid du Colombier
412*593dc095SDavid du Colombier /* Get a byte of input into the bit accumulator, or return from inflate()
413*593dc095SDavid du Colombier if there is no input available. */
414*593dc095SDavid du Colombier #define PULLBYTE() \
415*593dc095SDavid du Colombier do { \
416*593dc095SDavid du Colombier if (have == 0) goto inf_leave; \
417*593dc095SDavid du Colombier have--; \
418*593dc095SDavid du Colombier hold += (unsigned long)(*next++) << bits; \
419*593dc095SDavid du Colombier bits += 8; \
420*593dc095SDavid du Colombier } while (0)
421*593dc095SDavid du Colombier
422*593dc095SDavid du Colombier /* Assure that there are at least n bits in the bit accumulator. If there is
423*593dc095SDavid du Colombier not enough available input to do that, then return from inflate(). */
424*593dc095SDavid du Colombier #define NEEDBITS(n) \
425*593dc095SDavid du Colombier do { \
426*593dc095SDavid du Colombier while (bits < (unsigned)(n)) \
427*593dc095SDavid du Colombier PULLBYTE(); \
428*593dc095SDavid du Colombier } while (0)
429*593dc095SDavid du Colombier
430*593dc095SDavid du Colombier /* Return the low n bits of the bit accumulator (n < 16) */
431*593dc095SDavid du Colombier #define BITS(n) \
432*593dc095SDavid du Colombier ((unsigned)hold & ((1U << (n)) - 1))
433*593dc095SDavid du Colombier
434*593dc095SDavid du Colombier /* Remove n bits from the bit accumulator */
435*593dc095SDavid du Colombier #define DROPBITS(n) \
436*593dc095SDavid du Colombier do { \
437*593dc095SDavid du Colombier hold >>= (n); \
438*593dc095SDavid du Colombier bits -= (unsigned)(n); \
439*593dc095SDavid du Colombier } while (0)
440*593dc095SDavid du Colombier
441*593dc095SDavid du Colombier /* Remove zero to seven bits as needed to go to a byte boundary */
442*593dc095SDavid du Colombier #define BYTEBITS() \
443*593dc095SDavid du Colombier do { \
444*593dc095SDavid du Colombier hold >>= bits & 7; \
445*593dc095SDavid du Colombier bits -= bits & 7; \
446*593dc095SDavid du Colombier } while (0)
447*593dc095SDavid du Colombier
448*593dc095SDavid du Colombier /* Reverse the bytes in a 32-bit value */
449*593dc095SDavid du Colombier #define REVERSE(q) \
450*593dc095SDavid du Colombier ((((q) >> 24) & 0xff) + (((q) >> 8) & 0xff00) + \
451*593dc095SDavid du Colombier (((q) & 0xff00) << 8) + (((q) & 0xff) << 24))
452*593dc095SDavid du Colombier
453*593dc095SDavid du Colombier /*
454*593dc095SDavid du Colombier inflate() uses a state machine to process as much input data and generate as
455*593dc095SDavid du Colombier much output data as possible before returning. The state machine is
456*593dc095SDavid du Colombier structured roughly as follows:
457*593dc095SDavid du Colombier
458*593dc095SDavid du Colombier for (;;) switch (state) {
459*593dc095SDavid du Colombier ...
460*593dc095SDavid du Colombier case STATEn:
461*593dc095SDavid du Colombier if (not enough input data or output space to make progress)
462*593dc095SDavid du Colombier return;
463*593dc095SDavid du Colombier ... make progress ...
464*593dc095SDavid du Colombier state = STATEm;
465*593dc095SDavid du Colombier break;
466*593dc095SDavid du Colombier ...
467*593dc095SDavid du Colombier }
468*593dc095SDavid du Colombier
469*593dc095SDavid du Colombier so when inflate() is called again, the same case is attempted again, and
470*593dc095SDavid du Colombier if the appropriate resources are provided, the machine proceeds to the
471*593dc095SDavid du Colombier next state. The NEEDBITS() macro is usually the way the state evaluates
472*593dc095SDavid du Colombier whether it can proceed or should return. NEEDBITS() does the return if
473*593dc095SDavid du Colombier the requested bits are not available. The typical use of the BITS macros
474*593dc095SDavid du Colombier is:
475*593dc095SDavid du Colombier
476*593dc095SDavid du Colombier NEEDBITS(n);
477*593dc095SDavid du Colombier ... do something with BITS(n) ...
478*593dc095SDavid du Colombier DROPBITS(n);
479*593dc095SDavid du Colombier
480*593dc095SDavid du Colombier where NEEDBITS(n) either returns from inflate() if there isn't enough
481*593dc095SDavid du Colombier input left to load n bits into the accumulator, or it continues. BITS(n)
482*593dc095SDavid du Colombier gives the low n bits in the accumulator. When done, DROPBITS(n) drops
483*593dc095SDavid du Colombier the low n bits off the accumulator. INITBITS() clears the accumulator
484*593dc095SDavid du Colombier and sets the number of available bits to zero. BYTEBITS() discards just
485*593dc095SDavid du Colombier enough bits to put the accumulator on a byte boundary. After BYTEBITS()
486*593dc095SDavid du Colombier and a NEEDBITS(8), then BITS(8) would return the next byte in the stream.
487*593dc095SDavid du Colombier
488*593dc095SDavid du Colombier NEEDBITS(n) uses PULLBYTE() to get an available byte of input, or to return
489*593dc095SDavid du Colombier if there is no input available. The decoding of variable length codes uses
490*593dc095SDavid du Colombier PULLBYTE() directly in order to pull just enough bytes to decode the next
491*593dc095SDavid du Colombier code, and no more.
492*593dc095SDavid du Colombier
493*593dc095SDavid du Colombier Some states loop until they get enough input, making sure that enough
494*593dc095SDavid du Colombier state information is maintained to continue the loop where it left off
495*593dc095SDavid du Colombier if NEEDBITS() returns in the loop. For example, want, need, and keep
496*593dc095SDavid du Colombier would all have to actually be part of the saved state in case NEEDBITS()
497*593dc095SDavid du Colombier returns:
498*593dc095SDavid du Colombier
499*593dc095SDavid du Colombier case STATEw:
500*593dc095SDavid du Colombier while (want < need) {
501*593dc095SDavid du Colombier NEEDBITS(n);
502*593dc095SDavid du Colombier keep[want++] = BITS(n);
503*593dc095SDavid du Colombier DROPBITS(n);
504*593dc095SDavid du Colombier }
505*593dc095SDavid du Colombier state = STATEx;
506*593dc095SDavid du Colombier case STATEx:
507*593dc095SDavid du Colombier
508*593dc095SDavid du Colombier As shown above, if the next state is also the next case, then the break
509*593dc095SDavid du Colombier is omitted.
510*593dc095SDavid du Colombier
511*593dc095SDavid du Colombier A state may also return if there is not enough output space available to
512*593dc095SDavid du Colombier complete that state. Those states are copying stored data, writing a
513*593dc095SDavid du Colombier literal byte, and copying a matching string.
514*593dc095SDavid du Colombier
515*593dc095SDavid du Colombier When returning, a "goto inf_leave" is used to update the total counters,
516*593dc095SDavid du Colombier update the check value, and determine whether any progress has been made
517*593dc095SDavid du Colombier during that inflate() call in order to return the proper return code.
518*593dc095SDavid du Colombier Progress is defined as a change in either strm->avail_in or strm->avail_out.
519*593dc095SDavid du Colombier When there is a window, goto inf_leave will update the window with the last
520*593dc095SDavid du Colombier output written. If a goto inf_leave occurs in the middle of decompression
521*593dc095SDavid du Colombier and there is no window currently, goto inf_leave will create one and copy
522*593dc095SDavid du Colombier output to the window for the next call of inflate().
523*593dc095SDavid du Colombier
524*593dc095SDavid du Colombier In this implementation, the flush parameter of inflate() only affects the
525*593dc095SDavid du Colombier return code (per zlib.h). inflate() always writes as much as possible to
526*593dc095SDavid du Colombier strm->next_out, given the space available and the provided input--the effect
527*593dc095SDavid du Colombier documented in zlib.h of Z_SYNC_FLUSH. Furthermore, inflate() always defers
528*593dc095SDavid du Colombier the allocation of and copying into a sliding window until necessary, which
529*593dc095SDavid du Colombier provides the effect documented in zlib.h for Z_FINISH when the entire input
530*593dc095SDavid du Colombier stream available. So the only thing the flush parameter actually does is:
531*593dc095SDavid du Colombier when flush is set to Z_FINISH, inflate() cannot return Z_OK. Instead it
532*593dc095SDavid du Colombier will return Z_BUF_ERROR if it has not reached the end of the stream.
533*593dc095SDavid du Colombier */
534*593dc095SDavid du Colombier
inflate(strm,flush)535*593dc095SDavid du Colombier int ZEXPORT inflate(strm, flush)
536*593dc095SDavid du Colombier z_streamp strm;
537*593dc095SDavid du Colombier int flush;
538*593dc095SDavid du Colombier {
539*593dc095SDavid du Colombier struct inflate_state FAR *state;
540*593dc095SDavid du Colombier unsigned char FAR *next; /* next input */
541*593dc095SDavid du Colombier unsigned char FAR *put; /* next output */
542*593dc095SDavid du Colombier unsigned have, left; /* available input and output */
543*593dc095SDavid du Colombier unsigned long hold; /* bit buffer */
544*593dc095SDavid du Colombier unsigned bits; /* bits in bit buffer */
545*593dc095SDavid du Colombier unsigned in, out; /* save starting available input and output */
546*593dc095SDavid du Colombier unsigned copy; /* number of stored or match bytes to copy */
547*593dc095SDavid du Colombier unsigned char FAR *from; /* where to copy match bytes from */
548*593dc095SDavid du Colombier code this; /* current decoding table entry */
549*593dc095SDavid du Colombier code last; /* parent table entry */
550*593dc095SDavid du Colombier unsigned len; /* length to copy for repeats, bits to drop */
551*593dc095SDavid du Colombier int ret; /* return code */
552*593dc095SDavid du Colombier #ifdef GUNZIP
553*593dc095SDavid du Colombier unsigned char hbuf[4]; /* buffer for gzip header crc calculation */
554*593dc095SDavid du Colombier #endif
555*593dc095SDavid du Colombier static const unsigned short order[19] = /* permutation of code lengths */
556*593dc095SDavid du Colombier {16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15};
557*593dc095SDavid du Colombier
558*593dc095SDavid du Colombier if (strm == Z_NULL || strm->state == Z_NULL || strm->next_out == Z_NULL ||
559*593dc095SDavid du Colombier (strm->next_in == Z_NULL && strm->avail_in != 0))
5607dd7cddfSDavid du Colombier return Z_STREAM_ERROR;
561*593dc095SDavid du Colombier
562*593dc095SDavid du Colombier state = (struct inflate_state FAR *)strm->state;
563*593dc095SDavid du Colombier if (state->mode == TYPE) state->mode = TYPEDO; /* skip check */
564*593dc095SDavid du Colombier LOAD();
565*593dc095SDavid du Colombier in = have;
566*593dc095SDavid du Colombier out = left;
567*593dc095SDavid du Colombier ret = Z_OK;
568*593dc095SDavid du Colombier for (;;)
569*593dc095SDavid du Colombier switch (state->mode) {
570*593dc095SDavid du Colombier case HEAD:
571*593dc095SDavid du Colombier if (state->wrap == 0) {
572*593dc095SDavid du Colombier state->mode = TYPEDO;
5737dd7cddfSDavid du Colombier break;
5747dd7cddfSDavid du Colombier }
575*593dc095SDavid du Colombier NEEDBITS(16);
576*593dc095SDavid du Colombier #ifdef GUNZIP
577*593dc095SDavid du Colombier if ((state->wrap & 2) && hold == 0x8b1f) { /* gzip header */
578*593dc095SDavid du Colombier state->check = crc32(0L, Z_NULL, 0);
579*593dc095SDavid du Colombier CRC2(state->check, hold);
580*593dc095SDavid du Colombier INITBITS();
581*593dc095SDavid du Colombier state->mode = FLAGS;
5827dd7cddfSDavid du Colombier break;
5837dd7cddfSDavid du Colombier }
584*593dc095SDavid du Colombier state->flags = 0; /* expect zlib header */
585*593dc095SDavid du Colombier if (!(state->wrap & 1) || /* check if zlib header allowed */
586*593dc095SDavid du Colombier #else
587*593dc095SDavid du Colombier if (
588*593dc095SDavid du Colombier #endif
589*593dc095SDavid du Colombier ((BITS(8) << 8) + (hold >> 8)) % 31) {
590*593dc095SDavid du Colombier strm->msg = (char *)"incorrect header check";
591*593dc095SDavid du Colombier state->mode = BAD;
5927dd7cddfSDavid du Colombier break;
5937dd7cddfSDavid du Colombier }
594*593dc095SDavid du Colombier if (BITS(4) != Z_DEFLATED) {
595*593dc095SDavid du Colombier strm->msg = (char *)"unknown compression method";
596*593dc095SDavid du Colombier state->mode = BAD;
5977dd7cddfSDavid du Colombier break;
5987dd7cddfSDavid du Colombier }
599*593dc095SDavid du Colombier DROPBITS(4);
600*593dc095SDavid du Colombier if (BITS(4) + 8 > state->wbits) {
601*593dc095SDavid du Colombier strm->msg = (char *)"invalid window size";
602*593dc095SDavid du Colombier state->mode = BAD;
603*593dc095SDavid du Colombier break;
604*593dc095SDavid du Colombier }
605*593dc095SDavid du Colombier Tracev((stderr, "inflate: zlib header ok\n"));
606*593dc095SDavid du Colombier strm->adler = state->check = adler32(0L, Z_NULL, 0);
607*593dc095SDavid du Colombier state->mode = hold & 0x200 ? DICTID : TYPE;
608*593dc095SDavid du Colombier INITBITS();
609*593dc095SDavid du Colombier break;
610*593dc095SDavid du Colombier #ifdef GUNZIP
611*593dc095SDavid du Colombier case FLAGS:
612*593dc095SDavid du Colombier NEEDBITS(16);
613*593dc095SDavid du Colombier state->flags = (int)(hold);
614*593dc095SDavid du Colombier if ((state->flags & 0xff) != Z_DEFLATED) {
615*593dc095SDavid du Colombier strm->msg = (char *)"unknown compression method";
616*593dc095SDavid du Colombier state->mode = BAD;
617*593dc095SDavid du Colombier break;
618*593dc095SDavid du Colombier }
619*593dc095SDavid du Colombier if (state->flags & 0xe000) {
620*593dc095SDavid du Colombier strm->msg = (char *)"unknown header flags set";
621*593dc095SDavid du Colombier state->mode = BAD;
622*593dc095SDavid du Colombier break;
623*593dc095SDavid du Colombier }
624*593dc095SDavid du Colombier if (state->flags & 0x0200) CRC2(state->check, hold);
625*593dc095SDavid du Colombier INITBITS();
626*593dc095SDavid du Colombier state->mode = TIME;
627*593dc095SDavid du Colombier case TIME:
628*593dc095SDavid du Colombier NEEDBITS(32);
629*593dc095SDavid du Colombier if (state->flags & 0x0200) CRC4(state->check, hold);
630*593dc095SDavid du Colombier INITBITS();
631*593dc095SDavid du Colombier state->mode = OS;
632*593dc095SDavid du Colombier case OS:
633*593dc095SDavid du Colombier NEEDBITS(16);
634*593dc095SDavid du Colombier if (state->flags & 0x0200) CRC2(state->check, hold);
635*593dc095SDavid du Colombier INITBITS();
636*593dc095SDavid du Colombier state->mode = EXLEN;
637*593dc095SDavid du Colombier case EXLEN:
638*593dc095SDavid du Colombier if (state->flags & 0x0400) {
639*593dc095SDavid du Colombier NEEDBITS(16);
640*593dc095SDavid du Colombier state->length = (unsigned)(hold);
641*593dc095SDavid du Colombier if (state->flags & 0x0200) CRC2(state->check, hold);
642*593dc095SDavid du Colombier INITBITS();
643*593dc095SDavid du Colombier }
644*593dc095SDavid du Colombier state->mode = EXTRA;
645*593dc095SDavid du Colombier case EXTRA:
646*593dc095SDavid du Colombier if (state->flags & 0x0400) {
647*593dc095SDavid du Colombier copy = state->length;
648*593dc095SDavid du Colombier if (copy > have) copy = have;
649*593dc095SDavid du Colombier if (copy) {
650*593dc095SDavid du Colombier if (state->flags & 0x0200)
651*593dc095SDavid du Colombier state->check = crc32(state->check, next, copy);
652*593dc095SDavid du Colombier have -= copy;
653*593dc095SDavid du Colombier next += copy;
654*593dc095SDavid du Colombier state->length -= copy;
655*593dc095SDavid du Colombier }
656*593dc095SDavid du Colombier if (state->length) goto inf_leave;
657*593dc095SDavid du Colombier }
658*593dc095SDavid du Colombier state->mode = NAME;
659*593dc095SDavid du Colombier case NAME:
660*593dc095SDavid du Colombier if (state->flags & 0x0800) {
661*593dc095SDavid du Colombier if (have == 0) goto inf_leave;
662*593dc095SDavid du Colombier copy = 0;
663*593dc095SDavid du Colombier do {
664*593dc095SDavid du Colombier len = (unsigned)(next[copy++]);
665*593dc095SDavid du Colombier } while (len && copy < have);
666*593dc095SDavid du Colombier if (state->flags & 0x02000)
667*593dc095SDavid du Colombier state->check = crc32(state->check, next, copy);
668*593dc095SDavid du Colombier have -= copy;
669*593dc095SDavid du Colombier next += copy;
670*593dc095SDavid du Colombier if (len) goto inf_leave;
671*593dc095SDavid du Colombier }
672*593dc095SDavid du Colombier state->mode = COMMENT;
673*593dc095SDavid du Colombier case COMMENT:
674*593dc095SDavid du Colombier if (state->flags & 0x1000) {
675*593dc095SDavid du Colombier if (have == 0) goto inf_leave;
676*593dc095SDavid du Colombier copy = 0;
677*593dc095SDavid du Colombier do {
678*593dc095SDavid du Colombier len = (unsigned)(next[copy++]);
679*593dc095SDavid du Colombier } while (len && copy < have);
680*593dc095SDavid du Colombier if (state->flags & 0x02000)
681*593dc095SDavid du Colombier state->check = crc32(state->check, next, copy);
682*593dc095SDavid du Colombier have -= copy;
683*593dc095SDavid du Colombier next += copy;
684*593dc095SDavid du Colombier if (len) goto inf_leave;
685*593dc095SDavid du Colombier }
686*593dc095SDavid du Colombier state->mode = HCRC;
687*593dc095SDavid du Colombier case HCRC:
688*593dc095SDavid du Colombier if (state->flags & 0x0200) {
689*593dc095SDavid du Colombier NEEDBITS(16);
690*593dc095SDavid du Colombier if (hold != (state->check & 0xffff)) {
691*593dc095SDavid du Colombier strm->msg = (char *)"header crc mismatch";
692*593dc095SDavid du Colombier state->mode = BAD;
693*593dc095SDavid du Colombier break;
694*593dc095SDavid du Colombier }
695*593dc095SDavid du Colombier INITBITS();
696*593dc095SDavid du Colombier }
697*593dc095SDavid du Colombier strm->adler = state->check = crc32(0L, Z_NULL, 0);
698*593dc095SDavid du Colombier state->mode = TYPE;
699*593dc095SDavid du Colombier break;
700*593dc095SDavid du Colombier #endif
701*593dc095SDavid du Colombier case DICTID:
702*593dc095SDavid du Colombier NEEDBITS(32);
703*593dc095SDavid du Colombier strm->adler = state->check = REVERSE(hold);
704*593dc095SDavid du Colombier INITBITS();
705*593dc095SDavid du Colombier state->mode = DICT;
706*593dc095SDavid du Colombier case DICT:
707*593dc095SDavid du Colombier if (state->havedict == 0) {
708*593dc095SDavid du Colombier RESTORE();
7097dd7cddfSDavid du Colombier return Z_NEED_DICT;
710*593dc095SDavid du Colombier }
711*593dc095SDavid du Colombier strm->adler = state->check = adler32(0L, Z_NULL, 0);
712*593dc095SDavid du Colombier state->mode = TYPE;
713*593dc095SDavid du Colombier case TYPE:
714*593dc095SDavid du Colombier if (flush == Z_BLOCK) goto inf_leave;
715*593dc095SDavid du Colombier case TYPEDO:
716*593dc095SDavid du Colombier if (state->last) {
717*593dc095SDavid du Colombier BYTEBITS();
718*593dc095SDavid du Colombier state->mode = CHECK;
7197dd7cddfSDavid du Colombier break;
7207dd7cddfSDavid du Colombier }
721*593dc095SDavid du Colombier NEEDBITS(3);
722*593dc095SDavid du Colombier state->last = BITS(1);
723*593dc095SDavid du Colombier DROPBITS(1);
724*593dc095SDavid du Colombier switch (BITS(2)) {
725*593dc095SDavid du Colombier case 0: /* stored block */
726*593dc095SDavid du Colombier Tracev((stderr, "inflate: stored block%s\n",
727*593dc095SDavid du Colombier state->last ? " (last)" : ""));
728*593dc095SDavid du Colombier state->mode = STORED;
729*593dc095SDavid du Colombier break;
730*593dc095SDavid du Colombier case 1: /* fixed block */
731*593dc095SDavid du Colombier fixedtables(state);
732*593dc095SDavid du Colombier Tracev((stderr, "inflate: fixed codes block%s\n",
733*593dc095SDavid du Colombier state->last ? " (last)" : ""));
734*593dc095SDavid du Colombier state->mode = LEN; /* decode codes */
735*593dc095SDavid du Colombier break;
736*593dc095SDavid du Colombier case 2: /* dynamic block */
737*593dc095SDavid du Colombier Tracev((stderr, "inflate: dynamic codes block%s\n",
738*593dc095SDavid du Colombier state->last ? " (last)" : ""));
739*593dc095SDavid du Colombier state->mode = TABLE;
740*593dc095SDavid du Colombier break;
741*593dc095SDavid du Colombier case 3:
742*593dc095SDavid du Colombier strm->msg = (char *)"invalid block type";
743*593dc095SDavid du Colombier state->mode = BAD;
744*593dc095SDavid du Colombier }
745*593dc095SDavid du Colombier DROPBITS(2);
746*593dc095SDavid du Colombier break;
747*593dc095SDavid du Colombier case STORED:
748*593dc095SDavid du Colombier BYTEBITS(); /* go to byte boundary */
749*593dc095SDavid du Colombier NEEDBITS(32);
750*593dc095SDavid du Colombier if ((hold & 0xffff) != ((hold >> 16) ^ 0xffff)) {
751*593dc095SDavid du Colombier strm->msg = (char *)"invalid stored block lengths";
752*593dc095SDavid du Colombier state->mode = BAD;
7537dd7cddfSDavid du Colombier break;
7547dd7cddfSDavid du Colombier }
755*593dc095SDavid du Colombier state->length = (unsigned)hold & 0xffff;
756*593dc095SDavid du Colombier Tracev((stderr, "inflate: stored length %u\n",
757*593dc095SDavid du Colombier state->length));
758*593dc095SDavid du Colombier INITBITS();
759*593dc095SDavid du Colombier state->mode = COPY;
760*593dc095SDavid du Colombier case COPY:
761*593dc095SDavid du Colombier copy = state->length;
762*593dc095SDavid du Colombier if (copy) {
763*593dc095SDavid du Colombier if (copy > have) copy = have;
764*593dc095SDavid du Colombier if (copy > left) copy = left;
765*593dc095SDavid du Colombier if (copy == 0) goto inf_leave;
766*593dc095SDavid du Colombier zmemcpy(put, next, copy);
767*593dc095SDavid du Colombier have -= copy;
768*593dc095SDavid du Colombier next += copy;
769*593dc095SDavid du Colombier left -= copy;
770*593dc095SDavid du Colombier put += copy;
771*593dc095SDavid du Colombier state->length -= copy;
772*593dc095SDavid du Colombier break;
773*593dc095SDavid du Colombier }
774*593dc095SDavid du Colombier Tracev((stderr, "inflate: stored end\n"));
775*593dc095SDavid du Colombier state->mode = TYPE;
776*593dc095SDavid du Colombier break;
777*593dc095SDavid du Colombier case TABLE:
778*593dc095SDavid du Colombier NEEDBITS(14);
779*593dc095SDavid du Colombier state->nlen = BITS(5) + 257;
780*593dc095SDavid du Colombier DROPBITS(5);
781*593dc095SDavid du Colombier state->ndist = BITS(5) + 1;
782*593dc095SDavid du Colombier DROPBITS(5);
783*593dc095SDavid du Colombier state->ncode = BITS(4) + 4;
784*593dc095SDavid du Colombier DROPBITS(4);
785*593dc095SDavid du Colombier #ifndef PKZIP_BUG_WORKAROUND
786*593dc095SDavid du Colombier if (state->nlen > 286 || state->ndist > 30) {
787*593dc095SDavid du Colombier strm->msg = (char *)"too many length or distance symbols";
788*593dc095SDavid du Colombier state->mode = BAD;
789*593dc095SDavid du Colombier break;
790*593dc095SDavid du Colombier }
791*593dc095SDavid du Colombier #endif
792*593dc095SDavid du Colombier Tracev((stderr, "inflate: table sizes ok\n"));
793*593dc095SDavid du Colombier state->have = 0;
794*593dc095SDavid du Colombier state->mode = LENLENS;
795*593dc095SDavid du Colombier case LENLENS:
796*593dc095SDavid du Colombier while (state->have < state->ncode) {
797*593dc095SDavid du Colombier NEEDBITS(3);
798*593dc095SDavid du Colombier state->lens[order[state->have++]] = (unsigned short)BITS(3);
799*593dc095SDavid du Colombier DROPBITS(3);
800*593dc095SDavid du Colombier }
801*593dc095SDavid du Colombier while (state->have < 19)
802*593dc095SDavid du Colombier state->lens[order[state->have++]] = 0;
803*593dc095SDavid du Colombier state->next = state->codes;
804*593dc095SDavid du Colombier state->lencode = (code const FAR *)(state->next);
805*593dc095SDavid du Colombier state->lenbits = 7;
806*593dc095SDavid du Colombier ret = inflate_table(CODES, state->lens, 19, &(state->next),
807*593dc095SDavid du Colombier &(state->lenbits), state->work);
808*593dc095SDavid du Colombier if (ret) {
809*593dc095SDavid du Colombier strm->msg = (char *)"invalid code lengths set";
810*593dc095SDavid du Colombier state->mode = BAD;
811*593dc095SDavid du Colombier break;
812*593dc095SDavid du Colombier }
813*593dc095SDavid du Colombier Tracev((stderr, "inflate: code lengths ok\n"));
814*593dc095SDavid du Colombier state->have = 0;
815*593dc095SDavid du Colombier state->mode = CODELENS;
816*593dc095SDavid du Colombier case CODELENS:
817*593dc095SDavid du Colombier while (state->have < state->nlen + state->ndist) {
818*593dc095SDavid du Colombier for (;;) {
819*593dc095SDavid du Colombier this = state->lencode[BITS(state->lenbits)];
820*593dc095SDavid du Colombier if ((unsigned)(this.bits) <= bits) break;
821*593dc095SDavid du Colombier PULLBYTE();
822*593dc095SDavid du Colombier }
823*593dc095SDavid du Colombier if (this.val < 16) {
824*593dc095SDavid du Colombier NEEDBITS(this.bits);
825*593dc095SDavid du Colombier DROPBITS(this.bits);
826*593dc095SDavid du Colombier state->lens[state->have++] = this.val;
827*593dc095SDavid du Colombier }
828*593dc095SDavid du Colombier else {
829*593dc095SDavid du Colombier if (this.val == 16) {
830*593dc095SDavid du Colombier NEEDBITS(this.bits + 2);
831*593dc095SDavid du Colombier DROPBITS(this.bits);
832*593dc095SDavid du Colombier if (state->have == 0) {
833*593dc095SDavid du Colombier strm->msg = (char *)"invalid bit length repeat";
834*593dc095SDavid du Colombier state->mode = BAD;
835*593dc095SDavid du Colombier break;
836*593dc095SDavid du Colombier }
837*593dc095SDavid du Colombier len = state->lens[state->have - 1];
838*593dc095SDavid du Colombier copy = 3 + BITS(2);
839*593dc095SDavid du Colombier DROPBITS(2);
840*593dc095SDavid du Colombier }
841*593dc095SDavid du Colombier else if (this.val == 17) {
842*593dc095SDavid du Colombier NEEDBITS(this.bits + 3);
843*593dc095SDavid du Colombier DROPBITS(this.bits);
844*593dc095SDavid du Colombier len = 0;
845*593dc095SDavid du Colombier copy = 3 + BITS(3);
846*593dc095SDavid du Colombier DROPBITS(3);
847*593dc095SDavid du Colombier }
848*593dc095SDavid du Colombier else {
849*593dc095SDavid du Colombier NEEDBITS(this.bits + 7);
850*593dc095SDavid du Colombier DROPBITS(this.bits);
851*593dc095SDavid du Colombier len = 0;
852*593dc095SDavid du Colombier copy = 11 + BITS(7);
853*593dc095SDavid du Colombier DROPBITS(7);
854*593dc095SDavid du Colombier }
855*593dc095SDavid du Colombier if (state->have + copy > state->nlen + state->ndist) {
856*593dc095SDavid du Colombier strm->msg = (char *)"invalid bit length repeat";
857*593dc095SDavid du Colombier state->mode = BAD;
858*593dc095SDavid du Colombier break;
859*593dc095SDavid du Colombier }
860*593dc095SDavid du Colombier while (copy--)
861*593dc095SDavid du Colombier state->lens[state->have++] = (unsigned short)len;
862*593dc095SDavid du Colombier }
863*593dc095SDavid du Colombier }
8647dd7cddfSDavid du Colombier
865*593dc095SDavid du Colombier /* handle error breaks in while */
866*593dc095SDavid du Colombier if (state->mode == BAD) break;
867*593dc095SDavid du Colombier
868*593dc095SDavid du Colombier /* build code tables */
869*593dc095SDavid du Colombier state->next = state->codes;
870*593dc095SDavid du Colombier state->lencode = (code const FAR *)(state->next);
871*593dc095SDavid du Colombier state->lenbits = 9;
872*593dc095SDavid du Colombier ret = inflate_table(LENS, state->lens, state->nlen, &(state->next),
873*593dc095SDavid du Colombier &(state->lenbits), state->work);
874*593dc095SDavid du Colombier if (ret) {
875*593dc095SDavid du Colombier strm->msg = (char *)"invalid literal/lengths set";
876*593dc095SDavid du Colombier state->mode = BAD;
8777dd7cddfSDavid du Colombier break;
8787dd7cddfSDavid du Colombier }
879*593dc095SDavid du Colombier state->distcode = (code const FAR *)(state->next);
880*593dc095SDavid du Colombier state->distbits = 6;
881*593dc095SDavid du Colombier ret = inflate_table(DISTS, state->lens + state->nlen, state->ndist,
882*593dc095SDavid du Colombier &(state->next), &(state->distbits), state->work);
883*593dc095SDavid du Colombier if (ret) {
884*593dc095SDavid du Colombier strm->msg = (char *)"invalid distances set";
885*593dc095SDavid du Colombier state->mode = BAD;
886*593dc095SDavid du Colombier break;
887*593dc095SDavid du Colombier }
888*593dc095SDavid du Colombier Tracev((stderr, "inflate: codes ok\n"));
889*593dc095SDavid du Colombier state->mode = LEN;
890*593dc095SDavid du Colombier case LEN:
891*593dc095SDavid du Colombier if (have >= 6 && left >= 258) {
892*593dc095SDavid du Colombier RESTORE();
893*593dc095SDavid du Colombier inflate_fast(strm, out);
894*593dc095SDavid du Colombier LOAD();
895*593dc095SDavid du Colombier break;
896*593dc095SDavid du Colombier }
897*593dc095SDavid du Colombier for (;;) {
898*593dc095SDavid du Colombier this = state->lencode[BITS(state->lenbits)];
899*593dc095SDavid du Colombier if ((unsigned)(this.bits) <= bits) break;
900*593dc095SDavid du Colombier PULLBYTE();
901*593dc095SDavid du Colombier }
902*593dc095SDavid du Colombier if (this.op && (this.op & 0xf0) == 0) {
903*593dc095SDavid du Colombier last = this;
904*593dc095SDavid du Colombier for (;;) {
905*593dc095SDavid du Colombier this = state->lencode[last.val +
906*593dc095SDavid du Colombier (BITS(last.bits + last.op) >> last.bits)];
907*593dc095SDavid du Colombier if ((unsigned)(last.bits + this.bits) <= bits) break;
908*593dc095SDavid du Colombier PULLBYTE();
909*593dc095SDavid du Colombier }
910*593dc095SDavid du Colombier DROPBITS(last.bits);
911*593dc095SDavid du Colombier }
912*593dc095SDavid du Colombier DROPBITS(this.bits);
913*593dc095SDavid du Colombier state->length = (unsigned)this.val;
914*593dc095SDavid du Colombier if ((int)(this.op) == 0) {
915*593dc095SDavid du Colombier Tracevv((stderr, this.val >= 0x20 && this.val < 0x7f ?
916*593dc095SDavid du Colombier "inflate: literal '%c'\n" :
917*593dc095SDavid du Colombier "inflate: literal 0x%02x\n", this.val));
918*593dc095SDavid du Colombier state->mode = LIT;
919*593dc095SDavid du Colombier break;
920*593dc095SDavid du Colombier }
921*593dc095SDavid du Colombier if (this.op & 32) {
922*593dc095SDavid du Colombier Tracevv((stderr, "inflate: end of block\n"));
923*593dc095SDavid du Colombier state->mode = TYPE;
924*593dc095SDavid du Colombier break;
925*593dc095SDavid du Colombier }
926*593dc095SDavid du Colombier if (this.op & 64) {
927*593dc095SDavid du Colombier strm->msg = (char *)"invalid literal/length code";
928*593dc095SDavid du Colombier state->mode = BAD;
929*593dc095SDavid du Colombier break;
930*593dc095SDavid du Colombier }
931*593dc095SDavid du Colombier state->extra = (unsigned)(this.op) & 15;
932*593dc095SDavid du Colombier state->mode = LENEXT;
933*593dc095SDavid du Colombier case LENEXT:
934*593dc095SDavid du Colombier if (state->extra) {
935*593dc095SDavid du Colombier NEEDBITS(state->extra);
936*593dc095SDavid du Colombier state->length += BITS(state->extra);
937*593dc095SDavid du Colombier DROPBITS(state->extra);
938*593dc095SDavid du Colombier }
939*593dc095SDavid du Colombier Tracevv((stderr, "inflate: length %u\n", state->length));
940*593dc095SDavid du Colombier state->mode = DIST;
941*593dc095SDavid du Colombier case DIST:
942*593dc095SDavid du Colombier for (;;) {
943*593dc095SDavid du Colombier this = state->distcode[BITS(state->distbits)];
944*593dc095SDavid du Colombier if ((unsigned)(this.bits) <= bits) break;
945*593dc095SDavid du Colombier PULLBYTE();
946*593dc095SDavid du Colombier }
947*593dc095SDavid du Colombier if ((this.op & 0xf0) == 0) {
948*593dc095SDavid du Colombier last = this;
949*593dc095SDavid du Colombier for (;;) {
950*593dc095SDavid du Colombier this = state->distcode[last.val +
951*593dc095SDavid du Colombier (BITS(last.bits + last.op) >> last.bits)];
952*593dc095SDavid du Colombier if ((unsigned)(last.bits + this.bits) <= bits) break;
953*593dc095SDavid du Colombier PULLBYTE();
954*593dc095SDavid du Colombier }
955*593dc095SDavid du Colombier DROPBITS(last.bits);
956*593dc095SDavid du Colombier }
957*593dc095SDavid du Colombier DROPBITS(this.bits);
958*593dc095SDavid du Colombier if (this.op & 64) {
959*593dc095SDavid du Colombier strm->msg = (char *)"invalid distance code";
960*593dc095SDavid du Colombier state->mode = BAD;
961*593dc095SDavid du Colombier break;
962*593dc095SDavid du Colombier }
963*593dc095SDavid du Colombier state->offset = (unsigned)this.val;
964*593dc095SDavid du Colombier state->extra = (unsigned)(this.op) & 15;
965*593dc095SDavid du Colombier state->mode = DISTEXT;
966*593dc095SDavid du Colombier case DISTEXT:
967*593dc095SDavid du Colombier if (state->extra) {
968*593dc095SDavid du Colombier NEEDBITS(state->extra);
969*593dc095SDavid du Colombier state->offset += BITS(state->extra);
970*593dc095SDavid du Colombier DROPBITS(state->extra);
971*593dc095SDavid du Colombier }
972*593dc095SDavid du Colombier if (state->offset > state->whave + out - left) {
973*593dc095SDavid du Colombier strm->msg = (char *)"invalid distance too far back";
974*593dc095SDavid du Colombier state->mode = BAD;
975*593dc095SDavid du Colombier break;
976*593dc095SDavid du Colombier }
977*593dc095SDavid du Colombier Tracevv((stderr, "inflate: distance %u\n", state->offset));
978*593dc095SDavid du Colombier state->mode = MATCH;
979*593dc095SDavid du Colombier case MATCH:
980*593dc095SDavid du Colombier if (left == 0) goto inf_leave;
981*593dc095SDavid du Colombier copy = out - left;
982*593dc095SDavid du Colombier if (state->offset > copy) { /* copy from window */
983*593dc095SDavid du Colombier copy = state->offset - copy;
984*593dc095SDavid du Colombier if (copy > state->write) {
985*593dc095SDavid du Colombier copy -= state->write;
986*593dc095SDavid du Colombier from = state->window + (state->wsize - copy);
987*593dc095SDavid du Colombier }
988*593dc095SDavid du Colombier else
989*593dc095SDavid du Colombier from = state->window + (state->write - copy);
990*593dc095SDavid du Colombier if (copy > state->length) copy = state->length;
991*593dc095SDavid du Colombier }
992*593dc095SDavid du Colombier else { /* copy from output */
993*593dc095SDavid du Colombier from = put - state->offset;
994*593dc095SDavid du Colombier copy = state->length;
995*593dc095SDavid du Colombier }
996*593dc095SDavid du Colombier if (copy > left) copy = left;
997*593dc095SDavid du Colombier left -= copy;
998*593dc095SDavid du Colombier state->length -= copy;
999*593dc095SDavid du Colombier do {
1000*593dc095SDavid du Colombier *put++ = *from++;
1001*593dc095SDavid du Colombier } while (--copy);
1002*593dc095SDavid du Colombier if (state->length == 0) state->mode = LEN;
1003*593dc095SDavid du Colombier break;
1004*593dc095SDavid du Colombier case LIT:
1005*593dc095SDavid du Colombier if (left == 0) goto inf_leave;
1006*593dc095SDavid du Colombier *put++ = (unsigned char)(state->length);
1007*593dc095SDavid du Colombier left--;
1008*593dc095SDavid du Colombier state->mode = LEN;
1009*593dc095SDavid du Colombier break;
1010*593dc095SDavid du Colombier case CHECK:
1011*593dc095SDavid du Colombier if (state->wrap) {
1012*593dc095SDavid du Colombier NEEDBITS(32);
1013*593dc095SDavid du Colombier out -= left;
1014*593dc095SDavid du Colombier strm->total_out += out;
1015*593dc095SDavid du Colombier state->total += out;
1016*593dc095SDavid du Colombier if (out)
1017*593dc095SDavid du Colombier strm->adler = state->check =
1018*593dc095SDavid du Colombier UPDATE(state->check, put - out, out);
1019*593dc095SDavid du Colombier out = left;
1020*593dc095SDavid du Colombier if ((
1021*593dc095SDavid du Colombier #ifdef GUNZIP
1022*593dc095SDavid du Colombier state->flags ? hold :
1023*593dc095SDavid du Colombier #endif
1024*593dc095SDavid du Colombier REVERSE(hold)) != state->check) {
1025*593dc095SDavid du Colombier strm->msg = (char *)"incorrect data check";
1026*593dc095SDavid du Colombier state->mode = BAD;
1027*593dc095SDavid du Colombier break;
1028*593dc095SDavid du Colombier }
1029*593dc095SDavid du Colombier INITBITS();
1030*593dc095SDavid du Colombier Tracev((stderr, "inflate: check matches trailer\n"));
1031*593dc095SDavid du Colombier }
1032*593dc095SDavid du Colombier #ifdef GUNZIP
1033*593dc095SDavid du Colombier state->mode = LENGTH;
1034*593dc095SDavid du Colombier case LENGTH:
1035*593dc095SDavid du Colombier if (state->wrap && state->flags) {
1036*593dc095SDavid du Colombier NEEDBITS(32);
1037*593dc095SDavid du Colombier if (hold != (state->total & 0xffffffffUL)) {
1038*593dc095SDavid du Colombier strm->msg = (char *)"incorrect length check";
1039*593dc095SDavid du Colombier state->mode = BAD;
1040*593dc095SDavid du Colombier break;
1041*593dc095SDavid du Colombier }
1042*593dc095SDavid du Colombier INITBITS();
1043*593dc095SDavid du Colombier Tracev((stderr, "inflate: length matches trailer\n"));
1044*593dc095SDavid du Colombier }
1045*593dc095SDavid du Colombier #endif
1046*593dc095SDavid du Colombier state->mode = DONE;
10477dd7cddfSDavid du Colombier case DONE:
1048*593dc095SDavid du Colombier ret = Z_STREAM_END;
1049*593dc095SDavid du Colombier goto inf_leave;
10507dd7cddfSDavid du Colombier case BAD:
1051*593dc095SDavid du Colombier ret = Z_DATA_ERROR;
1052*593dc095SDavid du Colombier goto inf_leave;
1053*593dc095SDavid du Colombier case MEM:
1054*593dc095SDavid du Colombier return Z_MEM_ERROR;
1055*593dc095SDavid du Colombier case SYNC:
10567dd7cddfSDavid du Colombier default:
10577dd7cddfSDavid du Colombier return Z_STREAM_ERROR;
10587dd7cddfSDavid du Colombier }
1059*593dc095SDavid du Colombier
1060*593dc095SDavid du Colombier /*
1061*593dc095SDavid du Colombier Return from inflate(), updating the total counts and the check value.
1062*593dc095SDavid du Colombier If there was no progress during the inflate() call, return a buffer
1063*593dc095SDavid du Colombier error. Call updatewindow() to create and/or update the window state.
1064*593dc095SDavid du Colombier Note: a memory error from inflate() is non-recoverable.
1065*593dc095SDavid du Colombier */
1066*593dc095SDavid du Colombier inf_leave:
1067*593dc095SDavid du Colombier RESTORE();
1068*593dc095SDavid du Colombier if (state->wsize || (state->mode < CHECK && out != strm->avail_out))
1069*593dc095SDavid du Colombier if (updatewindow(strm, out)) {
1070*593dc095SDavid du Colombier state->mode = MEM;
1071*593dc095SDavid du Colombier return Z_MEM_ERROR;
1072*593dc095SDavid du Colombier }
1073*593dc095SDavid du Colombier in -= strm->avail_in;
1074*593dc095SDavid du Colombier out -= strm->avail_out;
1075*593dc095SDavid du Colombier strm->total_in += in;
1076*593dc095SDavid du Colombier strm->total_out += out;
1077*593dc095SDavid du Colombier state->total += out;
1078*593dc095SDavid du Colombier if (state->wrap && out)
1079*593dc095SDavid du Colombier strm->adler = state->check =
1080*593dc095SDavid du Colombier UPDATE(state->check, strm->next_out - out, out);
1081*593dc095SDavid du Colombier strm->data_type = state->bits + (state->last ? 64 : 0) +
1082*593dc095SDavid du Colombier (state->mode == TYPE ? 128 : 0);
1083*593dc095SDavid du Colombier if (((in == 0 && out == 0) || flush == Z_FINISH) && ret == Z_OK)
1084*593dc095SDavid du Colombier ret = Z_BUF_ERROR;
1085*593dc095SDavid du Colombier return ret;
10867dd7cddfSDavid du Colombier }
10877dd7cddfSDavid du Colombier
inflateEnd(strm)1088*593dc095SDavid du Colombier int ZEXPORT inflateEnd(strm)
1089*593dc095SDavid du Colombier z_streamp strm;
10907dd7cddfSDavid du Colombier {
1091*593dc095SDavid du Colombier struct inflate_state FAR *state;
1092*593dc095SDavid du Colombier if (strm == Z_NULL || strm->state == Z_NULL || strm->zfree == (free_func)0)
10937dd7cddfSDavid du Colombier return Z_STREAM_ERROR;
1094*593dc095SDavid du Colombier state = (struct inflate_state FAR *)strm->state;
1095*593dc095SDavid du Colombier if (state->window != Z_NULL) ZFREE(strm, state->window);
1096*593dc095SDavid du Colombier ZFREE(strm, strm->state);
1097*593dc095SDavid du Colombier strm->state = Z_NULL;
1098*593dc095SDavid du Colombier Tracev((stderr, "inflate: end\n"));
10997dd7cddfSDavid du Colombier return Z_OK;
11007dd7cddfSDavid du Colombier }
11017dd7cddfSDavid du Colombier
inflateSetDictionary(strm,dictionary,dictLength)1102*593dc095SDavid du Colombier int ZEXPORT inflateSetDictionary(strm, dictionary, dictLength)
1103*593dc095SDavid du Colombier z_streamp strm;
1104*593dc095SDavid du Colombier const Bytef *dictionary;
1105*593dc095SDavid du Colombier uInt dictLength;
11067dd7cddfSDavid du Colombier {
1107*593dc095SDavid du Colombier struct inflate_state FAR *state;
1108*593dc095SDavid du Colombier unsigned long id;
11097dd7cddfSDavid du Colombier
1110*593dc095SDavid du Colombier /* check state */
1111*593dc095SDavid du Colombier if (strm == Z_NULL || strm->state == Z_NULL) return Z_STREAM_ERROR;
1112*593dc095SDavid du Colombier state = (struct inflate_state FAR *)strm->state;
1113*593dc095SDavid du Colombier if (state->mode != DICT) return Z_STREAM_ERROR;
1114*593dc095SDavid du Colombier
1115*593dc095SDavid du Colombier /* check for correct dictionary id */
1116*593dc095SDavid du Colombier id = adler32(0L, Z_NULL, 0);
1117*593dc095SDavid du Colombier id = adler32(id, dictionary, dictLength);
1118*593dc095SDavid du Colombier if (id != state->check) return Z_DATA_ERROR;
1119*593dc095SDavid du Colombier
1120*593dc095SDavid du Colombier /* copy dictionary to window */
1121*593dc095SDavid du Colombier if (updatewindow(strm, strm->avail_out)) {
1122*593dc095SDavid du Colombier state->mode = MEM;
1123*593dc095SDavid du Colombier return Z_MEM_ERROR;
11247dd7cddfSDavid du Colombier }
1125*593dc095SDavid du Colombier if (dictLength > state->wsize) {
1126*593dc095SDavid du Colombier zmemcpy(state->window, dictionary + dictLength - state->wsize,
1127*593dc095SDavid du Colombier state->wsize);
1128*593dc095SDavid du Colombier state->whave = state->wsize;
1129*593dc095SDavid du Colombier }
1130*593dc095SDavid du Colombier else {
1131*593dc095SDavid du Colombier zmemcpy(state->window + state->wsize - dictLength, dictionary,
1132*593dc095SDavid du Colombier dictLength);
1133*593dc095SDavid du Colombier state->whave = dictLength;
1134*593dc095SDavid du Colombier }
1135*593dc095SDavid du Colombier state->havedict = 1;
1136*593dc095SDavid du Colombier Tracev((stderr, "inflate: dictionary set\n"));
1137*593dc095SDavid du Colombier return Z_OK;
1138*593dc095SDavid du Colombier }
11397dd7cddfSDavid du Colombier
1140*593dc095SDavid du Colombier /*
1141*593dc095SDavid du Colombier Search buf[0..len-1] for the pattern: 0, 0, 0xff, 0xff. Return when found
1142*593dc095SDavid du Colombier or when out of input. When called, *have is the number of pattern bytes
1143*593dc095SDavid du Colombier found in order so far, in 0..3. On return *have is updated to the new
1144*593dc095SDavid du Colombier state. If on return *have equals four, then the pattern was found and the
1145*593dc095SDavid du Colombier return value is how many bytes were read including the last byte of the
1146*593dc095SDavid du Colombier pattern. If *have is less than four, then the pattern has not been found
1147*593dc095SDavid du Colombier yet and the return value is len. In the latter case, syncsearch() can be
1148*593dc095SDavid du Colombier called again with more data and the *have state. *have is initialized to
1149*593dc095SDavid du Colombier zero for the first call.
1150*593dc095SDavid du Colombier */
syncsearch(have,buf,len)1151*593dc095SDavid du Colombier local unsigned syncsearch(have, buf, len)
1152*593dc095SDavid du Colombier unsigned FAR *have;
1153*593dc095SDavid du Colombier unsigned char FAR *buf;
1154*593dc095SDavid du Colombier unsigned len;
11557dd7cddfSDavid du Colombier {
1156*593dc095SDavid du Colombier unsigned got;
1157*593dc095SDavid du Colombier unsigned next;
1158*593dc095SDavid du Colombier
1159*593dc095SDavid du Colombier got = *have;
1160*593dc095SDavid du Colombier next = 0;
1161*593dc095SDavid du Colombier while (next < len && got < 4) {
1162*593dc095SDavid du Colombier if ((int)(buf[next]) == (got < 2 ? 0 : 0xff))
1163*593dc095SDavid du Colombier got++;
1164*593dc095SDavid du Colombier else if (buf[next])
1165*593dc095SDavid du Colombier got = 0;
11667dd7cddfSDavid du Colombier else
1167*593dc095SDavid du Colombier got = 4 - got;
1168*593dc095SDavid du Colombier next++;
1169*593dc095SDavid du Colombier }
1170*593dc095SDavid du Colombier *have = got;
1171*593dc095SDavid du Colombier return next;
11727dd7cddfSDavid du Colombier }
11737dd7cddfSDavid du Colombier
inflateSync(strm)1174*593dc095SDavid du Colombier int ZEXPORT inflateSync(strm)
1175*593dc095SDavid du Colombier z_streamp strm;
1176*593dc095SDavid du Colombier {
1177*593dc095SDavid du Colombier unsigned len; /* number of bytes to look at or looked at */
1178*593dc095SDavid du Colombier unsigned long in, out; /* temporary to save total_in and total_out */
1179*593dc095SDavid du Colombier unsigned char buf[4]; /* to restore bit buffer to byte string */
1180*593dc095SDavid du Colombier struct inflate_state FAR *state;
11817dd7cddfSDavid du Colombier
1182*593dc095SDavid du Colombier /* check parameters */
1183*593dc095SDavid du Colombier if (strm == Z_NULL || strm->state == Z_NULL) return Z_STREAM_ERROR;
1184*593dc095SDavid du Colombier state = (struct inflate_state FAR *)strm->state;
1185*593dc095SDavid du Colombier if (strm->avail_in == 0 && state->bits < 8) return Z_BUF_ERROR;
1186*593dc095SDavid du Colombier
1187*593dc095SDavid du Colombier /* if first time, start search in bit buffer */
1188*593dc095SDavid du Colombier if (state->mode != SYNC) {
1189*593dc095SDavid du Colombier state->mode = SYNC;
1190*593dc095SDavid du Colombier state->hold <<= state->bits & 7;
1191*593dc095SDavid du Colombier state->bits -= state->bits & 7;
1192*593dc095SDavid du Colombier len = 0;
1193*593dc095SDavid du Colombier while (state->bits >= 8) {
1194*593dc095SDavid du Colombier buf[len++] = (unsigned char)(state->hold);
1195*593dc095SDavid du Colombier state->hold >>= 8;
1196*593dc095SDavid du Colombier state->bits -= 8;
1197*593dc095SDavid du Colombier }
1198*593dc095SDavid du Colombier state->have = 0;
1199*593dc095SDavid du Colombier syncsearch(&(state->have), buf, len);
1200*593dc095SDavid du Colombier }
1201*593dc095SDavid du Colombier
1202*593dc095SDavid du Colombier /* search available input */
1203*593dc095SDavid du Colombier len = syncsearch(&(state->have), strm->next_in, strm->avail_in);
1204*593dc095SDavid du Colombier strm->avail_in -= len;
1205*593dc095SDavid du Colombier strm->next_in += len;
1206*593dc095SDavid du Colombier strm->total_in += len;
1207*593dc095SDavid du Colombier
1208*593dc095SDavid du Colombier /* return no joy or set up to restart inflate() on a new block */
1209*593dc095SDavid du Colombier if (state->have != 4) return Z_DATA_ERROR;
1210*593dc095SDavid du Colombier in = strm->total_in; out = strm->total_out;
1211*593dc095SDavid du Colombier inflateReset(strm);
1212*593dc095SDavid du Colombier strm->total_in = in; strm->total_out = out;
1213*593dc095SDavid du Colombier state->mode = TYPE;
1214*593dc095SDavid du Colombier return Z_OK;
1215*593dc095SDavid du Colombier }
1216*593dc095SDavid du Colombier
1217*593dc095SDavid du Colombier /*
1218*593dc095SDavid du Colombier Returns true if inflate is currently at the end of a block generated by
1219*593dc095SDavid du Colombier Z_SYNC_FLUSH or Z_FULL_FLUSH. This function is used by one PPP
1220*593dc095SDavid du Colombier implementation to provide an additional safety check. PPP uses
1221*593dc095SDavid du Colombier Z_SYNC_FLUSH but removes the length bytes of the resulting empty stored
1222*593dc095SDavid du Colombier block. When decompressing, PPP checks that at the end of input packet,
1223*593dc095SDavid du Colombier inflate is waiting for these length bytes.
1224*593dc095SDavid du Colombier */
inflateSyncPoint(strm)1225*593dc095SDavid du Colombier int ZEXPORT inflateSyncPoint(strm)
1226*593dc095SDavid du Colombier z_streamp strm;
1227*593dc095SDavid du Colombier {
1228*593dc095SDavid du Colombier struct inflate_state FAR *state;
1229*593dc095SDavid du Colombier
1230*593dc095SDavid du Colombier if (strm == Z_NULL || strm->state == Z_NULL) return Z_STREAM_ERROR;
1231*593dc095SDavid du Colombier state = (struct inflate_state FAR *)strm->state;
1232*593dc095SDavid du Colombier return state->mode == STORED && state->bits == 0;
1233*593dc095SDavid du Colombier }
1234*593dc095SDavid du Colombier
inflateCopy(dest,source)1235*593dc095SDavid du Colombier int ZEXPORT inflateCopy(dest, source)
1236*593dc095SDavid du Colombier z_streamp dest;
1237*593dc095SDavid du Colombier z_streamp source;
1238*593dc095SDavid du Colombier {
1239*593dc095SDavid du Colombier struct inflate_state FAR *state;
1240*593dc095SDavid du Colombier struct inflate_state FAR *copy;
1241*593dc095SDavid du Colombier unsigned char FAR *window;
1242*593dc095SDavid du Colombier
1243*593dc095SDavid du Colombier /* check input */
1244*593dc095SDavid du Colombier if (dest == Z_NULL || source == Z_NULL || source->state == Z_NULL ||
1245*593dc095SDavid du Colombier source->zalloc == (alloc_func)0 || source->zfree == (free_func)0)
1246*593dc095SDavid du Colombier return Z_STREAM_ERROR;
1247*593dc095SDavid du Colombier state = (struct inflate_state FAR *)source->state;
1248*593dc095SDavid du Colombier
1249*593dc095SDavid du Colombier /* allocate space */
1250*593dc095SDavid du Colombier copy = (struct inflate_state FAR *)
1251*593dc095SDavid du Colombier ZALLOC(source, 1, sizeof(struct inflate_state));
1252*593dc095SDavid du Colombier if (copy == Z_NULL) return Z_MEM_ERROR;
1253*593dc095SDavid du Colombier window = Z_NULL;
1254*593dc095SDavid du Colombier if (state->window != Z_NULL) {
1255*593dc095SDavid du Colombier window = (unsigned char FAR *)
1256*593dc095SDavid du Colombier ZALLOC(source, 1U << state->wbits, sizeof(unsigned char));
1257*593dc095SDavid du Colombier if (window == Z_NULL) {
1258*593dc095SDavid du Colombier ZFREE(source, copy);
1259*593dc095SDavid du Colombier return Z_MEM_ERROR;
1260*593dc095SDavid du Colombier }
1261*593dc095SDavid du Colombier }
1262*593dc095SDavid du Colombier
1263*593dc095SDavid du Colombier /* copy state */
1264*593dc095SDavid du Colombier *dest = *source;
1265*593dc095SDavid du Colombier *copy = *state;
1266*593dc095SDavid du Colombier copy->lencode = copy->codes + (state->lencode - state->codes);
1267*593dc095SDavid du Colombier copy->distcode = copy->codes + (state->distcode - state->codes);
1268*593dc095SDavid du Colombier copy->next = copy->codes + (state->next - state->codes);
1269*593dc095SDavid du Colombier if (window != Z_NULL)
1270*593dc095SDavid du Colombier zmemcpy(window, state->window, 1U << state->wbits);
1271*593dc095SDavid du Colombier copy->window = window;
1272*593dc095SDavid du Colombier dest->state = (voidpf)copy;
12737dd7cddfSDavid du Colombier return Z_OK;
12747dd7cddfSDavid du Colombier }
1275