xref: /plan9/sys/src/cmd/gs/zlib/inflate.c (revision 593dc095aefb2a85c828727bbfa9da139a49bdf4)
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