1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/ossrv/compressionlibs/ziplib/src/zlib/infback.cpp Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,655 @@
1.4 +/* Portions Copyright (c) 2007-2009 Nokia Corporation and/or its subsidiary(-ies).
1.5 + * All rights reserved.
1.6 + */
1.7 +
1.8 +/* infback.cpp -- inflate using a call-back interface
1.9 + * Copyright (C) 1995-2005 Mark Adler
1.10 + * For conditions of distribution and use, see copyright notice in zlib.h
1.11 + */
1.12 +
1.13 +/*
1.14 + This code is largely copied from inflate.c. Normally either infback.o or
1.15 + inflate.o would be linked into an application--not both. The interface
1.16 + with inffast.c is retained so that optimized assembler-coded versions of
1.17 + inflate_fast() can be used with either inflate.c or infback.c.
1.18 + */
1.19 +
1.20 +#include "zutil.h"
1.21 +#include "inftrees.h"
1.22 +#include "inflate.h"
1.23 +#include "inffast.h"
1.24 +
1.25 +/* function prototypes */
1.26 +local void fixedtables OF((struct inflate_state FAR *state));
1.27 +
1.28 +/*
1.29 + strm provides memory allocation functions in zalloc and zfree, or
1.30 + Z_NULL to use the library memory allocation functions.
1.31 +
1.32 + windowBits is in the range 8..15, and window is a user-supplied
1.33 + window and output buffer that is 2**windowBits bytes.
1.34 + */
1.35 +
1.36 +#ifdef __SYMBIAN32__
1.37 +EXPORT_C int inflateBackInit__r(z_streamp strm, int windowBits,unsigned char FAR * window,const char * version,int stream_size)
1.38 +#else
1.39 +int ZEXPORT inflateBackInit_(strm, windowBits, window, version, stream_size)
1.40 +z_streamp strm;
1.41 +int windowBits;
1.42 +unsigned char FAR *window;
1.43 +const char *version;
1.44 +int stream_size;
1.45 +#endif //__SYMBIAN32__
1.46 +{
1.47 + struct inflate_state FAR *state;
1.48 +
1.49 + if (version == Z_NULL || version[0] != ZLIB_VERSION[0] ||
1.50 + stream_size != (int)(sizeof(z_stream)))
1.51 + return Z_VERSION_ERROR;
1.52 + if (strm == Z_NULL || window == Z_NULL ||
1.53 + windowBits < 8 || windowBits > 15)
1.54 + return Z_STREAM_ERROR;
1.55 + strm->msg = Z_NULL; /* in case we return an error */
1.56 + if (strm->zalloc == (alloc_func)0) {
1.57 + strm->zalloc = zcalloc;
1.58 + strm->opaque = (voidpf)0;
1.59 + }
1.60 + if (strm->zfree == (free_func)0) strm->zfree = zcfree;
1.61 + state = (struct inflate_state FAR *)ZALLOC(strm, 1,
1.62 + sizeof(struct inflate_state));
1.63 + if (state == Z_NULL) return Z_MEM_ERROR;
1.64 + Tracev((stderr, "inflate: allocated\n"));
1.65 + strm->state = (struct internal_state FAR *)state;
1.66 + state->dmax = 32768U;
1.67 + state->wbits = windowBits;
1.68 + state->wsize = 1U << windowBits;
1.69 + state->window = window;
1.70 + state->write = 0;
1.71 + state->whave = 0;
1.72 + return Z_OK;
1.73 +}
1.74 +
1.75 +
1.76 +/*
1.77 + Return state with length and distance decoding tables and index sizes set to
1.78 + fixed code decoding. Normally this returns fixed tables from inffixed.h.
1.79 + If BUILDFIXED is defined, then instead this routine builds the tables the
1.80 + first time it's called, and returns those tables the first time and
1.81 + thereafter. This reduces the size of the code by about 2K bytes, in
1.82 + exchange for a little execution time. However, BUILDFIXED should not be
1.83 + used for threaded applications, since the rewriting of the tables and virgin
1.84 + may not be thread-safe.
1.85 + */
1.86 +
1.87 +#ifdef __SYMBIAN32__
1.88 +local void fixedtables(struct inflate_state FAR * state)
1.89 +#else
1.90 +local void fixedtables(state)
1.91 +struct inflate_state FAR *state;
1.92 +#endif //__SYMBIAN32__
1.93 +{
1.94 +#ifdef BUILDFIXED
1.95 + static int virgin = 1;
1.96 + static code *lenfix, *distfix;
1.97 + static code fixed[544];
1.98 +
1.99 + /* build fixed huffman tables if first call (may not be thread safe) */
1.100 + if (virgin) {
1.101 + unsigned sym, bits;
1.102 + static code *next;
1.103 +
1.104 + /* literal/length table */
1.105 + sym = 0;
1.106 + while (sym < 144) state->lens[sym++] = 8;
1.107 + while (sym < 256) state->lens[sym++] = 9;
1.108 + while (sym < 280) state->lens[sym++] = 7;
1.109 + while (sym < 288) state->lens[sym++] = 8;
1.110 + next = fixed;
1.111 + lenfix = next;
1.112 + bits = 9;
1.113 + inflate_table(LENS, state->lens, 288, &(next), &(bits), state->work);
1.114 +
1.115 + /* distance table */
1.116 + sym = 0;
1.117 + while (sym < 32) state->lens[sym++] = 5;
1.118 + distfix = next;
1.119 + bits = 5;
1.120 + inflate_table(DISTS, state->lens, 32, &(next), &(bits), state->work);
1.121 +
1.122 + /* do this just once */
1.123 + virgin = 0;
1.124 + }
1.125 +#else /* !BUILDFIXED */
1.126 +# include "inffixed.h"
1.127 +#endif /* BUILDFIXED */
1.128 + state->lencode = lenfix;
1.129 + state->lenbits = 9;
1.130 + state->distcode = distfix;
1.131 + state->distbits = 5;
1.132 +}
1.133 +
1.134 +
1.135 +/* Macros for inflateBack(): */
1.136 +
1.137 +/* Load returned state from inflate_fast() */
1.138 +#define LOAD() \
1.139 + do { \
1.140 + put = strm->next_out; \
1.141 + left = strm->avail_out; \
1.142 + next = strm->next_in; \
1.143 + have = strm->avail_in; \
1.144 + hold = state->hold; \
1.145 + bits = state->bits; \
1.146 + } while (0)
1.147 +
1.148 +/* Set state from registers for inflate_fast() */
1.149 +#define RESTORE() \
1.150 + do { \
1.151 + strm->next_out = put; \
1.152 + strm->avail_out = left; \
1.153 + strm->next_in = next; \
1.154 + strm->avail_in = have; \
1.155 + state->hold = hold; \
1.156 + state->bits = bits; \
1.157 + } while (0)
1.158 +
1.159 +/* Clear the input bit accumulator */
1.160 +#define INITBITS() \
1.161 + do { \
1.162 + hold = 0; \
1.163 + bits = 0; \
1.164 + } while (0)
1.165 +
1.166 +/* Assure that some input is available. If input is requested, but denied,
1.167 + then return a Z_BUF_ERROR from inflateBack(). */
1.168 +#define PULL() \
1.169 + do { \
1.170 + if (have == 0) { \
1.171 + have = in(in_desc, &next); \
1.172 + if (have == 0) { \
1.173 + next = Z_NULL; \
1.174 + ret = Z_BUF_ERROR; \
1.175 + goto inf_leave; \
1.176 + } \
1.177 + } \
1.178 + } while (0)
1.179 +
1.180 +/* Get a byte of input into the bit accumulator, or return from inflateBack()
1.181 + with an error if there is no input available. */
1.182 +#define PULLBYTE() \
1.183 + do { \
1.184 + PULL(); \
1.185 + have--; \
1.186 + hold += (unsigned long)(*next++) << bits; \
1.187 + bits += 8; \
1.188 + } while (0)
1.189 +
1.190 +/* Assure that there are at least n bits in the bit accumulator. If there is
1.191 + not enough available input to do that, then return from inflateBack() with
1.192 + an error. */
1.193 +#define NEEDBITS(n) \
1.194 + do { \
1.195 + while (bits < (unsigned)(n)) \
1.196 + PULLBYTE(); \
1.197 + } while (0)
1.198 +
1.199 +/* Return the low n bits of the bit accumulator (n < 16) */
1.200 +#define BITS(n) \
1.201 + ((unsigned)hold & ((1U << (n)) - 1))
1.202 +
1.203 +/* Remove n bits from the bit accumulator */
1.204 +#define DROPBITS(n) \
1.205 + do { \
1.206 + hold >>= (n); \
1.207 + bits -= (unsigned)(n); \
1.208 + } while (0)
1.209 +
1.210 +/* Remove zero to seven bits as needed to go to a byte boundary */
1.211 +#define BYTEBITS() \
1.212 + do { \
1.213 + hold >>= bits & 7; \
1.214 + bits -= bits & 7; \
1.215 + } while (0)
1.216 +
1.217 +/* Assure that some output space is available, by writing out the window
1.218 + if it's full. If the write fails, return from inflateBack() with a
1.219 + Z_BUF_ERROR. */
1.220 +#define ROOM() \
1.221 + do { \
1.222 + if (left == 0) { \
1.223 + put = state->window; \
1.224 + left = state->wsize; \
1.225 + state->whave = left; \
1.226 + if (out(out_desc, put, left)) { \
1.227 + ret = Z_BUF_ERROR; \
1.228 + goto inf_leave; \
1.229 + } \
1.230 + } \
1.231 + } while (0)
1.232 +
1.233 +/*
1.234 + strm provides the memory allocation functions and window buffer on input,
1.235 + and provides information on the unused input on return. For Z_DATA_ERROR
1.236 + returns, strm will also provide an error message.
1.237 +
1.238 + in() and out() are the call-back input and output functions. When
1.239 + inflateBack() needs more input, it calls in(). When inflateBack() has
1.240 + filled the window with output, or when it completes with data in the
1.241 + window, it calls out() to write out the data. The application must not
1.242 + change the provided input until in() is called again or inflateBack()
1.243 + returns. The application must not change the window/output buffer until
1.244 + inflateBack() returns.
1.245 +
1.246 + in() and out() are called with a descriptor parameter provided in the
1.247 + inflateBack() call. This parameter can be a structure that provides the
1.248 + information required to do the read or write, as well as accumulated
1.249 + information on the input and output such as totals and check values.
1.250 +
1.251 + in() should return zero on failure. out() should return non-zero on
1.252 + failure. If either in() or out() fails, than inflateBack() returns a
1.253 + Z_BUF_ERROR. strm->next_in can be checked for Z_NULL to see whether it
1.254 + was in() or out() that caused in the error. Otherwise, inflateBack()
1.255 + returns Z_STREAM_END on success, Z_DATA_ERROR for an deflate format
1.256 + error, or Z_MEM_ERROR if it could not allocate memory for the state.
1.257 + inflateBack() can also return Z_STREAM_ERROR if the input parameters
1.258 + are not correct, i.e. strm is Z_NULL or the state was not initialized.
1.259 + */
1.260 +
1.261 +#ifdef __SYMBIAN32__
1.262 +EXPORT_C int inflateBack_r(z_streamp strm,in_func in, void FAR * in_desc,out_func out,void FAR * out_desc)
1.263 +#else
1.264 +int ZEXPORT inflateBack(strm, in, in_desc, out, out_desc)
1.265 +z_streamp strm;
1.266 +in_func in;
1.267 +void FAR *in_desc;
1.268 +out_func out;
1.269 +void FAR *out_desc;
1.270 +#endif //__SYMBIAN32__
1.271 +{
1.272 + struct inflate_state FAR *state;
1.273 + unsigned char FAR *next; /* next input */
1.274 + unsigned char FAR *put; /* next output */
1.275 + unsigned have, left; /* available input and output */
1.276 + unsigned long hold; /* bit buffer */
1.277 + unsigned bits; /* bits in bit buffer */
1.278 + unsigned copy; /* number of stored or match bytes to copy */
1.279 + unsigned char FAR *from; /* where to copy match bytes from */
1.280 +
1.281 +/* Need to replace "this" variable with "current" as "this" is a reserved
1.282 + * keyword in C++ which is prefectly fine for a c code. As this file
1.283 + * has been changed to C++ "this" needs to be changed.
1.284 + */
1.285 +# define this current
1.286 + code this; /* current decoding table entry */
1.287 + code last; /* parent table entry */
1.288 + unsigned len; /* length to copy for repeats, bits to drop */
1.289 + int ret; /* return code */
1.290 + static const unsigned short order[19] = /* permutation of code lengths */
1.291 + {16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15};
1.292 +
1.293 + /* Check that the strm exists and that the state was initialized */
1.294 + if (strm == Z_NULL || strm->state == Z_NULL)
1.295 + return Z_STREAM_ERROR;
1.296 + state = (struct inflate_state FAR *)strm->state;
1.297 +
1.298 + /* Reset the state */
1.299 + strm->msg = Z_NULL;
1.300 + state->mode = TYPE;
1.301 + state->last = 0;
1.302 + state->whave = 0;
1.303 + next = strm->next_in;
1.304 + have = next != Z_NULL ? strm->avail_in : 0;
1.305 + hold = 0;
1.306 + bits = 0;
1.307 + put = state->window;
1.308 + left = state->wsize;
1.309 +
1.310 + /* Inflate until end of block marked as last */
1.311 + for (;;)
1.312 + switch (state->mode) {
1.313 + case TYPE:
1.314 + /* determine and dispatch block type */
1.315 + if (state->last) {
1.316 + BYTEBITS();
1.317 + state->mode = DONE;
1.318 + break;
1.319 + }
1.320 + NEEDBITS(3);
1.321 + state->last = BITS(1);
1.322 + DROPBITS(1);
1.323 + switch (BITS(2)) {
1.324 + case 0: /* stored block */
1.325 + Tracev((stderr, "inflate: stored block%s\n",
1.326 + state->last ? " (last)" : ""));
1.327 + state->mode = STORED;
1.328 + break;
1.329 + case 1: /* fixed block */
1.330 + fixedtables(state);
1.331 + Tracev((stderr, "inflate: fixed codes block%s\n",
1.332 + state->last ? " (last)" : ""));
1.333 + state->mode = LEN; /* decode codes */
1.334 + break;
1.335 + case 2: /* dynamic block */
1.336 + Tracev((stderr, "inflate: dynamic codes block%s\n",
1.337 + state->last ? " (last)" : ""));
1.338 + state->mode = TABLE;
1.339 + break;
1.340 + case 3:
1.341 + strm->msg = (char *)"invalid block type";
1.342 + state->mode = BAD;
1.343 + }
1.344 + DROPBITS(2);
1.345 + break;
1.346 +
1.347 + case STORED:
1.348 + /* get and verify stored block length */
1.349 + BYTEBITS(); /* go to byte boundary */
1.350 + NEEDBITS(32);
1.351 + if ((hold & 0xffff) != ((hold >> 16) ^ 0xffff)) {
1.352 + strm->msg = (char *)"invalid stored block lengths";
1.353 + state->mode = BAD;
1.354 + break;
1.355 + }
1.356 + state->length = (unsigned)hold & 0xffff;
1.357 + Tracev((stderr, "inflate: stored length %u\n",
1.358 + state->length));
1.359 + INITBITS();
1.360 +
1.361 + /* copy stored block from input to output */
1.362 + while (state->length != 0) {
1.363 + copy = state->length;
1.364 + PULL();
1.365 + ROOM();
1.366 + if (copy > have) copy = have;
1.367 + if (copy > left) copy = left;
1.368 + zmemcpy(put, next, copy);
1.369 + have -= copy;
1.370 + next += copy;
1.371 + left -= copy;
1.372 + put += copy;
1.373 + state->length -= copy;
1.374 + }
1.375 + Tracev((stderr, "inflate: stored end\n"));
1.376 + state->mode = TYPE;
1.377 + break;
1.378 +
1.379 + case TABLE:
1.380 + /* get dynamic table entries descriptor */
1.381 + NEEDBITS(14);
1.382 + state->nlen = BITS(5) + 257;
1.383 + DROPBITS(5);
1.384 + state->ndist = BITS(5) + 1;
1.385 + DROPBITS(5);
1.386 + state->ncode = BITS(4) + 4;
1.387 + DROPBITS(4);
1.388 +#ifndef PKZIP_BUG_WORKAROUND
1.389 + if (state->nlen > 286 || state->ndist > 30) {
1.390 + strm->msg = (char *)"too many length or distance symbols";
1.391 + state->mode = BAD;
1.392 + break;
1.393 + }
1.394 +#endif
1.395 + Tracev((stderr, "inflate: table sizes ok\n"));
1.396 +
1.397 + /* get code length code lengths (not a typo) */
1.398 + state->have = 0;
1.399 + while (state->have < state->ncode) {
1.400 + NEEDBITS(3);
1.401 + state->lens[order[state->have++]] = (unsigned short)BITS(3);
1.402 + DROPBITS(3);
1.403 + }
1.404 + while (state->have < 19)
1.405 + state->lens[order[state->have++]] = 0;
1.406 + state->next = state->codes;
1.407 + state->lencode = (code const FAR *)(state->next);
1.408 + state->lenbits = 7;
1.409 + ret = inflate_table(CODES, state->lens, 19, &(state->next),
1.410 + &(state->lenbits), state->work);
1.411 + if (ret) {
1.412 + strm->msg = (char *)"invalid code lengths set";
1.413 + state->mode = BAD;
1.414 + break;
1.415 + }
1.416 + Tracev((stderr, "inflate: code lengths ok\n"));
1.417 +
1.418 + /* get length and distance code code lengths */
1.419 + state->have = 0;
1.420 + while (state->have < state->nlen + state->ndist) {
1.421 + for (;;) {
1.422 + this = state->lencode[BITS(state->lenbits)];
1.423 + if ((unsigned)(this.bits) <= bits) break;
1.424 + PULLBYTE();
1.425 + }
1.426 + if (this.val < 16) {
1.427 + NEEDBITS(this.bits);
1.428 + DROPBITS(this.bits);
1.429 + state->lens[state->have++] = this.val;
1.430 + }
1.431 + else {
1.432 + if (this.val == 16) {
1.433 + NEEDBITS(this.bits + 2);
1.434 + DROPBITS(this.bits);
1.435 + if (state->have == 0) {
1.436 + strm->msg = (char *)"invalid bit length repeat";
1.437 + state->mode = BAD;
1.438 + break;
1.439 + }
1.440 + len = (unsigned)(state->lens[state->have - 1]);
1.441 + copy = 3 + BITS(2);
1.442 + DROPBITS(2);
1.443 + }
1.444 + else if (this.val == 17) {
1.445 + NEEDBITS(this.bits + 3);
1.446 + DROPBITS(this.bits);
1.447 + len = 0;
1.448 + copy = 3 + BITS(3);
1.449 + DROPBITS(3);
1.450 + }
1.451 + else {
1.452 + NEEDBITS(this.bits + 7);
1.453 + DROPBITS(this.bits);
1.454 + len = 0;
1.455 + copy = 11 + BITS(7);
1.456 + DROPBITS(7);
1.457 + }
1.458 + if (state->have + copy > state->nlen + state->ndist) {
1.459 + strm->msg = (char *)"invalid bit length repeat";
1.460 + state->mode = BAD;
1.461 + break;
1.462 + }
1.463 + while (copy--)
1.464 + state->lens[state->have++] = (unsigned short)len;
1.465 + }
1.466 + }
1.467 +
1.468 + /* handle error breaks in while */
1.469 + if (state->mode == BAD) break;
1.470 +
1.471 + /* build code tables */
1.472 + state->next = state->codes;
1.473 + state->lencode = (code const FAR *)(state->next);
1.474 + state->lenbits = 9;
1.475 + ret = inflate_table(LENS, state->lens, state->nlen, &(state->next),
1.476 + &(state->lenbits), state->work);
1.477 + if (ret) {
1.478 + strm->msg = (char *)"invalid literal/lengths set";
1.479 + state->mode = BAD;
1.480 + break;
1.481 + }
1.482 + state->distcode = (code const FAR *)(state->next);
1.483 + state->distbits = 6;
1.484 + ret = inflate_table(DISTS, state->lens + state->nlen, state->ndist,
1.485 + &(state->next), &(state->distbits), state->work);
1.486 + if (ret) {
1.487 + strm->msg = (char *)"invalid distances set";
1.488 + state->mode = BAD;
1.489 + break;
1.490 + }
1.491 + Tracev((stderr, "inflate: codes ok\n"));
1.492 + state->mode = LEN;
1.493 +
1.494 + case LEN:
1.495 + /* use inflate_fast() if we have enough input and output */
1.496 + if (have >= 6 && left >= 258) {
1.497 + RESTORE();
1.498 + if (state->whave < state->wsize)
1.499 + state->whave = state->wsize - left;
1.500 + inflate_fast(strm, state->wsize);
1.501 + LOAD();
1.502 + break;
1.503 + }
1.504 +
1.505 + /* get a literal, length, or end-of-block code */
1.506 + for (;;) {
1.507 + this = state->lencode[BITS(state->lenbits)];
1.508 + if ((unsigned)(this.bits) <= bits) break;
1.509 + PULLBYTE();
1.510 + }
1.511 + if (this.op && (this.op & 0xf0) == 0) {
1.512 + last = this;
1.513 + for (;;) {
1.514 + this = state->lencode[last.val +
1.515 + (BITS(last.bits + last.op) >> last.bits)];
1.516 + if ((unsigned)(last.bits + this.bits) <= bits) break;
1.517 + PULLBYTE();
1.518 + }
1.519 + DROPBITS(last.bits);
1.520 + }
1.521 + DROPBITS(this.bits);
1.522 + state->length = (unsigned)this.val;
1.523 +
1.524 + /* process literal */
1.525 + if (this.op == 0) {
1.526 + Tracevv((stderr, this.val >= 0x20 && this.val < 0x7f ?
1.527 + "inflate: literal '%c'\n" :
1.528 + "inflate: literal 0x%02x\n", this.val));
1.529 + ROOM();
1.530 + *put++ = (unsigned char)(state->length);
1.531 + left--;
1.532 + state->mode = LEN;
1.533 + break;
1.534 + }
1.535 +
1.536 + /* process end of block */
1.537 + if (this.op & 32) {
1.538 + Tracevv((stderr, "inflate: end of block\n"));
1.539 + state->mode = TYPE;
1.540 + break;
1.541 + }
1.542 +
1.543 + /* invalid code */
1.544 + if (this.op & 64) {
1.545 + strm->msg = (char *)"invalid literal/length code";
1.546 + state->mode = BAD;
1.547 + break;
1.548 + }
1.549 +
1.550 + /* length code -- get extra bits, if any */
1.551 + state->extra = (unsigned)(this.op) & 15;
1.552 + if (state->extra != 0) {
1.553 + NEEDBITS(state->extra);
1.554 + state->length += BITS(state->extra);
1.555 + DROPBITS(state->extra);
1.556 + }
1.557 + Tracevv((stderr, "inflate: length %u\n", state->length));
1.558 +
1.559 + /* get distance code */
1.560 + for (;;) {
1.561 + this = state->distcode[BITS(state->distbits)];
1.562 + if ((unsigned)(this.bits) <= bits) break;
1.563 + PULLBYTE();
1.564 + }
1.565 + if ((this.op & 0xf0) == 0) {
1.566 + last = this;
1.567 + for (;;) {
1.568 + this = state->distcode[last.val +
1.569 + (BITS(last.bits + last.op) >> last.bits)];
1.570 + if ((unsigned)(last.bits + this.bits) <= bits) break;
1.571 + PULLBYTE();
1.572 + }
1.573 + DROPBITS(last.bits);
1.574 + }
1.575 + DROPBITS(this.bits);
1.576 + if (this.op & 64) {
1.577 + strm->msg = (char *)"invalid distance code";
1.578 + state->mode = BAD;
1.579 + break;
1.580 + }
1.581 + state->offset = (unsigned)this.val;
1.582 +
1.583 + /* get distance extra bits, if any */
1.584 + state->extra = (unsigned)(this.op) & 15;
1.585 + if (state->extra != 0) {
1.586 + NEEDBITS(state->extra);
1.587 + state->offset += BITS(state->extra);
1.588 + DROPBITS(state->extra);
1.589 + }
1.590 + if (state->offset > state->wsize - (state->whave < state->wsize ?
1.591 + left : 0)) {
1.592 + strm->msg = (char *)"invalid distance too far back";
1.593 + state->mode = BAD;
1.594 + break;
1.595 + }
1.596 + Tracevv((stderr, "inflate: distance %u\n", state->offset));
1.597 +
1.598 + /* copy match from window to output */
1.599 + do {
1.600 + ROOM();
1.601 + copy = state->wsize - state->offset;
1.602 + if (copy < left) {
1.603 + from = put + copy;
1.604 + copy = left - copy;
1.605 + }
1.606 + else {
1.607 + from = put - state->offset;
1.608 + copy = left;
1.609 + }
1.610 + if (copy > state->length) copy = state->length;
1.611 + state->length -= copy;
1.612 + left -= copy;
1.613 + do {
1.614 + *put++ = *from++;
1.615 + } while (--copy);
1.616 + } while (state->length != 0);
1.617 + break;
1.618 +
1.619 + case DONE:
1.620 + /* inflate stream terminated properly -- write leftover output */
1.621 + ret = Z_STREAM_END;
1.622 + if (left < state->wsize) {
1.623 + if (out(out_desc, state->window, state->wsize - left))
1.624 + ret = Z_BUF_ERROR;
1.625 + }
1.626 + goto inf_leave;
1.627 +
1.628 + case BAD:
1.629 + ret = Z_DATA_ERROR;
1.630 + goto inf_leave;
1.631 +
1.632 + default: /* can't happen, but makes compilers happy */
1.633 + ret = Z_STREAM_ERROR;
1.634 + goto inf_leave;
1.635 + }
1.636 +
1.637 + /* Return unused input */
1.638 + inf_leave:
1.639 + strm->next_in = next;
1.640 + strm->avail_in = have;
1.641 + return ret;
1.642 +}
1.643 +
1.644 +
1.645 +#ifdef __SYMBIAN32__
1.646 +EXPORT_C int inflateBackEnd_r(z_streamp strm)
1.647 +#else
1.648 +int ZEXPORT inflateBackEnd(strm)
1.649 +z_streamp strm;
1.650 + #endif //__SYMBIAN32__
1.651 +{
1.652 + if (strm == Z_NULL || strm->state == Z_NULL || strm->zfree == (free_func)0)
1.653 + return Z_STREAM_ERROR;
1.654 + ZFREE(strm, strm->state);
1.655 + strm->state = Z_NULL;
1.656 + Tracev((stderr, "inflate: end\n"));
1.657 + return Z_OK;
1.658 +}