os/ossrv/compressionlibs/ziplib/src/zlib/inffast.cpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
     1 /* Portions Copyright (c) 2007-2009 Nokia Corporation and/or its subsidiary(-ies).
     2  * All rights reserved.
     3  */
     4 
     5 /* inffast.cpp -- fast decoding
     6  * Copyright (C) 1995-2004 Mark Adler
     7  * For conditions of distribution and use, see copyright notice in zlib.h
     8  */
     9 
    10 #if (__ARMCC_VERSION >= 300000)
    11 #pragma O2
    12 #endif
    13 
    14 #include "zutil.h"
    15 #include "inftrees.h"
    16 #include "inflate.h"
    17 #include "inffast.h"
    18 
    19 #ifndef ASMINF
    20 
    21 /* Allow machine dependent optimization for post-increment or pre-increment.
    22    Based on testing to date,
    23    Pre-increment preferred for:
    24    - PowerPC G3 (Adler)
    25    - MIPS R5000 (Randers-Pehrson)
    26    Post-increment preferred for:
    27    - none
    28    No measurable difference:
    29    - Pentium III (Anderson)
    30    - M68060 (Nikl)
    31  */
    32 #ifdef POSTINC
    33 #  define OFF 0
    34 #  define PUP(a) *(a)++
    35 #else
    36 #  define OFF 1
    37 #  define PUP(a) *++(a)
    38 #endif
    39 
    40 /*
    41    Decode literal, length, and distance codes and write out the resulting
    42    literal and match bytes until either not enough input or output is
    43    available, an end-of-block is encountered, or a data error is encountered.
    44    When large enough input and output buffers are supplied to inflate(), for
    45    example, a 16K input buffer and a 64K output buffer, more than 95% of the
    46    inflate execution time is spent in this routine.
    47 
    48    Entry assumptions:
    49 
    50         state->mode == LEN
    51         strm->avail_in >= 6
    52         strm->avail_out >= 258
    53         start >= strm->avail_out
    54         state->bits < 8
    55 
    56    On return, state->mode is one of:
    57 
    58         LEN -- ran out of enough output space or enough available input
    59         TYPE -- reached end of block code, inflate() to interpret next block
    60         BAD -- error in block data
    61 
    62    Notes:
    63 
    64     - The maximum input bits used by a length/distance pair is 15 bits for the
    65       length code, 5 bits for the length extra, 15 bits for the distance code,
    66       and 13 bits for the distance extra.  This totals 48 bits, or six bytes.
    67       Therefore if strm->avail_in >= 6, then there is enough input to avoid
    68       checking for available input while decoding.
    69 
    70     - The maximum bytes that a single length/distance pair can output is 258
    71       bytes, which is the maximum length that can be coded.  inflate_fast()
    72       requires strm->avail_out >= 258 for each loop to avoid checking for
    73       output space.
    74  */
    75 #ifdef __SYMBIAN32__
    76 void inflate_fast(z_streamp strm,unsigned start)
    77 #else
    78 void inflate_fast(strm, start)
    79 z_streamp strm;
    80 unsigned start;         /* inflate()'s starting value for strm->avail_out */
    81 #endif //__SYMBIAN32__
    82 {
    83     struct inflate_state FAR *state;
    84     unsigned char FAR *in;      /* local strm->next_in */
    85     unsigned char FAR *last;    /* while in < last, enough input available */
    86     unsigned char FAR *out;     /* local strm->next_out */
    87     unsigned char FAR *beg;     /* inflate()'s initial strm->next_out */
    88     unsigned char FAR *end;     /* while out < end, enough space available */
    89 #ifdef INFLATE_STRICT
    90     unsigned dmax;              /* maximum distance from zlib header */
    91 #endif
    92     unsigned wsize;             /* window size or zero if not using window */
    93     unsigned whave;             /* valid bytes in the window */
    94     unsigned write;             /* window write index */
    95     unsigned char FAR *window;  /* allocated sliding window, if wsize != 0 */
    96     unsigned long hold;         /* local strm->hold */
    97     unsigned bits;              /* local strm->bits */
    98     code const FAR *lcode;      /* local strm->lencode */
    99     code const FAR *dcode;      /* local strm->distcode */
   100     unsigned lmask;             /* mask for first level of length codes */
   101     unsigned dmask;             /* mask for first level of distance codes */
   102 /*  Need to replace "this" variable with "current" as "this" is a reserved 
   103  *  keyword in C++ which is prefectly fine for a c code. As this file
   104  *  has been changed to C++ "this" needs to be changed.
   105  */ 
   106 #   define this current 
   107     code this;                  /* retrieved table entry */
   108     unsigned op;                /* code bits, operation, extra bits, or */
   109                                 /*  window position, window bytes to copy */
   110     unsigned len;               /* match length, unused bytes */
   111     unsigned dist;              /* match distance */
   112     unsigned char FAR *from;    /* where to copy match from */
   113 
   114     /* copy state to local variables */
   115     state = (struct inflate_state FAR *)strm->state;
   116     in = strm->next_in - OFF;
   117     last = in + (strm->avail_in - 5);
   118     out = strm->next_out - OFF;
   119     beg = out - (start - strm->avail_out);
   120     end = out + (strm->avail_out - 257);
   121 #ifdef INFLATE_STRICT
   122     dmax = state->dmax;
   123 #endif
   124     wsize = state->wsize;
   125     whave = state->whave;
   126     write = state->write;
   127     window = state->window;
   128     hold = state->hold;
   129     bits = state->bits;
   130     lcode = state->lencode;
   131     dcode = state->distcode;
   132     lmask = (1U << state->lenbits) - 1;
   133     dmask = (1U << state->distbits) - 1;
   134 
   135     /* decode literals and length/distances until end-of-block or not enough
   136        input data or output space */
   137     do {
   138         if (bits < 15) {
   139             hold += (unsigned long)(PUP(in)) << bits;
   140             bits += 8;
   141             hold += (unsigned long)(PUP(in)) << bits;
   142             bits += 8;
   143         }
   144         this = lcode[hold & lmask];
   145       dolen:
   146         op = (unsigned)(this.bits);
   147         hold >>= op;
   148         bits -= op;
   149         op = (unsigned)(this.op);
   150         if (op == 0) {                          /* literal */
   151             Tracevv((stderr, this.val >= 0x20 && this.val < 0x7f ?
   152                     "inflate:         literal '%c'\n" :
   153                     "inflate:         literal 0x%02x\n", this.val));
   154             PUP(out) = (unsigned char)(this.val);
   155         }
   156         else if (op & 16) {                     /* length base */
   157             len = (unsigned)(this.val);
   158             op &= 15;                           /* number of extra bits */
   159             if (op) {
   160                 if (bits < op) {
   161                     hold += (unsigned long)(PUP(in)) << bits;
   162                     bits += 8;
   163                 }
   164                 len += (unsigned)hold & ((1U << op) - 1);
   165                 hold >>= op;
   166                 bits -= op;
   167             }
   168             Tracevv((stderr, "inflate:         length %u\n", len));
   169             if (bits < 15) {
   170                 hold += (unsigned long)(PUP(in)) << bits;
   171                 bits += 8;
   172                 hold += (unsigned long)(PUP(in)) << bits;
   173                 bits += 8;
   174             }
   175             this = dcode[hold & dmask];
   176           dodist:
   177             op = (unsigned)(this.bits);
   178             hold >>= op;
   179             bits -= op;
   180             op = (unsigned)(this.op);
   181             if (op & 16) {                      /* distance base */
   182                 dist = (unsigned)(this.val);
   183                 op &= 15;                       /* number of extra bits */
   184                 if (bits < op) {
   185                     hold += (unsigned long)(PUP(in)) << bits;
   186                     bits += 8;
   187                     if (bits < op) {
   188                         hold += (unsigned long)(PUP(in)) << bits;
   189                         bits += 8;
   190                     }
   191                 }
   192                 dist += (unsigned)hold & ((1U << op) - 1);
   193 #ifdef INFLATE_STRICT
   194                 if (dist > dmax) {
   195                     strm->msg = (char *)"invalid distance too far back";
   196                     state->mode = BAD;
   197                     break;
   198                 }
   199 #endif
   200                 hold >>= op;
   201                 bits -= op;
   202                 Tracevv((stderr, "inflate:         distance %u\n", dist));
   203                 op = (unsigned)(out - beg);     /* max distance in output */
   204                 if (dist > op) {                /* see if copy from window */
   205                     op = dist - op;             /* distance back in window */
   206                     if (op > whave) {
   207                         strm->msg = (char *)"invalid distance too far back";
   208                         state->mode = BAD;
   209                         break;
   210                     }
   211                     from = window - OFF;
   212                     if (write == 0) {           /* very common case */
   213                         from += wsize - op;
   214                         if (op < len) {         /* some from window */
   215                             len -= op;
   216                             do {
   217                                 PUP(out) = PUP(from);
   218                             } while (--op);
   219                             from = out - dist;  /* rest from output */
   220                         }
   221                     }
   222                     else if (write < op) {      /* wrap around window */
   223                         from += wsize + write - op;
   224                         op -= write;
   225                         if (op < len) {         /* some from end of window */
   226                             len -= op;
   227                             do {
   228                                 PUP(out) = PUP(from);
   229                             } while (--op);
   230                             from = window - OFF;
   231                             if (write < len) {  /* some from start of window */
   232                                 op = write;
   233                                 len -= op;
   234                                 do {
   235                                     PUP(out) = PUP(from);
   236                                 } while (--op);
   237                                 from = out - dist;      /* rest from output */
   238                             }
   239                         }
   240                     }
   241                     else {                      /* contiguous in window */
   242                         from += write - op;
   243                         if (op < len) {         /* some from window */
   244                             len -= op;
   245                             do {
   246                                 PUP(out) = PUP(from);
   247                             } while (--op);
   248                             from = out - dist;  /* rest from output */
   249                         }
   250                     }
   251                     while (len > 2) {
   252                         PUP(out) = PUP(from);
   253                         PUP(out) = PUP(from);
   254                         PUP(out) = PUP(from);
   255                         len -= 3;
   256                     }
   257                     if (len) {
   258                         PUP(out) = PUP(from);
   259                         if (len > 1)
   260                             PUP(out) = PUP(from);
   261                     }
   262                 }
   263                 else {
   264                     from = out - dist;          /* copy direct from output */
   265                     do {                        /* minimum length is three */
   266                         PUP(out) = PUP(from);
   267                         PUP(out) = PUP(from);
   268                         PUP(out) = PUP(from);
   269                         len -= 3;
   270                     } while (len > 2);
   271                     if (len) {
   272                         PUP(out) = PUP(from);
   273                         if (len > 1)
   274                             PUP(out) = PUP(from);
   275                     }
   276                 }
   277             }
   278             else if ((op & 64) == 0) {          /* 2nd level distance code */
   279                 this = dcode[this.val + (hold & ((1U << op) - 1))];
   280                 goto dodist;
   281             }
   282             else {
   283                 strm->msg = (char *)"invalid distance code";
   284                 state->mode = BAD;
   285                 break;
   286             }
   287         }
   288         else if ((op & 64) == 0) {              /* 2nd level length code */
   289             this = lcode[this.val + (hold & ((1U << op) - 1))];
   290             goto dolen;
   291         }
   292         else if (op & 32) {                     /* end-of-block */
   293             Tracevv((stderr, "inflate:         end of block\n"));
   294             state->mode = TYPE;
   295             break;
   296         }
   297         else {
   298             strm->msg = (char *)"invalid literal/length code";
   299             state->mode = BAD;
   300             break;
   301         }
   302     } while (in < last && out < end);
   303 
   304     /* return unused bytes (on entry, bits < 8, so in won't go too far back) */
   305     len = bits >> 3;
   306     in -= len;
   307     bits -= len << 3;
   308     hold &= (1U << bits) - 1;
   309 
   310     /* update state and return */
   311     strm->next_in = in + OFF;
   312     strm->next_out = out + OFF;
   313     strm->avail_in = (unsigned)(in < last ? 5 + (last - in) : 5 - (in - last));
   314     strm->avail_out = (unsigned)(out < end ?
   315                                  257 + (end - out) : 257 - (out - end));
   316     state->hold = hold;
   317     state->bits = bits;
   318     return;
   319 }
   320 
   321 /*
   322    inflate_fast() speedups that turned out slower (on a PowerPC G3 750CXe):
   323    - Using bit fields for code structure
   324    - Different op definition to avoid & for extra bits (do & for table bits)
   325    - Three separate decoding do-loops for direct, window, and write == 0
   326    - Special case for distance > 1 copies to do overlapped load and store copy
   327    - Explicit branch predictions (based on measured branch probabilities)
   328    - Deferring match copy and interspersed it with decoding subsequent codes
   329    - Swapping literal/length else
   330    - Swapping window/direct else
   331    - Larger unrolled copy loops (three is about right)
   332    - Moving len -= 3 statement into middle of loop
   333  */
   334 
   335 #endif /* !ASMINF */
   336 
   337 
   338 
   339 
   340