changeset 357:46c2115ae1c8

update jzlib to a21be20213d66eff15904d925e9b721956a01ef7
author Carl Byington <carl@five-ten-sg.com>
date Fri, 01 Aug 2014 13:34:58 -0700
parents 5e91b559b5fe
children 37f4a3b506d9
files TODO src/com/jcraft/jzlib/Adler32.java src/com/jcraft/jzlib/CRC32.java src/com/jcraft/jzlib/Checksum.java src/com/jcraft/jzlib/Deflate.java src/com/jcraft/jzlib/Deflater.java src/com/jcraft/jzlib/DeflaterOutputStream.java src/com/jcraft/jzlib/GZIPException.java src/com/jcraft/jzlib/GZIPHeader.java src/com/jcraft/jzlib/GZIPInputStream.java src/com/jcraft/jzlib/GZIPOutputStream.java src/com/jcraft/jzlib/InfBlocks.java src/com/jcraft/jzlib/InfCodes.java src/com/jcraft/jzlib/InfTree.java src/com/jcraft/jzlib/Inflate.java src/com/jcraft/jzlib/Inflater.java src/com/jcraft/jzlib/InflaterInputStream.java src/com/jcraft/jzlib/JZlib.java src/com/jcraft/jzlib/StaticTree.java src/com/jcraft/jzlib/Tree.java src/com/jcraft/jzlib/ZInputStream.java src/com/jcraft/jzlib/ZOutputStream.java src/com/jcraft/jzlib/ZStream.java src/com/jcraft/jzlib/ZStreamException.java
diffstat 24 files changed, 6223 insertions(+), 4152 deletions(-) [+]
line wrap: on
line diff
--- a/TODO	Fri Aug 01 11:57:17 2014 -0700
+++ b/TODO	Fri Aug 01 13:34:58 2014 -0700
@@ -54,6 +54,13 @@
 ==================================
 
 updated jsocks to 2011-03-19 version
+updated jzlib to a21be20213d66eff15904d925e9b721956a01ef7
+
+git clone https://github.com/ymnk/jzlib.git jzlib
+commit a21be20213d66eff15904d925e9b721956a01ef7
+Merge: 9d0fcb9 3e3dae9
+Author: Atsuhiko Yamanaka <atsuhiko.yamanaka@gmail.com>
+Date:   Tue Nov 19 19:03:18 2013 -0800
 
 ==================================
 
--- a/src/com/jcraft/jzlib/Adler32.java	Fri Aug 01 11:57:17 2014 -0700
+++ b/src/com/jcraft/jzlib/Adler32.java	Fri Aug 01 13:34:58 2014 -0700
@@ -1,6 +1,6 @@
 /* -*-mode:java; c-basic-offset:2; -*- */
 /*
-Copyright (c) 2000,2001,2002,2003 ymnk, JCraft,Inc. All rights reserved.
+Copyright (c) 2000-2011 ymnk, JCraft,Inc. All rights reserved.
 
 Redistribution and use in source and binary forms, with or without
 modification, are permitted provided that the following conditions are met:
@@ -8,8 +8,8 @@
   1. Redistributions of source code must retain the above copyright notice,
      this list of conditions and the following disclaimer.
 
-  2. Redistributions in binary form must reproduce the above copyright
-     notice, this list of conditions and the following disclaimer in
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
      the documentation and/or other materials provided with the distribution.
 
   3. The names of the authors may not be used to endorse or promote products
@@ -34,65 +34,106 @@
 
 package com.jcraft.jzlib;
 
-final class Adler32 {
+final public class Adler32 implements Checksum {
 
-    // largest prime smaller than 65536
-    static final private int BASE = 65521;
-    // NMAX is the largest n such that 255n(n+1)/2 + (n+1)(BASE-1) <= 2^32-1
-    static final private int NMAX = 5552;
+  // largest prime smaller than 65536
+  static final private int BASE=65521; 
+  // NMAX is the largest n such that 255n(n+1)/2 + (n+1)(BASE-1) <= 2^32-1
+  static final private int NMAX=5552;
+
+  private long s1=1L;
+  private long s2=0L;
 
-    long adler32(long adler, byte[] buf, int index, int len) {
-        if (buf == null) { return 1L; }
+  public void reset(long init){
+    s1=init&0xffff;
+    s2=(init>>16)&0xffff;
+  }
 
-        long s1 = adler & 0xffff;
-        long s2 = (adler >> 16) & 0xffff;
-        int k;
-
-        while (len > 0) {
-            k = len < NMAX ? len : NMAX;
-            len -= k;
+  public void reset(){
+    s1=1L;
+    s2=0L;
+  }
 
-            while (k >= 16) {
-                s1 += buf[index++] & 0xff; s2 += s1;
-                s1 += buf[index++] & 0xff; s2 += s1;
-                s1 += buf[index++] & 0xff; s2 += s1;
-                s1 += buf[index++] & 0xff; s2 += s1;
-                s1 += buf[index++] & 0xff; s2 += s1;
-                s1 += buf[index++] & 0xff; s2 += s1;
-                s1 += buf[index++] & 0xff; s2 += s1;
-                s1 += buf[index++] & 0xff; s2 += s1;
-                s1 += buf[index++] & 0xff; s2 += s1;
-                s1 += buf[index++] & 0xff; s2 += s1;
-                s1 += buf[index++] & 0xff; s2 += s1;
-                s1 += buf[index++] & 0xff; s2 += s1;
-                s1 += buf[index++] & 0xff; s2 += s1;
-                s1 += buf[index++] & 0xff; s2 += s1;
-                s1 += buf[index++] & 0xff; s2 += s1;
-                s1 += buf[index++] & 0xff; s2 += s1;
-                k -= 16;
-            }
+  public long getValue(){
+    return ((s2<<16)|s1);
+  }
+
+  public void update(byte[] buf, int index, int len){
 
-            if (k != 0) {
-                do {
-                    s1 += buf[index++] & 0xff; s2 += s1;
-                }
-                while (--k != 0);
-            }
+    if(len==1){
+      s1+=buf[index++]&0xff; s2+=s1;
+      s1%=BASE;
+      s2%=BASE;
+      return;
+    }
 
-            s1 %= BASE;
-            s2 %= BASE;
-        }
-
-        return (s2 << 16) | s1;
+    int len1 = len/NMAX;
+    int len2 = len%NMAX;
+    while(len1-->0) {
+      int k=NMAX;
+      len-=k;
+      while(k-->0){
+	s1+=buf[index++]&0xff; s2+=s1;
+      }
+      s1%=BASE;
+      s2%=BASE;
     }
 
-    /*
-    private java.util.zip.Adler32 adler=new java.util.zip.Adler32();
-    long adler32(long value, byte[] buf, int index, int len){
-      if(value==1) {adler.reset();}
-      if(buf==null) {adler.reset();}
-      else{adler.update(buf, index, len);}
-      return adler.getValue();
+    int k=len2;
+    len-=k;
+    while(k-->0){
+      s1+=buf[index++]&0xff; s2+=s1;
     }
-    */
+    s1%=BASE;
+    s2%=BASE;
+  }
+
+  public Adler32 copy(){
+    Adler32 foo = new Adler32();
+    foo.s1 = this.s1;
+    foo.s2 = this.s2;
+    return foo;
+  }
+
+  // The following logic has come from zlib.1.2.
+  static long combine(long adler1, long adler2, long len2){
+    long BASEL = (long)BASE;
+    long sum1;
+    long sum2;
+    long rem;  // unsigned int
+
+    rem = len2 % BASEL;
+    sum1 = adler1 & 0xffffL;
+    sum2 = rem * sum1;
+    sum2 %= BASEL; // MOD(sum2);
+    sum1 += (adler2 & 0xffffL) + BASEL - 1;
+    sum2 += ((adler1 >> 16) & 0xffffL) + ((adler2 >> 16) & 0xffffL) + BASEL - rem;
+    if (sum1 >= BASEL) sum1 -= BASEL;
+    if (sum1 >= BASEL) sum1 -= BASEL;
+    if (sum2 >= (BASEL << 1)) sum2 -= (BASEL << 1);
+    if (sum2 >= BASEL) sum2 -= BASEL;
+    return sum1 | (sum2 << 16);
+  }
+
+/*
+  private java.util.zip.Adler32 adler=new java.util.zip.Adler32();
+  public void update(byte[] buf, int index, int len){
+    if(buf==null) {adler.reset();}
+    else{adler.update(buf, index, len);}
+  }
+  public void reset(){
+    adler.reset();
+  }
+  public void reset(long init){
+    if(init==1L){
+      adler.reset();
+    }
+    else{
+      System.err.println("unsupported operation");
+    }
+  }
+  public long getValue(){
+    return adler.getValue();
+  }
+*/
 }
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/com/jcraft/jzlib/CRC32.java	Fri Aug 01 13:34:58 2014 -0700
@@ -0,0 +1,179 @@
+/* -*-mode:java; c-basic-offset:2; -*- */
+/*
+Copyright (c) 2011 ymnk, JCraft,Inc. All rights reserved.
+
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions are met:
+
+  1. Redistributions of source code must retain the above copyright notice,
+     this list of conditions and the following disclaimer.
+
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
+     the documentation and/or other materials provided with the distribution.
+
+  3. The names of the authors may not be used to endorse or promote products
+     derived from this software without specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES,
+INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
+FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JCRAFT,
+INC. OR ANY CONTRIBUTORS TO THIS SOFTWARE BE LIABLE FOR ANY DIRECT, INDIRECT,
+INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
+OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
+LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
+EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+/*
+ * This program is based on zlib-1.1.3, so all credit should go authors
+ * Jean-loup Gailly(jloup@gzip.org) and Mark Adler(madler@alumni.caltech.edu)
+ * and contributors of zlib.
+ */
+
+package com.jcraft.jzlib;
+
+final public class CRC32 implements Checksum {
+
+  /*
+   *  The following logic has come from RFC1952.
+   */
+  private int v = 0;
+  private static int[] crc_table = null;
+  static {
+    crc_table = new int[256];
+    for (int n = 0; n < 256; n++) {
+      int c = n;
+      for (int k = 8;  --k >= 0; ) {
+        if ((c & 1) != 0)
+	  c = 0xedb88320 ^ (c >>> 1);
+        else
+          c = c >>> 1;
+      }
+      crc_table[n] = c;
+    }
+  }
+
+  public void update (byte[] buf, int index, int len) {
+    int c = ~v;
+    while (--len >= 0)
+      c = crc_table[(c^buf[index++])&0xff]^(c >>> 8);
+    v = ~c;
+  }
+
+  public void reset(){
+    v = 0;
+  }
+
+  public void reset(long vv){
+    v = (int)(vv&0xffffffffL);
+  }
+
+  public long getValue(){
+    return (long)(v&0xffffffffL);
+  }
+
+  // The following logic has come from zlib.1.2.
+  private static final int GF2_DIM = 32;
+  static long combine(long crc1, long crc2, long len2){
+    long row;
+    long[] even = new long[GF2_DIM];
+    long[] odd = new long[GF2_DIM];
+
+    // degenerate case (also disallow negative lengths)
+    if (len2 <= 0)
+      return crc1;
+
+    // put operator for one zero bit in odd
+    odd[0] = 0xedb88320L;          // CRC-32 polynomial
+    row = 1;
+    for (int n = 1; n < GF2_DIM; n++) {
+        odd[n] = row;
+        row <<= 1;
+    }
+
+    // put operator for two zero bits in even
+    gf2_matrix_square(even, odd);
+
+    // put operator for four zero bits in odd
+    gf2_matrix_square(odd, even);
+
+    // apply len2 zeros to crc1 (first square will put the operator for one
+    // zero byte, eight zero bits, in even)
+    do {
+      // apply zeros operator for this bit of len2
+      gf2_matrix_square(even, odd);
+      if ((len2 & 1)!=0)
+        crc1 = gf2_matrix_times(even, crc1);
+      len2 >>= 1;
+
+      // if no more bits set, then done
+      if (len2 == 0)
+        break;
+
+      // another iteration of the loop with odd and even swapped
+      gf2_matrix_square(odd, even);
+      if ((len2 & 1)!=0)
+        crc1 = gf2_matrix_times(odd, crc1);
+      len2 >>= 1;
+
+      // if no more bits set, then done
+    } while (len2 != 0);
+
+    /* return combined crc */
+    crc1 ^= crc2;
+    return crc1;
+  }
+
+  private static long gf2_matrix_times(long[] mat, long vec){
+    long sum = 0;
+    int index = 0;
+    while (vec!=0) {
+      if ((vec & 1)!=0)
+        sum ^= mat[index];
+      vec >>= 1;
+      index++;
+    }
+    return sum;
+  }
+
+  static final void gf2_matrix_square(long[] square, long[] mat) {
+    for (int n = 0; n < GF2_DIM; n++)
+      square[n] = gf2_matrix_times(mat, mat[n]);
+  }
+
+  /*
+  private java.util.zip.CRC32 crc32 = new java.util.zip.CRC32();
+
+  public void update(byte[] buf, int index, int len){
+    if(buf==null) {crc32.reset();}
+    else{crc32.update(buf, index, len);}
+  }
+  public void reset(){
+    crc32.reset();
+  }
+  public void reset(long init){
+    if(init==0L){
+      crc32.reset();
+    }
+    else{
+      System.err.println("unsupported operation");
+    }
+  }
+  public long getValue(){
+    return crc32.getValue();
+  }
+*/
+  public CRC32 copy(){
+    CRC32 foo = new CRC32();
+    foo.v = this.v;
+    return foo;
+  }
+
+  public static int[] getCRC32Table(){
+    int[] tmp = new int[crc_table.length];
+    System.arraycopy(crc_table, 0, tmp, 0, tmp.length);
+    return tmp;
+  }
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/com/jcraft/jzlib/Checksum.java	Fri Aug 01 13:34:58 2014 -0700
@@ -0,0 +1,43 @@
+/* -*-mode:java; c-basic-offset:2; -*- */
+/*
+Copyright (c) 2011 ymnk, JCraft,Inc. All rights reserved.
+
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions are met:
+
+  1. Redistributions of source code must retain the above copyright notice,
+     this list of conditions and the following disclaimer.
+
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
+     the documentation and/or other materials provided with the distribution.
+
+  3. The names of the authors may not be used to endorse or promote products
+     derived from this software without specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES,
+INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
+FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JCRAFT,
+INC. OR ANY CONTRIBUTORS TO THIS SOFTWARE BE LIABLE FOR ANY DIRECT, INDIRECT,
+INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
+OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
+LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
+EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+/*
+ * This program is based on zlib-1.1.3, so all credit should go authors
+ * Jean-loup Gailly(jloup@gzip.org) and Mark Adler(madler@alumni.caltech.edu)
+ * and contributors of zlib.
+ */
+
+package com.jcraft.jzlib;
+
+interface Checksum {
+  void update(byte[] buf, int index, int len);
+  void reset();
+  void reset(long init);
+  long getValue();
+  Checksum copy();
+}
--- a/src/com/jcraft/jzlib/Deflate.java	Fri Aug 01 11:57:17 2014 -0700
+++ b/src/com/jcraft/jzlib/Deflate.java	Fri Aug 01 13:34:58 2014 -0700
@@ -1,6 +1,6 @@
 /* -*-mode:java; c-basic-offset:2; -*- */
 /*
-Copyright (c) 2000,2001,2002,2003 ymnk, JCraft,Inc. All rights reserved.
+Copyright (c) 2000-2011 ymnk, JCraft,Inc. All rights reserved.
 
 Redistribution and use in source and binary forms, with or without
 modification, are permitted provided that the following conditions are met:
@@ -8,8 +8,8 @@
   1. Redistributions of source code must retain the above copyright notice,
      this list of conditions and the following disclaimer.
 
-  2. Redistributions in binary form must reproduce the above copyright
-     notice, this list of conditions and the following disclaimer in
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
      the documentation and/or other materials provided with the distribution.
 
   3. The names of the authors may not be used to endorse or promote products
@@ -34,1608 +34,1724 @@
 
 package com.jcraft.jzlib;
 
-public
-final class Deflate {
-
-    static final private int MAX_MEM_LEVEL = 9;
+public 
+final class Deflate implements Cloneable {
 
-    static final private int Z_DEFAULT_COMPRESSION = -1;
-
-    static final private int MAX_WBITS = 15;          // 32K LZ77 window
-    static final private int DEF_MEM_LEVEL = 8;
+  static final private int MAX_MEM_LEVEL=9;
 
-    static class Config {
-        int good_length; // reduce lazy search above this match length
-        int max_lazy;    // do not perform lazy search above this match length
-        int nice_length; // quit search above this match length
-        int max_chain;
-        int func;
-        Config(int good_length, int max_lazy,
-               int nice_length, int max_chain, int func) {
-            this.good_length = good_length;
-            this.max_lazy = max_lazy;
-            this.nice_length = nice_length;
-            this.max_chain = max_chain;
-            this.func = func;
-        }
-    }
+  static final private int Z_DEFAULT_COMPRESSION=-1;
+
+  static final private int MAX_WBITS=15;            // 32K LZ77 window
+  static final private int DEF_MEM_LEVEL=8;
 
-    static final private int STORED = 0;
-    static final private int FAST = 1;
-    static final private int SLOW = 2;
-    static final private Config[] config_table;
-    static {
-        config_table = new Config[10];
-        //                         good  lazy  nice  chain
-        config_table[0] = new Config(0,    0,    0,    0, STORED);
-        config_table[1] = new Config(4,    4,    8,    4, FAST);
-        config_table[2] = new Config(4,    5,   16,    8, FAST);
-        config_table[3] = new Config(4,    6,   32,   32, FAST);
-        config_table[4] = new Config(4,    4,   16,   16, SLOW);
-        config_table[5] = new Config(8,   16,   32,   32, SLOW);
-        config_table[6] = new Config(8,   16,  128,  128, SLOW);
-        config_table[7] = new Config(8,   32,  128,  256, SLOW);
-        config_table[8] = new Config(32, 128,  258, 1024, SLOW);
-        config_table[9] = new Config(32, 258,  258, 4096, SLOW);
+  static class Config{
+    int good_length; // reduce lazy search above this match length
+    int max_lazy;    // do not perform lazy search above this match length
+    int nice_length; // quit search above this match length
+    int max_chain;
+    int func;
+    Config(int good_length, int max_lazy, 
+	   int nice_length, int max_chain, int func){
+      this.good_length=good_length;
+      this.max_lazy=max_lazy;
+      this.nice_length=nice_length;
+      this.max_chain=max_chain;
+      this.func=func;
     }
+  }
+  
+  static final private int STORED=0;
+  static final private int FAST=1;
+  static final private int SLOW=2;
+  static final private Config[] config_table;    
+  static{
+    config_table=new Config[10];
+    //                         good  lazy  nice  chain
+    config_table[0]=new Config(0,    0,    0,    0, STORED);
+    config_table[1]=new Config(4,    4,    8,    4, FAST);
+    config_table[2]=new Config(4,    5,   16,    8, FAST);
+    config_table[3]=new Config(4,    6,   32,   32, FAST);
 
-    static final private String[] z_errmsg = {
-        "need dictionary",     // Z_NEED_DICT       2
-        "stream end",          // Z_STREAM_END      1
-        "",                    // Z_OK              0
-        "file error",          // Z_ERRNO         (-1)
-        "stream error",        // Z_STREAM_ERROR  (-2)
-        "data error",          // Z_DATA_ERROR    (-3)
-        "insufficient memory", // Z_MEM_ERROR     (-4)
-        "buffer error",        // Z_BUF_ERROR     (-5)
-        "incompatible version",// Z_VERSION_ERROR (-6)
-        ""
-    };
-
-    // block not completed, need more input or more output
-    static final private int NeedMore = 0;
-
-    // block flush performed
-    static final private int BlockDone = 1;
+    config_table[4]=new Config(4,    4,   16,   16, SLOW);
+    config_table[5]=new Config(8,   16,   32,   32, SLOW);
+    config_table[6]=new Config(8,   16,  128,  128, SLOW);
+    config_table[7]=new Config(8,   32,  128,  256, SLOW);
+    config_table[8]=new Config(32, 128,  258, 1024, SLOW);
+    config_table[9]=new Config(32, 258,  258, 4096, SLOW);
+  }
 
-    // finish started, need only more output at next deflate
-    static final private int FinishStarted = 2;
-
-    // finish done, accept no more input or output
-    static final private int FinishDone = 3;
-
-    // preset dictionary flag in zlib header
-    static final private int PRESET_DICT = 0x20;
+  static final private String[] z_errmsg = {
+    "need dictionary",     // Z_NEED_DICT       2
+    "stream end",          // Z_STREAM_END      1
+    "",                    // Z_OK              0
+    "file error",          // Z_ERRNO         (-1)
+    "stream error",        // Z_STREAM_ERROR  (-2)
+    "data error",          // Z_DATA_ERROR    (-3)
+    "insufficient memory", // Z_MEM_ERROR     (-4)
+    "buffer error",        // Z_BUF_ERROR     (-5)
+    "incompatible version",// Z_VERSION_ERROR (-6)
+    ""
+  };
 
-    static final private int Z_FILTERED = 1;
-    static final private int Z_HUFFMAN_ONLY = 2;
-    static final private int Z_DEFAULT_STRATEGY = 0;
+  // block not completed, need more input or more output
+  static final private int NeedMore=0; 
+
+  // block flush performed
+  static final private int BlockDone=1; 
 
-    static final private int Z_NO_FLUSH = 0;
-    static final private int Z_PARTIAL_FLUSH = 1;
-    static final private int Z_SYNC_FLUSH = 2;
-    static final private int Z_FULL_FLUSH = 3;
-    static final private int Z_FINISH = 4;
+  // finish started, need only more output at next deflate
+  static final private int FinishStarted=2;
+
+  // finish done, accept no more input or output
+  static final private int FinishDone=3;
+
+  // preset dictionary flag in zlib header
+  static final private int PRESET_DICT=0x20;
 
-    static final private int Z_OK = 0;
-    static final private int Z_STREAM_END = 1;
-    static final private int Z_NEED_DICT = 2;
-    static final private int Z_ERRNO = -1;
-    static final private int Z_STREAM_ERROR = -2;
-    static final private int Z_DATA_ERROR = -3;
-    static final private int Z_MEM_ERROR = -4;
-    static final private int Z_BUF_ERROR = -5;
-    static final private int Z_VERSION_ERROR = -6;
+  static final private int Z_FILTERED=1;
+  static final private int Z_HUFFMAN_ONLY=2;
+  static final private int Z_DEFAULT_STRATEGY=0;
 
-    static final private int INIT_STATE = 42;
-    static final private int BUSY_STATE = 113;
-    static final private int FINISH_STATE = 666;
-
-    // The deflate compression method
-    static final private int Z_DEFLATED = 8;
-
-    static final private int STORED_BLOCK = 0;
-    static final private int STATIC_TREES = 1;
-    static final private int DYN_TREES = 2;
+  static final private int Z_NO_FLUSH=0;
+  static final private int Z_PARTIAL_FLUSH=1;
+  static final private int Z_SYNC_FLUSH=2;
+  static final private int Z_FULL_FLUSH=3;
+  static final private int Z_FINISH=4;
 
-    // The three kinds of block type
-    static final private int Z_BINARY = 0;
-    static final private int Z_ASCII = 1;
-    static final private int Z_UNKNOWN = 2;
+  static final private int Z_OK=0;
+  static final private int Z_STREAM_END=1;
+  static final private int Z_NEED_DICT=2;
+  static final private int Z_ERRNO=-1;
+  static final private int Z_STREAM_ERROR=-2;
+  static final private int Z_DATA_ERROR=-3;
+  static final private int Z_MEM_ERROR=-4;
+  static final private int Z_BUF_ERROR=-5;
+  static final private int Z_VERSION_ERROR=-6;
 
-    static final private int Buf_size = 8 * 2;
-
-    // repeat previous bit length 3-6 times (2 bits of repeat count)
-    static final private int REP_3_6 = 16;
-
-    // repeat a zero length 3-10 times  (3 bits of repeat count)
-    static final private int REPZ_3_10 = 17;
+  static final private int INIT_STATE=42;
+  static final private int BUSY_STATE=113;
+  static final private int FINISH_STATE=666;
 
-    // repeat a zero length 11-138 times  (7 bits of repeat count)
-    static final private int REPZ_11_138 = 18;
+  // The deflate compression method
+  static final private int Z_DEFLATED=8;
 
-    static final private int MIN_MATCH = 3;
-    static final private int MAX_MATCH = 258;
-    static final private int MIN_LOOKAHEAD = (MAX_MATCH + MIN_MATCH + 1);
+  static final private int STORED_BLOCK=0;
+  static final private int STATIC_TREES=1;
+  static final private int DYN_TREES=2;
 
-    static final private int MAX_BITS = 15;
-    static final private int D_CODES = 30;
-    static final private int BL_CODES = 19;
-    static final private int LENGTH_CODES = 29;
-    static final private int LITERALS = 256;
-    static final private int L_CODES = (LITERALS + 1 + LENGTH_CODES);
-    static final private int HEAP_SIZE = (2 * L_CODES + 1);
+  // The three kinds of block type
+  static final private int Z_BINARY=0;
+  static final private int Z_ASCII=1;
+  static final private int Z_UNKNOWN=2;
 
-    static final private int END_BLOCK = 256;
+  static final private int Buf_size=8*2;
 
-    ZStream strm;         // pointer back to this zlib stream
-    int status;           // as the name implies
-    byte[] pending_buf;   // output still pending
-    int pending_buf_size; // size of pending_buf
-    int pending_out;      // next pending byte to output to the stream
-    int pending;          // nb of bytes in the pending buffer
-    int noheader;         // suppress zlib header and adler32
-    byte data_type;       // UNKNOWN, BINARY or ASCII
-    byte method;          // STORED (for zip only) or DEFLATED
-    int last_flush;       // value of flush param for previous deflate call
+  // repeat previous bit length 3-6 times (2 bits of repeat count)
+  static final private int REP_3_6=16; 
+
+  // repeat a zero length 3-10 times  (3 bits of repeat count)
+  static final private int REPZ_3_10=17; 
 
-    int w_size;           // LZ77 window size (32K by default)
-    int w_bits;           // log2(w_size)  (8..16)
-    int w_mask;           // w_size - 1
+  // repeat a zero length 11-138 times  (7 bits of repeat count)
+  static final private int REPZ_11_138=18; 
 
-    byte[] window;
-    // Sliding window. Input bytes are read into the second half of the window,
-    // and move to the first half later to keep a dictionary of at least wSize
-    // bytes. With this organization, matches are limited to a distance of
-    // wSize-MAX_MATCH bytes, but this ensures that IO is always
-    // performed with a length multiple of the block size. Also, it limits
-    // the window size to 64K, which is quite useful on MSDOS.
-    // To do: use the user input buffer as sliding window.
+  static final private int MIN_MATCH=3;
+  static final private int MAX_MATCH=258;
+  static final private int MIN_LOOKAHEAD=(MAX_MATCH+MIN_MATCH+1);
 
-    int window_size;
-    // Actual size of window: 2*wSize, except when the user input buffer
-    // is directly used as sliding window.
+  static final private int MAX_BITS=15;
+  static final private int D_CODES=30;
+  static final private int BL_CODES=19;
+  static final private int LENGTH_CODES=29;
+  static final private int LITERALS=256;
+  static final private int L_CODES=(LITERALS+1+LENGTH_CODES);
+  static final private int HEAP_SIZE=(2*L_CODES+1);
 
-    short[] prev;
-    // Link to older string with same hash index. To limit the size of this
-    // array to 64K, this link is maintained only for the last 32K strings.
-    // An index in this array is thus a window index modulo 32K.
-
-    short[] head; // Heads of the hash chains or NIL.
+  static final private int END_BLOCK=256;
 
-    int ins_h;          // hash index of string to be inserted
-    int hash_size;      // number of elements in hash table
-    int hash_bits;      // log2(hash_size)
-    int hash_mask;      // hash_size-1
+  ZStream strm;        // pointer back to this zlib stream
+  int status;           // as the name implies
+  byte[] pending_buf;   // output still pending
+  int pending_buf_size; // size of pending_buf
+  int pending_out;      // next pending byte to output to the stream
+  int pending;          // nb of bytes in the pending buffer
+  int wrap = 1;
+  byte data_type;       // UNKNOWN, BINARY or ASCII
+  byte method;          // STORED (for zip only) or DEFLATED
+  int last_flush;       // value of flush param for previous deflate call
 
-    // Number of bits by which ins_h must be shifted at each input
-    // step. It must be such that after MIN_MATCH steps, the oldest
-    // byte no longer takes part in the hash key, that is:
-    // hash_shift * MIN_MATCH >= hash_bits
-    int hash_shift;
-
-    // Window position at the beginning of the current output block. Gets
-    // negative when the window is moved backwards.
-
-    int block_start;
+  int w_size;           // LZ77 window size (32K by default)
+  int w_bits;           // log2(w_size)  (8..16)
+  int w_mask;           // w_size - 1
 
-    int match_length;           // length of best match
-    int prev_match;             // previous match
-    int match_available;        // set if previous match exists
-    int strstart;               // start of string to insert
-    int match_start;            // start of matching string
-    int lookahead;              // number of valid bytes ahead in window
+  byte[] window;
+  // Sliding window. Input bytes are read into the second half of the window,
+  // and move to the first half later to keep a dictionary of at least wSize
+  // bytes. With this organization, matches are limited to a distance of
+  // wSize-MAX_MATCH bytes, but this ensures that IO is always
+  // performed with a length multiple of the block size. Also, it limits
+  // the window size to 64K, which is quite useful on MSDOS.
+  // To do: use the user input buffer as sliding window.
 
-    // Length of the best match at previous step. Matches not greater than this
-    // are discarded. This is used in the lazy match evaluation.
-    int prev_length;
+  int window_size;
+  // Actual size of window: 2*wSize, except when the user input buffer
+  // is directly used as sliding window.
 
-    // To speed up deflation, hash chains are never searched beyond this
-    // length.  A higher limit improves compression ratio but degrades the speed.
-    int max_chain_length;
-
-    // Attempt to find a better match only when the current match is strictly
-    // smaller than this value. This mechanism is used only for compression
-    // levels >= 4.
-    int max_lazy_match;
+  short[] prev;
+  // Link to older string with same hash index. To limit the size of this
+  // array to 64K, this link is maintained only for the last 32K strings.
+  // An index in this array is thus a window index modulo 32K.
 
-    // Insert new strings in the hash table only if the match length is not
-    // greater than this length. This saves time but degrades compression.
-    // max_insert_length is used only for compression levels <= 3.
+  short[] head; // Heads of the hash chains or NIL.
 
-    int level;    // compression level (1..9)
-    int strategy; // favor or force Huffman coding
-
-    // Use a faster search when the previous match is longer than this
-    int good_match;
+  int ins_h;          // hash index of string to be inserted
+  int hash_size;      // number of elements in hash table
+  int hash_bits;      // log2(hash_size)
+  int hash_mask;      // hash_size-1
 
-    // Stop searching when current match exceeds this
-    int nice_match;
-
-    short[] dyn_ltree;       // literal and length tree
-    short[] dyn_dtree;       // distance tree
-    short[] bl_tree;         // Huffman tree for bit lengths
+  // Number of bits by which ins_h must be shifted at each input
+  // step. It must be such that after MIN_MATCH steps, the oldest
+  // byte no longer takes part in the hash key, that is:
+  // hash_shift * MIN_MATCH >= hash_bits
+  int hash_shift;
 
-    Tree l_desc = new Tree(); // desc for literal tree
-    Tree d_desc = new Tree(); // desc for distance tree
-    Tree bl_desc = new Tree(); // desc for bit length tree
+  // Window position at the beginning of the current output block. Gets
+  // negative when the window is moved backwards.
 
-    // number of codes at each bit length for an optimal tree
-    short[] bl_count = new short[MAX_BITS + 1];
+  int block_start;
 
-    // heap used to build the Huffman trees
-    int[] heap = new int[2 * L_CODES + 1];
+  int match_length;           // length of best match
+  int prev_match;             // previous match
+  int match_available;        // set if previous match exists
+  int strstart;               // start of string to insert
+  int match_start;            // start of matching string
+  int lookahead;              // number of valid bytes ahead in window
 
-    int heap_len;               // number of elements in the heap
-    int heap_max;               // element of largest frequency
-    // The sons of heap[n] are heap[2*n] and heap[2*n+1]. heap[0] is not used.
-    // The same heap array is used to build all trees.
+  // Length of the best match at previous step. Matches not greater than this
+  // are discarded. This is used in the lazy match evaluation.
+  int prev_length;
 
-    // Depth of each subtree used as tie breaker for trees of equal frequency
-    byte[] depth = new byte[2 * L_CODES + 1];
-
-    int l_buf;               // index for literals or lengths */
+  // To speed up deflation, hash chains are never searched beyond this
+  // length.  A higher limit improves compression ratio but degrades the speed.
+  int max_chain_length;
 
-    // Size of match buffer for literals/lengths.  There are 4 reasons for
-    // limiting lit_bufsize to 64K:
-    //   - frequencies can be kept in 16 bit counters
-    //   - if compression is not successful for the first block, all input
-    //     data is still in the window so we can still emit a stored block even
-    //     when input comes from standard input.  (This can also be done for
-    //     all blocks if lit_bufsize is not greater than 32K.)
-    //   - if compression is not successful for a file smaller than 64K, we can
-    //     even emit a stored file instead of a stored block (saving 5 bytes).
-    //     This is applicable only for zip (not gzip or zlib).
-    //   - creating new Huffman trees less frequently may not provide fast
-    //     adaptation to changes in the input data statistics. (Take for
-    //     example a binary file with poorly compressible code followed by
-    //     a highly compressible string table.) Smaller buffer sizes give
-    //     fast adaptation but have of course the overhead of transmitting
-    //     trees more frequently.
-    //   - I can't count above 4
-    int lit_bufsize;
+  // Attempt to find a better match only when the current match is strictly
+  // smaller than this value. This mechanism is used only for compression
+  // levels >= 4.
+  int max_lazy_match;
+
+  // Insert new strings in the hash table only if the match length is not
+  // greater than this length. This saves time but degrades compression.
+  // max_insert_length is used only for compression levels <= 3.
 
-    int last_lit;      // running index in l_buf
+  int level;    // compression level (1..9)
+  int strategy; // favor or force Huffman coding
 
-    // Buffer for distances. To simplify the code, d_buf and l_buf have
-    // the same number of elements. To use different lengths, an extra flag
-    // array would be necessary.
-
-    int d_buf;         // index of pendig_buf
+  // Use a faster search when the previous match is longer than this
+  int good_match;
 
-    int opt_len;        // bit length of current block with optimal trees
-    int static_len;     // bit length of current block with static trees
-    int matches;        // number of string matches in current block
-    int last_eob_len;   // bit length of EOB code for last block
+  // Stop searching when current match exceeds this
+  int nice_match;
 
-    // Output buffer. bits are inserted starting at the bottom (least
-    // significant bits).
-    short bi_buf;
+  short[] dyn_ltree;       // literal and length tree
+  short[] dyn_dtree;       // distance tree
+  short[] bl_tree;         // Huffman tree for bit lengths
 
-    // Number of valid bits in bi_buf.  All bits above the last valid bit
-    // are always zero.
-    int bi_valid;
+  Tree l_desc=new Tree();  // desc for literal tree
+  Tree d_desc=new Tree();  // desc for distance tree
+  Tree bl_desc=new Tree(); // desc for bit length tree
 
-    Deflate() {
-        dyn_ltree = new short[HEAP_SIZE * 2];
-        dyn_dtree = new short[(2 * D_CODES + 1) * 2];  // distance tree
-        bl_tree = new short[(2 * BL_CODES + 1) * 2];  // Huffman tree for bit lengths
-    }
-
-    void lm_init() {
-        window_size = 2 * w_size;
-        head[hash_size - 1] = 0;
-
-        for (int i = 0; i < hash_size - 1; i++) {
-            head[i] = 0;
-        }
+  // number of codes at each bit length for an optimal tree
+  short[] bl_count=new short[MAX_BITS+1];
+  // working area to be used in Tree#gen_codes()
+  short[] next_code=new short[MAX_BITS+1];
 
-        // Set the default configuration parameters:
-        max_lazy_match   = Deflate.config_table[level].max_lazy;
-        good_match       = Deflate.config_table[level].good_length;
-        nice_match       = Deflate.config_table[level].nice_length;
-        max_chain_length = Deflate.config_table[level].max_chain;
-        strstart = 0;
-        block_start = 0;
-        lookahead = 0;
-        match_length = prev_length = MIN_MATCH - 1;
-        match_available = 0;
-        ins_h = 0;
-    }
+  // heap used to build the Huffman trees
+  int[] heap=new int[2*L_CODES+1];
 
-    // Initialize the tree data structures for a new zlib stream.
-    void tr_init() {
-        l_desc.dyn_tree = dyn_ltree;
-        l_desc.stat_desc = StaticTree.static_l_desc;
-        d_desc.dyn_tree = dyn_dtree;
-        d_desc.stat_desc = StaticTree.static_d_desc;
-        bl_desc.dyn_tree = bl_tree;
-        bl_desc.stat_desc = StaticTree.static_bl_desc;
-        bi_buf = 0;
-        bi_valid = 0;
-        last_eob_len = 8; // enough lookahead for inflate
-        // Initialize the first block of the first file:
-        init_block();
-    }
+  int heap_len;               // number of elements in the heap
+  int heap_max;               // element of largest frequency
+  // The sons of heap[n] are heap[2*n] and heap[2*n+1]. heap[0] is not used.
+  // The same heap array is used to build all trees.
+
+  // Depth of each subtree used as tie breaker for trees of equal frequency
+  byte[] depth=new byte[2*L_CODES+1];
+
+  byte[] l_buf;               // index for literals or lengths */
 
-    void init_block() {
-        // Initialize the trees.
-        for (int i = 0; i < L_CODES; i++) dyn_ltree[i * 2] = 0;
-
-        for (int i = 0; i < D_CODES; i++) dyn_dtree[i * 2] = 0;
-
-        for (int i = 0; i < BL_CODES; i++) bl_tree[i * 2] = 0;
+  // Size of match buffer for literals/lengths.  There are 4 reasons for
+  // limiting lit_bufsize to 64K:
+  //   - frequencies can be kept in 16 bit counters
+  //   - if compression is not successful for the first block, all input
+  //     data is still in the window so we can still emit a stored block even
+  //     when input comes from standard input.  (This can also be done for
+  //     all blocks if lit_bufsize is not greater than 32K.)
+  //   - if compression is not successful for a file smaller than 64K, we can
+  //     even emit a stored file instead of a stored block (saving 5 bytes).
+  //     This is applicable only for zip (not gzip or zlib).
+  //   - creating new Huffman trees less frequently may not provide fast
+  //     adaptation to changes in the input data statistics. (Take for
+  //     example a binary file with poorly compressible code followed by
+  //     a highly compressible string table.) Smaller buffer sizes give
+  //     fast adaptation but have of course the overhead of transmitting
+  //     trees more frequently.
+  //   - I can't count above 4
+  int lit_bufsize;
 
-        dyn_ltree[END_BLOCK * 2] = 1;
-        opt_len = static_len = 0;
-        last_lit = matches = 0;
-    }
+  int last_lit;      // running index in l_buf
 
-    // Restore the heap property by moving down the tree starting at node k,
-    // exchanging a node with the smallest of its two sons if necessary, stopping
-    // when the heap property is re-established (each father smaller than its
-    // two sons).
-    void pqdownheap(short[] tree,  // the tree to restore
-                    int k          // node to move down
-                   ) {
-        int v = heap[k];
-        int j = k << 1;  // left son of k
+  // Buffer for distances. To simplify the code, d_buf and l_buf have
+  // the same number of elements. To use different lengths, an extra flag
+  // array would be necessary.
+
+  int d_buf;         // index of pendig_buf
 
-        while (j <= heap_len) {
-            // Set j to the smallest of the two sons:
-            if (j < heap_len &&
-                    smaller(tree, heap[j + 1], heap[j], depth)) {
-                j++;
-            }
+  int opt_len;        // bit length of current block with optimal trees
+  int static_len;     // bit length of current block with static trees
+  int matches;        // number of string matches in current block
+  int last_eob_len;   // bit length of EOB code for last block
 
-            // Exit if v is smaller than both sons
-            if (smaller(tree, v, heap[j], depth)) break;
+  // Output buffer. bits are inserted starting at the bottom (least
+  // significant bits).
+  short bi_buf;
+
+  // Number of valid bits in bi_buf.  All bits above the last valid bit
+  // are always zero.
+  int bi_valid;
 
-            // Exchange v with the smallest son
-            heap[k] = heap[j];  k = j;
-            // And continue down the tree, setting j to the left son of k
-            j <<= 1;
-        }
+  GZIPHeader gheader = null;
 
-        heap[k] = v;
-    }
+  Deflate(ZStream strm){
+    this.strm=strm;
+    dyn_ltree=new short[HEAP_SIZE*2];
+    dyn_dtree=new short[(2*D_CODES+1)*2]; // distance tree
+    bl_tree=new short[(2*BL_CODES+1)*2];  // Huffman tree for bit lengths
+  }
 
-    static boolean smaller(short[] tree, int n, int m, byte[] depth) {
-        short tn2 = tree[n * 2];
-        short tm2 = tree[m * 2];
-        return (tn2 < tm2 ||
-                (tn2 == tm2 && depth[n] <= depth[m]));
+  void lm_init() {
+    window_size=2*w_size;
+
+    head[hash_size-1]=0;
+    for(int i=0; i<hash_size-1; i++){
+      head[i]=0;
     }
 
-    // Scan a literal or distance tree to determine the frequencies of the codes
-    // in the bit length tree.
-    void scan_tree(short[] tree, // the tree to be scanned
-                   int max_code // and its largest code of non zero frequency
-                  ) {
-        int n;                     // iterates over all tree elements
-        int prevlen = -1;          // last emitted length
-        int curlen;                // length of current code
-        int nextlen = tree[0 * 2 + 1]; // length of next code
-        int count = 0;             // repeat count of the current code
-        int max_count = 7;         // max repeat count
-        int min_count = 4;         // min repeat count
+    // Set the default configuration parameters:
+    max_lazy_match   = Deflate.config_table[level].max_lazy;
+    good_match       = Deflate.config_table[level].good_length;
+    nice_match       = Deflate.config_table[level].nice_length;
+    max_chain_length = Deflate.config_table[level].max_chain;
+
+    strstart = 0;
+    block_start = 0;
+    lookahead = 0;
+    match_length = prev_length = MIN_MATCH-1;
+    match_available = 0;
+    ins_h = 0;
+  }
+
+  // Initialize the tree data structures for a new zlib stream.
+  void tr_init(){
 
-        if (nextlen == 0) { max_count = 138; min_count = 3; }
+    l_desc.dyn_tree = dyn_ltree;
+    l_desc.stat_desc = StaticTree.static_l_desc;
+
+    d_desc.dyn_tree = dyn_dtree;
+    d_desc.stat_desc = StaticTree.static_d_desc;
 
-        tree[(max_code + 1) * 2 + 1] = (short)0xffff;  // guard
+    bl_desc.dyn_tree = bl_tree;
+    bl_desc.stat_desc = StaticTree.static_bl_desc;
+
+    bi_buf = 0;
+    bi_valid = 0;
+    last_eob_len = 8; // enough lookahead for inflate
+
+    // Initialize the first block of the first file:
+    init_block();
+  }
 
-        for (n = 0; n <= max_code; n++) {
-            curlen = nextlen; nextlen = tree[(n + 1) * 2 + 1];
+  void init_block(){
+    // Initialize the trees.
+    for(int i = 0; i < L_CODES; i++) dyn_ltree[i*2] = 0;
+    for(int i= 0; i < D_CODES; i++) dyn_dtree[i*2] = 0;
+    for(int i= 0; i < BL_CODES; i++) bl_tree[i*2] = 0;
+
+    dyn_ltree[END_BLOCK*2] = 1;
+    opt_len = static_len = 0;
+    last_lit = matches = 0;
+  }
 
-            if (++count < max_count && curlen == nextlen) {
-                continue;
-            }
-            else if (count < min_count) {
-                bl_tree[curlen * 2] += count;
-            }
-            else if (curlen != 0) {
-                if (curlen != prevlen) bl_tree[curlen * 2]++;
+  // Restore the heap property by moving down the tree starting at node k,
+  // exchanging a node with the smallest of its two sons if necessary, stopping
+  // when the heap property is re-established (each father smaller than its
+  // two sons).
+  void pqdownheap(short[] tree,  // the tree to restore
+		  int k          // node to move down
+		  ){
+    int v = heap[k];
+    int j = k << 1;  // left son of k
+    while (j <= heap_len) {
+      // Set j to the smallest of the two sons:
+      if (j < heap_len &&
+	  smaller(tree, heap[j+1], heap[j], depth)){
+	j++;
+      }
+      // Exit if v is smaller than both sons
+      if(smaller(tree, v, heap[j], depth)) break;
 
-                bl_tree[REP_3_6 * 2]++;
-            }
-            else if (count <= 10) {
-                bl_tree[REPZ_3_10 * 2]++;
-            }
-            else {
-                bl_tree[REPZ_11_138 * 2]++;
-            }
+      // Exchange v with the smallest son
+      heap[k]=heap[j];  k = j;
+      // And continue down the tree, setting j to the left son of k
+      j <<= 1;
+    }
+    heap[k] = v;
+  }
 
-            count = 0; prevlen = curlen;
+  static boolean smaller(short[] tree, int n, int m, byte[] depth){
+    short tn2=tree[n*2];
+    short tm2=tree[m*2];
+    return (tn2<tm2 ||
+	    (tn2==tm2 && depth[n] <= depth[m]));
+  }
 
-            if (nextlen == 0) {
-                max_count = 138; min_count = 3;
-            }
-            else if (curlen == nextlen) {
-                max_count = 6; min_count = 3;
-            }
-            else {
-                max_count = 7; min_count = 4;
-            }
-        }
-    }
+  // Scan a literal or distance tree to determine the frequencies of the codes
+  // in the bit length tree.
+  void scan_tree (short[] tree,// the tree to be scanned
+		  int max_code // and its largest code of non zero frequency
+		  ){
+    int n;                     // iterates over all tree elements
+    int prevlen = -1;          // last emitted length
+    int curlen;                // length of current code
+    int nextlen = tree[0*2+1]; // length of next code
+    int count = 0;             // repeat count of the current code
+    int max_count = 7;         // max repeat count
+    int min_count = 4;         // min repeat count
+
+    if (nextlen == 0){ max_count = 138; min_count = 3; }
+    tree[(max_code+1)*2+1] = (short)0xffff; // guard
 
-    // Construct the Huffman tree for the bit lengths and return the index in
-    // bl_order of the last bit length code to send.
-    int build_bl_tree() {
-        int max_blindex;  // index of last bit length code of non zero freq
-        // Determine the bit length frequencies for literal and distance trees
-        scan_tree(dyn_ltree, l_desc.max_code);
-        scan_tree(dyn_dtree, d_desc.max_code);
-        // Build the bit length tree:
-        bl_desc.build_tree(this);
+    for(n = 0; n <= max_code; n++) {
+      curlen = nextlen; nextlen = tree[(n+1)*2+1];
+      if(++count < max_count && curlen == nextlen) {
+	continue;
+      }
+      else if(count < min_count) {
+	bl_tree[curlen*2] += count;
+      }
+      else if(curlen != 0) {
+	if(curlen != prevlen) bl_tree[curlen*2]++;
+	bl_tree[REP_3_6*2]++;
+      }
+      else if(count <= 10) {
+	bl_tree[REPZ_3_10*2]++;
+      }
+      else{
+	bl_tree[REPZ_11_138*2]++;
+      }
+      count = 0; prevlen = curlen;
+      if(nextlen == 0) {
+	max_count = 138; min_count = 3;
+      }
+      else if(curlen == nextlen) {
+	max_count = 6; min_count = 3;
+      }
+      else{
+	max_count = 7; min_count = 4;
+      }
+    }
+  }
 
-        // opt_len now includes the length of the tree representations, except
-        // the lengths of the bit lengths codes and the 5+5+4 bits for the counts.
-        // Determine the number of bit length codes to send. The pkzip format
-        // requires that at least 4 bit length codes be sent. (appnote.txt says
-        // 3 but the actual value used is 4.)
-        for (max_blindex = BL_CODES - 1; max_blindex >= 3; max_blindex--) {
-            if (bl_tree[Tree.bl_order[max_blindex] * 2 + 1] != 0) break;
-        }
+  // Construct the Huffman tree for the bit lengths and return the index in
+  // bl_order of the last bit length code to send.
+  int build_bl_tree(){
+    int max_blindex;  // index of last bit length code of non zero freq
+
+    // Determine the bit length frequencies for literal and distance trees
+    scan_tree(dyn_ltree, l_desc.max_code);
+    scan_tree(dyn_dtree, d_desc.max_code);
 
-        // Update opt_len to include the bit length tree and counts
-        opt_len += 3 * (max_blindex + 1) + 5 + 5 + 4;
-        return max_blindex;
+    // Build the bit length tree:
+    bl_desc.build_tree(this);
+    // opt_len now includes the length of the tree representations, except
+    // the lengths of the bit lengths codes and the 5+5+4 bits for the counts.
+
+    // Determine the number of bit length codes to send. The pkzip format
+    // requires that at least 4 bit length codes be sent. (appnote.txt says
+    // 3 but the actual value used is 4.)
+    for (max_blindex = BL_CODES-1; max_blindex >= 3; max_blindex--) {
+      if (bl_tree[Tree.bl_order[max_blindex]*2+1] != 0) break;
     }
+    // Update opt_len to include the bit length tree and counts
+    opt_len += 3*(max_blindex+1) + 5+5+4;
+
+    return max_blindex;
+  }
 
 
-    // Send the header for a block using dynamic Huffman trees: the counts, the
-    // lengths of the bit length codes, the literal tree and the distance tree.
-    // IN assertion: lcodes >= 257, dcodes >= 1, blcodes >= 4.
-    void send_all_trees(int lcodes, int dcodes, int blcodes) {
-        int rank;                    // index in bl_order
-        send_bits(lcodes - 257, 5);  // not +255 as stated in appnote.txt
-        send_bits(dcodes - 1,   5);
-        send_bits(blcodes - 4,  4);  // not -3 as stated in appnote.txt
+  // Send the header for a block using dynamic Huffman trees: the counts, the
+  // lengths of the bit length codes, the literal tree and the distance tree.
+  // IN assertion: lcodes >= 257, dcodes >= 1, blcodes >= 4.
+  void send_all_trees(int lcodes, int dcodes, int blcodes){
+    int rank;                    // index in bl_order
 
-        for (rank = 0; rank < blcodes; rank++) {
-            send_bits(bl_tree[Tree.bl_order[rank] * 2 + 1], 3);
-        }
+    send_bits(lcodes-257, 5); // not +255 as stated in appnote.txt
+    send_bits(dcodes-1,   5);
+    send_bits(blcodes-4,  4); // not -3 as stated in appnote.txt
+    for (rank = 0; rank < blcodes; rank++) {
+      send_bits(bl_tree[Tree.bl_order[rank]*2+1], 3);
+    }
+    send_tree(dyn_ltree, lcodes-1); // literal tree
+    send_tree(dyn_dtree, dcodes-1); // distance tree
+  }
 
-        send_tree(dyn_ltree, lcodes - 1);  // literal tree
-        send_tree(dyn_dtree, dcodes - 1);  // distance tree
-    }
+  // Send a literal or distance tree in compressed form, using the codes in
+  // bl_tree.
+  void send_tree (short[] tree,// the tree to be sent
+		  int max_code // and its largest code of non zero frequency
+		  ){
+    int n;                     // iterates over all tree elements
+    int prevlen = -1;          // last emitted length
+    int curlen;                // length of current code
+    int nextlen = tree[0*2+1]; // length of next code
+    int count = 0;             // repeat count of the current code
+    int max_count = 7;         // max repeat count
+    int min_count = 4;         // min repeat count
+
+    if (nextlen == 0){ max_count = 138; min_count = 3; }
 
-    // Send a literal or distance tree in compressed form, using the codes in
-    // bl_tree.
-    void send_tree(short[] tree, // the tree to be sent
-                   int max_code // and its largest code of non zero frequency
-                  ) {
-        int n;                     // iterates over all tree elements
-        int prevlen = -1;          // last emitted length
-        int curlen;                // length of current code
-        int nextlen = tree[0 * 2 + 1]; // length of next code
-        int count = 0;             // repeat count of the current code
-        int max_count = 7;         // max repeat count
-        int min_count = 4;         // min repeat count
-
-        if (nextlen == 0) { max_count = 138; min_count = 3; }
+    for (n = 0; n <= max_code; n++) {
+      curlen = nextlen; nextlen = tree[(n+1)*2+1];
+      if(++count < max_count && curlen == nextlen) {
+	continue;
+      }
+      else if(count < min_count) {
+	do { send_code(curlen, bl_tree); } while (--count != 0);
+      }
+      else if(curlen != 0){
+	if(curlen != prevlen){
+	  send_code(curlen, bl_tree); count--;
+	}
+	send_code(REP_3_6, bl_tree); 
+	send_bits(count-3, 2);
+      }
+      else if(count <= 10){
+	send_code(REPZ_3_10, bl_tree); 
+	send_bits(count-3, 3);
+      }
+      else{
+	send_code(REPZ_11_138, bl_tree);
+	send_bits(count-11, 7);
+      }
+      count = 0; prevlen = curlen;
+      if(nextlen == 0){
+	max_count = 138; min_count = 3;
+      }
+      else if(curlen == nextlen){
+	max_count = 6; min_count = 3;
+      }
+      else{
+	max_count = 7; min_count = 4;
+      }
+    }
+  }
 
-        for (n = 0; n <= max_code; n++) {
-            curlen = nextlen; nextlen = tree[(n + 1) * 2 + 1];
-
-            if (++count < max_count && curlen == nextlen) {
-                continue;
-            }
-            else if (count < min_count) {
-                do { send_code(curlen, bl_tree); }
-                while (--count != 0);
-            }
-            else if (curlen != 0) {
-                if (curlen != prevlen) {
-                    send_code(curlen, bl_tree); count--;
-                }
+  // Output a byte on the stream.
+  // IN assertion: there is enough room in pending_buf.
+  final void put_byte(byte[] p, int start, int len){
+    System.arraycopy(p, start, pending_buf, pending, len);
+    pending+=len;
+  }
 
-                send_code(REP_3_6, bl_tree);
-                send_bits(count - 3, 2);
-            }
-            else if (count <= 10) {
-                send_code(REPZ_3_10, bl_tree);
-                send_bits(count - 3, 3);
-            }
-            else {
-                send_code(REPZ_11_138, bl_tree);
-                send_bits(count - 11, 7);
-            }
-
-            count = 0; prevlen = curlen;
+  final void put_byte(byte c){
+    pending_buf[pending++]=c;
+  }
+  final void put_short(int w) {
+    put_byte((byte)(w/*&0xff*/));
+    put_byte((byte)(w>>>8));
+  }
+  final void putShortMSB(int b){
+    put_byte((byte)(b>>8));
+    put_byte((byte)(b/*&0xff*/));
+  }   
 
-            if (nextlen == 0) {
-                max_count = 138; min_count = 3;
-            }
-            else if (curlen == nextlen) {
-                max_count = 6; min_count = 3;
-            }
-            else {
-                max_count = 7; min_count = 4;
-            }
-        }
+  final void send_code(int c, short[] tree){
+    int c2=c*2;
+    send_bits((tree[c2]&0xffff), (tree[c2+1]&0xffff));
+  }
+
+  void send_bits(int value, int length){
+    int len = length;
+    if (bi_valid > (int)Buf_size - len) {
+      int val = value;
+//      bi_buf |= (val << bi_valid);
+      bi_buf |= ((val << bi_valid)&0xffff);
+      put_short(bi_buf);
+      bi_buf = (short)(val >>> (Buf_size - bi_valid));
+      bi_valid += len - Buf_size;
+    } else {
+//      bi_buf |= (value) << bi_valid;
+      bi_buf |= (((value) << bi_valid)&0xffff);
+      bi_valid += len;
     }
-
-    // Output a byte on the stream.
-    // IN assertion: there is enough room in pending_buf.
-    final void put_byte(byte[] p, int start, int len) {
-        System.arraycopy(p, start, pending_buf, pending, len);
-        pending += len;
-    }
+  }
 
-    final void put_byte(byte c) {
-        pending_buf[pending++] = c;
-    }
-    final void put_short(int w) {
-        put_byte((byte)(w/*&0xff*/));
-        put_byte((byte)(w >>> 8));
-    }
-    final void putShortMSB(int b) {
-        put_byte((byte)(b >> 8));
-        put_byte((byte)(b/*&0xff*/));
-    }
+  // Send one empty static block to give enough lookahead for inflate.
+  // This takes 10 bits, of which 7 may remain in the bit buffer.
+  // The current inflate code requires 9 bits of lookahead. If the
+  // last two codes for the previous block (real code plus EOB) were coded
+  // on 5 bits or less, inflate may have only 5+3 bits of lookahead to decode
+  // the last real code. In this case we send two empty static blocks instead
+  // of one. (There are no problems if the previous block is stored or fixed.)
+  // To simplify the code, we assume the worst case of last real code encoded
+  // on one bit only.
+  void _tr_align(){
+    send_bits(STATIC_TREES<<1, 3);
+    send_code(END_BLOCK, StaticTree.static_ltree);
 
-    final void send_code(int c, short[] tree) {
-        int c2 = c * 2;
-        send_bits((tree[c2] & 0xffff), (tree[c2 + 1] & 0xffff));
-    }
+    bi_flush();
 
-    void send_bits(int value, int length) {
-        int len = length;
+    // Of the 10 bits for the empty block, we have already sent
+    // (10 - bi_valid) bits. The lookahead for the last real code (before
+    // the EOB of the previous block) was thus at least one plus the length
+    // of the EOB plus what we have just sent of the empty static block.
+    if (1 + last_eob_len + 10 - bi_valid < 9) {
+      send_bits(STATIC_TREES<<1, 3);
+      send_code(END_BLOCK, StaticTree.static_ltree);
+      bi_flush();
+    }
+    last_eob_len = 7;
+  }
+
 
-        if (bi_valid > (int)Buf_size - len) {
-            int val = value;
-//      bi_buf |= (val << bi_valid);
-            bi_buf |= ((val << bi_valid) & 0xffff);
-            put_short(bi_buf);
-            bi_buf = (short)(val >>> (Buf_size - bi_valid));
-            bi_valid += len - Buf_size;
-        }
-        else {
-//      bi_buf |= (value) << bi_valid;
-            bi_buf |= (((value) << bi_valid) & 0xffff);
-            bi_valid += len;
-        }
+  // Save the match info and tally the frequency counts. Return true if
+  // the current block must be flushed.
+  boolean _tr_tally (int dist, // distance of matched string
+		     int lc // match length-MIN_MATCH or unmatched char (if dist==0)
+		     ){
+
+    pending_buf[d_buf+last_lit*2] = (byte)(dist>>>8);
+    pending_buf[d_buf+last_lit*2+1] = (byte)dist;
+
+    l_buf[last_lit] = (byte)lc; last_lit++;
+
+    if (dist == 0) {
+      // lc is the unmatched char
+      dyn_ltree[lc*2]++;
+    } 
+    else {
+      matches++;
+      // Here, lc is the match length - MIN_MATCH
+      dist--;             // dist = match distance - 1
+      dyn_ltree[(Tree._length_code[lc]+LITERALS+1)*2]++;
+      dyn_dtree[Tree.d_code(dist)*2]++;
     }
 
-    // Send one empty static block to give enough lookahead for inflate.
-    // This takes 10 bits, of which 7 may remain in the bit buffer.
-    // The current inflate code requires 9 bits of lookahead. If the
-    // last two codes for the previous block (real code plus EOB) were coded
-    // on 5 bits or less, inflate may have only 5+3 bits of lookahead to decode
-    // the last real code. In this case we send two empty static blocks instead
-    // of one. (There are no problems if the previous block is stored or fixed.)
-    // To simplify the code, we assume the worst case of last real code encoded
-    // on one bit only.
-    void _tr_align() {
-        send_bits(STATIC_TREES << 1, 3);
-        send_code(END_BLOCK, StaticTree.static_ltree);
-        bi_flush();
-
-        // Of the 10 bits for the empty block, we have already sent
-        // (10 - bi_valid) bits. The lookahead for the last real code (before
-        // the EOB of the previous block) was thus at least one plus the length
-        // of the EOB plus what we have just sent of the empty static block.
-        if (1 + last_eob_len + 10 - bi_valid < 9) {
-            send_bits(STATIC_TREES << 1, 3);
-            send_code(END_BLOCK, StaticTree.static_ltree);
-            bi_flush();
-        }
-
-        last_eob_len = 7;
+    if ((last_lit & 0x1fff) == 0 && level > 2) {
+      // Compute an upper bound for the compressed length
+      int out_length = last_lit*8;
+      int in_length = strstart - block_start;
+      int dcode;
+      for (dcode = 0; dcode < D_CODES; dcode++) {
+	out_length += (int)dyn_dtree[dcode*2] *
+	  (5L+Tree.extra_dbits[dcode]);
+      }
+      out_length >>>= 3;
+      if ((matches < (last_lit/2)) && out_length < in_length/2) return true;
     }
 
+    return (last_lit == lit_bufsize-1);
+    // We avoid equality with lit_bufsize because of wraparound at 64K
+    // on 16 bit machines and because stored blocks are restricted to
+    // 64K-1 bytes.
+  }
 
-    // Save the match info and tally the frequency counts. Return true if
-    // the current block must be flushed.
-    boolean _tr_tally(int dist,  // distance of matched string
-                      int lc // match length-MIN_MATCH or unmatched char (if dist==0)
-                     ) {
-        pending_buf[d_buf + last_lit * 2] = (byte)(dist >>> 8);
-        pending_buf[d_buf + last_lit * 2 + 1] = (byte)dist;
-        pending_buf[l_buf + last_lit] = (byte)lc; last_lit++;
+  // Send the block data compressed using the given Huffman trees
+  void compress_block(short[] ltree, short[] dtree){
+    int  dist;      // distance of matched string
+    int lc;         // match length or unmatched char (if dist == 0)
+    int lx = 0;     // running index in l_buf
+    int code;       // the code to send
+    int extra;      // number of extra bits to send
+
+    if (last_lit != 0){
+      do{
+	dist=((pending_buf[d_buf+lx*2]<<8)&0xff00)|
+	  (pending_buf[d_buf+lx*2+1]&0xff);
+	lc=(l_buf[lx])&0xff; lx++;
 
-        if (dist == 0) {
-            // lc is the unmatched char
-            dyn_ltree[lc * 2]++;
-        }
-        else {
-            matches++;
-            // Here, lc is the match length - MIN_MATCH
-            dist--;             // dist = match distance - 1
-            dyn_ltree[(Tree._length_code[lc] + LITERALS + 1) * 2]++;
-            dyn_dtree[Tree.d_code(dist) * 2]++;
-        }
+	if(dist == 0){
+	  send_code(lc, ltree); // send a literal byte
+	} 
+	else{
+	  // Here, lc is the match length - MIN_MATCH
+	  code = Tree._length_code[lc];
 
-        if ((last_lit & 0x1fff) == 0 && level > 2) {
-            // Compute an upper bound for the compressed length
-            int out_length = last_lit * 8;
-            int in_length = strstart - block_start;
-            int dcode;
+	  send_code(code+LITERALS+1, ltree); // send the length code
+	  extra = Tree.extra_lbits[code];
+	  if(extra != 0){
+	    lc -= Tree.base_length[code];
+	    send_bits(lc, extra);       // send the extra length bits
+	  }
+	  dist--; // dist is now the match distance - 1
+	  code = Tree.d_code(dist);
 
-            for (dcode = 0; dcode < D_CODES; dcode++) {
-                out_length += (int)dyn_dtree[dcode * 2] *
-                              (5L + Tree.extra_dbits[dcode]);
-            }
-
-            out_length >>>= 3;
+	  send_code(code, dtree);       // send the distance code
+	  extra = Tree.extra_dbits[code];
+	  if (extra != 0) {
+	    dist -= Tree.base_dist[code];
+	    send_bits(dist, extra);   // send the extra distance bits
+	  }
+	} // literal or match pair ?
 
-            if ((matches < (last_lit / 2)) && out_length < in_length / 2) return true;
-        }
-
-        return (last_lit == lit_bufsize - 1);
-        // We avoid equality with lit_bufsize because of wraparound at 64K
-        // on 16 bit machines and because stored blocks are restricted to
-        // 64K-1 bytes.
+	// Check that the overlay between pending_buf and d_buf+l_buf is ok:
+      }
+      while (lx < last_lit);
     }
 
-    // Send the block data compressed using the given Huffman trees
-    void compress_block(short[] ltree, short[] dtree) {
-        int  dist;      // distance of matched string
-        int lc;         // match length or unmatched char (if dist == 0)
-        int lx = 0;     // running index in l_buf
-        int code;       // the code to send
-        int extra;      // number of extra bits to send
+    send_code(END_BLOCK, ltree);
+    last_eob_len = ltree[END_BLOCK*2+1];
+  }
 
-        if (last_lit != 0) {
-            do {
-                dist = ((pending_buf[d_buf + lx * 2] << 8) & 0xff00) |
-                       (pending_buf[d_buf + lx * 2 + 1] & 0xff);
-                lc = (pending_buf[l_buf + lx]) & 0xff; lx++;
-
-                if (dist == 0) {
-                    send_code(lc, ltree);  // send a literal byte
-                }
-                else {
-                    // Here, lc is the match length - MIN_MATCH
-                    code = Tree._length_code[lc];
-                    send_code(code + LITERALS + 1, ltree);  // send the length code
-                    extra = Tree.extra_lbits[code];
+  // Set the data type to ASCII or BINARY, using a crude approximation:
+  // binary if more than 20% of the bytes are <= 6 or >= 128, ascii otherwise.
+  // IN assertion: the fields freq of dyn_ltree are set and the total of all
+  // frequencies does not exceed 64K (to fit in an int on 16 bit machines).
+  void set_data_type(){
+    int n = 0;
+    int  ascii_freq = 0;
+    int  bin_freq = 0;
+    while(n<7){ bin_freq += dyn_ltree[n*2]; n++;}
+    while(n<128){ ascii_freq += dyn_ltree[n*2]; n++;}
+    while(n<LITERALS){ bin_freq += dyn_ltree[n*2]; n++;}
+    data_type=(byte)(bin_freq > (ascii_freq >>> 2) ? Z_BINARY : Z_ASCII);
+  }
 
-                    if (extra != 0) {
-                        lc -= Tree.base_length[code];
-                        send_bits(lc, extra);       // send the extra length bits
-                    }
-
-                    dist--; // dist is now the match distance - 1
-                    code = Tree.d_code(dist);
-                    send_code(code, dtree);       // send the distance code
-                    extra = Tree.extra_dbits[code];
+  // Flush the bit buffer, keeping at most 7 bits in it.
+  void bi_flush(){
+    if (bi_valid == 16) {
+      put_short(bi_buf);
+      bi_buf=0;
+      bi_valid=0;
+    }
+    else if (bi_valid >= 8) {
+      put_byte((byte)bi_buf);
+      bi_buf>>>=8;
+      bi_valid-=8;
+    }
+  }
 
-                    if (extra != 0) {
-                        dist -= Tree.base_dist[code];
-                        send_bits(dist, extra);   // send the extra distance bits
-                    }
-                } // literal or match pair ?
+  // Flush the bit buffer and align the output on a byte boundary
+  void bi_windup(){
+    if (bi_valid > 8) {
+      put_short(bi_buf);
+    } else if (bi_valid > 0) {
+      put_byte((byte)bi_buf);
+    }
+    bi_buf = 0;
+    bi_valid = 0;
+  }
 
-                // Check that the overlay between pending_buf and d_buf+l_buf is ok:
-            }
-            while (lx < last_lit);
-        }
+  // Copy a stored block, storing first the length and its
+  // one's complement if requested.
+  void copy_block(int buf,         // the input data
+		  int len,         // its length
+		  boolean header   // true if block header must be written
+		  ){
+    int index=0;
+    bi_windup();      // align on byte boundary
+    last_eob_len = 8; // enough lookahead for inflate
 
-        send_code(END_BLOCK, ltree);
-        last_eob_len = ltree[END_BLOCK * 2 + 1];
+    if (header) {
+      put_short((short)len);   
+      put_short((short)~len);
     }
 
-    // Set the data type to ASCII or BINARY, using a crude approximation:
-    // binary if more than 20% of the bytes are <= 6 or >= 128, ascii otherwise.
-    // IN assertion: the fields freq of dyn_ltree are set and the total of all
-    // frequencies does not exceed 64K (to fit in an int on 16 bit machines).
-    void set_data_type() {
-        int n = 0;
-        int  ascii_freq = 0;
-        int  bin_freq = 0;
+    //  while(len--!=0) {
+    //    put_byte(window[buf+index]);
+    //    index++;
+    //  }
+    put_byte(window, buf, len);
+  }
 
-        while (n < 7) { bin_freq += dyn_ltree[n * 2]; n++;}
-
-        while (n < 128) { ascii_freq += dyn_ltree[n * 2]; n++;}
-
-        while (n < LITERALS) { bin_freq += dyn_ltree[n * 2]; n++;}
+  void flush_block_only(boolean eof){
+    _tr_flush_block(block_start>=0 ? block_start : -1,
+		    strstart-block_start,
+		    eof);
+    block_start=strstart;
+    strm.flush_pending();
+  }
 
-        data_type = (byte)(bin_freq > (ascii_freq >>> 2) ? Z_BINARY : Z_ASCII);
-    }
+  // Copy without compression as much as possible from the input stream, return
+  // the current block state.
+  // This function does not insert new strings in the dictionary since
+  // uncompressible data is probably not useful. This function is used
+  // only for the level=0 compression option.
+  // NOTE: this function should be optimized to avoid extra copying from
+  // window to pending_buf.
+  int deflate_stored(int flush){
+    // Stored blocks are limited to 0xffff bytes, pending_buf is limited
+    // to pending_buf_size, and each stored block has a 5 byte header:
 
-    // Flush the bit buffer, keeping at most 7 bits in it.
-    void bi_flush() {
-        if (bi_valid == 16) {
-            put_short(bi_buf);
-            bi_buf = 0;
-            bi_valid = 0;
-        }
-        else if (bi_valid >= 8) {
-            put_byte((byte)bi_buf);
-            bi_buf >>>= 8;
-            bi_valid -= 8;
-        }
+    int max_block_size = 0xffff;
+    int max_start;
+
+    if(max_block_size > pending_buf_size - 5) {
+      max_block_size = pending_buf_size - 5;
     }
 
-    // Flush the bit buffer and align the output on a byte boundary
-    void bi_windup() {
-        if (bi_valid > 8) {
-            put_short(bi_buf);
-        }
-        else if (bi_valid > 0) {
-            put_byte((byte)bi_buf);
-        }
+    // Copy as much as possible from input to output:
+    while(true){
+      // Fill the window as much as possible:
+      if(lookahead<=1){
+	fill_window();
+	if(lookahead==0 && flush==Z_NO_FLUSH) return NeedMore;
+	if(lookahead==0) break; // flush the current block
+      }
+
+      strstart+=lookahead;
+      lookahead=0;
 
-        bi_buf = 0;
-        bi_valid = 0;
+      // Emit a stored block if pending_buf will be full:
+      max_start=block_start+max_block_size;
+      if(strstart==0|| strstart>=max_start) {
+	// strstart == 0 is possible when wraparound on 16-bit machine
+	lookahead = (int)(strstart-max_start);
+	strstart = (int)max_start;
+      
+	flush_block_only(false);
+	if(strm.avail_out==0) return NeedMore;
+
+      }
+
+      // Flush if we may have to slide, otherwise block_start may become
+      // negative and the data will be gone:
+      if(strstart-block_start >= w_size-MIN_LOOKAHEAD) {
+	flush_block_only(false);
+	if(strm.avail_out==0) return NeedMore;
+      }
     }
 
-    // Copy a stored block, storing first the length and its
-    // one's complement if requested.
-    void copy_block(int buf,         // the input data
-                    int len,         // its length
-                    boolean header   // true if block header must be written
-                   ) {
-        int index = 0;
-        bi_windup();      // align on byte boundary
-        last_eob_len = 8; // enough lookahead for inflate
+    flush_block_only(flush == Z_FINISH);
+    if(strm.avail_out==0)
+      return (flush == Z_FINISH) ? FinishStarted : NeedMore;
+
+    return flush == Z_FINISH ? FinishDone : BlockDone;
+  }
+
+  // Send a stored block
+  void _tr_stored_block(int buf,        // input block
+			int stored_len, // length of input block
+			boolean eof     // true if this is the last block for a file
+			){
+    send_bits((STORED_BLOCK<<1)+(eof?1:0), 3);  // send block type
+    copy_block(buf, stored_len, true);          // with header
+  }
+
+  // Determine the best encoding for the current block: dynamic trees, static
+  // trees or store, and output the encoded block to the zip file.
+  void _tr_flush_block(int buf,        // input block, or NULL if too old
+		       int stored_len, // length of input block
+		       boolean eof     // true if this is the last block for a file
+		       ) {
+    int opt_lenb, static_lenb;// opt_len and static_len in bytes
+    int max_blindex = 0;      // index of last bit length code of non zero freq
 
-        if (header) {
-            put_short((short)len);
-            put_short((short)~len);
-        }
+    // Build the Huffman trees unless a stored block is forced
+    if(level > 0) {
+      // Check if the file is ascii or binary
+      if(data_type == Z_UNKNOWN) set_data_type();
+
+      // Construct the literal and distance trees
+      l_desc.build_tree(this);
+
+      d_desc.build_tree(this);
+
+      // At this point, opt_len and static_len are the total bit lengths of
+      // the compressed block data, excluding the tree representations.
 
-        //  while(len--!=0) {
-        //    put_byte(window[buf+index]);
-        //    index++;
-        //  }
-        put_byte(window, buf, len);
-    }
+      // Build the bit length tree for the above two trees, and get the index
+      // in bl_order of the last bit length code to send.
+      max_blindex=build_bl_tree();
 
-    void flush_block_only(boolean eof) {
-        _tr_flush_block(block_start >= 0 ? block_start : -1,
-                        strstart - block_start,
-                        eof);
-        block_start = strstart;
-        strm.flush_pending();
+      // Determine the best encoding. Compute first the block length in bytes
+      opt_lenb=(opt_len+3+7)>>>3;
+      static_lenb=(static_len+3+7)>>>3;
+
+      if(static_lenb<=opt_lenb) opt_lenb=static_lenb;
+    }
+    else {
+      opt_lenb=static_lenb=stored_len+5; // force a stored block
     }
 
-    // Copy without compression as much as possible from the input stream, return
-    // the current block state.
-    // This function does not insert new strings in the dictionary since
-    // uncompressible data is probably not useful. This function is used
-    // only for the level=0 compression option.
-    // NOTE: this function should be optimized to avoid extra copying from
-    // window to pending_buf.
-    int deflate_stored(int flush) {
-        // Stored blocks are limited to 0xffff bytes, pending_buf is limited
-        // to pending_buf_size, and each stored block has a 5 byte header:
-        int max_block_size = 0xffff;
-        int max_start;
-
-        if (max_block_size > pending_buf_size - 5) {
-            max_block_size = pending_buf_size - 5;
-        }
-
-        // Copy as much as possible from input to output:
-        while (true) {
-            // Fill the window as much as possible:
-            if (lookahead <= 1) {
-                fill_window();
-
-                if (lookahead == 0 && flush == Z_NO_FLUSH) return NeedMore;
-
-                if (lookahead == 0) break; // flush the current block
-            }
-
-            strstart += lookahead;
-            lookahead = 0;
-            // Emit a stored block if pending_buf will be full:
-            max_start = block_start + max_block_size;
-
-            if (strstart == 0 || strstart >= max_start) {
-                // strstart == 0 is possible when wraparound on 16-bit machine
-                lookahead = (int)(strstart - max_start);
-                strstart = (int)max_start;
-                flush_block_only(false);
-
-                if (strm.avail_out == 0) return NeedMore;
-            }
-
-            // Flush if we may have to slide, otherwise block_start may become
-            // negative and the data will be gone:
-            if (strstart - block_start >= w_size - MIN_LOOKAHEAD) {
-                flush_block_only(false);
-
-                if (strm.avail_out == 0) return NeedMore;
-            }
-        }
-
-        flush_block_only(flush == Z_FINISH);
-
-        if (strm.avail_out == 0)
-            return (flush == Z_FINISH) ? FinishStarted : NeedMore;
-
-        return flush == Z_FINISH ? FinishDone : BlockDone;
+    if(stored_len+4<=opt_lenb && buf != -1){
+      // 4: two words for the lengths
+      // The test buf != NULL is only necessary if LIT_BUFSIZE > WSIZE.
+      // Otherwise we can't have processed more than WSIZE input bytes since
+      // the last block flush, because compression would have been
+      // successful. If LIT_BUFSIZE <= WSIZE, it is never too late to
+      // transform a block into a stored block.
+      _tr_stored_block(buf, stored_len, eof);
     }
-
-    // Send a stored block
-    void _tr_stored_block(int buf,        // input block
-                          int stored_len, // length of input block
-                          boolean eof     // true if this is the last block for a file
-                         ) {
-        send_bits((STORED_BLOCK << 1) + (eof ? 1 : 0), 3);  // send block type
-        copy_block(buf, stored_len, true);          // with header
+    else if(static_lenb == opt_lenb){
+      send_bits((STATIC_TREES<<1)+(eof?1:0), 3);
+      compress_block(StaticTree.static_ltree, StaticTree.static_dtree);
     }
-
-    // Determine the best encoding for the current block: dynamic trees, static
-    // trees or store, and output the encoded block to the zip file.
-    void _tr_flush_block(int buf,        // input block, or NULL if too old
-                         int stored_len, // length of input block
-                         boolean eof     // true if this is the last block for a file
-                        ) {
-        int opt_lenb, static_lenb;// opt_len and static_len in bytes
-        int max_blindex = 0;      // index of last bit length code of non zero freq
-
-        // Build the Huffman trees unless a stored block is forced
-        if (level > 0) {
-            // Check if the file is ascii or binary
-            if (data_type == Z_UNKNOWN) set_data_type();
-
-            // Construct the literal and distance trees
-            l_desc.build_tree(this);
-            d_desc.build_tree(this);
-            // At this point, opt_len and static_len are the total bit lengths of
-            // the compressed block data, excluding the tree representations.
-            // Build the bit length tree for the above two trees, and get the index
-            // in bl_order of the last bit length code to send.
-            max_blindex = build_bl_tree();
-            // Determine the best encoding. Compute first the block length in bytes
-            opt_lenb = (opt_len + 3 + 7) >>> 3;
-            static_lenb = (static_len + 3 + 7) >>> 3;
-
-            if (static_lenb <= opt_lenb) opt_lenb = static_lenb;
-        }
-        else {
-            opt_lenb = static_lenb = stored_len + 5; // force a stored block
-        }
-
-        if (stored_len + 4 <= opt_lenb && buf != -1) {
-            // 4: two words for the lengths
-            // The test buf != NULL is only necessary if LIT_BUFSIZE > WSIZE.
-            // Otherwise we can't have processed more than WSIZE input bytes since
-            // the last block flush, because compression would have been
-            // successful. If LIT_BUFSIZE <= WSIZE, it is never too late to
-            // transform a block into a stored block.
-            _tr_stored_block(buf, stored_len, eof);
-        }
-        else if (static_lenb == opt_lenb) {
-            send_bits((STATIC_TREES << 1) + (eof ? 1 : 0), 3);
-            compress_block(StaticTree.static_ltree, StaticTree.static_dtree);
-        }
-        else {
-            send_bits((DYN_TREES << 1) + (eof ? 1 : 0), 3);
-            send_all_trees(l_desc.max_code + 1, d_desc.max_code + 1, max_blindex + 1);
-            compress_block(dyn_ltree, dyn_dtree);
-        }
-
-        // The above check is made mod 2^32, for files larger than 512 MB
-        // and uLong implemented on 32 bits.
-        init_block();
-
-        if (eof) {
-            bi_windup();
-        }
+    else{
+      send_bits((DYN_TREES<<1)+(eof?1:0), 3);
+      send_all_trees(l_desc.max_code+1, d_desc.max_code+1, max_blindex+1);
+      compress_block(dyn_ltree, dyn_dtree);
     }
 
-    // Fill the window when the lookahead becomes insufficient.
-    // Updates strstart and lookahead.
-    //
-    // IN assertion: lookahead < MIN_LOOKAHEAD
-    // OUT assertions: strstart <= window_size-MIN_LOOKAHEAD
-    //    At least one byte has been read, or avail_in == 0; reads are
-    //    performed for at least two bytes (required for the zip translate_eol
-    //    option -- not supported here).
-    void fill_window() {
-        int n, m;
-        int p;
-        int more;    // Amount of free space at the end of the window.
+    // The above check is made mod 2^32, for files larger than 512 MB
+    // and uLong implemented on 32 bits.
+
+    init_block();
+
+    if(eof){
+      bi_windup();
+    }
+  }
 
-        do {
-            more = (window_size - lookahead - strstart);
+  // Fill the window when the lookahead becomes insufficient.
+  // Updates strstart and lookahead.
+  //
+  // IN assertion: lookahead < MIN_LOOKAHEAD
+  // OUT assertions: strstart <= window_size-MIN_LOOKAHEAD
+  //    At least one byte has been read, or avail_in == 0; reads are
+  //    performed for at least two bytes (required for the zip translate_eol
+  //    option -- not supported here).
+  void fill_window(){
+    int n, m;
+    int p;
+    int more;    // Amount of free space at the end of the window.
+
+    do{
+      more = (window_size-lookahead-strstart);
+
+      // Deal with !@#$% 64K limit:
+      if(more==0 && strstart==0 && lookahead==0){
+	more = w_size;
+      } 
+      else if(more==-1) {
+	// Very unlikely, but possible on 16 bit machine if strstart == 0
+	// and lookahead == 1 (input done one byte at time)
+	more--;
+
+	// If the window is almost full and there is insufficient lookahead,
+	// move the upper half to the lower one to make room in the upper half.
+      }
+      else if(strstart >= w_size+ w_size-MIN_LOOKAHEAD) {
+	System.arraycopy(window, w_size, window, 0, w_size);
+	match_start-=w_size;
+	strstart-=w_size; // we now have strstart >= MAX_DIST
+	block_start-=w_size;
 
-            // Deal with !@#$% 64K limit:
-            if (more == 0 && strstart == 0 && lookahead == 0) {
-                more = w_size;
-            }
-            else if (more == -1) {
-                // Very unlikely, but possible on 16 bit machine if strstart == 0
-                // and lookahead == 1 (input done one byte at time)
-                more--;
-                // If the window is almost full and there is insufficient lookahead,
-                // move the upper half to the lower one to make room in the upper half.
-            }
-            else if (strstart >= w_size + w_size - MIN_LOOKAHEAD) {
-                System.arraycopy(window, w_size, window, 0, w_size);
-                match_start -= w_size;
-                strstart -= w_size; // we now have strstart >= MAX_DIST
-                block_start -= w_size;
-                // Slide the hash table (could be avoided with 32 bit values
-                // at the expense of memory usage). We slide even when level == 0
-                // to keep the hash table consistent if we switch back to level > 0
-                // later. (Using level 0 permanently is not an optimal usage of
-                // zlib, so we don't care about this pathological case.)
-                n = hash_size;
-                p = n;
+	// Slide the hash table (could be avoided with 32 bit values
+	// at the expense of memory usage). We slide even when level == 0
+	// to keep the hash table consistent if we switch back to level > 0
+	// later. (Using level 0 permanently is not an optimal usage of
+	// zlib, so we don't care about this pathological case.)
+
+	n = hash_size;
+	p=n;
+	do {
+	  m = (head[--p]&0xffff);
+	  head[p]=(m>=w_size ? (short)(m-w_size) : 0);
+	}
+	while (--n != 0);
+
+	n = w_size;
+	p = n;
+	do {
+	  m = (prev[--p]&0xffff);
+	  prev[p] = (m >= w_size ? (short)(m-w_size) : 0);
+	  // If n is not on any hash chain, prev[n] is garbage but
+	  // its value will never be used.
+	}
+	while (--n!=0);
+	more += w_size;
+      }
+
+      if (strm.avail_in == 0) return;
+
+      // If there was no sliding:
+      //    strstart <= WSIZE+MAX_DIST-1 && lookahead <= MIN_LOOKAHEAD - 1 &&
+      //    more == window_size - lookahead - strstart
+      // => more >= window_size - (MIN_LOOKAHEAD-1 + WSIZE + MAX_DIST-1)
+      // => more >= window_size - 2*WSIZE + 2
+      // In the BIG_MEM or MMAP case (not yet supported),
+      //   window_size == input_size + MIN_LOOKAHEAD  &&
+      //   strstart + s->lookahead <= input_size => more >= MIN_LOOKAHEAD.
+      // Otherwise, window_size == 2*WSIZE so more >= 2.
+      // If there was sliding, more >= WSIZE. So in all cases, more >= 2.
+
+      n = strm.read_buf(window, strstart + lookahead, more);
+      lookahead += n;
+
+      // Initialize the hash value now that we have some input:
+      if(lookahead >= MIN_MATCH) {
+	ins_h = window[strstart]&0xff;
+	ins_h=(((ins_h)<<hash_shift)^(window[strstart+1]&0xff))&hash_mask;
+      }
+      // If the whole input has less than MIN_MATCH bytes, ins_h is garbage,
+      // but this is not important since only literal bytes will be emitted.
+    }
+    while (lookahead < MIN_LOOKAHEAD && strm.avail_in != 0);
+  }
 
-                do {
-                    m = (head[--p] & 0xffff);
-                    head[p] = (m >= w_size ? (short)(m - w_size) : 0);
-                }
-                while (--n != 0);
-
-                n = w_size;
-                p = n;
+  // Compress as much as possible from the input stream, return the current
+  // block state.
+  // This function does not perform lazy evaluation of matches and inserts
+  // new strings in the dictionary only for unmatched strings or for short
+  // matches. It is used only for the fast compression options.
+  int deflate_fast(int flush){
+//    short hash_head = 0; // head of the hash chain
+    int hash_head = 0; // head of the hash chain
+    boolean bflush;      // set if current block must be flushed
 
-                do {
-                    m = (prev[--p] & 0xffff);
-                    prev[p] = (m >= w_size ? (short)(m - w_size) : 0);
-                    // If n is not on any hash chain, prev[n] is garbage but
-                    // its value will never be used.
-                }
-                while (--n != 0);
+    while(true){
+      // Make sure that we always have enough lookahead, except
+      // at the end of the input file. We need MAX_MATCH bytes
+      // for the next match, plus MIN_MATCH bytes to insert the
+      // string following the next match.
+      if(lookahead < MIN_LOOKAHEAD){
+	fill_window();
+	if(lookahead < MIN_LOOKAHEAD && flush == Z_NO_FLUSH){
+	  return NeedMore;
+	}
+	if(lookahead == 0) break; // flush the current block
+      }
 
-                more += w_size;
-            }
+      // Insert the string window[strstart .. strstart+2] in the
+      // dictionary, and set hash_head to the head of the hash chain:
+      if(lookahead >= MIN_MATCH){
+	ins_h=(((ins_h)<<hash_shift)^(window[(strstart)+(MIN_MATCH-1)]&0xff))&hash_mask;
 
-            if (strm.avail_in == 0) return;
+//	prev[strstart&w_mask]=hash_head=head[ins_h];
+        hash_head=(head[ins_h]&0xffff);
+	prev[strstart&w_mask]=head[ins_h];
+	head[ins_h]=(short)strstart;
+      }
+
+      // Find the longest match, discarding those <= prev_length.
+      // At this point we have always match_length < MIN_MATCH
 
-            // If there was no sliding:
-            //    strstart <= WSIZE+MAX_DIST-1 && lookahead <= MIN_LOOKAHEAD - 1 &&
-            //    more == window_size - lookahead - strstart
-            // => more >= window_size - (MIN_LOOKAHEAD-1 + WSIZE + MAX_DIST-1)
-            // => more >= window_size - 2*WSIZE + 2
-            // In the BIG_MEM or MMAP case (not yet supported),
-            //   window_size == input_size + MIN_LOOKAHEAD  &&
-            //   strstart + s->lookahead <= input_size => more >= MIN_LOOKAHEAD.
-            // Otherwise, window_size == 2*WSIZE so more >= 2.
-            // If there was sliding, more >= WSIZE. So in all cases, more >= 2.
-            n = strm.read_buf(window, strstart + lookahead, more);
-            lookahead += n;
+      if(hash_head!=0L && 
+	 ((strstart-hash_head)&0xffff) <= w_size-MIN_LOOKAHEAD
+	 ){
+	// To simplify the code, we prevent matches with the string
+	// of window index 0 (in particular we have to avoid a match
+	// of the string with itself at the start of the input file).
+	if(strategy != Z_HUFFMAN_ONLY){
+	  match_length=longest_match (hash_head);
+	}
+	// longest_match() sets match_start
+      }
+      if(match_length>=MIN_MATCH){
+	//        check_match(strstart, match_start, match_length);
+
+	bflush=_tr_tally(strstart-match_start, match_length-MIN_MATCH);
+
+	lookahead -= match_length;
+
+	// Insert new strings in the hash table only if the match length
+	// is not too large. This saves time but degrades compression.
+	if(match_length <= max_lazy_match &&
+	   lookahead >= MIN_MATCH) {
+	  match_length--; // string at strstart already in hash table
+	  do{
+	    strstart++;
 
-            // Initialize the hash value now that we have some input:
-            if (lookahead >= MIN_MATCH) {
-                ins_h = window[strstart] & 0xff;
-                ins_h = (((ins_h) << hash_shift) ^ (window[strstart + 1] & 0xff)) & hash_mask;
-            }
+	    ins_h=((ins_h<<hash_shift)^(window[(strstart)+(MIN_MATCH-1)]&0xff))&hash_mask;
+//	    prev[strstart&w_mask]=hash_head=head[ins_h];
+	    hash_head=(head[ins_h]&0xffff);
+	    prev[strstart&w_mask]=head[ins_h];
+	    head[ins_h]=(short)strstart;
+
+	    // strstart never exceeds WSIZE-MAX_MATCH, so there are
+	    // always MIN_MATCH bytes ahead.
+	  }
+	  while (--match_length != 0);
+	  strstart++; 
+	}
+	else{
+	  strstart += match_length;
+	  match_length = 0;
+	  ins_h = window[strstart]&0xff;
 
-            // If the whole input has less than MIN_MATCH bytes, ins_h is garbage,
-            // but this is not important since only literal bytes will be emitted.
-        }
-        while (lookahead < MIN_LOOKAHEAD && strm.avail_in != 0);
+	  ins_h=(((ins_h)<<hash_shift)^(window[strstart+1]&0xff))&hash_mask;
+	  // If lookahead < MIN_MATCH, ins_h is garbage, but it does not
+	  // matter since it will be recomputed at next deflate call.
+	}
+      }
+      else {
+	// No match, output a literal byte
+
+	bflush=_tr_tally(0, window[strstart]&0xff);
+	lookahead--;
+	strstart++; 
+      }
+      if (bflush){
+
+	flush_block_only(false);
+	if(strm.avail_out==0) return NeedMore;
+      }
     }
 
-    // Compress as much as possible from the input stream, return the current
-    // block state.
-    // This function does not perform lazy evaluation of matches and inserts
-    // new strings in the dictionary only for unmatched strings or for short
-    // matches. It is used only for the fast compression options.
-    int deflate_fast(int flush) {
-//    short hash_head = 0; // head of the hash chain
-        int hash_head = 0; // head of the hash chain
-        boolean bflush;      // set if current block must be flushed
+    flush_block_only(flush == Z_FINISH);
+    if(strm.avail_out==0){
+      if(flush == Z_FINISH) return FinishStarted;
+      else return NeedMore;
+    }
+    return flush==Z_FINISH ? FinishDone : BlockDone;
+  }
+
+  // Same as above, but achieves better compression. We use a lazy
+  // evaluation for matches: a match is finally adopted only if there is
+  // no better match at the next window position.
+  int deflate_slow(int flush){
+//    short hash_head = 0;    // head of hash chain
+    int hash_head = 0;    // head of hash chain
+    boolean bflush;         // set if current block must be flushed
 
-        while (true) {
-            // Make sure that we always have enough lookahead, except
-            // at the end of the input file. We need MAX_MATCH bytes
-            // for the next match, plus MIN_MATCH bytes to insert the
-            // string following the next match.
-            if (lookahead < MIN_LOOKAHEAD) {
-                fill_window();
+    // Process the input block.
+    while(true){
+      // Make sure that we always have enough lookahead, except
+      // at the end of the input file. We need MAX_MATCH bytes
+      // for the next match, plus MIN_MATCH bytes to insert the
+      // string following the next match.
 
-                if (lookahead < MIN_LOOKAHEAD && flush == Z_NO_FLUSH) {
-                    return NeedMore;
-                }
+      if (lookahead < MIN_LOOKAHEAD) {
+	fill_window();
+	if(lookahead < MIN_LOOKAHEAD && flush == Z_NO_FLUSH) {
+	  return NeedMore;
+	}
+	if(lookahead == 0) break; // flush the current block
+      }
 
-                if (lookahead == 0) break; // flush the current block
-            }
+      // Insert the string window[strstart .. strstart+2] in the
+      // dictionary, and set hash_head to the head of the hash chain:
 
-            // Insert the string window[strstart .. strstart+2] in the
-            // dictionary, and set hash_head to the head of the hash chain:
-            if (lookahead >= MIN_MATCH) {
-                ins_h = (((ins_h) << hash_shift) ^ (window[(strstart) + (MIN_MATCH - 1)] & 0xff)) & hash_mask;
-//  prev[strstart&w_mask]=hash_head=head[ins_h];
-                hash_head = (head[ins_h] & 0xffff);
-                prev[strstart & w_mask] = head[ins_h];
-                head[ins_h] = (short)strstart;
-            }
+      if(lookahead >= MIN_MATCH) {
+	ins_h=(((ins_h)<<hash_shift)^(window[(strstart)+(MIN_MATCH-1)]&0xff)) & hash_mask;
+//	prev[strstart&w_mask]=hash_head=head[ins_h];
+	hash_head=(head[ins_h]&0xffff);
+	prev[strstart&w_mask]=head[ins_h];
+	head[ins_h]=(short)strstart;
+      }
+
+      // Find the longest match, discarding those <= prev_length.
+      prev_length = match_length; prev_match = match_start;
+      match_length = MIN_MATCH-1;
+
+      if (hash_head != 0 && prev_length < max_lazy_match &&
+	  ((strstart-hash_head)&0xffff) <= w_size-MIN_LOOKAHEAD
+	  ){
+	// To simplify the code, we prevent matches with the string
+	// of window index 0 (in particular we have to avoid a match
+	// of the string with itself at the start of the input file).
 
-            // Find the longest match, discarding those <= prev_length.
-            // At this point we have always match_length < MIN_MATCH
-            if (hash_head != 0L &&
-                    ((strstart - hash_head) & 0xffff) <= w_size - MIN_LOOKAHEAD
-               ) {
-                // To simplify the code, we prevent matches with the string
-                // of window index 0 (in particular we have to avoid a match
-                // of the string with itself at the start of the input file).
-                if (strategy != Z_HUFFMAN_ONLY) {
-                    match_length = longest_match(hash_head);
-                }
+	if(strategy != Z_HUFFMAN_ONLY) {
+	  match_length = longest_match(hash_head);
+	}
+	// longest_match() sets match_start
+
+	if (match_length <= 5 && (strategy == Z_FILTERED ||
+				  (match_length == MIN_MATCH &&
+				   strstart - match_start > 4096))) {
 
-                // longest_match() sets match_start
-            }
+	  // If prev_match is also MIN_MATCH, match_start is garbage
+	  // but we will ignore the current match anyway.
+	  match_length = MIN_MATCH-1;
+	}
+      }
+
+      // If there was a match at the previous step and the current
+      // match is not better, output the previous match:
+      if(prev_length >= MIN_MATCH && match_length <= prev_length) {
+	int max_insert = strstart + lookahead - MIN_MATCH;
+	// Do not insert strings in hash table beyond this.
+
+	//          check_match(strstart-1, prev_match, prev_length);
+
+	bflush=_tr_tally(strstart-1-prev_match, prev_length - MIN_MATCH);
 
-            if (match_length >= MIN_MATCH) {
-                //        check_match(strstart, match_start, match_length);
-                bflush = _tr_tally(strstart - match_start, match_length - MIN_MATCH);
-                lookahead -= match_length;
-
-                // Insert new strings in the hash table only if the match length
-                // is not too large. This saves time but degrades compression.
-                if (match_length <= max_lazy_match &&
-                        lookahead >= MIN_MATCH) {
-                    match_length--; // string at strstart already in hash table
+	// Insert in hash table all strings up to the end of the match.
+	// strstart-1 and strstart are already inserted. If there is not
+	// enough lookahead, the last two strings are not inserted in
+	// the hash table.
+	lookahead -= prev_length-1;
+	prev_length -= 2;
+	do{
+	  if(++strstart <= max_insert) {
+	    ins_h=(((ins_h)<<hash_shift)^(window[(strstart)+(MIN_MATCH-1)]&0xff))&hash_mask;
+	    //prev[strstart&w_mask]=hash_head=head[ins_h];
+	    hash_head=(head[ins_h]&0xffff);
+	    prev[strstart&w_mask]=head[ins_h];
+	    head[ins_h]=(short)strstart;
+	  }
+	}
+	while(--prev_length != 0);
+	match_available = 0;
+	match_length = MIN_MATCH-1;
+	strstart++;
 
-                    do {
-                        strstart++;
-                        ins_h = ((ins_h << hash_shift) ^ (window[(strstart) + (MIN_MATCH - 1)] & 0xff)) & hash_mask;
-//      prev[strstart&w_mask]=hash_head=head[ins_h];
-                        hash_head = (head[ins_h] & 0xffff);
-                        prev[strstart & w_mask] = head[ins_h];
-                        head[ins_h] = (short)strstart;
-                        // strstart never exceeds WSIZE-MAX_MATCH, so there are
-                        // always MIN_MATCH bytes ahead.
-                    }
-                    while (--match_length != 0);
+	if (bflush){
+	  flush_block_only(false);
+	  if(strm.avail_out==0) return NeedMore;
+	}
+      } else if (match_available!=0) {
+
+	// If there was no match at the previous position, output a
+	// single literal. If there was a match but the current match
+	// is longer, truncate the previous match to a single literal.
+
+	bflush=_tr_tally(0, window[strstart-1]&0xff);
+
+	if (bflush) {
+	  flush_block_only(false);
+	}
+	strstart++;
+	lookahead--;
+	if(strm.avail_out == 0) return NeedMore;
+      } else {
+	// There is no previous match to compare with, wait for
+	// the next step to decide.
 
-                    strstart++;
-                }
-                else {
-                    strstart += match_length;
-                    match_length = 0;
-                    ins_h = window[strstart] & 0xff;
-                    ins_h = (((ins_h) << hash_shift) ^ (window[strstart + 1] & 0xff)) & hash_mask;
-                    // If lookahead < MIN_MATCH, ins_h is garbage, but it does not
-                    // matter since it will be recomputed at next deflate call.
-                }
-            }
-            else {
-                // No match, output a literal byte
-                bflush = _tr_tally(0, window[strstart] & 0xff);
-                lookahead--;
-                strstart++;
-            }
+	match_available = 1;
+	strstart++;
+	lookahead--;
+      }
+    }
+
+    if(match_available!=0) {
+      bflush=_tr_tally(0, window[strstart-1]&0xff);
+      match_available = 0;
+    }
+    flush_block_only(flush == Z_FINISH);
+
+    if(strm.avail_out==0){
+      if(flush == Z_FINISH) return FinishStarted;
+      else return NeedMore;
+    }
+
+    return flush == Z_FINISH ? FinishDone : BlockDone;
+  }
 
-            if (bflush) {
-                flush_block_only(false);
+  int longest_match(int cur_match){
+    int chain_length = max_chain_length; // max hash chain length
+    int scan = strstart;                 // current string
+    int match;                           // matched string
+    int len;                             // length of current match
+    int best_len = prev_length;          // best match length so far
+    int limit = strstart>(w_size-MIN_LOOKAHEAD) ?
+      strstart-(w_size-MIN_LOOKAHEAD) : 0;
+    int nice_match=this.nice_match;
 
-                if (strm.avail_out == 0) return NeedMore;
-            }
-        }
+    // Stop when cur_match becomes <= limit. To simplify the code,
+    // we prevent matches with the string of window index 0.
 
-        flush_block_only(flush == Z_FINISH);
+    int wmask = w_mask;
+
+    int strend = strstart + MAX_MATCH;
+    byte scan_end1 = window[scan+best_len-1];
+    byte scan_end = window[scan+best_len];
 
-        if (strm.avail_out == 0) {
-            if (flush == Z_FINISH) return FinishStarted;
-            else return NeedMore;
-        }
+    // The code is optimized for HASH_BITS >= 8 and MAX_MATCH-2 multiple of 16.
+    // It is easy to get rid of this optimization if necessary.
 
-        return flush == Z_FINISH ? FinishDone : BlockDone;
+    // Do not waste too much time if we already have a good match:
+    if (prev_length >= good_match) {
+      chain_length >>= 2;
     }
 
-    // Same as above, but achieves better compression. We use a lazy
-    // evaluation for matches: a match is finally adopted only if there is
-    // no better match at the next window position.
-    int deflate_slow(int flush) {
-//    short hash_head = 0;    // head of hash chain
-        int hash_head = 0;    // head of hash chain
-        boolean bflush;         // set if current block must be flushed
+    // Do not look for matches beyond the end of the input. This is necessary
+    // to make deflate deterministic.
+    if (nice_match > lookahead) nice_match = lookahead;
+
+    do {
+      match = cur_match;
 
-        // Process the input block.
-        while (true) {
-            // Make sure that we always have enough lookahead, except
-            // at the end of the input file. We need MAX_MATCH bytes
-            // for the next match, plus MIN_MATCH bytes to insert the
-            // string following the next match.
-            if (lookahead < MIN_LOOKAHEAD) {
-                fill_window();
+      // Skip to next match if the match length cannot increase
+      // or if the match length is less than 2:
+      if (window[match+best_len]   != scan_end  ||
+	  window[match+best_len-1] != scan_end1 ||
+	  window[match]       != window[scan]     ||
+	  window[++match]     != window[scan+1])      continue;
 
-                if (lookahead < MIN_LOOKAHEAD && flush == Z_NO_FLUSH) {
-                    return NeedMore;
-                }
-
-                if (lookahead == 0) break; // flush the current block
-            }
+      // The check at best_len-1 can be removed because it will be made
+      // again later. (This heuristic is not always a win.)
+      // It is not necessary to compare scan[2] and match[2] since they
+      // are always equal when the other bytes match, given that
+      // the hash keys are equal and that HASH_BITS >= 8.
+      scan += 2; match++;
 
-            // Insert the string window[strstart .. strstart+2] in the
-            // dictionary, and set hash_head to the head of the hash chain:
-            if (lookahead >= MIN_MATCH) {
-                ins_h = (((ins_h) << hash_shift) ^ (window[(strstart) + (MIN_MATCH - 1)] & 0xff)) & hash_mask;
-//  prev[strstart&w_mask]=hash_head=head[ins_h];
-                hash_head = (head[ins_h] & 0xffff);
-                prev[strstart & w_mask] = head[ins_h];
-                head[ins_h] = (short)strstart;
-            }
-
-            // Find the longest match, discarding those <= prev_length.
-            prev_length = match_length; prev_match = match_start;
-            match_length = MIN_MATCH - 1;
+      // We check for insufficient lookahead only every 8th comparison;
+      // the 256th check will be made at strstart+258.
+      do {
+      } while (window[++scan] == window[++match] &&
+	       window[++scan] == window[++match] &&
+	       window[++scan] == window[++match] &&
+	       window[++scan] == window[++match] &&
+	       window[++scan] == window[++match] &&
+	       window[++scan] == window[++match] &&
+	       window[++scan] == window[++match] &&
+	       window[++scan] == window[++match] &&
+	       scan < strend);
 
-            if (hash_head != 0 && prev_length < max_lazy_match &&
-                    ((strstart - hash_head) & 0xffff) <= w_size - MIN_LOOKAHEAD
-               ) {
-                // To simplify the code, we prevent matches with the string
-                // of window index 0 (in particular we have to avoid a match
-                // of the string with itself at the start of the input file).
-                if (strategy != Z_HUFFMAN_ONLY) {
-                    match_length = longest_match(hash_head);
-                }
+      len = MAX_MATCH - (int)(strend - scan);
+      scan = strend - MAX_MATCH;
 
-                // longest_match() sets match_start
-                if (match_length <= 5 && (strategy == Z_FILTERED ||
-                                          (match_length == MIN_MATCH &&
-                                           strstart - match_start > 4096))) {
-                    // If prev_match is also MIN_MATCH, match_start is garbage
-                    // but we will ignore the current match anyway.
-                    match_length = MIN_MATCH - 1;
-                }
-            }
+      if(len>best_len) {
+	match_start = cur_match;
+	best_len = len;
+	if (len >= nice_match) break;
+	scan_end1  = window[scan+best_len-1];
+	scan_end   = window[scan+best_len];
+      }
 
-            // If there was a match at the previous step and the current
-            // match is not better, output the previous match:
-            if (prev_length >= MIN_MATCH && match_length <= prev_length) {
-                int max_insert = strstart + lookahead - MIN_MATCH;
-                // Do not insert strings in hash table beyond this.
-                //          check_match(strstart-1, prev_match, prev_length);
-                bflush = _tr_tally(strstart - 1 - prev_match, prev_length - MIN_MATCH);
-                // Insert in hash table all strings up to the end of the match.
-                // strstart-1 and strstart are already inserted. If there is not
-                // enough lookahead, the last two strings are not inserted in
-                // the hash table.
-                lookahead -= prev_length - 1;
-                prev_length -= 2;
+    } while ((cur_match = (prev[cur_match & wmask]&0xffff)) > limit
+	     && --chain_length != 0);
+
+    if (best_len <= lookahead) return best_len;
+    return lookahead;
+  }
 
-                do {
-                    if (++strstart <= max_insert) {
-                        ins_h = (((ins_h) << hash_shift) ^ (window[(strstart) + (MIN_MATCH - 1)] & 0xff)) & hash_mask;
-                        //prev[strstart&w_mask]=hash_head=head[ins_h];
-                        hash_head = (head[ins_h] & 0xffff);
-                        prev[strstart & w_mask] = head[ins_h];
-                        head[ins_h] = (short)strstart;
-                    }
-                }
-                while (--prev_length != 0);
-
-                match_available = 0;
-                match_length = MIN_MATCH - 1;
-                strstart++;
-
-                if (bflush) {
-                    flush_block_only(false);
+  int deflateInit(int level, int bits, int memlevel){
+    return deflateInit(level, Z_DEFLATED, bits, memlevel,
+			Z_DEFAULT_STRATEGY);
+  }
+    
+  int deflateInit(int level, int bits){
+    return deflateInit(level, Z_DEFLATED, bits, DEF_MEM_LEVEL,
+			Z_DEFAULT_STRATEGY);
+  }
+  int deflateInit(int level){
+    return deflateInit(level, MAX_WBITS);
+  }
+  private int deflateInit(int level, int method,  int windowBits,
+			  int memLevel, int strategy){
+    int wrap = 1;
+    //    byte[] my_version=ZLIB_VERSION;
 
-                    if (strm.avail_out == 0) return NeedMore;
-                }
-            }
-            else if (match_available != 0) {
-                // If there was no match at the previous position, output a
-                // single literal. If there was a match but the current match
-                // is longer, truncate the previous match to a single literal.
-                bflush = _tr_tally(0, window[strstart - 1] & 0xff);
+    //
+    //  if (version == null || version[0] != my_version[0]
+    //  || stream_size != sizeof(z_stream)) {
+    //  return Z_VERSION_ERROR;
+    //  }
 
-                if (bflush) {
-                    flush_block_only(false);
-                }
+    strm.msg = null;
 
-                strstart++;
-                lookahead--;
+    if (level == Z_DEFAULT_COMPRESSION) level = 6;
 
-                if (strm.avail_out == 0) return NeedMore;
-            }
-            else {
-                // There is no previous match to compare with, wait for
-                // the next step to decide.
-                match_available = 1;
-                strstart++;
-                lookahead--;
-            }
-        }
+    if (windowBits < 0) { // undocumented feature: suppress zlib header
+      wrap = 0;
+      windowBits = -windowBits;
+    }
+    else if(windowBits > 15){
+      wrap = 2;
+      windowBits -= 16;
+      strm.adler=new CRC32();
+    }
 
-        if (match_available != 0) {
-            bflush = _tr_tally(0, window[strstart - 1] & 0xff);
-            match_available = 0;
-        }
-
-        flush_block_only(flush == Z_FINISH);
-
-        if (strm.avail_out == 0) {
-            if (flush == Z_FINISH) return FinishStarted;
-            else return NeedMore;
-        }
-
-        return flush == Z_FINISH ? FinishDone : BlockDone;
+    if (memLevel < 1 || memLevel > MAX_MEM_LEVEL || 
+	method != Z_DEFLATED ||
+	windowBits < 9 || windowBits > 15 || level < 0 || level > 9 ||
+        strategy < 0 || strategy > Z_HUFFMAN_ONLY) {
+      return Z_STREAM_ERROR;
     }
 
-    int longest_match(int cur_match) {
-        int chain_length = max_chain_length; // max hash chain length
-        int scan = strstart;                 // current string
-        int match;                           // matched string
-        int len;                             // length of current match
-        int best_len = prev_length;          // best match length so far
-        int limit = strstart > (w_size - MIN_LOOKAHEAD) ?
-                    strstart - (w_size - MIN_LOOKAHEAD) : 0;
-        int nice_match = this.nice_match;
-        // Stop when cur_match becomes <= limit. To simplify the code,
-        // we prevent matches with the string of window index 0.
-        int wmask = w_mask;
-        int strend = strstart + MAX_MATCH;
-        byte scan_end1 = window[scan + best_len - 1];
-        byte scan_end = window[scan + best_len];
+    strm.dstate = (Deflate)this;
+
+    this.wrap = wrap;
+    w_bits = windowBits;
+    w_size = 1 << w_bits;
+    w_mask = w_size - 1;
+
+    hash_bits = memLevel + 7;
+    hash_size = 1 << hash_bits;
+    hash_mask = hash_size - 1;
+    hash_shift = ((hash_bits+MIN_MATCH-1)/MIN_MATCH);
+
+    window = new byte[w_size*2];
+    prev = new short[w_size];
+    head = new short[hash_size];
+
+    lit_bufsize = 1 << (memLevel + 6); // 16K elements by default
+
+    // We overlay pending_buf and d_buf+l_buf. This works since the average
+    // output size for (length,distance) codes is <= 24 bits.
+    pending_buf = new byte[lit_bufsize*3];
+    pending_buf_size = lit_bufsize*3;
+
+    d_buf = lit_bufsize;
+    l_buf = new byte[lit_bufsize];
+
+    this.level = level;
+
+    this.strategy = strategy;
+    this.method = (byte)method;
+
+    return deflateReset();
+  }
 
-        // The code is optimized for HASH_BITS >= 8 and MAX_MATCH-2 multiple of 16.
-        // It is easy to get rid of this optimization if necessary.
-        // Do not waste too much time if we already have a good match:
-        if (prev_length >= good_match) {
-            chain_length >>= 2;
-        }
+  int deflateReset(){
+    strm.total_in = strm.total_out = 0;
+    strm.msg = null; //
+    strm.data_type = Z_UNKNOWN;
+
+    pending = 0;
+    pending_out = 0;
+
+    if(wrap < 0){
+      wrap = -wrap;
+    }
+    status = (wrap==0) ? BUSY_STATE : INIT_STATE;
+    strm.adler.reset();
+
+    last_flush = Z_NO_FLUSH;
 
-        // Do not look for matches beyond the end of the input. This is necessary
-        // to make deflate deterministic.
-        if (nice_match > lookahead) nice_match = lookahead;
-
-        do {
-            match = cur_match;
+    tr_init();
+    lm_init();
+    return Z_OK;
+  }
 
-            // Skip to next match if the match length cannot increase
-            // or if the match length is less than 2:
-            if (window[match + best_len]   != scan_end  ||
-                    window[match + best_len - 1] != scan_end1 ||
-                    window[match]       != window[scan]     ||
-                    window[++match]     != window[scan + 1])      continue;
+  int deflateEnd(){
+    if(status!=INIT_STATE && status!=BUSY_STATE && status!=FINISH_STATE){
+      return Z_STREAM_ERROR;
+    }
+    // Deallocate in reverse order of allocations:
+    pending_buf=null;
+    l_buf=null;
+    head=null;
+    prev=null;
+    window=null;
+    // free
+    // dstate=null;
+    return status == BUSY_STATE ? Z_DATA_ERROR : Z_OK;
+  }
+
+  int deflateParams(int _level, int _strategy){
+    int err=Z_OK;
 
-            // The check at best_len-1 can be removed because it will be made
-            // again later. (This heuristic is not always a win.)
-            // It is not necessary to compare scan[2] and match[2] since they
-            // are always equal when the other bytes match, given that
-            // the hash keys are equal and that HASH_BITS >= 8.
-            scan += 2; match++;
+    if(_level == Z_DEFAULT_COMPRESSION){
+      _level = 6;
+    }
+    if(_level < 0 || _level > 9 || 
+       _strategy < 0 || _strategy > Z_HUFFMAN_ONLY) {
+      return Z_STREAM_ERROR;
+    }
+
+    if(config_table[level].func!=config_table[_level].func &&
+       strm.total_in != 0) {
+      // Flush the last buffer:
+      err = strm.deflate(Z_PARTIAL_FLUSH);
+    }
+
+    if(level != _level) {
+      level = _level;
+      max_lazy_match   = config_table[level].max_lazy;
+      good_match       = config_table[level].good_length;
+      nice_match       = config_table[level].nice_length;
+      max_chain_length = config_table[level].max_chain;
+    }
+    strategy = _strategy;
+    return err;
+  }
+
+  int deflateSetDictionary (byte[] dictionary, int dictLength){
+    int length = dictLength;
+    int index=0;
+
+    if(dictionary == null || status != INIT_STATE)
+      return Z_STREAM_ERROR;
+
+    strm.adler.update(dictionary, 0, dictLength);
 
-            // We check for insufficient lookahead only every 8th comparison;
-            // the 256th check will be made at strstart+258.
-            do {
-            }
-            while (window[++scan] == window[++match] &&
-                    window[++scan] == window[++match] &&
-                    window[++scan] == window[++match] &&
-                    window[++scan] == window[++match] &&
-                    window[++scan] == window[++match] &&
-                    window[++scan] == window[++match] &&
-                    window[++scan] == window[++match] &&
-                    window[++scan] == window[++match] &&
-                    scan < strend);
+    if(length < MIN_MATCH) return Z_OK;
+    if(length > w_size-MIN_LOOKAHEAD){
+      length = w_size-MIN_LOOKAHEAD;
+      index=dictLength-length; // use the tail of the dictionary
+    }
+    System.arraycopy(dictionary, index, window, 0, length);
+    strstart = length;
+    block_start = length;
 
-            len = MAX_MATCH - (int)(strend - scan);
-            scan = strend - MAX_MATCH;
+    // Insert all strings in the hash table (except for the last two bytes).
+    // s->lookahead stays null, so s->ins_h will be recomputed at the next
+    // call of fill_window.
+
+    ins_h = window[0]&0xff;
+    ins_h=(((ins_h)<<hash_shift)^(window[1]&0xff))&hash_mask;
 
-            if (len > best_len) {
-                match_start = cur_match;
-                best_len = len;
+    for(int n=0; n<=length-MIN_MATCH; n++){
+      ins_h=(((ins_h)<<hash_shift)^(window[(n)+(MIN_MATCH-1)]&0xff))&hash_mask;
+      prev[n&w_mask]=head[ins_h];
+      head[ins_h]=(short)n;
+    }
+    return Z_OK;
+  }
 
-                if (len >= nice_match) break;
+  int deflate(int flush){
+    int old_flush;
 
-                scan_end1  = window[scan + best_len - 1];
-                scan_end   = window[scan + best_len];
-            }
-        }
-        while ((cur_match = (prev[cur_match & wmask] & 0xffff)) > limit
-                && --chain_length != 0);
+    if(flush>Z_FINISH || flush<0){
+      return Z_STREAM_ERROR;
+    }
 
-        if (best_len <= lookahead) return best_len;
-
-        return lookahead;
+    if(strm.next_out == null ||
+       (strm.next_in == null && strm.avail_in != 0) ||
+       (status == FINISH_STATE && flush != Z_FINISH)) {
+      strm.msg=z_errmsg[Z_NEED_DICT-(Z_STREAM_ERROR)];
+      return Z_STREAM_ERROR;
+    }
+    if(strm.avail_out == 0){
+      strm.msg=z_errmsg[Z_NEED_DICT-(Z_BUF_ERROR)];
+      return Z_BUF_ERROR;
     }
 
-    int deflateInit(ZStream strm, int level, int bits) {
-        return deflateInit2(strm, level, Z_DEFLATED, bits, DEF_MEM_LEVEL,
-                            Z_DEFAULT_STRATEGY);
-    }
-    int deflateInit(ZStream strm, int level) {
-        return deflateInit(strm, level, MAX_WBITS);
-    }
-    int deflateInit2(ZStream strm, int level, int method,  int windowBits,
-                     int memLevel, int strategy) {
-        int noheader = 0;
-        //    byte[] my_version=ZLIB_VERSION;
-        //
-        //  if (version == null || version[0] != my_version[0]
-        //  || stream_size != sizeof(z_stream)) {
-        //  return Z_VERSION_ERROR;
-        //  }
-        strm.msg = null;
+    old_flush = last_flush;
+    last_flush = flush;
 
-        if (level == Z_DEFAULT_COMPRESSION) level = 6;
-
-        if (windowBits < 0) { // undocumented feature: suppress zlib header
-            noheader = 1;
-            windowBits = -windowBits;
-        }
+    // Write the zlib header
+    if(status == INIT_STATE) {
+      if(wrap == 2){
+        getGZIPHeader().put(this);
+        status=BUSY_STATE;
+        strm.adler.reset();
+      }
+      else{
+      int header = (Z_DEFLATED+((w_bits-8)<<4))<<8;
+      int level_flags=((level-1)&0xff)>>1;
 
-        if (memLevel < 1 || memLevel > MAX_MEM_LEVEL ||
-                method != Z_DEFLATED ||
-                windowBits < 9 || windowBits > 15 || level < 0 || level > 9 ||
-                strategy < 0 || strategy > Z_HUFFMAN_ONLY) {
-            return Z_STREAM_ERROR;
-        }
+      if(level_flags>3) level_flags=3;
+      header |= (level_flags<<6);
+      if(strstart!=0) header |= PRESET_DICT;
+      header+=31-(header % 31);
+
+      status=BUSY_STATE;
+      putShortMSB(header);
+
 
-        strm.dstate = (Deflate)this;
-        this.noheader = noheader;
-        w_bits = windowBits;
-        w_size = 1 << w_bits;
-        w_mask = w_size - 1;
-        hash_bits = memLevel + 7;
-        hash_size = 1 << hash_bits;
-        hash_mask = hash_size - 1;
-        hash_shift = ((hash_bits + MIN_MATCH - 1) / MIN_MATCH);
-        window = new byte[w_size * 2];
-        prev = new short[w_size];
-        head = new short[hash_size];
-        lit_bufsize = 1 << (memLevel + 6); // 16K elements by default
-        // We overlay pending_buf and d_buf+l_buf. This works since the average
-        // output size for (length,distance) codes is <= 24 bits.
-        pending_buf = new byte[lit_bufsize * 4];
-        pending_buf_size = lit_bufsize * 4;
-        d_buf = lit_bufsize / 2;
-        l_buf = (1 + 2) * lit_bufsize;
-        this.level = level;
-//System.out.println("level="+level);
-        this.strategy = strategy;
-        this.method = (byte)method;
-        return deflateReset(strm);
+      // Save the adler32 of the preset dictionary:
+      if(strstart!=0){
+        long adler=strm.adler.getValue();
+        putShortMSB((int)(adler>>>16));
+        putShortMSB((int)(adler&0xffff));
+      }
+      strm.adler.reset();
+      }
     }
 
-    int deflateReset(ZStream strm) {
-        strm.total_in = strm.total_out = 0;
-        strm.msg = null; //
-        strm.data_type = Z_UNKNOWN;
-        pending = 0;
-        pending_out = 0;
+    // Flush as much pending output as possible
+    if(pending != 0) {
+      strm.flush_pending();
+      if(strm.avail_out == 0) {
+	// Since avail_out is 0, deflate will be called again with
+	// more output space, but possibly with both pending and
+	// avail_in equal to zero. There won't be anything to do,
+	// but this is not an error situation so make sure we
+	// return OK instead of BUF_ERROR at next call of deflate:
+	last_flush = -1;
+	return Z_OK;
+      }
 
-        if (noheader < 0) {
-            noheader = 0; // was set to -1 by deflate(..., Z_FINISH);
-        }
-
-        status = (noheader != 0) ? BUSY_STATE : INIT_STATE;
-        strm.adler = strm._adler.adler32(0, null, 0, 0);
-        last_flush = Z_NO_FLUSH;
-        tr_init();
-        lm_init();
-        return Z_OK;
+      // Make sure there is something to do and avoid duplicate consecutive
+      // flushes. For repeated and useless calls with Z_FINISH, we keep
+      // returning Z_STREAM_END instead of Z_BUFF_ERROR.
+    }
+    else if(strm.avail_in==0 && flush <= old_flush &&
+	    flush != Z_FINISH) {
+      strm.msg=z_errmsg[Z_NEED_DICT-(Z_BUF_ERROR)];
+      return Z_BUF_ERROR;
     }
 
-    int deflateEnd() {
-        if (status != INIT_STATE && status != BUSY_STATE && status != FINISH_STATE) {
-            return Z_STREAM_ERROR;
-        }
-
-        // Deallocate in reverse order of allocations:
-        pending_buf = null;
-        head = null;
-        prev = null;
-        window = null;
-        // free
-        // dstate=null;
-        return status == BUSY_STATE ? Z_DATA_ERROR : Z_OK;
+    // User must not provide more input after the first FINISH:
+    if(status == FINISH_STATE && strm.avail_in != 0) {
+      strm.msg=z_errmsg[Z_NEED_DICT-(Z_BUF_ERROR)];
+      return Z_BUF_ERROR;
     }
 
-    int deflateParams(ZStream strm, int _level, int _strategy) {
-        int err = Z_OK;
-
-        if (_level == Z_DEFAULT_COMPRESSION) {
-            _level = 6;
-        }
-
-        if (_level < 0 || _level > 9 ||
-                _strategy < 0 || _strategy > Z_HUFFMAN_ONLY) {
-            return Z_STREAM_ERROR;
-        }
-
-        if (config_table[level].func != config_table[_level].func &&
-                strm.total_in != 0) {
-            // Flush the last buffer:
-            err = strm.deflate(Z_PARTIAL_FLUSH);
-        }
-
-        if (level != _level) {
-            level = _level;
-            max_lazy_match   = config_table[level].max_lazy;
-            good_match       = config_table[level].good_length;
-            nice_match       = config_table[level].nice_length;
-            max_chain_length = config_table[level].max_chain;
-        }
-
-        strategy = _strategy;
-        return err;
-    }
+    // Start a new block or continue the current one.
+    if(strm.avail_in!=0 || lookahead!=0 ||
+       (flush != Z_NO_FLUSH && status != FINISH_STATE)) {
+      int bstate=-1;
+      switch(config_table[level].func){
+      case STORED: 
+	bstate = deflate_stored(flush);
+	break;
+      case FAST: 
+	bstate = deflate_fast(flush);
+	break;
+      case SLOW: 
+	bstate = deflate_slow(flush);
+	break;
+      default:
+      }
 
-    int deflateSetDictionary(ZStream strm, byte[] dictionary, int dictLength) {
-        int length = dictLength;
-        int index = 0;
-
-        if (dictionary == null || status != INIT_STATE)
-            return Z_STREAM_ERROR;
-
-        strm.adler = strm._adler.adler32(strm.adler, dictionary, 0, dictLength);
-
-        if (length < MIN_MATCH) return Z_OK;
-
-        if (length > w_size - MIN_LOOKAHEAD) {
-            length = w_size - MIN_LOOKAHEAD;
-            index = dictLength - length; // use the tail of the dictionary
-        }
+      if (bstate==FinishStarted || bstate==FinishDone) {
+	status = FINISH_STATE;
+      }
+      if (bstate==NeedMore || bstate==FinishStarted) {
+	if(strm.avail_out == 0) {
+	  last_flush = -1; // avoid BUF_ERROR next call, see above
+	}
+	return Z_OK;
+	// If flush != Z_NO_FLUSH && avail_out == 0, the next call
+	// of deflate should use the same flush parameter to make sure
+	// that the flush is complete. So we don't have to output an
+	// empty block here, this will be done at next call. This also
+	// ensures that for a very small output buffer, we emit at most
+	// one empty block.
+      }
 
-        System.arraycopy(dictionary, index, window, 0, length);
-        strstart = length;
-        block_start = length;
-        // Insert all strings in the hash table (except for the last two bytes).
-        // s->lookahead stays null, so s->ins_h will be recomputed at the next
-        // call of fill_window.
-        ins_h = window[0] & 0xff;
-        ins_h = (((ins_h) << hash_shift) ^ (window[1] & 0xff)) & hash_mask;
-
-        for (int n = 0; n <= length - MIN_MATCH; n++) {
-            ins_h = (((ins_h) << hash_shift) ^ (window[(n) + (MIN_MATCH - 1)] & 0xff)) & hash_mask;
-            prev[n & w_mask] = head[ins_h];
-            head[ins_h] = (short)n;
-        }
-
-        return Z_OK;
+      if (bstate==BlockDone) {
+	if(flush == Z_PARTIAL_FLUSH) {
+	  _tr_align();
+	} 
+	else { // FULL_FLUSH or SYNC_FLUSH
+	  _tr_stored_block(0, 0, false);
+	  // For a full flush, this empty block will be recognized
+	  // as a special marker by inflate_sync().
+	  if(flush == Z_FULL_FLUSH) {
+	    //state.head[s.hash_size-1]=0;
+	    for(int i=0; i<hash_size/*-1*/; i++)  // forget history
+	      head[i]=0;
+	  }
+	}
+	strm.flush_pending();
+	if(strm.avail_out == 0) {
+	  last_flush = -1; // avoid BUF_ERROR at next call, see above
+	  return Z_OK;
+	}
+      }
     }
 
-    int deflate(ZStream strm, int flush) {
-        int old_flush;
-
-        if (flush > Z_FINISH || flush < 0) {
-            return Z_STREAM_ERROR;
-        }
+    if(flush!=Z_FINISH) return Z_OK;
+    if(wrap<=0) return Z_STREAM_END;
 
-        if (strm.next_out == null ||
-                (strm.next_in == null && strm.avail_in != 0) ||
-                (status == FINISH_STATE && flush != Z_FINISH)) {
-            strm.msg = z_errmsg[Z_NEED_DICT - (Z_STREAM_ERROR)];
-            return Z_STREAM_ERROR;
-        }
-
-        if (strm.avail_out == 0) {
-            strm.msg = z_errmsg[Z_NEED_DICT - (Z_BUF_ERROR)];
-            return Z_BUF_ERROR;
-        }
+    if(wrap==2){
+      long adler=strm.adler.getValue();
+      put_byte((byte)(adler&0xff));
+      put_byte((byte)((adler>>8)&0xff));
+      put_byte((byte)((adler>>16)&0xff));
+      put_byte((byte)((adler>>24)&0xff));
+      put_byte((byte)(strm.total_in&0xff));
+      put_byte((byte)((strm.total_in>>8)&0xff));
+      put_byte((byte)((strm.total_in>>16)&0xff));
+      put_byte((byte)((strm.total_in>>24)&0xff));
 
-        this.strm = strm; // just in case
-        old_flush = last_flush;
-        last_flush = flush;
-
-        // Write the zlib header
-        if (status == INIT_STATE) {
-            int header = (Z_DEFLATED + ((w_bits - 8) << 4)) << 8;
-            int level_flags = ((level - 1) & 0xff) >> 1;
+      getGZIPHeader().setCRC(adler);
+    } 
+    else{
+      // Write the zlib trailer (adler32)
+      long adler=strm.adler.getValue();
+      putShortMSB((int)(adler>>>16));
+      putShortMSB((int)(adler&0xffff));
+    }
 
-            if (level_flags > 3) level_flags = 3;
-
-            header |= (level_flags << 6);
+    strm.flush_pending();
 
-            if (strstart != 0) header |= PRESET_DICT;
-
-            header += 31 - (header % 31);
-            status = BUSY_STATE;
-            putShortMSB(header);
+    // If avail_out is zero, the application will call deflate again
+    // to flush the rest.
 
-            // Save the adler32 of the preset dictionary:
-            if (strstart != 0) {
-                putShortMSB((int)(strm.adler >>> 16));
-                putShortMSB((int)(strm.adler & 0xffff));
-            }
+    if(wrap > 0) wrap = -wrap; // write the trailer only once!
+    return pending != 0 ? Z_OK : Z_STREAM_END;
+  }
 
-            strm.adler = strm._adler.adler32(0, null, 0, 0);
-        }
+  static int deflateCopy(ZStream dest, ZStream src){
 
-        // Flush as much pending output as possible
-        if (pending != 0) {
-            strm.flush_pending();
+    if(src.dstate == null){
+      return Z_STREAM_ERROR;
+    }
 
-            if (strm.avail_out == 0) {
-                //System.out.println("  avail_out==0");
-                // Since avail_out is 0, deflate will be called again with
-                // more output space, but possibly with both pending and
-                // avail_in equal to zero. There won't be anything to do,
-                // but this is not an error situation so make sure we
-                // return OK instead of BUF_ERROR at next call of deflate:
-                last_flush = -1;
-                return Z_OK;
-            }
+    if(src.next_in!=null){
+      dest.next_in = new byte[src.next_in.length];
+      System.arraycopy(src.next_in, 0, dest.next_in, 0, src.next_in.length);
+    }
+    dest.next_in_index = src.next_in_index;
+    dest.avail_in = src.avail_in;
+    dest.total_in = src.total_in;
 
-            // Make sure there is something to do and avoid duplicate consecutive
-            // flushes. For repeated and useless calls with Z_FINISH, we keep
-            // returning Z_STREAM_END instead of Z_BUFF_ERROR.
-        }
-        else if (strm.avail_in == 0 && flush <= old_flush &&
-                 flush != Z_FINISH) {
-            strm.msg = z_errmsg[Z_NEED_DICT - (Z_BUF_ERROR)];
-            return Z_BUF_ERROR;
-        }
+    if(src.next_out!=null){
+      dest.next_out = new byte[src.next_out.length];
+      System.arraycopy(src.next_out, 0, dest.next_out ,0 , src.next_out.length);
+    }
 
-        // User must not provide more input after the first FINISH:
-        if (status == FINISH_STATE && strm.avail_in != 0) {
-            strm.msg = z_errmsg[Z_NEED_DICT - (Z_BUF_ERROR)];
-            return Z_BUF_ERROR;
-        }
+    dest.next_out_index = src.next_out_index;
+    dest.avail_out = src.avail_out;
+    dest.total_out = src.total_out;
+
+    dest.msg = src.msg;
+    dest.data_type = src.data_type;
+    dest.adler = src.adler.copy();
 
-        // Start a new block or continue the current one.
-        if (strm.avail_in != 0 || lookahead != 0 ||
-                (flush != Z_NO_FLUSH && status != FINISH_STATE)) {
-            int bstate = -1;
+    try{
+      dest.dstate = (Deflate)src.dstate.clone();
+      dest.dstate.strm = dest;
+    }
+    catch(CloneNotSupportedException e){
+      //
+    }
+    return Z_OK;
+  }
 
-            switch (config_table[level].func) {
-                case STORED:
-                    bstate = deflate_stored(flush);
-                    break;
-
-                case FAST:
-                    bstate = deflate_fast(flush);
-                    break;
-
-                case SLOW:
-                    bstate = deflate_slow(flush);
-                    break;
+  public Object clone() throws CloneNotSupportedException {
+    Deflate dest = (Deflate)super.clone();
 
-                default:
-            }
-
-            if (bstate == FinishStarted || bstate == FinishDone) {
-                status = FINISH_STATE;
-            }
+    dest.pending_buf = dup(dest.pending_buf);
+    dest.d_buf = dest.d_buf;
+    dest.l_buf = dup(dest.l_buf);
+    dest.window = dup(dest.window);
 
-            if (bstate == NeedMore || bstate == FinishStarted) {
-                if (strm.avail_out == 0) {
-                    last_flush = -1; // avoid BUF_ERROR next call, see above
-                }
+    dest.prev = dup(dest.prev);
+    dest.head = dup(dest.head);
+    dest.dyn_ltree = dup(dest.dyn_ltree);
+    dest.dyn_dtree = dup(dest.dyn_dtree);
+    dest.bl_tree = dup(dest.bl_tree);
 
-                return Z_OK;
-                // If flush != Z_NO_FLUSH && avail_out == 0, the next call
-                // of deflate should use the same flush parameter to make sure
-                // that the flush is complete. So we don't have to output an
-                // empty block here, this will be done at next call. This also
-                // ensures that for a very small output buffer, we emit at most
-                // one empty block.
-            }
+    dest.bl_count = dup(dest.bl_count);
+    dest.next_code = dup(dest.next_code);
+    dest.heap = dup(dest.heap);
+    dest.depth = dup(dest.depth);
+
+    dest.l_desc.dyn_tree = dest.dyn_ltree;
+    dest.d_desc.dyn_tree = dest.dyn_dtree;
+    dest.bl_desc.dyn_tree = dest.bl_tree;
 
-            if (bstate == BlockDone) {
-                if (flush == Z_PARTIAL_FLUSH) {
-                    _tr_align();
-                }
-                else { // FULL_FLUSH or SYNC_FLUSH
-                    _tr_stored_block(0, 0, false);
+    /*
+    dest.l_desc.stat_desc = StaticTree.static_l_desc;
+    dest.d_desc.stat_desc = StaticTree.static_d_desc;
+    dest.bl_desc.stat_desc = StaticTree.static_bl_desc;
+    */
 
-                    // For a full flush, this empty block will be recognized
-                    // as a special marker by inflate_sync().
-                    if (flush == Z_FULL_FLUSH) {
-                        //state.head[s.hash_size-1]=0;
-                        for (int i = 0; i < hash_size/*-1*/; i++) // forget history
-                            head[i] = 0;
-                    }
-                }
+    if(dest.gheader!=null){
+      dest.gheader = (GZIPHeader)dest.gheader.clone();
+    }
 
-                strm.flush_pending();
+    return dest;
+  }
 
-                if (strm.avail_out == 0) {
-                    last_flush = -1; // avoid BUF_ERROR at next call, see above
-                    return Z_OK;
-                }
-            }
-        }
-
-        if (flush != Z_FINISH) return Z_OK;
-
-        if (noheader != 0) return Z_STREAM_END;
+  private byte[] dup(byte[] buf){
+    byte[] foo = new byte[buf.length];
+    System.arraycopy(buf, 0, foo, 0, foo.length);
+    return foo;
+  }
+  private short[] dup(short[] buf){
+    short[] foo = new short[buf.length];
+    System.arraycopy(buf, 0, foo, 0, foo.length);
+    return foo;
+  }
+  private int[] dup(int[] buf){
+    int[] foo = new int[buf.length];
+    System.arraycopy(buf, 0, foo, 0, foo.length);
+    return foo;
+  }
 
-        // Write the zlib trailer (adler32)
-        putShortMSB((int)(strm.adler >>> 16));
-        putShortMSB((int)(strm.adler & 0xffff));
-        strm.flush_pending();
-        // If avail_out is zero, the application will call deflate again
-        // to flush the rest.
-        noheader = -1; // write the trailer only once!
-        return pending != 0 ? Z_OK : Z_STREAM_END;
+  synchronized GZIPHeader getGZIPHeader(){
+    if(gheader==null){
+      gheader = new GZIPHeader();
     }
+    return gheader;
+  }
 }
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/com/jcraft/jzlib/Deflater.java	Fri Aug 01 13:34:58 2014 -0700
@@ -0,0 +1,171 @@
+/* -*-mode:java; c-basic-offset:2; indent-tabs-mode:nil -*- */
+/*
+Copyright (c) 2011 ymnk, JCraft,Inc. All rights reserved.
+
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions are met:
+
+  1. Redistributions of source code must retain the above copyright notice,
+     this list of conditions and the following disclaimer.
+
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
+     the documentation and/or other materials provided with the distribution.
+
+  3. The names of the authors may not be used to endorse or promote products
+     derived from this software without specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES,
+INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
+FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JCRAFT,
+INC. OR ANY CONTRIBUTORS TO THIS SOFTWARE BE LIABLE FOR ANY DIRECT, INDIRECT,
+INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
+OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
+LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
+EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+/*
+ * This program is based on zlib-1.1.3, so all credit should go authors
+ * Jean-loup Gailly(jloup@gzip.org) and Mark Adler(madler@alumni.caltech.edu)
+ * and contributors of zlib.
+ */
+
+package com.jcraft.jzlib;
+
+final public class Deflater extends ZStream{
+
+  static final private int MAX_WBITS=15;        // 32K LZ77 window
+  static final private int DEF_WBITS=MAX_WBITS;
+
+  static final private int Z_NO_FLUSH=0;
+  static final private int Z_PARTIAL_FLUSH=1;
+  static final private int Z_SYNC_FLUSH=2;
+  static final private int Z_FULL_FLUSH=3;
+  static final private int Z_FINISH=4;
+
+  static final private int MAX_MEM_LEVEL=9;
+
+  static final private int Z_OK=0;
+  static final private int Z_STREAM_END=1;
+  static final private int Z_NEED_DICT=2;
+  static final private int Z_ERRNO=-1;
+  static final private int Z_STREAM_ERROR=-2;
+  static final private int Z_DATA_ERROR=-3;
+  static final private int Z_MEM_ERROR=-4;
+  static final private int Z_BUF_ERROR=-5;
+  static final private int Z_VERSION_ERROR=-6;
+
+  private boolean finished = false;
+
+  public Deflater(){
+    super();
+  }
+
+  public Deflater(int level) throws GZIPException {
+    this(level, MAX_WBITS);
+  }
+
+  public Deflater(int level, boolean nowrap) throws GZIPException {
+    this(level, MAX_WBITS, nowrap);
+  }
+
+  public Deflater(int level, int bits) throws GZIPException {
+    this(level, bits, false);
+  }
+
+  public Deflater(int level, int bits, boolean nowrap) throws GZIPException {
+    super();
+    int ret = init(level, bits, nowrap);
+    if(ret!=Z_OK)
+      throw new GZIPException(ret+": "+msg);
+  }
+
+  public Deflater(int level, int bits, int memlevel, JZlib.WrapperType wrapperType) throws GZIPException {
+    super();
+    int ret = init(level, bits, memlevel, wrapperType);
+    if(ret!=Z_OK)
+      throw new GZIPException(ret+": "+msg);
+  }
+
+  public Deflater(int level, int bits, int memlevel) throws GZIPException {
+    super();
+    int ret = init(level, bits, memlevel);
+    if(ret!=Z_OK)
+      throw new GZIPException(ret+": "+msg);
+  }
+
+  public int init(int level){
+    return init(level, MAX_WBITS);
+  }
+  public int init(int level, boolean nowrap){
+    return init(level, MAX_WBITS, nowrap);
+  }
+  public int init(int level, int bits){
+    return init(level, bits, false);
+  }
+  public int init(int level, int bits, int memlevel, JZlib.WrapperType wrapperType){
+    if(bits < 9 || bits > 15){
+      return Z_STREAM_ERROR;
+    }
+    if(wrapperType == JZlib.W_NONE) {
+      bits *= -1;
+    }
+    else if(wrapperType == JZlib.W_GZIP) {
+        bits += 16;
+    }
+    else if(wrapperType == JZlib.W_ANY) {
+        return Z_STREAM_ERROR;
+    }
+    else if(wrapperType == JZlib.W_ZLIB) {
+    }
+    return init(level, bits, memlevel);
+  }
+  public int init(int level, int bits, int memlevel){
+    finished = false;
+    dstate=new Deflate(this);
+    return dstate.deflateInit(level, bits, memlevel);
+  }
+  public int init(int level, int bits, boolean nowrap){
+    finished = false;
+    dstate=new Deflate(this);
+    return dstate.deflateInit(level, nowrap?-bits:bits);
+  }
+
+  public int deflate(int flush){
+    if(dstate==null){
+      return Z_STREAM_ERROR;
+    }
+    int ret = dstate.deflate(flush);
+    if(ret == Z_STREAM_END)
+      finished = true;
+    return ret;
+  }
+  public int end(){
+    finished = true;
+    if(dstate==null) return Z_STREAM_ERROR;
+    int ret=dstate.deflateEnd();
+    dstate=null;
+    free();
+    return ret;
+  }
+  public int params(int level, int strategy){
+    if(dstate==null) return Z_STREAM_ERROR;
+    return dstate.deflateParams(level, strategy);
+  }
+  public int setDictionary (byte[] dictionary, int dictLength){
+    if(dstate == null)
+      return Z_STREAM_ERROR;
+    return dstate.deflateSetDictionary(dictionary, dictLength);
+  }
+
+  public boolean finished(){
+    return finished;
+  }
+
+  public int copy(Deflater src){
+    this.finished = src.finished;
+    return Deflate.deflateCopy(this, src);
+  }
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/com/jcraft/jzlib/DeflaterOutputStream.java	Fri Aug 01 13:34:58 2014 -0700
@@ -0,0 +1,181 @@
+/* -*-mode:java; c-basic-offset:2; indent-tabs-mode:nil -*- */
+/*
+Copyright (c) 2011 ymnk, JCraft,Inc. All rights reserved.
+
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions are met:
+
+  1. Redistributions of source code must retain the above copyright notice,
+     this list of conditions and the following disclaimer.
+
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
+     the documentation and/or other materials provided with the distribution.
+
+  3. The names of the authors may not be used to endorse or promote products
+     derived from this software without specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES,
+INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
+FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JCRAFT,
+INC. OR ANY CONTRIBUTORS TO THIS SOFTWARE BE LIABLE FOR ANY DIRECT, INDIRECT,
+INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
+OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
+LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
+EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+package com.jcraft.jzlib;
+import java.io.*;
+
+public class DeflaterOutputStream extends FilterOutputStream {
+
+  protected final Deflater deflater;
+
+  protected byte[] buffer;
+
+  private boolean closed = false;
+
+  private boolean syncFlush = false;
+
+  private final byte[] buf1 = new byte[1];
+
+  protected boolean mydeflater = false;
+
+  private boolean close_out = true;
+
+  protected static final int DEFAULT_BUFSIZE = 512;
+
+  public DeflaterOutputStream(OutputStream out) throws IOException {
+    this(out, 
+         new Deflater(JZlib.Z_DEFAULT_COMPRESSION),
+         DEFAULT_BUFSIZE, true);
+    mydeflater = true;
+  }
+
+  public DeflaterOutputStream(OutputStream out, Deflater def) throws IOException {
+    this(out, def, DEFAULT_BUFSIZE, true);
+  }
+
+  public DeflaterOutputStream(OutputStream out,
+                              Deflater deflater,
+                              int size) throws IOException {
+    this(out, deflater, size, true);
+  }
+  public DeflaterOutputStream(OutputStream out,
+                              Deflater deflater,
+                              int size,
+                              boolean close_out)  throws IOException {
+    super(out);
+    if (out == null || deflater == null) {
+      throw new NullPointerException();
+    }
+    else if (size <= 0) {
+      throw new IllegalArgumentException("buffer size must be greater than 0");
+    }
+    this.deflater = deflater;
+    buffer = new byte[size];
+    this.close_out = close_out;
+  }
+
+  public void write(int b) throws IOException {
+    buf1[0] = (byte)(b & 0xff);
+    write(buf1, 0, 1);
+  }
+
+  public void write(byte[] b, int off, int len) throws IOException {
+    if (deflater.finished()) {
+      throw new IOException("finished");
+    }
+    else if (off<0 | len<0 | off+len>b.length) {
+      throw new IndexOutOfBoundsException();
+    }
+    else if (len == 0) {
+      return;
+    }
+    else {
+      int flush = syncFlush ? JZlib.Z_SYNC_FLUSH : JZlib.Z_NO_FLUSH;
+      deflater.setInput(b, off, len, true);
+      while (deflater.avail_in>0) {
+        int err = deflate(flush);
+        if (err == JZlib.Z_STREAM_END)
+          break;
+      }
+    }
+  }
+
+  public void finish() throws IOException {
+    while (!deflater.finished()) {
+      deflate(JZlib.Z_FINISH);
+    }
+  }
+
+  public void close() throws IOException {
+    if (!closed) {
+      finish();
+      if (mydeflater){
+        deflater.end();
+      }
+      if(close_out)
+        out.close();
+      closed = true;
+    }
+  }
+
+  protected int deflate(int flush) throws IOException {
+    deflater.setOutput(buffer, 0, buffer.length);
+    int err = deflater.deflate(flush);
+    switch(err) {
+      case JZlib.Z_OK:
+      case JZlib.Z_STREAM_END:
+        break;
+      case JZlib.Z_BUF_ERROR:
+        if(deflater.avail_in<=0 && flush!=JZlib.Z_FINISH){
+          // flush() without any data
+          break;
+        }
+      default:
+        throw new IOException("failed to deflate: error="+err+" avail_out="+deflater.avail_out);
+    }
+    int len = deflater.next_out_index;
+    if (len > 0) {
+      out.write(buffer, 0, len);
+    }
+    return err;
+  }
+
+  public void flush() throws IOException {
+    if (syncFlush && !deflater.finished()) {
+      while (true) {
+        int err = deflate(JZlib.Z_SYNC_FLUSH);
+        if (deflater.next_out_index < buffer.length)
+          break;
+        if (err == JZlib.Z_STREAM_END)
+          break;
+      }
+    }
+    out.flush();
+  }
+
+  public long getTotalIn() {
+    return deflater.getTotalIn();
+  }
+
+  public long getTotalOut() {
+    return deflater.getTotalOut();
+  }
+
+  public void setSyncFlush(boolean syncFlush){
+    this.syncFlush = syncFlush;
+  }
+
+  public boolean getSyncFlush(){
+    return this.syncFlush;
+  }
+
+  public Deflater getDeflater(){
+    return deflater;
+  }
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/com/jcraft/jzlib/GZIPException.java	Fri Aug 01 13:34:58 2014 -0700
@@ -0,0 +1,44 @@
+/* -*-mode:java; c-basic-offset:2; -*- */
+/*
+Copyright (c) 2011 ymnk, JCraft,Inc. All rights reserved.
+
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions are met:
+
+  1. Redistributions of source code must retain the above copyright notice,
+     this list of conditions and the following disclaimer.
+
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
+     the documentation and/or other materials provided with the distribution.
+
+  3. The names of the authors may not be used to endorse or promote products
+     derived from this software without specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES,
+INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
+FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JCRAFT,
+INC. OR ANY CONTRIBUTORS TO THIS SOFTWARE BE LIABLE FOR ANY DIRECT, INDIRECT,
+INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
+OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
+LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
+EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+/*
+ * This program is based on zlib-1.1.3, so all credit should go authors
+ * Jean-loup Gailly(jloup@gzip.org) and Mark Adler(madler@alumni.caltech.edu)
+ * and contributors of zlib.
+ */
+
+package com.jcraft.jzlib;
+
+public class GZIPException extends java.io.IOException {
+  public GZIPException() {
+    super();
+  }
+  public GZIPException(String s) {
+    super(s);
+  }
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/com/jcraft/jzlib/GZIPHeader.java	Fri Aug 01 13:34:58 2014 -0700
@@ -0,0 +1,214 @@
+/* -*-mode:java; c-basic-offset:2; -*- */
+/*
+Copyright (c) 2011 ymnk, JCraft,Inc. All rights reserved.
+
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions are met:
+
+  1. Redistributions of source code must retain the above copyright notice,
+     this list of conditions and the following disclaimer.
+
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
+     the documentation and/or other materials provided with the distribution.
+
+  3. The names of the authors may not be used to endorse or promote products
+     derived from this software without specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES,
+INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
+FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JCRAFT,
+INC. OR ANY CONTRIBUTORS TO THIS SOFTWARE BE LIABLE FOR ANY DIRECT, INDIRECT,
+INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
+OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
+LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
+EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+/*
+ * This program is based on zlib-1.1.3, so all credit should go authors
+ * Jean-loup Gailly(jloup@gzip.org) and Mark Adler(madler@alumni.caltech.edu)
+ * and contributors of zlib.
+ */
+
+package com.jcraft.jzlib;
+
+import java.io.UnsupportedEncodingException;
+
+/**
+ * @see "http://www.ietf.org/rfc/rfc1952.txt"
+ */
+public class GZIPHeader implements Cloneable {
+
+  public static final byte OS_MSDOS = (byte) 0x00;
+  public static final byte OS_AMIGA = (byte) 0x01;
+  public static final byte OS_VMS = (byte) 0x02;
+  public static final byte OS_UNIX = (byte) 0x03;
+  public static final byte OS_ATARI = (byte) 0x05;
+  public static final byte OS_OS2 = (byte) 0x06;
+  public static final byte OS_MACOS = (byte) 0x07;
+  public static final byte OS_TOPS20 = (byte) 0x0a;
+  public static final byte OS_WIN32 = (byte) 0x0b;
+  public static final byte OS_VMCMS = (byte) 0x04;
+  public static final byte OS_ZSYSTEM = (byte) 0x08;
+  public static final byte OS_CPM = (byte) 0x09;
+  public static final byte OS_QDOS = (byte) 0x0c;
+  public static final byte OS_RISCOS = (byte) 0x0d;
+  public static final byte OS_UNKNOWN = (byte) 0xff;
+
+  boolean text = false;
+  private boolean fhcrc = false;
+  long time;
+  int xflags;
+  int os = 255;
+  byte[] extra;
+  byte[] name;
+  byte[] comment;
+  int hcrc;
+  long crc;
+  boolean done = false;
+  long mtime = 0;
+
+  public void setModifiedTime(long mtime) {
+    this.mtime = mtime;
+  }
+
+  public long getModifiedTime() {
+    return mtime;
+  }
+
+  public void setOS(int os) {
+    if((0<=os && os <=13) || os==255)
+      this.os=os;
+    else
+      throw new IllegalArgumentException("os: "+os);
+  }
+
+  public int getOS(){
+    return os;
+  }
+
+  public void setName(String name) {
+    try{
+      this.name=name.getBytes("ISO-8859-1");
+    }
+    catch(UnsupportedEncodingException e){
+      throw new IllegalArgumentException("name must be in ISO-8859-1 "+name);
+    }
+  }
+
+  public String getName(){
+    if(name==null) return "";
+    try {
+      return new String(name, "ISO-8859-1");
+    }
+    catch (UnsupportedEncodingException e) {
+      throw new InternalError(e.toString());
+    }
+  }
+
+  public void setComment(String comment) {
+    try{
+      this.comment=comment.getBytes("ISO-8859-1");
+    }
+    catch(UnsupportedEncodingException e){
+      throw new IllegalArgumentException("comment must be in ISO-8859-1 "+name);
+    }
+  }
+
+  public String getComment(){
+    if(comment==null) return "";
+    try {
+      return new String(comment, "ISO-8859-1");
+    }
+    catch (UnsupportedEncodingException e) {
+      throw new InternalError(e.toString());
+    }
+  }
+
+  public void setCRC(long crc){
+    this.crc = crc;
+  }
+
+  public long getCRC(){
+    return crc;
+  }
+
+  void put(Deflate d){
+    int flag = 0;
+    if(text){
+      flag |= 1;     // FTEXT
+    }
+    if(fhcrc){
+      flag |= 2;     // FHCRC
+    }
+    if(extra!=null){
+      flag |= 4;     // FEXTRA
+    }
+    if(name!=null){
+      flag |= 8;    // FNAME
+    }
+    if(comment!=null){
+      flag |= 16;   // FCOMMENT
+    }
+    int xfl = 0;
+    if(d.level == JZlib.Z_BEST_SPEED){
+      xfl |= 4;
+    }
+    else if (d.level == JZlib.Z_BEST_COMPRESSION){
+      xfl |= 2;
+    }
+
+    d.put_short((short)0x8b1f);  // ID1 ID2
+    d.put_byte((byte)8);         // CM(Compression Method)
+    d.put_byte((byte)flag);
+    d.put_byte((byte)mtime);
+    d.put_byte((byte)(mtime>>8));
+    d.put_byte((byte)(mtime>>16));
+    d.put_byte((byte)(mtime>>24));
+    d.put_byte((byte)xfl);
+    d.put_byte((byte)os);
+
+    if(extra!=null){
+      d.put_byte((byte)extra.length);
+      d.put_byte((byte)(extra.length>>8));
+      d.put_byte(extra, 0, extra.length);
+    }
+
+    if(name!=null){
+      d.put_byte(name, 0, name.length);
+      d.put_byte((byte)0);
+    }
+
+    if(comment!=null){
+      d.put_byte(comment, 0, comment.length);
+      d.put_byte((byte)0);
+    }
+  }
+
+  @Override
+  public Object clone() throws CloneNotSupportedException {
+    GZIPHeader gheader = (GZIPHeader)super.clone();
+    byte[] tmp;
+    if(gheader.extra!=null){
+      tmp=new byte[gheader.extra.length];
+      System.arraycopy(gheader.extra, 0, tmp, 0, tmp.length);
+      gheader.extra = tmp;
+    }
+
+    if(gheader.name!=null){
+      tmp=new byte[gheader.name.length];
+      System.arraycopy(gheader.name, 0, tmp, 0, tmp.length);
+      gheader.name = tmp;
+    }
+
+    if(gheader.comment!=null){
+      tmp=new byte[gheader.comment.length];
+      System.arraycopy(gheader.comment, 0, tmp, 0, tmp.length);
+      gheader.comment = tmp;
+    }
+
+    return gheader;
+  }
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/com/jcraft/jzlib/GZIPInputStream.java	Fri Aug 01 13:34:58 2014 -0700
@@ -0,0 +1,145 @@
+/* -*-mode:java; c-basic-offset:2; indent-tabs-mode:nil -*- */
+/*
+Copyright (c) 2011 ymnk, JCraft,Inc. All rights reserved.
+
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions are met:
+
+  1. Redistributions of source code must retain the above copyright notice,
+     this list of conditions and the following disclaimer.
+
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
+     the documentation and/or other materials provided with the distribution.
+
+  3. The names of the authors may not be used to endorse or promote products
+     derived from this software without specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES,
+INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
+FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JCRAFT,
+INC. OR ANY CONTRIBUTORS TO THIS SOFTWARE BE LIABLE FOR ANY DIRECT, INDIRECT,
+INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
+OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
+LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
+EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+package com.jcraft.jzlib;
+import java.io.*;
+
+public class GZIPInputStream extends InflaterInputStream {
+
+  public GZIPInputStream(InputStream in) throws IOException {
+    this(in, DEFAULT_BUFSIZE, true);
+  }
+
+  public GZIPInputStream(InputStream in,
+                         int size,
+                         boolean close_in) throws IOException {
+    this(in, new Inflater(15+16), size, close_in);
+    myinflater = true;
+  }
+
+  public GZIPInputStream(InputStream in, 
+                         Inflater inflater,
+                         int size,
+                         boolean close_in) throws IOException {
+    super(in, inflater, size, close_in);
+  }
+
+  public long getModifiedtime() {
+    return inflater.istate.getGZIPHeader().getModifiedTime();
+  }
+
+  public int getOS() {
+    return inflater.istate.getGZIPHeader().getOS();
+  }
+
+  public String getName() {
+    return inflater.istate.getGZIPHeader().getName();
+  }
+
+  public String getComment() {
+    return inflater.istate.getGZIPHeader().getComment();
+  }
+
+  public long getCRC() throws GZIPException {
+    if(inflater.istate.mode != 12 /*DONE*/)
+      throw new GZIPException("checksum is not calculated yet.");
+    return inflater.istate.getGZIPHeader().getCRC();
+  }
+
+  public void readHeader() throws IOException {
+
+    byte[] empty = "".getBytes();
+    inflater.setOutput(empty, 0, 0);
+    inflater.setInput(empty, 0, 0, false);
+
+    byte[] b = new byte[10];
+
+    int n = fill(b);
+    if(n!=10){
+      if(n>0){
+        inflater.setInput(b, 0, n, false);
+        //inflater.next_in_index = n;
+        inflater.next_in_index = 0;
+        inflater.avail_in = n;
+      }
+      throw new IOException("no input");
+    }
+
+    inflater.setInput(b, 0, n, false);
+
+    byte[] b1 = new byte[1];
+    do{
+      if(inflater.avail_in<=0){
+        int i = in.read(b1);
+        if(i<=0)
+          throw new IOException("no input");
+        inflater.setInput(b1, 0, 1, true);
+      }
+
+      int err = inflater.inflate(JZlib.Z_NO_FLUSH);
+
+      if(err!=0/*Z_OK*/){
+        int len = 2048-inflater.next_in.length;
+        if(len>0){
+          byte[] tmp = new byte[len];
+          n = fill(tmp);
+          if(n>0){
+            inflater.avail_in += inflater.next_in_index;
+            inflater.next_in_index = 0;
+            inflater.setInput(tmp, 0, n, true);
+          }
+        }
+        //inflater.next_in_index = inflater.next_in.length;
+        inflater.avail_in += inflater.next_in_index;
+        inflater.next_in_index = 0;
+        throw new IOException(inflater.msg);
+      }
+    }
+    while(inflater.istate.inParsingHeader());
+  }
+
+  private int fill(byte[] buf) {
+    int len = buf.length;
+    int n = 0;
+    do{
+      int i = -1;
+      try {
+        i = in.read(buf, n, buf.length - n);
+      }
+      catch(IOException e){
+      }
+      if(i == -1){
+        break;
+      }
+      n+=i;
+    }
+    while(n<len);
+    return n;
+  }
+}
\ No newline at end of file
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/com/jcraft/jzlib/GZIPOutputStream.java	Fri Aug 01 13:34:58 2014 -0700
@@ -0,0 +1,90 @@
+/* -*-mode:java; c-basic-offset:2; indent-tabs-mode:nil -*- */
+/*
+Copyright (c) 2011 ymnk, JCraft,Inc. All rights reserved.
+
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions are met:
+
+  1. Redistributions of source code must retain the above copyright notice,
+     this list of conditions and the following disclaimer.
+
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
+     the documentation and/or other materials provided with the distribution.
+
+  3. The names of the authors may not be used to endorse or promote products
+     derived from this software without specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES,
+INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
+FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JCRAFT,
+INC. OR ANY CONTRIBUTORS TO THIS SOFTWARE BE LIABLE FOR ANY DIRECT, INDIRECT,
+INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
+OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
+LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
+EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+package com.jcraft.jzlib;
+import java.io.*;
+
+public class GZIPOutputStream extends DeflaterOutputStream {
+
+  public GZIPOutputStream(OutputStream out) throws IOException {
+    this(out, DEFAULT_BUFSIZE);
+  }
+
+  public GZIPOutputStream(OutputStream out, int size) throws IOException {
+    this(out, size, true);
+  }
+
+  public GZIPOutputStream(OutputStream out, 
+                          int size,
+                          boolean close_out) throws IOException {
+    this(out,
+         new Deflater(JZlib.Z_DEFAULT_COMPRESSION, 15+16),
+         size, close_out);
+    mydeflater=true; 
+  }
+
+  public GZIPOutputStream(OutputStream out, 
+                          Deflater deflater,
+                          int size,
+                          boolean close_out) throws IOException{
+    super(out, deflater, size, close_out);
+  }
+
+
+  private void check() throws GZIPException {
+    if(deflater.dstate.status != 42 /*INIT_STATUS*/)
+      throw new GZIPException("header is already written.");
+  }
+
+  public void setModifiedTime(long mtime) throws GZIPException {
+    check();
+    deflater.dstate.getGZIPHeader().setModifiedTime(mtime);
+  }
+
+  public void setOS(int os) throws GZIPException {
+    check();
+    deflater.dstate.getGZIPHeader().setOS(os);
+  }
+
+  public void setName(String name) throws GZIPException {
+    check();
+    deflater.dstate.getGZIPHeader().setName(name);
+  }
+
+  public void setComment(String comment) throws GZIPException {
+    check();
+    deflater.dstate.getGZIPHeader().setComment(comment);
+  }
+
+  public long getCRC() throws GZIPException {
+    if(deflater.dstate.status != 666 /*FINISH_STATE*/)
+      throw new GZIPException("checksum is not calculated yet.");
+    return deflater.dstate.getGZIPHeader().getCRC();
+  }
+}
--- a/src/com/jcraft/jzlib/InfBlocks.java	Fri Aug 01 11:57:17 2014 -0700
+++ b/src/com/jcraft/jzlib/InfBlocks.java	Fri Aug 01 13:34:58 2014 -0700
@@ -1,6 +1,6 @@
 /* -*-mode:java; c-basic-offset:2; -*- */
 /*
-Copyright (c) 2000,2001,2002,2003 ymnk, JCraft,Inc. All rights reserved.
+Copyright (c) 2011 ymnk, JCraft,Inc. All rights reserved.
 
 Redistribution and use in source and binary forms, with or without
 modification, are permitted provided that the following conditions are met:
@@ -8,8 +8,8 @@
   1. Redistributions of source code must retain the above copyright notice,
      this list of conditions and the following disclaimer.
 
-  2. Redistributions in binary form must reproduce the above copyright
-     notice, this list of conditions and the following disclaimer in
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
      the documentation and/or other materials provided with the distribution.
 
   3. The names of the authors may not be used to endorse or promote products
@@ -34,619 +34,581 @@
 
 package com.jcraft.jzlib;
 
-final class InfBlocks {
-    static final private int MANY = 1440;
+final class InfBlocks{
+  static final private int MANY=1440;
+
+  // And'ing with mask[n] masks the lower n bits
+  static final private int[] inflate_mask = {
+    0x00000000, 0x00000001, 0x00000003, 0x00000007, 0x0000000f,
+    0x0000001f, 0x0000003f, 0x0000007f, 0x000000ff, 0x000001ff,
+    0x000003ff, 0x000007ff, 0x00000fff, 0x00001fff, 0x00003fff,
+    0x00007fff, 0x0000ffff
+  };
+
+  // Table for deflate from PKZIP's appnote.txt.
+  static final int[] border = { // Order of the bit length code lengths
+    16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15
+  };
+
+  static final private int Z_OK=0;
+  static final private int Z_STREAM_END=1;
+  static final private int Z_NEED_DICT=2;
+  static final private int Z_ERRNO=-1;
+  static final private int Z_STREAM_ERROR=-2;
+  static final private int Z_DATA_ERROR=-3;
+  static final private int Z_MEM_ERROR=-4;
+  static final private int Z_BUF_ERROR=-5;
+  static final private int Z_VERSION_ERROR=-6;
+
+  static final private int TYPE=0;  // get type bits (3, including end bit)
+  static final private int LENS=1;  // get lengths for stored
+  static final private int STORED=2;// processing stored block
+  static final private int TABLE=3; // get table lengths
+  static final private int BTREE=4; // get bit lengths tree for a dynamic block
+  static final private int DTREE=5; // get length, distance trees for a dynamic block
+  static final private int CODES=6; // processing fixed or dynamic block
+  static final private int DRY=7;   // output remaining window bytes
+  static final private int DONE=8;  // finished last block, done
+  static final private int BAD=9;   // ot a data error--stuck here
+
+  int mode;            // current inflate_block mode 
+
+  int left;            // if STORED, bytes left to copy 
+
+  int table;           // table lengths (14 bits) 
+  int index;           // index into blens (or border) 
+  int[] blens;         // bit lengths of codes 
+  int[] bb=new int[1]; // bit length tree depth 
+  int[] tb=new int[1]; // bit length decoding tree 
+
+  int[] bl=new int[1];
+  int[] bd=new int[1];
+
+  int[][] tl=new int[1][];
+  int[][] td=new int[1][];
+  int[] tli=new int[1]; // tl_index
+  int[] tdi=new int[1]; // td_index
+
+  private final InfCodes codes;      // if CODES, current state 
 
-    // And'ing with mask[n] masks the lower n bits
-    static final private int[] inflate_mask = {
-        0x00000000, 0x00000001, 0x00000003, 0x00000007, 0x0000000f,
-        0x0000001f, 0x0000003f, 0x0000007f, 0x000000ff, 0x000001ff,
-        0x000003ff, 0x000007ff, 0x00000fff, 0x00001fff, 0x00003fff,
-        0x00007fff, 0x0000ffff
-    };
+  int last;            // true if this block is the last block 
+
+  // mode independent information 
+  int bitk;            // bits in bit buffer 
+  int bitb;            // bit buffer 
+  int[] hufts;         // single malloc for tree space 
+  byte[] window;       // sliding window 
+  int end;             // one byte after sliding window 
+  int read;            // window read pointer 
+  int write;           // window write pointer 
+  private boolean check;
+
+  private final InfTree inftree=new InfTree();
+
+  private final ZStream z; 
+
+  InfBlocks(ZStream z, int w){
+    this.z=z;
+    this.codes=new InfCodes(this.z, this);
+    hufts=new int[MANY*3];
+    window=new byte[w];
+    end=w;
+    this.check = (z.istate.wrap==0) ? false : true;
+    mode = TYPE;
+    reset();
+  }
 
-    // Table for deflate from PKZIP's appnote.txt.
-    static final int[] border = { // Order of the bit length code lengths
-        16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15
-    };
+  void reset(){
+    if(mode==BTREE || mode==DTREE){
+    }
+    if(mode==CODES){
+      codes.free(z);
+    }
+    mode=TYPE;
+    bitk=0;
+    bitb=0;
+    read=write=0;
+    if(check){
+      z.adler.reset();
+    }
+  }
+
+  int proc(int r){
+    int t;              // temporary storage
+    int b;              // bit buffer
+    int k;              // bits in bit buffer
+    int p;              // input data pointer
+    int n;              // bytes available there
+    int q;              // output window write pointer
+    int m;              // bytes to end of window or read pointer
+
+    // copy input/output information to locals (UPDATE macro restores)
+    {p=z.next_in_index;n=z.avail_in;b=bitb;k=bitk;}
+    {q=write;m=(int)(q<read?read-q-1:end-q);}
+
+    // process input based on current state
+    while(true){
+      switch (mode){
+      case TYPE:
 
-    static final private int Z_OK = 0;
-    static final private int Z_STREAM_END = 1;
-    static final private int Z_NEED_DICT = 2;
-    static final private int Z_ERRNO = -1;
-    static final private int Z_STREAM_ERROR = -2;
-    static final private int Z_DATA_ERROR = -3;
-    static final private int Z_MEM_ERROR = -4;
-    static final private int Z_BUF_ERROR = -5;
-    static final private int Z_VERSION_ERROR = -6;
+	while(k<(3)){
+	  if(n!=0){
+	    r=Z_OK;
+	  }
+	  else{
+	    bitb=b; bitk=k; 
+	    z.avail_in=n;
+	    z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	    write=q;
+	    return inflate_flush(r);
+	  };
+	  n--;
+	  b|=(z.next_in[p++]&0xff)<<k;
+	  k+=8;
+	}
+	t = (int)(b & 7);
+	last = t & 1;
+
+	switch (t >>> 1){
+        case 0:                         // stored 
+          {b>>>=(3);k-=(3);}
+          t = k & 7;                    // go to byte boundary
+
+          {b>>>=(t);k-=(t);}
+          mode = LENS;                  // get length of stored block
+          break;
+        case 1:                         // fixed
+          InfTree.inflate_trees_fixed(bl, bd, tl, td, z);
+          codes.init(bl[0], bd[0], tl[0], 0, td[0], 0);
+
+          {b>>>=(3);k-=(3);}
+
+          mode = CODES;
+          break;
+        case 2:                         // dynamic
+
+          {b>>>=(3);k-=(3);}
+
+          mode = TABLE;
+          break;
+        case 3:                         // illegal
+
+          {b>>>=(3);k-=(3);}
+          mode = BAD;
+          z.msg = "invalid block type";
+          r = Z_DATA_ERROR;
+
+	  bitb=b; bitk=k; 
+	  z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	  write=q;
+	  return inflate_flush(r);
+	}
+	break;
+      case LENS:
+
+	while(k<(32)){
+	  if(n!=0){
+	    r=Z_OK;
+	  }
+	  else{
+	    bitb=b; bitk=k; 
+	    z.avail_in=n;
+	    z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	    write=q;
+	    return inflate_flush(r);
+	  };
+	  n--;
+	  b|=(z.next_in[p++]&0xff)<<k;
+	  k+=8;
+	}
 
-    static final private int TYPE = 0; // get type bits (3, including end bit)
-    static final private int LENS = 1; // get lengths for stored
-    static final private int STORED = 2; // processing stored block
-    static final private int TABLE = 3; // get table lengths
-    static final private int BTREE = 4; // get bit lengths tree for a dynamic block
-    static final private int DTREE = 5; // get length, distance trees for a dynamic block
-    static final private int CODES = 6; // processing fixed or dynamic block
-    static final private int DRY = 7; // output remaining window bytes
-    static final private int DONE = 8; // finished last block, done
-    static final private int BAD = 9; // ot a data error--stuck here
+	if ((((~b) >>> 16) & 0xffff) != (b & 0xffff)){
+	  mode = BAD;
+	  z.msg = "invalid stored block lengths";
+	  r = Z_DATA_ERROR;
+
+	  bitb=b; bitk=k; 
+	  z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	  write=q;
+	  return inflate_flush(r);
+	}
+	left = (b & 0xffff);
+	b = k = 0;                       // dump bits
+	mode = left!=0 ? STORED : (last!=0 ? DRY : TYPE);
+	break;
+      case STORED:
+	if (n == 0){
+	  bitb=b; bitk=k; 
+	  z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	  write=q;
+	  return inflate_flush(r);
+	}
 
-    int mode;            // current inflate_block mode
+	if(m==0){
+	  if(q==end&&read!=0){
+	    q=0; m=(int)(q<read?read-q-1:end-q);
+	  }
+	  if(m==0){
+	    write=q; 
+	    r=inflate_flush(r);
+	    q=write;m=(int)(q<read?read-q-1:end-q);
+	    if(q==end&&read!=0){
+	      q=0; m=(int)(q<read?read-q-1:end-q);
+	    }
+	    if(m==0){
+	      bitb=b; bitk=k; 
+	      z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	      write=q;
+	      return inflate_flush(r);
+	    }
+	  }
+	}
+	r=Z_OK;
 
-    int left;            // if STORED, bytes left to copy
+	t = left;
+	if(t>n) t = n;
+	if(t>m) t = m;
+	System.arraycopy(z.next_in, p, window, q, t);
+	p += t;  n -= t;
+	q += t;  m -= t;
+	if ((left -= t) != 0)
+	  break;
+	mode = last!=0 ? DRY : TYPE;
+	break;
+      case TABLE:
+
+	while(k<(14)){
+	  if(n!=0){
+	    r=Z_OK;
+	  }
+	  else{
+	    bitb=b; bitk=k; 
+	    z.avail_in=n;
+	    z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	    write=q;
+	    return inflate_flush(r);
+	  };
+	  n--;
+	  b|=(z.next_in[p++]&0xff)<<k;
+	  k+=8;
+	}
+
+	table = t = (b & 0x3fff);
+	if ((t & 0x1f) > 29 || ((t >> 5) & 0x1f) > 29)
+	  {
+	    mode = BAD;
+	    z.msg = "too many length or distance symbols";
+	    r = Z_DATA_ERROR;
 
-    int table;           // table lengths (14 bits)
-    int index;           // index into blens (or border)
-    int[] blens;         // bit lengths of codes
-    int[] bb = new int[1]; // bit length tree depth
-    int[] tb = new int[1]; // bit length decoding tree
+	    bitb=b; bitk=k; 
+	    z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	    write=q;
+	    return inflate_flush(r);
+	  }
+	t = 258 + (t & 0x1f) + ((t >> 5) & 0x1f);
+	if(blens==null || blens.length<t){
+	  blens=new int[t];
+	}
+	else{
+	  for(int i=0; i<t; i++){blens[i]=0;}
+	}
+
+	{b>>>=(14);k-=(14);}
 
-    InfCodes codes = new InfCodes();    // if CODES, current state
+	index = 0;
+	mode = BTREE;
+      case BTREE:
+	while (index < 4 + (table >>> 10)){
+	  while(k<(3)){
+	    if(n!=0){
+	      r=Z_OK;
+	    }
+	    else{
+	      bitb=b; bitk=k; 
+	      z.avail_in=n;
+	      z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	      write=q;
+	      return inflate_flush(r);
+	    };
+	    n--;
+	    b|=(z.next_in[p++]&0xff)<<k;
+	    k+=8;
+	  }
+
+	  blens[border[index++]] = b&7;
 
-    int last;            // true if this block is the last block
+	  {b>>>=(3);k-=(3);}
+	}
+
+	while(index < 19){
+	  blens[border[index++]] = 0;
+	}
+
+	bb[0] = 7;
+	t = inftree.inflate_trees_bits(blens, bb, tb, hufts, z);
+	if (t != Z_OK){
+	  r = t;
+	  if (r == Z_DATA_ERROR){
+	    blens=null;
+	    mode = BAD;
+	  }
+
+	  bitb=b; bitk=k; 
+	  z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	  write=q;
+	  return inflate_flush(r);
+	}
 
-    // mode independent information
-    int bitk;            // bits in bit buffer
-    int bitb;            // bit buffer
-    int[] hufts;         // single malloc for tree space
-    byte[] window;       // sliding window
-    int end;             // one byte after sliding window
-    int read;            // window read pointer
-    int write;           // window write pointer
-    Object checkfn;      // check function
-    long check;          // check on output
+	index = 0;
+	mode = DTREE;
+      case DTREE:
+	while (true){
+	  t = table;
+	  if(!(index < 258 + (t & 0x1f) + ((t >> 5) & 0x1f))){
+	    break;
+	  }
+
+	  int[] h;
+	  int i, j, c;
+
+	  t = bb[0];
+
+	  while(k<(t)){
+	    if(n!=0){
+	      r=Z_OK;
+	    }
+	    else{
+	      bitb=b; bitk=k; 
+	      z.avail_in=n;
+	      z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	      write=q;
+	      return inflate_flush(r);
+	    };
+	    n--;
+	    b|=(z.next_in[p++]&0xff)<<k;
+	    k+=8;
+	  }
+
+	  if(tb[0]==-1){
+            //System.err.println("null...");
+	  }
+
+	  t=hufts[(tb[0]+(b&inflate_mask[t]))*3+1];
+	  c=hufts[(tb[0]+(b&inflate_mask[t]))*3+2];
 
-    InfTree inftree = new InfTree();
+	  if (c < 16){
+	    b>>>=(t);k-=(t);
+	    blens[index++] = c;
+	  }
+	  else { // c == 16..18
+	    i = c == 18 ? 7 : c - 14;
+	    j = c == 18 ? 11 : 3;
+
+	    while(k<(t+i)){
+	      if(n!=0){
+		r=Z_OK;
+	      }
+	      else{
+		bitb=b; bitk=k; 
+		z.avail_in=n;
+		z.total_in+=p-z.next_in_index;z.next_in_index=p;
+		write=q;
+		return inflate_flush(r);
+	      };
+	      n--;
+	      b|=(z.next_in[p++]&0xff)<<k;
+	      k+=8;
+	    }
+
+	    b>>>=(t);k-=(t);
+
+	    j += (b & inflate_mask[i]);
+
+	    b>>>=(i);k-=(i);
+
+	    i = index;
+	    t = table;
+	    if (i + j > 258 + (t & 0x1f) + ((t >> 5) & 0x1f) ||
+		(c == 16 && i < 1)){
+	      blens=null;
+	      mode = BAD;
+	      z.msg = "invalid bit length repeat";
+	      r = Z_DATA_ERROR;
 
-    InfBlocks(ZStream z, Object checkfn, int w) {
-        hufts = new int[MANY * 3];
-        window = new byte[w];
-        end = w;
-        this.checkfn = checkfn;
-        mode = TYPE;
-        reset(z, null);
-    }
+	      bitb=b; bitk=k; 
+	      z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	      write=q;
+	      return inflate_flush(r);
+	    }
+
+	    c = c == 16 ? blens[i-1] : 0;
+	    do{
+	      blens[i++] = c;
+	    }
+	    while (--j!=0);
+	    index = i;
+	  }
+	}
+
+	tb[0]=-1;
+	{
+	  bl[0] = 9;         // must be <= 9 for lookahead assumptions
+	  bd[0] = 6;         // must be <= 9 for lookahead assumptions
+	  t = table;
+	  t = inftree.inflate_trees_dynamic(257 + (t & 0x1f), 
+					    1 + ((t >> 5) & 0x1f),
+					    blens, bl, bd, tli, tdi, hufts, z);
 
-    void reset(ZStream z, long[] c) {
-        if (c != null) c[0] = check;
+	  if (t != Z_OK){
+	    if (t == Z_DATA_ERROR){
+	      blens=null;
+	      mode = BAD;
+	    }
+	    r = t;
+
+	    bitb=b; bitk=k; 
+	    z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	    write=q;
+	    return inflate_flush(r);
+	  }
+	  codes.init(bl[0], bd[0], hufts, tli[0], hufts, tdi[0]);
+	}
+	mode = CODES;
+      case CODES:
+	bitb=b; bitk=k;
+	z.avail_in=n; z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	write=q;
+
+	if ((r = codes.proc(r)) != Z_STREAM_END){
+	  return inflate_flush(r);
+	}
+	r = Z_OK;
+	codes.free(z);
+
+	p=z.next_in_index; n=z.avail_in;b=bitb;k=bitk;
+	q=write;m=(int)(q<read?read-q-1:end-q);
 
-        if (mode == BTREE || mode == DTREE) {
-        }
+	if (last==0){
+	  mode = TYPE;
+	  break;
+	}
+	mode = DRY;
+      case DRY:
+	write=q; 
+	r=inflate_flush(r); 
+	q=write; m=(int)(q<read?read-q-1:end-q);
+	if (read != write){
+	  bitb=b; bitk=k; 
+	  z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	  write=q;
+	  return inflate_flush(r);
+	}
+	mode = DONE;
+      case DONE:
+	r = Z_STREAM_END;
 
-        if (mode == CODES) {
-            codes.free(z);
-        }
+	bitb=b; bitk=k; 
+	z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	write=q;
+	return inflate_flush(r);
+      case BAD:
+	r = Z_DATA_ERROR;
+
+	bitb=b; bitk=k; 
+	z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	write=q;
+	return inflate_flush(r);
+
+      default:
+	r = Z_STREAM_ERROR;
 
-        mode = TYPE;
-        bitk = 0;
-        bitb = 0;
-        read = write = 0;
+	bitb=b; bitk=k; 
+	z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	write=q;
+	return inflate_flush(r);
+      }
+    }
+  }
+
+  void free(){
+    reset();
+    window=null;
+    hufts=null;
+    //ZFREE(z, s);
+  }
+
+  void set_dictionary(byte[] d, int start, int n){
+    System.arraycopy(d, start, window, 0, n);
+    read = write = n;
+  }
 
-        if (checkfn != null)
-            z.adler = check = z._adler.adler32(0L, null, 0, 0);
+  // Returns true if inflate is currently at the end of a block generated
+  // by Z_SYNC_FLUSH or Z_FULL_FLUSH. 
+  int sync_point(){
+    return mode == LENS ? 1 : 0;
+  }
+
+  // copy as much as possible from the sliding window to the output area
+  int inflate_flush(int r){
+    int n;
+    int p;
+    int q;
+
+    // local copies of source and destination pointers
+    p = z.next_out_index;
+    q = read;
+
+    // compute number of bytes to copy as far as end of window
+    n = (int)((q <= write ? write : end) - q);
+    if(n > z.avail_out) n = z.avail_out;
+    if(n!=0 && r == Z_BUF_ERROR) r = Z_OK;
+
+    // update counters
+    z.avail_out -= n;
+    z.total_out += n;
+
+    // update check information
+    if(check && n>0){
+      z.adler.update(window, q, n);
     }
 
-    int proc(ZStream z, int r) {
-        int t;              // temporary storage
-        int b;              // bit buffer
-        int k;              // bits in bit buffer
-        int p;              // input data pointer
-        int n;              // bytes available there
-        int q;              // output window write pointer
-        int m;              // bytes to end of window or read pointer
-        // copy input/output information to locals (UPDATE macro restores)
-        {p = z.next_in_index; n = z.avail_in; b = bitb; k = bitk;}
-        {q = write; m = (int)(q < read ? read - q - 1 : end - q);}
-
-        // process input based on current state
-        while (true) {
-            switch (mode) {
-                case TYPE:
-                    while (k < (3)) {
-                        if (n != 0) {
-                            r = Z_OK;
-                        }
-                        else {
-                            bitb = b; bitk = k;
-                            z.avail_in = n;
-                            z.total_in += p - z.next_in_index; z.next_in_index = p;
-                            write = q;
-                            return inflate_flush(z, r);
-                        };
-
-                        n--;
-
-                        b |= (z.next_in[p++] & 0xff) << k;
-
-                        k += 8;
-                    }
-
-                    t = (int)(b & 7);
-                    last = t & 1;
-
-                    switch (t >>> 1) {
-                        case 0:                         // stored
-                            {b >>>= (3); k -= (3);}
-
-                            t = k & 7;                    // go to byte boundary
-                            {b >>>= (t); k -= (t);}
-                            mode = LENS;                  // get length of stored block
-                            break;
-
-                        case 1: {                       // fixed
-                                int[] bl = new int[1];
-                                int[] bd = new int[1];
-                                int[][] tl = new int[1][];
-                                int[][] td = new int[1][];
-                                InfTree.inflate_trees_fixed(bl, bd, tl, td, z);
-                                codes.init(bl[0], bd[0], tl[0], 0, td[0], 0, z);
-                            }
-
-                            {b >>>= (3); k -= (3);}
-
-                            mode = CODES;
-                            break;
-
-                        case 2:                         // dynamic
-                            {b >>>= (3); k -= (3);}
-
-                            mode = TABLE;
-                            break;
-
-                        case 3:                         // illegal
-                            {b >>>= (3); k -= (3);}
-
-                            mode = BAD;
-                            z.msg = "invalid block type";
-                            r = Z_DATA_ERROR;
-                            bitb = b; bitk = k;
-                            z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                            write = q;
-                            return inflate_flush(z, r);
-                    }
-
-                    break;
-
-                case LENS:
-                    while (k < (32)) {
-                        if (n != 0) {
-                            r = Z_OK;
-                        }
-                        else {
-                            bitb = b; bitk = k;
-                            z.avail_in = n;
-                            z.total_in += p - z.next_in_index; z.next_in_index = p;
-                            write = q;
-                            return inflate_flush(z, r);
-                        };
-
-                        n--;
-
-                        b |= (z.next_in[p++] & 0xff) << k;
-
-                        k += 8;
-                    }
+    // copy as far as end of window
+    System.arraycopy(window, q, z.next_out, p, n);
+    p += n;
+    q += n;
 
-                    if ((((~b) >>> 16) & 0xffff) != (b & 0xffff)) {
-                        mode = BAD;
-                        z.msg = "invalid stored block lengths";
-                        r = Z_DATA_ERROR;
-                        bitb = b; bitk = k;
-                        z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                        write = q;
-                        return inflate_flush(z, r);
-                    }
-
-                    left = (b & 0xffff);
-                    b = k = 0;                       // dump bits
-                    mode = left != 0 ? STORED : (last != 0 ? DRY : TYPE);
-                    break;
-
-                case STORED:
-                    if (n == 0) {
-                        bitb = b; bitk = k;
-                        z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                        write = q;
-                        return inflate_flush(z, r);
-                    }
-
-                    if (m == 0) {
-                        if (q == end && read != 0) {
-                            q = 0; m = (int)(q < read ? read - q - 1 : end - q);
-                        }
-
-                        if (m == 0) {
-                            write = q;
-                            r = inflate_flush(z, r);
-                            q = write; m = (int)(q < read ? read - q - 1 : end - q);
-
-                            if (q == end && read != 0) {
-                                q = 0; m = (int)(q < read ? read - q - 1 : end - q);
-                            }
-
-                            if (m == 0) {
-                                bitb = b; bitk = k;
-                                z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                                write = q;
-                                return inflate_flush(z, r);
-                            }
-                        }
-                    }
-
-                    r = Z_OK;
-                    t = left;
-
-                    if (t > n) t = n;
-
-                    if (t > m) t = m;
-
-                    System.arraycopy(z.next_in, p, window, q, t);
-                    p += t;  n -= t;
-                    q += t;  m -= t;
-
-                    if ((left -= t) != 0)
-                        break;
-
-                    mode = last != 0 ? DRY : TYPE;
-                    break;
-
-                case TABLE:
-                    while (k < (14)) {
-                        if (n != 0) {
-                            r = Z_OK;
-                        }
-                        else {
-                            bitb = b; bitk = k;
-                            z.avail_in = n;
-                            z.total_in += p - z.next_in_index; z.next_in_index = p;
-                            write = q;
-                            return inflate_flush(z, r);
-                        };
-
-                        n--;
-
-                        b |= (z.next_in[p++] & 0xff) << k;
-
-                        k += 8;
-                    }
-
-                    table = t = (b & 0x3fff);
-
-                    if ((t & 0x1f) > 29 || ((t >> 5) & 0x1f) > 29) {
-                        mode = BAD;
-                        z.msg = "too many length or distance symbols";
-                        r = Z_DATA_ERROR;
-                        bitb = b; bitk = k;
-                        z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                        write = q;
-                        return inflate_flush(z, r);
-                    }
-
-                    t = 258 + (t & 0x1f) + ((t >> 5) & 0x1f);
-
-                    if (blens == null || blens.length < t) {
-                        blens = new int[t];
-                    }
-                    else {
-                        for (int i = 0; i < t; i++) {blens[i] = 0;}
-                    }
-
-                    {b >>>= (14); k -= (14);}
-
-                    index = 0;
-                    mode = BTREE;
+    // see if more to copy at beginning of window
+    if (q == end){
+      // wrap pointers
+      q = 0;
+      if (write == end)
+        write = 0;
 
-                case BTREE:
-                    while (index < 4 + (table >>> 10)) {
-                        while (k < (3)) {
-                            if (n != 0) {
-                                r = Z_OK;
-                            }
-                            else {
-                                bitb = b; bitk = k;
-                                z.avail_in = n;
-                                z.total_in += p - z.next_in_index; z.next_in_index = p;
-                                write = q;
-                                return inflate_flush(z, r);
-                            };
-
-                            n--;
-
-                            b |= (z.next_in[p++] & 0xff) << k;
-
-                            k += 8;
-                        }
-
-                        blens[border[index++]] = b & 7;
-                        {b >>>= (3); k -= (3);}
-                    }
-
-                    while (index < 19) {
-                        blens[border[index++]] = 0;
-                    }
-
-                    bb[0] = 7;
-                    t = inftree.inflate_trees_bits(blens, bb, tb, hufts, z);
-
-                    if (t != Z_OK) {
-                        r = t;
-
-                        if (r == Z_DATA_ERROR) {
-                            blens = null;
-                            mode = BAD;
-                        }
-
-                        bitb = b; bitk = k;
-                        z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                        write = q;
-                        return inflate_flush(z, r);
-                    }
-
-                    index = 0;
-                    mode = DTREE;
-
-                case DTREE:
-                    while (true) {
-                        t = table;
-
-                        if (!(index < 258 + (t & 0x1f) + ((t >> 5) & 0x1f))) {
-                            break;
-                        }
+      // compute bytes to copy
+      n = write - q;
+      if (n > z.avail_out) n = z.avail_out;
+      if (n!=0 && r == Z_BUF_ERROR) r = Z_OK;
 
-                        int[] h;
-                        int i, j, c;
-                        t = bb[0];
-
-                        while (k < (t)) {
-                            if (n != 0) {
-                                r = Z_OK;
-                            }
-                            else {
-                                bitb = b; bitk = k;
-                                z.avail_in = n;
-                                z.total_in += p - z.next_in_index; z.next_in_index = p;
-                                write = q;
-                                return inflate_flush(z, r);
-                            };
-
-                            n--;
-
-                            b |= (z.next_in[p++] & 0xff) << k;
-
-                            k += 8;
-                        }
-
-                        if (tb[0] == -1) {
-                            //System.err.println("null...");
-                        }
-
-                        t = hufts[(tb[0] + (b & inflate_mask[t])) * 3 + 1];
-                        c = hufts[(tb[0] + (b & inflate_mask[t])) * 3 + 2];
-
-                        if (c < 16) {
-                            b >>>= (t); k -= (t);
-                            blens[index++] = c;
-                        }
-                        else { // c == 16..18
-                            i = c == 18 ? 7 : c - 14;
-                            j = c == 18 ? 11 : 3;
-
-                            while (k < (t + i)) {
-                                if (n != 0) {
-                                    r = Z_OK;
-                                }
-                                else {
-                                    bitb = b; bitk = k;
-                                    z.avail_in = n;
-                                    z.total_in += p - z.next_in_index; z.next_in_index = p;
-                                    write = q;
-                                    return inflate_flush(z, r);
-                                };
-
-                                n--;
-
-                                b |= (z.next_in[p++] & 0xff) << k;
-
-                                k += 8;
-                            }
+      // update counters
+      z.avail_out -= n;
+      z.total_out += n;
 
-                            b >>>= (t); k -= (t);
-                            j += (b & inflate_mask[i]);
-                            b >>>= (i); k -= (i);
-                            i = index;
-                            t = table;
-
-                            if (i + j > 258 + (t & 0x1f) + ((t >> 5) & 0x1f) ||
-                                    (c == 16 && i < 1)) {
-                                blens = null;
-                                mode = BAD;
-                                z.msg = "invalid bit length repeat";
-                                r = Z_DATA_ERROR;
-                                bitb = b; bitk = k;
-                                z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                                write = q;
-                                return inflate_flush(z, r);
-                            }
-
-                            c = c == 16 ? blens[i - 1] : 0;
-
-                            do {
-                                blens[i++] = c;
-                            }
-                            while (--j != 0);
-
-                            index = i;
-                        }
-                    }
-
-                    tb[0] = -1;
-                    {
-                        int[] bl = new int[1];
-                        int[] bd = new int[1];
-                        int[] tl = new int[1];
-                        int[] td = new int[1];
-                        bl[0] = 9;         // must be <= 9 for lookahead assumptions
-                        bd[0] = 6;         // must be <= 9 for lookahead assumptions
-                        t = table;
-                        t = inftree.inflate_trees_dynamic(257 + (t & 0x1f),
-                                                          1 + ((t >> 5) & 0x1f),
-                                                          blens, bl, bd, tl, td, hufts, z);
-
-                        if (t != Z_OK) {
-                            if (t == Z_DATA_ERROR) {
-                                blens = null;
-                                mode = BAD;
-                            }
-
-                            r = t;
-                            bitb = b; bitk = k;
-                            z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                            write = q;
-                            return inflate_flush(z, r);
-                        }
+      // update check information
+      if(check && n>0){
+	z.adler.update(window, q, n);
+      }
 
-                        codes.init(bl[0], bd[0], hufts, tl[0], hufts, td[0], z);
-                    }
-                    mode = CODES;
-
-                case CODES:
-                    bitb = b; bitk = k;
-                    z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                    write = q;
-
-                    if ((r = codes.proc(this, z, r)) != Z_STREAM_END) {
-                        return inflate_flush(z, r);
-                    }
-
-                    r = Z_OK;
-                    codes.free(z);
-                    p = z.next_in_index; n = z.avail_in; b = bitb; k = bitk;
-                    q = write; m = (int)(q < read ? read - q - 1 : end - q);
-
-                    if (last == 0) {
-                        mode = TYPE;
-                        break;
-                    }
-
-                    mode = DRY;
-
-                case DRY:
-                    write = q;
-                    r = inflate_flush(z, r);
-                    q = write; m = (int)(q < read ? read - q - 1 : end - q);
-
-                    if (read != write) {
-                        bitb = b; bitk = k;
-                        z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                        write = q;
-                        return inflate_flush(z, r);
-                    }
-
-                    mode = DONE;
-
-                case DONE:
-                    r = Z_STREAM_END;
-                    bitb = b; bitk = k;
-                    z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                    write = q;
-                    return inflate_flush(z, r);
-
-                case BAD:
-                    r = Z_DATA_ERROR;
-                    bitb = b; bitk = k;
-                    z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                    write = q;
-                    return inflate_flush(z, r);
-
-                default:
-                    r = Z_STREAM_ERROR;
-                    bitb = b; bitk = k;
-                    z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                    write = q;
-                    return inflate_flush(z, r);
-            }
-        }
+      // copy
+      System.arraycopy(window, q, z.next_out, p, n);
+      p += n;
+      q += n;
     }
 
-    void free(ZStream z) {
-        reset(z, null);
-        window = null;
-        hufts = null;
-        //ZFREE(z, s);
-    }
-
-    void set_dictionary(byte[] d, int start, int n) {
-        System.arraycopy(d, start, window, 0, n);
-        read = write = n;
-    }
-
-    // Returns true if inflate is currently at the end of a block generated
-    // by Z_SYNC_FLUSH or Z_FULL_FLUSH.
-    int sync_point() {
-        return mode == LENS ? 1 : 0;
-    }
-
-    // copy as much as possible from the sliding window to the output area
-    int inflate_flush(ZStream z, int r) {
-        int n;
-        int p;
-        int q;
-        // local copies of source and destination pointers
-        p = z.next_out_index;
-        q = read;
-        // compute number of bytes to copy as far as end of window
-        n = (int)((q <= write ? write : end) - q);
-
-        if (n > z.avail_out) n = z.avail_out;
-
-        if (n != 0 && r == Z_BUF_ERROR) r = Z_OK;
-
-        // update counters
-        z.avail_out -= n;
-        z.total_out += n;
-
-        // update check information
-        if (checkfn != null)
-            z.adler = check = z._adler.adler32(check, window, q, n);
+    // update pointers
+    z.next_out_index = p;
+    read = q;
 
-        // copy as far as end of window
-        System.arraycopy(window, q, z.next_out, p, n);
-        p += n;
-        q += n;
-
-        // see if more to copy at beginning of window
-        if (q == end) {
-            // wrap pointers
-            q = 0;
-
-            if (write == end)
-                write = 0;
-
-            // compute bytes to copy
-            n = write - q;
-
-            if (n > z.avail_out) n = z.avail_out;
-
-            if (n != 0 && r == Z_BUF_ERROR) r = Z_OK;
-
-            // update counters
-            z.avail_out -= n;
-            z.total_out += n;
-
-            // update check information
-            if (checkfn != null)
-                z.adler = check = z._adler.adler32(check, window, q, n);
-
-            // copy
-            System.arraycopy(window, q, z.next_out, p, n);
-            p += n;
-            q += n;
-        }
-
-        // update pointers
-        z.next_out_index = p;
-        read = q;
-        // done
-        return r;
-    }
+    // done
+    return r;
+  }
 }
--- a/src/com/jcraft/jzlib/InfCodes.java	Fri Aug 01 11:57:17 2014 -0700
+++ b/src/com/jcraft/jzlib/InfCodes.java	Fri Aug 01 13:34:58 2014 -0700
@@ -8,8 +8,8 @@
   1. Redistributions of source code must retain the above copyright notice,
      this list of conditions and the following disclaimer.
 
-  2. Redistributions in binary form must reproduce the above copyright
-     notice, this list of conditions and the following disclaimer in
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
      the documentation and/or other materials provided with the distribution.
 
   3. The names of the authors may not be used to endorse or promote products
@@ -34,566 +34,577 @@
 
 package com.jcraft.jzlib;
 
-final class InfCodes {
+final class InfCodes{
 
-    static final private int[] inflate_mask = {
-        0x00000000, 0x00000001, 0x00000003, 0x00000007, 0x0000000f,
-        0x0000001f, 0x0000003f, 0x0000007f, 0x000000ff, 0x000001ff,
-        0x000003ff, 0x000007ff, 0x00000fff, 0x00001fff, 0x00003fff,
-        0x00007fff, 0x0000ffff
-    };
+  static final private int[] inflate_mask = {
+    0x00000000, 0x00000001, 0x00000003, 0x00000007, 0x0000000f,
+    0x0000001f, 0x0000003f, 0x0000007f, 0x000000ff, 0x000001ff,
+    0x000003ff, 0x000007ff, 0x00000fff, 0x00001fff, 0x00003fff,
+    0x00007fff, 0x0000ffff
+  };
 
-    static final private int Z_OK = 0;
-    static final private int Z_STREAM_END = 1;
-    static final private int Z_NEED_DICT = 2;
-    static final private int Z_ERRNO = -1;
-    static final private int Z_STREAM_ERROR = -2;
-    static final private int Z_DATA_ERROR = -3;
-    static final private int Z_MEM_ERROR = -4;
-    static final private int Z_BUF_ERROR = -5;
-    static final private int Z_VERSION_ERROR = -6;
-
-    // waiting for "i:"=input,
-    //             "o:"=output,
-    //             "x:"=nothing
-    static final private int START = 0; // x: set up for LEN
-    static final private int LEN = 1;  // i: get length/literal/eob next
-    static final private int LENEXT = 2; // i: getting length extra (have base)
-    static final private int DIST = 3; // i: get distance next
-    static final private int DISTEXT = 4; // i: getting distance extra
-    static final private int COPY = 5; // o: copying bytes in window, waiting for space
-    static final private int LIT = 6;  // o: got literal, waiting for output space
-    static final private int WASH = 7; // o: got eob, possibly still output waiting
-    static final private int END = 8;  // x: got eob and all data flushed
-    static final private int BADCODE = 9; // x: got error
-
-    int mode;      // current inflate_codes mode
+  static final private int Z_OK=0;
+  static final private int Z_STREAM_END=1;
+  static final private int Z_NEED_DICT=2;
+  static final private int Z_ERRNO=-1;
+  static final private int Z_STREAM_ERROR=-2;
+  static final private int Z_DATA_ERROR=-3;
+  static final private int Z_MEM_ERROR=-4;
+  static final private int Z_BUF_ERROR=-5;
+  static final private int Z_VERSION_ERROR=-6;
 
-    // mode dependent information
-    int len;
+  // waiting for "i:"=input,
+  //             "o:"=output,
+  //             "x:"=nothing
+  static final private int START=0;  // x: set up for LEN
+  static final private int LEN=1;    // i: get length/literal/eob next
+  static final private int LENEXT=2; // i: getting length extra (have base)
+  static final private int DIST=3;   // i: get distance next
+  static final private int DISTEXT=4;// i: getting distance extra
+  static final private int COPY=5;   // o: copying bytes in window, waiting for space
+  static final private int LIT=6;    // o: got literal, waiting for output space
+  static final private int WASH=7;   // o: got eob, possibly still output waiting
+  static final private int END=8;    // x: got eob and all data flushed
+  static final private int BADCODE=9;// x: got error
 
-    int[] tree; // pointer into tree
-    int tree_index = 0;
-    int need;   // bits needed
+  int mode;      // current inflate_codes mode
 
-    int lit;
-
-    // if EXT or COPY, where and how much
-    int get;              // bits to get for extra
-    int dist;             // distance back to copy from
+  // mode dependent information
+  int len;
 
-    byte lbits;           // ltree bits decoded per branch
-    byte dbits;           // dtree bits decoder per branch
-    int[] ltree;          // literal/length/eob tree
-    int ltree_index;      // literal/length/eob tree
-    int[] dtree;          // distance tree
-    int dtree_index;      // distance tree
+  int[] tree; // pointer into tree
+  int tree_index=0;
+  int need;   // bits needed
+
+  int lit;
+
+  // if EXT or COPY, where and how much
+  int get;              // bits to get for extra
+  int dist;             // distance back to copy from
 
-    InfCodes() {
-    }
-    void init(int bl, int bd,
-              int[] tl, int tl_index,
-              int[] td, int td_index, ZStream z) {
-        mode = START;
-        lbits = (byte)bl;
-        dbits = (byte)bd;
-        ltree = tl;
-        ltree_index = tl_index;
-        dtree = td;
-        dtree_index = td_index;
-        tree = null;
-    }
+  byte lbits;           // ltree bits decoded per branch
+  byte dbits;           // dtree bits decoder per branch
+  int[] ltree;          // literal/length/eob tree
+  int ltree_index;      // literal/length/eob tree
+  int[] dtree;          // distance tree
+  int dtree_index;      // distance tree
+
+  private final ZStream z;
+  private final InfBlocks s;
+  InfCodes(ZStream z, InfBlocks s){
+    this.z=z; 
+    this.s=s; 
+  }
 
-    int proc(InfBlocks s, ZStream z, int r) {
-        int j;              // temporary storage
-        int[] t;            // temporary pointer
-        int tindex;         // temporary pointer
-        int e;              // extra bits or operation
-        int b = 0;          // bit buffer
-        int k = 0;          // bits in bit buffer
-        int p = 0;          // input data pointer
-        int n;              // bytes available there
-        int q;              // output window write pointer
-        int m;              // bytes to end of window or read pointer
-        int f;              // pointer to copy strings from
-        // copy input/output information to locals (UPDATE macro restores)
-        p = z.next_in_index; n = z.avail_in; b = s.bitb; k = s.bitk;
-        q = s.write; m = q < s.read ? s.read - q - 1 : s.end - q;
+  void init(int bl, int bd,
+	   int[] tl, int tl_index,
+	   int[] td, int td_index){
+    mode=START;
+    lbits=(byte)bl;
+    dbits=(byte)bd;
+    ltree=tl;
+    ltree_index=tl_index;
+    dtree = td;
+    dtree_index=td_index;
+    tree=null;
+  }
 
-        // process input and output based on current state
-        while (true) {
-            switch (mode) {
-                // waiting for "i:"=input, "o:"=output, "x:"=nothing
-                case START:         // x: set up for LEN
-                    if (m >= 258 && n >= 10) {
-                        s.bitb = b; s.bitk = k;
-                        z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                        s.write = q;
-                        r = inflate_fast(lbits, dbits,
-                                         ltree, ltree_index,
-                                         dtree, dtree_index,
-                                         s, z);
-                        p = z.next_in_index; n = z.avail_in; b = s.bitb; k = s.bitk;
-                        q = s.write; m = q < s.read ? s.read - q - 1 : s.end - q;
+  int proc(int r){ 
+    int j;              // temporary storage
+    int[] t;            // temporary pointer
+    int tindex;         // temporary pointer
+    int e;              // extra bits or operation
+    int b=0;            // bit buffer
+    int k=0;            // bits in bit buffer
+    int p=0;            // input data pointer
+    int n;              // bytes available there
+    int q;              // output window write pointer
+    int m;              // bytes to end of window or read pointer
+    int f;              // pointer to copy strings from
+
+    // copy input/output information to locals (UPDATE macro restores)
+    p=z.next_in_index;n=z.avail_in;b=s.bitb;k=s.bitk;
+    q=s.write;m=q<s.read?s.read-q-1:s.end-q;
+
+    // process input and output based on current state
+    while (true){
+      switch (mode){
+	// waiting for "i:"=input, "o:"=output, "x:"=nothing
+      case START:         // x: set up for LEN
+	if (m >= 258 && n >= 10){
 
-                        if (r != Z_OK) {
-                            mode = r == Z_STREAM_END ? WASH : BADCODE;
-                            break;
-                        }
-                    }
+	  s.bitb=b;s.bitk=k;
+	  z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	  s.write=q;
+	  r = inflate_fast(lbits, dbits, 
+			   ltree, ltree_index, 
+			   dtree, dtree_index,
+			   s, z);
 
-                    need = lbits;
-                    tree = ltree;
-                    tree_index = ltree_index;
-                    mode = LEN;
+	  p=z.next_in_index;n=z.avail_in;b=s.bitb;k=s.bitk;
+	  q=s.write;m=q<s.read?s.read-q-1:s.end-q;
 
-                case LEN:           // i: get length/literal/eob next
-                    j = need;
+	  if (r != Z_OK){
+	    mode = r == Z_STREAM_END ? WASH : BADCODE;
+	    break;
+	  }
+	}
+	need = lbits;
+	tree = ltree;
+	tree_index=ltree_index;
 
-                    while (k < (j)) {
-                        if (n != 0)r = Z_OK;
-                        else {
-                            s.bitb = b; s.bitk = k;
-                            z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                            s.write = q;
-                            return s.inflate_flush(z, r);
-                        }
+	mode = LEN;
+      case LEN:           // i: get length/literal/eob next
+	j = need;
+
+	while(k<(j)){
+	  if(n!=0)r=Z_OK;
+	  else{
 
-                        n--;
-                        b |= (z.next_in[p++] & 0xff) << k;
-                        k += 8;
-                    }
+	    s.bitb=b;s.bitk=k;
+	    z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	    s.write=q;
+	    return s.inflate_flush(r);
+	  }
+	  n--;
+	  b|=(z.next_in[p++]&0xff)<<k;
+	  k+=8;
+	}
 
-                    tindex = (tree_index + (b & inflate_mask[j])) * 3;
-                    b >>>= (tree[tindex + 1]);
-                    k -= (tree[tindex + 1]);
-                    e = tree[tindex];
+	tindex=(tree_index+(b&inflate_mask[j]))*3;
 
-                    if (e == 0) {             // literal
-                        lit = tree[tindex + 2];
-                        mode = LIT;
-                        break;
-                    }
+	b>>>=(tree[tindex+1]);
+	k-=(tree[tindex+1]);
 
-                    if ((e & 16) != 0) {       // length
-                        get = e & 15;
-                        len = tree[tindex + 2];
-                        mode = LENEXT;
-                        break;
-                    }
-
-                    if ((e & 64) == 0) {       // next table
-                        need = e;
-                        tree_index = tindex / 3 + tree[tindex + 2];
-                        break;
-                    }
+	e=tree[tindex];
 
-                    if ((e & 32) != 0) {            // end of block
-                        mode = WASH;
-                        break;
-                    }
+	if(e == 0){               // literal
+	  lit = tree[tindex+2];
+	  mode = LIT;
+	  break;
+	}
+	if((e & 16)!=0 ){          // length
+	  get = e & 15;
+	  len = tree[tindex+2];
+	  mode = LENEXT;
+	  break;
+	}
+	if ((e & 64) == 0){        // next table
+	  need = e;
+	  tree_index = tindex/3+tree[tindex+2];
+	  break;
+	}
+	if ((e & 32)!=0){               // end of block
+	  mode = WASH;
+	  break;
+	}
+	mode = BADCODE;        // invalid code
+	z.msg = "invalid literal/length code";
+	r = Z_DATA_ERROR;
 
-                    mode = BADCODE;        // invalid code
-                    z.msg = "invalid literal/length code";
-                    r = Z_DATA_ERROR;
-                    s.bitb = b; s.bitk = k;
-                    z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                    s.write = q;
-                    return s.inflate_flush(z, r);
+	s.bitb=b;s.bitk=k;
+	z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	s.write=q;
+	return s.inflate_flush(r);
 
-                case LENEXT:        // i: getting length extra (have base)
-                    j = get;
+      case LENEXT:        // i: getting length extra (have base)
+	j = get;
 
-                    while (k < (j)) {
-                        if (n != 0)r = Z_OK;
-                        else {
-                            s.bitb = b; s.bitk = k;
-                            z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                            s.write = q;
-                            return s.inflate_flush(z, r);
-                        }
+	while(k<(j)){
+	  if(n!=0)r=Z_OK;
+	  else{
 
-                        n--; b |= (z.next_in[p++] & 0xff) << k;
-                        k += 8;
-                    }
+	    s.bitb=b;s.bitk=k;
+	    z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	    s.write=q;
+	    return s.inflate_flush(r);
+	  }
+	  n--; b|=(z.next_in[p++]&0xff)<<k;
+	  k+=8;
+	}
+
+	len += (b & inflate_mask[j]);
+
+	b>>=j;
+	k-=j;
 
-                    len += (b & inflate_mask[j]);
-                    b >>= j;
-                    k -= j;
-                    need = dbits;
-                    tree = dtree;
-                    tree_index = dtree_index;
-                    mode = DIST;
-
-                case DIST:          // i: get distance next
-                    j = need;
+	need = dbits;
+	tree = dtree;
+	tree_index=dtree_index;
+	mode = DIST;
+      case DIST:          // i: get distance next
+	j = need;
 
-                    while (k < (j)) {
-                        if (n != 0)r = Z_OK;
-                        else {
-                            s.bitb = b; s.bitk = k;
-                            z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                            s.write = q;
-                            return s.inflate_flush(z, r);
-                        }
+	while(k<(j)){
+	  if(n!=0)r=Z_OK;
+	  else{
 
-                        n--; b |= (z.next_in[p++] & 0xff) << k;
-                        k += 8;
-                    }
+	    s.bitb=b;s.bitk=k;
+	    z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	    s.write=q;
+	    return s.inflate_flush(r);
+	  }
+	  n--; b|=(z.next_in[p++]&0xff)<<k;
+	  k+=8;
+	}
 
-                    tindex = (tree_index + (b & inflate_mask[j])) * 3;
-                    b >>= tree[tindex + 1];
-                    k -= tree[tindex + 1];
-                    e = (tree[tindex]);
+	tindex=(tree_index+(b & inflate_mask[j]))*3;
 
-                    if ((e & 16) != 0) {           // distance
-                        get = e & 15;
-                        dist = tree[tindex + 2];
-                        mode = DISTEXT;
-                        break;
-                    }
+	b>>=tree[tindex+1];
+	k-=tree[tindex+1];
 
-                    if ((e & 64) == 0) {       // next table
-                        need = e;
-                        tree_index = tindex / 3 + tree[tindex + 2];
-                        break;
-                    }
+	e = (tree[tindex]);
+	if((e & 16)!=0){               // distance
+	  get = e & 15;
+	  dist = tree[tindex+2];
+	  mode = DISTEXT;
+	  break;
+	}
+	if ((e & 64) == 0){        // next table
+	  need = e;
+	  tree_index = tindex/3 + tree[tindex+2];
+	  break;
+	}
+	mode = BADCODE;        // invalid code
+	z.msg = "invalid distance code";
+	r = Z_DATA_ERROR;
 
-                    mode = BADCODE;        // invalid code
-                    z.msg = "invalid distance code";
-                    r = Z_DATA_ERROR;
-                    s.bitb = b; s.bitk = k;
-                    z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                    s.write = q;
-                    return s.inflate_flush(z, r);
+	s.bitb=b;s.bitk=k;
+	z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	s.write=q;
+	return s.inflate_flush(r);
 
-                case DISTEXT:       // i: getting distance extra
-                    j = get;
+      case DISTEXT:       // i: getting distance extra
+	j = get;
+
+	while(k<(j)){
+	  if(n!=0)r=Z_OK;
+	  else{
 
-                    while (k < (j)) {
-                        if (n != 0)r = Z_OK;
-                        else {
-                            s.bitb = b; s.bitk = k;
-                            z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                            s.write = q;
-                            return s.inflate_flush(z, r);
-                        }
+	    s.bitb=b;s.bitk=k;
+	    z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	    s.write=q;
+	    return s.inflate_flush(r);
+	  }
+	  n--; b|=(z.next_in[p++]&0xff)<<k;
+	  k+=8;
+	}
 
-                        n--; b |= (z.next_in[p++] & 0xff) << k;
-                        k += 8;
-                    }
+	dist += (b & inflate_mask[j]);
 
-                    dist += (b & inflate_mask[j]);
-                    b >>= j;
-                    k -= j;
-                    mode = COPY;
-
-                case COPY:          // o: copying bytes in window, waiting for space
-                    f = q - dist;
+	b>>=j;
+	k-=j;
 
-                    while (f < 0) {   // modulo window size-"while" instead
-                        f += s.end;     // of "if" handles invalid distances
-                    }
-
-                    while (len != 0) {
-                        if (m == 0) {
-                            if (q == s.end && s.read != 0) {q = 0; m = q < s.read ? s.read - q - 1 : s.end - q;}
-
-                            if (m == 0) {
-                                s.write = q; r = s.inflate_flush(z, r);
-                                q = s.write; m = q < s.read ? s.read - q - 1 : s.end - q;
+	mode = COPY;
+      case COPY:          // o: copying bytes in window, waiting for space
+        f = q - dist;
+        while(f < 0){     // modulo window size-"while" instead
+          f += s.end;     // of "if" handles invalid distances
+	}
+	while (len!=0){
 
-                                if (q == s.end && s.read != 0) {q = 0; m = q < s.read ? s.read - q - 1 : s.end - q;}
+	  if(m==0){
+	    if(q==s.end&&s.read!=0){q=0;m=q<s.read?s.read-q-1:s.end-q;}
+	    if(m==0){
+	      s.write=q; r=s.inflate_flush(r);
+	      q=s.write;m=q<s.read?s.read-q-1:s.end-q;
+
+	      if(q==s.end&&s.read!=0){q=0;m=q<s.read?s.read-q-1:s.end-q;}
 
-                                if (m == 0) {
-                                    s.bitb = b; s.bitk = k;
-                                    z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                                    s.write = q;
-                                    return s.inflate_flush(z, r);
-                                }
-                            }
-                        }
-
-                        s.window[q++] = s.window[f++]; m--;
+	      if(m==0){
+		s.bitb=b;s.bitk=k;
+		z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+		s.write=q;
+		return s.inflate_flush(r);
+	      }  
+	    }
+	  }
 
-                        if (f == s.end)
-                            f = 0;
-
-                        len--;
-                    }
-
-                    mode = START;
-                    break;
-
-                case LIT:           // o: got literal, waiting for output space
-                    if (m == 0) {
-                        if (q == s.end && s.read != 0) {q = 0; m = q < s.read ? s.read - q - 1 : s.end - q;}
+	  s.window[q++]=s.window[f++]; m--;
 
-                        if (m == 0) {
-                            s.write = q; r = s.inflate_flush(z, r);
-                            q = s.write; m = q < s.read ? s.read - q - 1 : s.end - q;
-
-                            if (q == s.end && s.read != 0) {q = 0; m = q < s.read ? s.read - q - 1 : s.end - q;}
+	  if (f == s.end)
+            f = 0;
+	  len--;
+	}
+	mode = START;
+	break;
+      case LIT:           // o: got literal, waiting for output space
+	if(m==0){
+	  if(q==s.end&&s.read!=0){q=0;m=q<s.read?s.read-q-1:s.end-q;}
+	  if(m==0){
+	    s.write=q; r=s.inflate_flush(r);
+	    q=s.write;m=q<s.read?s.read-q-1:s.end-q;
 
-                            if (m == 0) {
-                                s.bitb = b; s.bitk = k;
-                                z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                                s.write = q;
-                                return s.inflate_flush(z, r);
-                            }
-                        }
-                    }
+	    if(q==s.end&&s.read!=0){q=0;m=q<s.read?s.read-q-1:s.end-q;}
+	    if(m==0){
+	      s.bitb=b;s.bitk=k;
+	      z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	      s.write=q;
+	      return s.inflate_flush(r);
+	    }
+	  }
+	}
+	r=Z_OK;
 
-                    r = Z_OK;
-                    s.window[q++] = (byte)lit; m--;
-                    mode = START;
-                    break;
+	s.window[q++]=(byte)lit; m--;
 
-                case WASH:           // o: got eob, possibly more output
-                    if (k > 7) {       // return unused byte, if any
-                        k -= 8;
-                        n++;
-                        p--;             // can always return one
-                    }
+	mode = START;
+	break;
+      case WASH:           // o: got eob, possibly more output
+	if (k > 7){        // return unused byte, if any
+	  k -= 8;
+	  n++;
+	  p--;             // can always return one
+	}
 
-                    s.write = q; r = s.inflate_flush(z, r);
-                    q = s.write; m = q < s.read ? s.read - q - 1 : s.end - q;
+	s.write=q; r=s.inflate_flush(r);
+	q=s.write;m=q<s.read?s.read-q-1:s.end-q;
 
-                    if (s.read != s.write) {
-                        s.bitb = b; s.bitk = k;
-                        z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                        s.write = q;
-                        return s.inflate_flush(z, r);
-                    }
-
-                    mode = END;
+	if (s.read != s.write){
+	  s.bitb=b;s.bitk=k;
+	  z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	  s.write=q;
+	  return s.inflate_flush(r);
+	}
+	mode = END;
+      case END:
+	r = Z_STREAM_END;
+	s.bitb=b;s.bitk=k;
+	z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	s.write=q;
+	return s.inflate_flush(r);
 
-                case END:
-                    r = Z_STREAM_END;
-                    s.bitb = b; s.bitk = k;
-                    z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                    s.write = q;
-                    return s.inflate_flush(z, r);
+      case BADCODE:       // x: got error
+
+	r = Z_DATA_ERROR;
 
-                case BADCODE:       // x: got error
-                    r = Z_DATA_ERROR;
-                    s.bitb = b; s.bitk = k;
-                    z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                    s.write = q;
-                    return s.inflate_flush(z, r);
+	s.bitb=b;s.bitk=k;
+	z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	s.write=q;
+	return s.inflate_flush(r);
+
+      default:
+	r = Z_STREAM_ERROR;
 
-                default:
-                    r = Z_STREAM_ERROR;
-                    s.bitb = b; s.bitk = k;
-                    z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                    s.write = q;
-                    return s.inflate_flush(z, r);
-            }
-        }
+	s.bitb=b;s.bitk=k;
+	z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	s.write=q;
+	return s.inflate_flush(r);
+      }
     }
+  }
 
-    void free(ZStream z) {
-        //  ZFREE(z, c);
-    }
+  void free(ZStream z){
+    //  ZFREE(z, c);
+  }
+
+  // Called with number of bytes left to write in window at least 258
+  // (the maximum string length) and number of input bytes available
+  // at least ten.  The ten bytes are six bytes for the longest length/
+  // distance pair plus four bytes for overloading the bit buffer.
 
-    // Called with number of bytes left to write in window at least 258
-    // (the maximum string length) and number of input bytes available
-    // at least ten.  The ten bytes are six bytes for the longest length/
-    // distance pair plus four bytes for overloading the bit buffer.
+  int inflate_fast(int bl, int bd, 
+		   int[] tl, int tl_index,
+		   int[] td, int td_index,
+		   InfBlocks s, ZStream z){
+    int t;                // temporary pointer
+    int[] tp;             // temporary pointer
+    int tp_index;         // temporary pointer
+    int e;                // extra bits or operation
+    int b;                // bit buffer
+    int k;                // bits in bit buffer
+    int p;                // input data pointer
+    int n;                // bytes available there
+    int q;                // output window write pointer
+    int m;                // bytes to end of window or read pointer
+    int ml;               // mask for literal/length tree
+    int md;               // mask for distance tree
+    int c;                // bytes to copy
+    int d;                // distance back to copy from
+    int r;                // copy source pointer
 
-    int inflate_fast(int bl, int bd,
-                     int[] tl, int tl_index,
-                     int[] td, int td_index,
-                     InfBlocks s, ZStream z) {
-        int t;                // temporary pointer
-        int[] tp;             // temporary pointer
-        int tp_index;         // temporary pointer
-        int e;                // extra bits or operation
-        int b;                // bit buffer
-        int k;                // bits in bit buffer
-        int p;                // input data pointer
-        int n;                // bytes available there
-        int q;                // output window write pointer
-        int m;                // bytes to end of window or read pointer
-        int ml;               // mask for literal/length tree
-        int md;               // mask for distance tree
-        int c;                // bytes to copy
-        int d;                // distance back to copy from
-        int r;                // copy source pointer
-        int tp_index_t_3;     // (tp_index+t)*3
-        // load input, output, bit values
-        p = z.next_in_index; n = z.avail_in; b = s.bitb; k = s.bitk;
-        q = s.write; m = q < s.read ? s.read - q - 1 : s.end - q;
-        // initialize masks
-        ml = inflate_mask[bl];
-        md = inflate_mask[bd];
+    int tp_index_t_3;     // (tp_index+t)*3
+
+    // load input, output, bit values
+    p=z.next_in_index;n=z.avail_in;b=s.bitb;k=s.bitk;
+    q=s.write;m=q<s.read?s.read-q-1:s.end-q;
+
+    // initialize masks
+    ml = inflate_mask[bl];
+    md = inflate_mask[bd];
+
+    // do until not enough input or output space for fast loop
+    do {                          // assume called with m >= 258 && n >= 10
+      // get literal/length code
+      while(k<(20)){              // max bits for literal/length code
+	n--;
+	b|=(z.next_in[p++]&0xff)<<k;k+=8;
+      }
 
-        // do until not enough input or output space for fast loop
-        do {                          // assume called with m >= 258 && n >= 10
-            // get literal/length code
-            while (k < (20)) {          // max bits for literal/length code
-                n--;
-                b |= (z.next_in[p++] & 0xff) << k; k += 8;
-            }
-
-            t = b & ml;
-            tp = tl;
-            tp_index = tl_index;
-            tp_index_t_3 = (tp_index + t) * 3;
+      t= b&ml;
+      tp=tl; 
+      tp_index=tl_index;
+      tp_index_t_3=(tp_index+t)*3;
+      if ((e = tp[tp_index_t_3]) == 0){
+	b>>=(tp[tp_index_t_3+1]); k-=(tp[tp_index_t_3+1]);
 
-            if ((e = tp[tp_index_t_3]) == 0) {
-                b >>= (tp[tp_index_t_3 + 1]); k -= (tp[tp_index_t_3 + 1]);
-                s.window[q++] = (byte)tp[tp_index_t_3 + 2];
-                m--;
-                continue;
-            }
+	s.window[q++] = (byte)tp[tp_index_t_3+2];
+	m--;
+	continue;
+      }
+      do {
 
-            do {
-                b >>= (tp[tp_index_t_3 + 1]); k -= (tp[tp_index_t_3 + 1]);
+	b>>=(tp[tp_index_t_3+1]); k-=(tp[tp_index_t_3+1]);
 
-                if ((e & 16) != 0) {
-                    e &= 15;
-                    c = tp[tp_index_t_3 + 2] + ((int)b & inflate_mask[e]);
-                    b >>= e; k -= e;
+	if((e&16)!=0){
+	  e &= 15;
+	  c = tp[tp_index_t_3+2] + ((int)b & inflate_mask[e]);
+
+	  b>>=e; k-=e;
 
-                    // decode distance base of block to copy
-                    while (k < (15)) {       // max bits for distance code
-                        n--;
-                        b |= (z.next_in[p++] & 0xff) << k; k += 8;
-                    }
+	  // decode distance base of block to copy
+	  while(k<(15)){           // max bits for distance code
+	    n--;
+	    b|=(z.next_in[p++]&0xff)<<k;k+=8;
+	  }
 
-                    t = b & md;
-                    tp = td;
-                    tp_index = td_index;
-                    tp_index_t_3 = (tp_index + t) * 3;
-                    e = tp[tp_index_t_3];
+	  t= b&md;
+	  tp=td;
+	  tp_index=td_index;
+          tp_index_t_3=(tp_index+t)*3;
+	  e = tp[tp_index_t_3];
 
-                    do {
-                        b >>= (tp[tp_index_t_3 + 1]); k -= (tp[tp_index_t_3 + 1]);
+	  do {
 
-                        if ((e & 16) != 0) {
-                            // get extra bits to add to distance base
-                            e &= 15;
-
-                            while (k < (e)) {     // get extra bits (up to 13)
-                                n--;
-                                b |= (z.next_in[p++] & 0xff) << k; k += 8;
-                            }
+	    b>>=(tp[tp_index_t_3+1]); k-=(tp[tp_index_t_3+1]);
 
-                            d = tp[tp_index_t_3 + 2] + (b & inflate_mask[e]);
-                            b >>= (e); k -= (e);
-                            // do the copy
-                            m -= c;
+	    if((e&16)!=0){
+	      // get extra bits to add to distance base
+	      e &= 15;
+	      while(k<(e)){         // get extra bits (up to 13)
+		n--;
+		b|=(z.next_in[p++]&0xff)<<k;k+=8;
+	      }
 
-                            if (q >= d) {               // offset before dest
-                                //  just copy
-                                r = q - d;
+	      d = tp[tp_index_t_3+2] + (b&inflate_mask[e]);
+
+	      b>>=(e); k-=(e);
 
-                                if (q - r > 0 && 2 > (q - r)) {
-                                    s.window[q++] = s.window[r++]; // minimum count is three,
-                                    s.window[q++] = s.window[r++]; // so unroll loop a little
-                                    c -= 2;
-                                }
-                                else {
-                                    System.arraycopy(s.window, r, s.window, q, 2);
-                                    q += 2; r += 2; c -= 2;
-                                }
-                            }
-                            else {                 // else offset after destination
-                                r = q - d;
-
-                                do {
-                                    r += s.end;        // force pointer in window
-                                }
-                                while (r < 0);       // covers invalid distances
+	      // do the copy
+	      m -= c;
+	      if (q >= d){                // offset before dest
+		//  just copy
+		r=q-d;
+		if(q-r>0 && 2>(q-r)){           
+		  s.window[q++]=s.window[r++]; // minimum count is three,
+		  s.window[q++]=s.window[r++]; // so unroll loop a little
+		  c-=2;
+		}
+		else{
+		  System.arraycopy(s.window, r, s.window, q, 2);
+		  q+=2; r+=2; c-=2;
+		}
+	      }
+	      else{                  // else offset after destination
+                r=q-d;
+                do{
+                  r+=s.end;          // force pointer in window
+                }while(r<0);         // covers invalid distances
+		e=s.end-r;
+		if(c>e){             // if source crosses,
+		  c-=e;              // wrapped copy
+		  if(q-r>0 && e>(q-r)){           
+		    do{s.window[q++] = s.window[r++];}
+		    while(--e!=0);
+		  }
+		  else{
+		    System.arraycopy(s.window, r, s.window, q, e);
+		    q+=e; r+=e; e=0;
+		  }
+		  r = 0;                  // copy rest from start of window
+		}
 
-                                e = s.end - r;
-
-                                if (c > e) {         // if source crosses,
-                                    c -= e;            // wrapped copy
-
-                                    if (q - r > 0 && e > (q - r)) {
-                                        do {s.window[q++] = s.window[r++];}
-                                        while (--e != 0);
-                                    }
-                                    else {
-                                        System.arraycopy(s.window, r, s.window, q, e);
-                                        q += e; r += e; e = 0;
-                                    }
-
-                                    r = 0;                  // copy rest from start of window
-                                }
-                            }
+	      }
 
-                            // copy all or what's left
-                            if (q - r > 0 && c > (q - r)) {
-                                do {s.window[q++] = s.window[r++];}
-                                while (--c != 0);
-                            }
-                            else {
-                                System.arraycopy(s.window, r, s.window, q, c);
-                                q += c; r += c; c = 0;
-                            }
+	      // copy all or what's left
+	      if(q-r>0 && c>(q-r)){           
+		do{s.window[q++] = s.window[r++];}
+		while(--c!=0);
+	      }
+	      else{
+		System.arraycopy(s.window, r, s.window, q, c);
+		q+=c; r+=c; c=0;
+	      }
+	      break;
+	    }
+	    else if((e&64)==0){
+	      t+=tp[tp_index_t_3+2];
+	      t+=(b&inflate_mask[e]);
+	      tp_index_t_3=(tp_index+t)*3;
+	      e=tp[tp_index_t_3];
+	    }
+	    else{
+	      z.msg = "invalid distance code";
 
-                            break;
-                        }
-                        else if ((e & 64) == 0) {
-                            t += tp[tp_index_t_3 + 2];
-                            t += (b & inflate_mask[e]);
-                            tp_index_t_3 = (tp_index + t) * 3;
-                            e = tp[tp_index_t_3];
-                        }
-                        else {
-                            z.msg = "invalid distance code";
-                            c = z.avail_in - n; c = (k >> 3)<c ? k >> 3: c; n += c; p -= c; k -= c << 3;
-                            s.bitb = b; s.bitk = k;
-                            z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                            s.write = q;
-                            return Z_DATA_ERROR;
-                        }
-                    }
-                    while (true);
+	      c=z.avail_in-n;c=(k>>3)<c?k>>3:c;n+=c;p-=c;k-=c<<3;
+
+	      s.bitb=b;s.bitk=k;
+	      z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	      s.write=q;
 
-                    break;
-                }
+	      return Z_DATA_ERROR;
+	    }
+	  }
+	  while(true);
+	  break;
+	}
+
+	if((e&64)==0){
+	  t+=tp[tp_index_t_3+2];
+	  t+=(b&inflate_mask[e]);
+	  tp_index_t_3=(tp_index+t)*3;
+	  if((e=tp[tp_index_t_3])==0){
+
+	    b>>=(tp[tp_index_t_3+1]); k-=(tp[tp_index_t_3+1]);
 
-                if ((e & 64) == 0) {
-                    t += tp[tp_index_t_3 + 2];
-                    t += (b & inflate_mask[e]);
-                    tp_index_t_3 = (tp_index + t) * 3;
+	    s.window[q++]=(byte)tp[tp_index_t_3+2];
+	    m--;
+	    break;
+	  }
+	}
+	else if((e&32)!=0){
+
+	  c=z.avail_in-n;c=(k>>3)<c?k>>3:c;n+=c;p-=c;k-=c<<3;
+ 
+	  s.bitb=b;s.bitk=k;
+	  z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	  s.write=q;
+
+	  return Z_STREAM_END;
+	}
+	else{
+	  z.msg="invalid literal/length code";
+
+	  c=z.avail_in-n;c=(k>>3)<c?k>>3:c;n+=c;p-=c;k-=c<<3;
 
-                    if ((e = tp[tp_index_t_3]) == 0) {
-                        b >>= (tp[tp_index_t_3 + 1]); k -= (tp[tp_index_t_3 + 1]);
-                        s.window[q++] = (byte)tp[tp_index_t_3 + 2];
-                        m--;
-                        break;
-                    }
-                }
-                else if ((e & 32) != 0) {
-                    c = z.avail_in - n; c = (k >> 3)<c ? k >> 3: c; n += c; p -= c; k -= c << 3;
-                    s.bitb = b; s.bitk = k;
-                    z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                    s.write = q;
-                    return Z_STREAM_END;
-                }
-                else {
-                    z.msg = "invalid literal/length code";
-                    c = z.avail_in - n; c = (k >> 3)<c ? k >> 3: c; n += c; p -= c; k -= c << 3;
-                    s.bitb = b; s.bitk = k;
-                    z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-                    s.write = q;
-                    return Z_DATA_ERROR;
-                }
-            }
-            while (true);
-        }
-        while (m >= 258 && n >= 10);
+	  s.bitb=b;s.bitk=k;
+	  z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+	  s.write=q;
+
+	  return Z_DATA_ERROR;
+	}
+      } 
+      while(true);
+    } 
+    while(m>=258 && n>= 10);
 
-        // not enough input or output--restore pointers and return
-        c = z.avail_in - n; c = (k >> 3)<c ? k >> 3: c; n += c; p -= c; k -= c << 3;
-        s.bitb = b; s.bitk = k;
-        z.avail_in = n; z.total_in += p - z.next_in_index; z.next_in_index = p;
-        s.write = q;
-        return Z_OK;
-    }
+    // not enough input or output--restore pointers and return
+    c=z.avail_in-n;c=(k>>3)<c?k>>3:c;n+=c;p-=c;k-=c<<3;
+
+    s.bitb=b;s.bitk=k;
+    z.avail_in=n;z.total_in+=p-z.next_in_index;z.next_in_index=p;
+    s.write=q;
+
+    return Z_OK;
+  }
 }
--- a/src/com/jcraft/jzlib/InfTree.java	Fri Aug 01 11:57:17 2014 -0700
+++ b/src/com/jcraft/jzlib/InfTree.java	Fri Aug 01 13:34:58 2014 -0700
@@ -8,8 +8,8 @@
   1. Redistributions of source code must retain the above copyright notice,
      this list of conditions and the following disclaimer.
 
-  2. Redistributions in binary form must reproduce the above copyright
-     notice, this list of conditions and the following disclaimer in
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
      the documentation and/or other materials provided with the distribution.
 
   3. The names of the authors may not be used to endorse or promote products
@@ -34,516 +34,485 @@
 
 package com.jcraft.jzlib;
 
-final class InfTree {
+final class InfTree{
 
-    static final private int MANY = 1440;
+  static final private int MANY=1440;
 
-    static final private int Z_OK = 0;
-    static final private int Z_STREAM_END = 1;
-    static final private int Z_NEED_DICT = 2;
-    static final private int Z_ERRNO = -1;
-    static final private int Z_STREAM_ERROR = -2;
-    static final private int Z_DATA_ERROR = -3;
-    static final private int Z_MEM_ERROR = -4;
-    static final private int Z_BUF_ERROR = -5;
-    static final private int Z_VERSION_ERROR = -6;
+  static final private int Z_OK=0;
+  static final private int Z_STREAM_END=1;
+  static final private int Z_NEED_DICT=2;
+  static final private int Z_ERRNO=-1;
+  static final private int Z_STREAM_ERROR=-2;
+  static final private int Z_DATA_ERROR=-3;
+  static final private int Z_MEM_ERROR=-4;
+  static final private int Z_BUF_ERROR=-5;
+  static final private int Z_VERSION_ERROR=-6;
 
-    static final int fixed_bl = 9;
-    static final int fixed_bd = 5;
+  static final int fixed_bl = 9;
+  static final int fixed_bd = 5;
 
-    static final int[] fixed_tl = {
-        96, 7, 256, 0, 8, 80, 0, 8, 16, 84, 8, 115,
-        82, 7, 31, 0, 8, 112, 0, 8, 48, 0, 9, 192,
-        80, 7, 10, 0, 8, 96, 0, 8, 32, 0, 9, 160,
-        0, 8, 0, 0, 8, 128, 0, 8, 64, 0, 9, 224,
-        80, 7, 6, 0, 8, 88, 0, 8, 24, 0, 9, 144,
-        83, 7, 59, 0, 8, 120, 0, 8, 56, 0, 9, 208,
-        81, 7, 17, 0, 8, 104, 0, 8, 40, 0, 9, 176,
-        0, 8, 8, 0, 8, 136, 0, 8, 72, 0, 9, 240,
-        80, 7, 4, 0, 8, 84, 0, 8, 20, 85, 8, 227,
-        83, 7, 43, 0, 8, 116, 0, 8, 52, 0, 9, 200,
-        81, 7, 13, 0, 8, 100, 0, 8, 36, 0, 9, 168,
-        0, 8, 4, 0, 8, 132, 0, 8, 68, 0, 9, 232,
-        80, 7, 8, 0, 8, 92, 0, 8, 28, 0, 9, 152,
-        84, 7, 83, 0, 8, 124, 0, 8, 60, 0, 9, 216,
-        82, 7, 23, 0, 8, 108, 0, 8, 44, 0, 9, 184,
-        0, 8, 12, 0, 8, 140, 0, 8, 76, 0, 9, 248,
-        80, 7, 3, 0, 8, 82, 0, 8, 18, 85, 8, 163,
-        83, 7, 35, 0, 8, 114, 0, 8, 50, 0, 9, 196,
-        81, 7, 11, 0, 8, 98, 0, 8, 34, 0, 9, 164,
-        0, 8, 2, 0, 8, 130, 0, 8, 66, 0, 9, 228,
-        80, 7, 7, 0, 8, 90, 0, 8, 26, 0, 9, 148,
-        84, 7, 67, 0, 8, 122, 0, 8, 58, 0, 9, 212,
-        82, 7, 19, 0, 8, 106, 0, 8, 42, 0, 9, 180,
-        0, 8, 10, 0, 8, 138, 0, 8, 74, 0, 9, 244,
-        80, 7, 5, 0, 8, 86, 0, 8, 22, 192, 8, 0,
-        83, 7, 51, 0, 8, 118, 0, 8, 54, 0, 9, 204,
-        81, 7, 15, 0, 8, 102, 0, 8, 38, 0, 9, 172,
-        0, 8, 6, 0, 8, 134, 0, 8, 70, 0, 9, 236,
-        80, 7, 9, 0, 8, 94, 0, 8, 30, 0, 9, 156,
-        84, 7, 99, 0, 8, 126, 0, 8, 62, 0, 9, 220,
-        82, 7, 27, 0, 8, 110, 0, 8, 46, 0, 9, 188,
-        0, 8, 14, 0, 8, 142, 0, 8, 78, 0, 9, 252,
-        96, 7, 256, 0, 8, 81, 0, 8, 17, 85, 8, 131,
-        82, 7, 31, 0, 8, 113, 0, 8, 49, 0, 9, 194,
-        80, 7, 10, 0, 8, 97, 0, 8, 33, 0, 9, 162,
-        0, 8, 1, 0, 8, 129, 0, 8, 65, 0, 9, 226,
-        80, 7, 6, 0, 8, 89, 0, 8, 25, 0, 9, 146,
-        83, 7, 59, 0, 8, 121, 0, 8, 57, 0, 9, 210,
-        81, 7, 17, 0, 8, 105, 0, 8, 41, 0, 9, 178,
-        0, 8, 9, 0, 8, 137, 0, 8, 73, 0, 9, 242,
-        80, 7, 4, 0, 8, 85, 0, 8, 21, 80, 8, 258,
-        83, 7, 43, 0, 8, 117, 0, 8, 53, 0, 9, 202,
-        81, 7, 13, 0, 8, 101, 0, 8, 37, 0, 9, 170,
-        0, 8, 5, 0, 8, 133, 0, 8, 69, 0, 9, 234,
-        80, 7, 8, 0, 8, 93, 0, 8, 29, 0, 9, 154,
-        84, 7, 83, 0, 8, 125, 0, 8, 61, 0, 9, 218,
-        82, 7, 23, 0, 8, 109, 0, 8, 45, 0, 9, 186,
-        0, 8, 13, 0, 8, 141, 0, 8, 77, 0, 9, 250,
-        80, 7, 3, 0, 8, 83, 0, 8, 19, 85, 8, 195,
-        83, 7, 35, 0, 8, 115, 0, 8, 51, 0, 9, 198,
-        81, 7, 11, 0, 8, 99, 0, 8, 35, 0, 9, 166,
-        0, 8, 3, 0, 8, 131, 0, 8, 67, 0, 9, 230,
-        80, 7, 7, 0, 8, 91, 0, 8, 27, 0, 9, 150,
-        84, 7, 67, 0, 8, 123, 0, 8, 59, 0, 9, 214,
-        82, 7, 19, 0, 8, 107, 0, 8, 43, 0, 9, 182,
-        0, 8, 11, 0, 8, 139, 0, 8, 75, 0, 9, 246,
-        80, 7, 5, 0, 8, 87, 0, 8, 23, 192, 8, 0,
-        83, 7, 51, 0, 8, 119, 0, 8, 55, 0, 9, 206,
-        81, 7, 15, 0, 8, 103, 0, 8, 39, 0, 9, 174,
-        0, 8, 7, 0, 8, 135, 0, 8, 71, 0, 9, 238,
-        80, 7, 9, 0, 8, 95, 0, 8, 31, 0, 9, 158,
-        84, 7, 99, 0, 8, 127, 0, 8, 63, 0, 9, 222,
-        82, 7, 27, 0, 8, 111, 0, 8, 47, 0, 9, 190,
-        0, 8, 15, 0, 8, 143, 0, 8, 79, 0, 9, 254,
-        96, 7, 256, 0, 8, 80, 0, 8, 16, 84, 8, 115,
-        82, 7, 31, 0, 8, 112, 0, 8, 48, 0, 9, 193,
+  static final int[] fixed_tl = {
+    96,7,256, 0,8,80, 0,8,16, 84,8,115,
+    82,7,31, 0,8,112, 0,8,48, 0,9,192,
+    80,7,10, 0,8,96, 0,8,32, 0,9,160,
+    0,8,0, 0,8,128, 0,8,64, 0,9,224,
+    80,7,6, 0,8,88, 0,8,24, 0,9,144,
+    83,7,59, 0,8,120, 0,8,56, 0,9,208,
+    81,7,17, 0,8,104, 0,8,40, 0,9,176,
+    0,8,8, 0,8,136, 0,8,72, 0,9,240,
+    80,7,4, 0,8,84, 0,8,20, 85,8,227,
+    83,7,43, 0,8,116, 0,8,52, 0,9,200,
+    81,7,13, 0,8,100, 0,8,36, 0,9,168,
+    0,8,4, 0,8,132, 0,8,68, 0,9,232,
+    80,7,8, 0,8,92, 0,8,28, 0,9,152,
+    84,7,83, 0,8,124, 0,8,60, 0,9,216,
+    82,7,23, 0,8,108, 0,8,44, 0,9,184,
+    0,8,12, 0,8,140, 0,8,76, 0,9,248,
+    80,7,3, 0,8,82, 0,8,18, 85,8,163,
+    83,7,35, 0,8,114, 0,8,50, 0,9,196,
+    81,7,11, 0,8,98, 0,8,34, 0,9,164,
+    0,8,2, 0,8,130, 0,8,66, 0,9,228,
+    80,7,7, 0,8,90, 0,8,26, 0,9,148,
+    84,7,67, 0,8,122, 0,8,58, 0,9,212,
+    82,7,19, 0,8,106, 0,8,42, 0,9,180,
+    0,8,10, 0,8,138, 0,8,74, 0,9,244,
+    80,7,5, 0,8,86, 0,8,22, 192,8,0,
+    83,7,51, 0,8,118, 0,8,54, 0,9,204,
+    81,7,15, 0,8,102, 0,8,38, 0,9,172,
+    0,8,6, 0,8,134, 0,8,70, 0,9,236,
+    80,7,9, 0,8,94, 0,8,30, 0,9,156,
+    84,7,99, 0,8,126, 0,8,62, 0,9,220,
+    82,7,27, 0,8,110, 0,8,46, 0,9,188,
+    0,8,14, 0,8,142, 0,8,78, 0,9,252,
+    96,7,256, 0,8,81, 0,8,17, 85,8,131,
+    82,7,31, 0,8,113, 0,8,49, 0,9,194,
+    80,7,10, 0,8,97, 0,8,33, 0,9,162,
+    0,8,1, 0,8,129, 0,8,65, 0,9,226,
+    80,7,6, 0,8,89, 0,8,25, 0,9,146,
+    83,7,59, 0,8,121, 0,8,57, 0,9,210,
+    81,7,17, 0,8,105, 0,8,41, 0,9,178,
+    0,8,9, 0,8,137, 0,8,73, 0,9,242,
+    80,7,4, 0,8,85, 0,8,21, 80,8,258,
+    83,7,43, 0,8,117, 0,8,53, 0,9,202,
+    81,7,13, 0,8,101, 0,8,37, 0,9,170,
+    0,8,5, 0,8,133, 0,8,69, 0,9,234,
+    80,7,8, 0,8,93, 0,8,29, 0,9,154,
+    84,7,83, 0,8,125, 0,8,61, 0,9,218,
+    82,7,23, 0,8,109, 0,8,45, 0,9,186,
+    0,8,13, 0,8,141, 0,8,77, 0,9,250,
+    80,7,3, 0,8,83, 0,8,19, 85,8,195,
+    83,7,35, 0,8,115, 0,8,51, 0,9,198,
+    81,7,11, 0,8,99, 0,8,35, 0,9,166,
+    0,8,3, 0,8,131, 0,8,67, 0,9,230,
+    80,7,7, 0,8,91, 0,8,27, 0,9,150,
+    84,7,67, 0,8,123, 0,8,59, 0,9,214,
+    82,7,19, 0,8,107, 0,8,43, 0,9,182,
+    0,8,11, 0,8,139, 0,8,75, 0,9,246,
+    80,7,5, 0,8,87, 0,8,23, 192,8,0,
+    83,7,51, 0,8,119, 0,8,55, 0,9,206,
+    81,7,15, 0,8,103, 0,8,39, 0,9,174,
+    0,8,7, 0,8,135, 0,8,71, 0,9,238,
+    80,7,9, 0,8,95, 0,8,31, 0,9,158,
+    84,7,99, 0,8,127, 0,8,63, 0,9,222,
+    82,7,27, 0,8,111, 0,8,47, 0,9,190,
+    0,8,15, 0,8,143, 0,8,79, 0,9,254,
+    96,7,256, 0,8,80, 0,8,16, 84,8,115,
+    82,7,31, 0,8,112, 0,8,48, 0,9,193,
 
-        80, 7, 10, 0, 8, 96, 0, 8, 32, 0, 9, 161,
-        0, 8, 0, 0, 8, 128, 0, 8, 64, 0, 9, 225,
-        80, 7, 6, 0, 8, 88, 0, 8, 24, 0, 9, 145,
-        83, 7, 59, 0, 8, 120, 0, 8, 56, 0, 9, 209,
-        81, 7, 17, 0, 8, 104, 0, 8, 40, 0, 9, 177,
-        0, 8, 8, 0, 8, 136, 0, 8, 72, 0, 9, 241,
-        80, 7, 4, 0, 8, 84, 0, 8, 20, 85, 8, 227,
-        83, 7, 43, 0, 8, 116, 0, 8, 52, 0, 9, 201,
-        81, 7, 13, 0, 8, 100, 0, 8, 36, 0, 9, 169,
-        0, 8, 4, 0, 8, 132, 0, 8, 68, 0, 9, 233,
-        80, 7, 8, 0, 8, 92, 0, 8, 28, 0, 9, 153,
-        84, 7, 83, 0, 8, 124, 0, 8, 60, 0, 9, 217,
-        82, 7, 23, 0, 8, 108, 0, 8, 44, 0, 9, 185,
-        0, 8, 12, 0, 8, 140, 0, 8, 76, 0, 9, 249,
-        80, 7, 3, 0, 8, 82, 0, 8, 18, 85, 8, 163,
-        83, 7, 35, 0, 8, 114, 0, 8, 50, 0, 9, 197,
-        81, 7, 11, 0, 8, 98, 0, 8, 34, 0, 9, 165,
-        0, 8, 2, 0, 8, 130, 0, 8, 66, 0, 9, 229,
-        80, 7, 7, 0, 8, 90, 0, 8, 26, 0, 9, 149,
-        84, 7, 67, 0, 8, 122, 0, 8, 58, 0, 9, 213,
-        82, 7, 19, 0, 8, 106, 0, 8, 42, 0, 9, 181,
-        0, 8, 10, 0, 8, 138, 0, 8, 74, 0, 9, 245,
-        80, 7, 5, 0, 8, 86, 0, 8, 22, 192, 8, 0,
-        83, 7, 51, 0, 8, 118, 0, 8, 54, 0, 9, 205,
-        81, 7, 15, 0, 8, 102, 0, 8, 38, 0, 9, 173,
-        0, 8, 6, 0, 8, 134, 0, 8, 70, 0, 9, 237,
-        80, 7, 9, 0, 8, 94, 0, 8, 30, 0, 9, 157,
-        84, 7, 99, 0, 8, 126, 0, 8, 62, 0, 9, 221,
-        82, 7, 27, 0, 8, 110, 0, 8, 46, 0, 9, 189,
-        0, 8, 14, 0, 8, 142, 0, 8, 78, 0, 9, 253,
-        96, 7, 256, 0, 8, 81, 0, 8, 17, 85, 8, 131,
-        82, 7, 31, 0, 8, 113, 0, 8, 49, 0, 9, 195,
-        80, 7, 10, 0, 8, 97, 0, 8, 33, 0, 9, 163,
-        0, 8, 1, 0, 8, 129, 0, 8, 65, 0, 9, 227,
-        80, 7, 6, 0, 8, 89, 0, 8, 25, 0, 9, 147,
-        83, 7, 59, 0, 8, 121, 0, 8, 57, 0, 9, 211,
-        81, 7, 17, 0, 8, 105, 0, 8, 41, 0, 9, 179,
-        0, 8, 9, 0, 8, 137, 0, 8, 73, 0, 9, 243,
-        80, 7, 4, 0, 8, 85, 0, 8, 21, 80, 8, 258,
-        83, 7, 43, 0, 8, 117, 0, 8, 53, 0, 9, 203,
-        81, 7, 13, 0, 8, 101, 0, 8, 37, 0, 9, 171,
-        0, 8, 5, 0, 8, 133, 0, 8, 69, 0, 9, 235,
-        80, 7, 8, 0, 8, 93, 0, 8, 29, 0, 9, 155,
-        84, 7, 83, 0, 8, 125, 0, 8, 61, 0, 9, 219,
-        82, 7, 23, 0, 8, 109, 0, 8, 45, 0, 9, 187,
-        0, 8, 13, 0, 8, 141, 0, 8, 77, 0, 9, 251,
-        80, 7, 3, 0, 8, 83, 0, 8, 19, 85, 8, 195,
-        83, 7, 35, 0, 8, 115, 0, 8, 51, 0, 9, 199,
-        81, 7, 11, 0, 8, 99, 0, 8, 35, 0, 9, 167,
-        0, 8, 3, 0, 8, 131, 0, 8, 67, 0, 9, 231,
-        80, 7, 7, 0, 8, 91, 0, 8, 27, 0, 9, 151,
-        84, 7, 67, 0, 8, 123, 0, 8, 59, 0, 9, 215,
-        82, 7, 19, 0, 8, 107, 0, 8, 43, 0, 9, 183,
-        0, 8, 11, 0, 8, 139, 0, 8, 75, 0, 9, 247,
-        80, 7, 5, 0, 8, 87, 0, 8, 23, 192, 8, 0,
-        83, 7, 51, 0, 8, 119, 0, 8, 55, 0, 9, 207,
-        81, 7, 15, 0, 8, 103, 0, 8, 39, 0, 9, 175,
-        0, 8, 7, 0, 8, 135, 0, 8, 71, 0, 9, 239,
-        80, 7, 9, 0, 8, 95, 0, 8, 31, 0, 9, 159,
-        84, 7, 99, 0, 8, 127, 0, 8, 63, 0, 9, 223,
-        82, 7, 27, 0, 8, 111, 0, 8, 47, 0, 9, 191,
-        0, 8, 15, 0, 8, 143, 0, 8, 79, 0, 9, 255
-    };
-    static final int[] fixed_td = {
-        80, 5, 1, 87, 5, 257, 83, 5, 17, 91, 5, 4097,
-        81, 5, 5, 89, 5, 1025, 85, 5, 65, 93, 5, 16385,
-        80, 5, 3, 88, 5, 513, 84, 5, 33, 92, 5, 8193,
-        82, 5, 9, 90, 5, 2049, 86, 5, 129, 192, 5, 24577,
-        80, 5, 2, 87, 5, 385, 83, 5, 25, 91, 5, 6145,
-        81, 5, 7, 89, 5, 1537, 85, 5, 97, 93, 5, 24577,
-        80, 5, 4, 88, 5, 769, 84, 5, 49, 92, 5, 12289,
-        82, 5, 13, 90, 5, 3073, 86, 5, 193, 192, 5, 24577
-    };
+    80,7,10, 0,8,96, 0,8,32, 0,9,161,
+    0,8,0, 0,8,128, 0,8,64, 0,9,225,
+    80,7,6, 0,8,88, 0,8,24, 0,9,145,
+    83,7,59, 0,8,120, 0,8,56, 0,9,209,
+    81,7,17, 0,8,104, 0,8,40, 0,9,177,
+    0,8,8, 0,8,136, 0,8,72, 0,9,241,
+    80,7,4, 0,8,84, 0,8,20, 85,8,227,
+    83,7,43, 0,8,116, 0,8,52, 0,9,201,
+    81,7,13, 0,8,100, 0,8,36, 0,9,169,
+    0,8,4, 0,8,132, 0,8,68, 0,9,233,
+    80,7,8, 0,8,92, 0,8,28, 0,9,153,
+    84,7,83, 0,8,124, 0,8,60, 0,9,217,
+    82,7,23, 0,8,108, 0,8,44, 0,9,185,
+    0,8,12, 0,8,140, 0,8,76, 0,9,249,
+    80,7,3, 0,8,82, 0,8,18, 85,8,163,
+    83,7,35, 0,8,114, 0,8,50, 0,9,197,
+    81,7,11, 0,8,98, 0,8,34, 0,9,165,
+    0,8,2, 0,8,130, 0,8,66, 0,9,229,
+    80,7,7, 0,8,90, 0,8,26, 0,9,149,
+    84,7,67, 0,8,122, 0,8,58, 0,9,213,
+    82,7,19, 0,8,106, 0,8,42, 0,9,181,
+    0,8,10, 0,8,138, 0,8,74, 0,9,245,
+    80,7,5, 0,8,86, 0,8,22, 192,8,0,
+    83,7,51, 0,8,118, 0,8,54, 0,9,205,
+    81,7,15, 0,8,102, 0,8,38, 0,9,173,
+    0,8,6, 0,8,134, 0,8,70, 0,9,237,
+    80,7,9, 0,8,94, 0,8,30, 0,9,157,
+    84,7,99, 0,8,126, 0,8,62, 0,9,221,
+    82,7,27, 0,8,110, 0,8,46, 0,9,189,
+    0,8,14, 0,8,142, 0,8,78, 0,9,253,
+    96,7,256, 0,8,81, 0,8,17, 85,8,131,
+    82,7,31, 0,8,113, 0,8,49, 0,9,195,
+    80,7,10, 0,8,97, 0,8,33, 0,9,163,
+    0,8,1, 0,8,129, 0,8,65, 0,9,227,
+    80,7,6, 0,8,89, 0,8,25, 0,9,147,
+    83,7,59, 0,8,121, 0,8,57, 0,9,211,
+    81,7,17, 0,8,105, 0,8,41, 0,9,179,
+    0,8,9, 0,8,137, 0,8,73, 0,9,243,
+    80,7,4, 0,8,85, 0,8,21, 80,8,258,
+    83,7,43, 0,8,117, 0,8,53, 0,9,203,
+    81,7,13, 0,8,101, 0,8,37, 0,9,171,
+    0,8,5, 0,8,133, 0,8,69, 0,9,235,
+    80,7,8, 0,8,93, 0,8,29, 0,9,155,
+    84,7,83, 0,8,125, 0,8,61, 0,9,219,
+    82,7,23, 0,8,109, 0,8,45, 0,9,187,
+    0,8,13, 0,8,141, 0,8,77, 0,9,251,
+    80,7,3, 0,8,83, 0,8,19, 85,8,195,
+    83,7,35, 0,8,115, 0,8,51, 0,9,199,
+    81,7,11, 0,8,99, 0,8,35, 0,9,167,
+    0,8,3, 0,8,131, 0,8,67, 0,9,231,
+    80,7,7, 0,8,91, 0,8,27, 0,9,151,
+    84,7,67, 0,8,123, 0,8,59, 0,9,215,
+    82,7,19, 0,8,107, 0,8,43, 0,9,183,
+    0,8,11, 0,8,139, 0,8,75, 0,9,247,
+    80,7,5, 0,8,87, 0,8,23, 192,8,0,
+    83,7,51, 0,8,119, 0,8,55, 0,9,207,
+    81,7,15, 0,8,103, 0,8,39, 0,9,175,
+    0,8,7, 0,8,135, 0,8,71, 0,9,239,
+    80,7,9, 0,8,95, 0,8,31, 0,9,159,
+    84,7,99, 0,8,127, 0,8,63, 0,9,223,
+    82,7,27, 0,8,111, 0,8,47, 0,9,191,
+    0,8,15, 0,8,143, 0,8,79, 0,9,255
+  };
+  static final int[] fixed_td = {
+    80,5,1, 87,5,257, 83,5,17, 91,5,4097,
+    81,5,5, 89,5,1025, 85,5,65, 93,5,16385,
+    80,5,3, 88,5,513, 84,5,33, 92,5,8193,
+    82,5,9, 90,5,2049, 86,5,129, 192,5,24577,
+    80,5,2, 87,5,385, 83,5,25, 91,5,6145,
+    81,5,7, 89,5,1537, 85,5,97, 93,5,24577,
+    80,5,4, 88,5,769, 84,5,49, 92,5,12289,
+    82,5,13, 90,5,3073, 86,5,193, 192,5,24577
+  };
 
-    // Tables for deflate from PKZIP's appnote.txt.
-    static final int[] cplens = { // Copy lengths for literal codes 257..285
+  // Tables for deflate from PKZIP's appnote.txt.
+  static final int[] cplens = { // Copy lengths for literal codes 257..285
         3, 4, 5, 6, 7, 8, 9, 10, 11, 13, 15, 17, 19, 23, 27, 31,
         35, 43, 51, 59, 67, 83, 99, 115, 131, 163, 195, 227, 258, 0, 0
-    };
+  };
 
-    // see note #13 above about 258
-    static final int[] cplext = { // Extra bits for literal codes 257..285
+  // see note #13 above about 258
+  static final int[] cplext = { // Extra bits for literal codes 257..285
         0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 2, 2, 2, 2,
         3, 3, 3, 3, 4, 4, 4, 4, 5, 5, 5, 5, 0, 112, 112  // 112==invalid
-    };
+  };
 
-    static final int[] cpdist = { // Copy offsets for distance codes 0..29
+  static final int[] cpdist = { // Copy offsets for distance codes 0..29
         1, 2, 3, 4, 5, 7, 9, 13, 17, 25, 33, 49, 65, 97, 129, 193,
         257, 385, 513, 769, 1025, 1537, 2049, 3073, 4097, 6145,
         8193, 12289, 16385, 24577
-    };
+  };
 
-    static final int[] cpdext = { // Extra bits for distance codes
+  static final int[] cpdext = { // Extra bits for distance codes
         0, 0, 0, 0, 1, 1, 2, 2, 3, 3, 4, 4, 5, 5, 6, 6,
         7, 7, 8, 8, 9, 9, 10, 10, 11, 11,
-        12, 12, 13, 13
-    };
+        12, 12, 13, 13};
+
+  // If BMAX needs to be larger than 16, then h and x[] should be uLong.
+  static final int BMAX=15;         // maximum bit length of any code
 
-    // If BMAX needs to be larger than 16, then h and x[] should be uLong.
-    static final int BMAX = 15;       // maximum bit length of any code
+  int[] hn = null;  // hufts used in space
+  int[] v = null;   // work area for huft_build 
+  int[] c = null;   // bit length count table
+  int[] r = null;   // table entry for structure assignment
+  int[] u = null;   // table stack
+  int[] x = null;   // bit offsets, then code stack
 
-    int[] hn = null;  // hufts used in space
-    int[] v = null;   // work area for huft_build
-    int[] c = null;   // bit length count table
-    int[] r = null;   // table entry for structure assignment
-    int[] u = null;   // table stack
-    int[] x = null;   // bit offsets, then code stack
+  private int huft_build(int[] b, // code lengths in bits (all assumed <= BMAX)
+                         int bindex, 
+                         int n,   // number of codes (assumed <= 288)
+                         int s,   // number of simple-valued codes (0..s-1)
+                         int[] d, // list of base values for non-simple codes
+                         int[] e, // list of extra bits for non-simple codes
+                         int[] t, // result: starting table
+                         int[] m, // maximum lookup bits, returns actual
+                         int[] hp,// space for trees
+                         int[] hn,// hufts used in space
+                         int[] v  // working area: values in order of bit length
+                         ){
+    // Given a list of code lengths and a maximum table size, make a set of
+    // tables to decode that set of codes.  Return Z_OK on success, Z_BUF_ERROR
+    // if the given code set is incomplete (the tables are still built in this
+    // case), Z_DATA_ERROR if the input is invalid (an over-subscribed set of
+    // lengths), or Z_MEM_ERROR if not enough memory.
 
-    private int huft_build(int[] b,  // code lengths in bits (all assumed <= BMAX)
-                           int bindex,
-                           int n,   // number of codes (assumed <= 288)
-                           int s,   // number of simple-valued codes (0..s-1)
-                           int[] d, // list of base values for non-simple codes
-                           int[] e, // list of extra bits for non-simple codes
-                           int[] t, // result: starting table
-                           int[] m, // maximum lookup bits, returns actual
-                           int[] hp,// space for trees
-                           int[] hn,// hufts used in space
-                           int[] v  // working area: values in order of bit length
-                          ) {
-        // Given a list of code lengths and a maximum table size, make a set of
-        // tables to decode that set of codes.  Return Z_OK on success, Z_BUF_ERROR
-        // if the given code set is incomplete (the tables are still built in this
-        // case), Z_DATA_ERROR if the input is invalid (an over-subscribed set of
-        // lengths), or Z_MEM_ERROR if not enough memory.
-        int a;                       // counter for codes of length k
-        int f;                       // i repeats in table every f entries
-        int g;                       // maximum code length
-        int h;                       // table level
-        int i;                       // counter, current code
-        int j;                       // counter
-        int k;                       // number of bits in current code
-        int l;                       // bits per table (returned in m)
-        int mask;                    // (1 << w) - 1, to avoid cc -O bug on HP
-        int p;                       // pointer into c[], b[], or v[]
-        int q;                       // points to current table
-        int w;                       // bits before this table == (l * h)
-        int xp;                      // pointer into x
-        int y;                       // number of dummy codes added
-        int z;                       // number of entries in current table
-        // Generate counts for each bit length
-        p = 0; i = n;
+    int a;                       // counter for codes of length k
+    int f;                       // i repeats in table every f entries
+    int g;                       // maximum code length
+    int h;                       // table level
+    int i;                       // counter, current code
+    int j;                       // counter
+    int k;                       // number of bits in current code
+    int l;                       // bits per table (returned in m)
+    int mask;                    // (1 << w) - 1, to avoid cc -O bug on HP
+    int p;                       // pointer into c[], b[], or v[]
+    int q;                       // points to current table
+    int w;                       // bits before this table == (l * h)
+    int xp;                      // pointer into x
+    int y;                       // number of dummy codes added
+    int z;                       // number of entries in current table
+
+    // Generate counts for each bit length
+
+    p = 0; i = n;
+    do {
+      c[b[bindex+p]]++; p++; i--;   // assume all entries <= BMAX
+    }while(i!=0);
+
+    if(c[0] == n){                // null input--all zero length codes
+      t[0] = -1;
+      m[0] = 0;
+      return Z_OK;
+    }
+
+    // Find minimum and maximum length, bound *m by those
+    l = m[0];
+    for (j = 1; j <= BMAX; j++)
+      if(c[j]!=0) break;
+    k = j;                        // minimum code length
+    if(l < j){
+      l = j;
+    }
+    for (i = BMAX; i!=0; i--){
+      if(c[i]!=0) break;
+    }
+    g = i;                        // maximum code length
+    if(l > i){
+      l = i;
+    }
+    m[0] = l;
 
-        do {
-            c[b[bindex + p]]++; p++; i--; // assume all entries <= BMAX
-        }
-        while (i != 0);
+    // Adjust last length count to fill out codes, if needed
+    for (y = 1 << j; j < i; j++, y <<= 1){
+      if ((y -= c[j]) < 0){
+        return Z_DATA_ERROR;
+      }
+    }
+    if ((y -= c[i]) < 0){
+      return Z_DATA_ERROR;
+    }
+    c[i] += y;
 
-        if (c[0] == n) {              // null input--all zero length codes
-            t[0] = -1;
-            m[0] = 0;
-            return Z_OK;
-        }
+    // Generate starting offsets into the value table for each length
+    x[1] = j = 0;
+    p = 1;  xp = 2;
+    while (--i!=0) {                 // note that i == g from above
+      x[xp] = (j += c[p]);
+      xp++;
+      p++;
+    }
 
-        // Find minimum and maximum length, bound *m by those
-        l = m[0];
-
-        for (j = 1; j <= BMAX; j++)
-            if (c[j] != 0) break;
-
-        k = j;                        // minimum code length
+    // Make a table of values in order of bit lengths
+    i = 0; p = 0;
+    do {
+      if ((j = b[bindex+p]) != 0){
+        v[x[j]++] = i;
+      }
+      p++;
+    }
+    while (++i < n);
+    n = x[g];                     // set n to length of v
 
-        if (l < j) {
-            l = j;
-        }
-
-        for (i = BMAX; i != 0; i--) {
-            if (c[i] != 0) break;
-        }
+    // Generate the Huffman codes and for each, make the table entries
+    x[0] = i = 0;                 // first Huffman code is zero
+    p = 0;                        // grab values in bit order
+    h = -1;                       // no tables yet--level -1
+    w = -l;                       // bits decoded == (l * h)
+    u[0] = 0;                     // just to keep compilers happy
+    q = 0;                        // ditto
+    z = 0;                        // ditto
 
-        g = i;                        // maximum code length
-
-        if (l > i) {
-            l = i;
-        }
-
-        m[0] = l;
+    // go through the bit lengths (k already is bits in shortest code)
+    for (; k <= g; k++){
+      a = c[k];
+      while (a--!=0){
+	// here i is the Huffman code of length k bits for value *p
+	// make tables up to required level
+        while (k > w + l){
+          h++;
+          w += l;                 // previous table always l bits
+	  // compute minimum size table less than or equal to l bits
+          z = g - w;
+          z = (z > l) ? l : z;        // table size upper limit
+          if((f=1<<(j=k-w))>a+1){     // try a k-w bit table
+                                      // too few codes for k-w bit table
+            f -= a + 1;               // deduct codes from patterns left
+            xp = k;
+            if(j < z){
+              while (++j < z){        // try smaller tables up to z bits
+                if((f <<= 1) <= c[++xp])
+                  break;              // enough codes to use up j bits
+                f -= c[xp];           // else deduct codes from patterns
+              }
+	    }
+          }
+          z = 1 << j;                 // table entries for j-bit table
 
-        // Adjust last length count to fill out codes, if needed
-        for (y = 1 << j; j < i; j++, y <<= 1) {
-            if ((y -= c[j]) < 0) {
-                return Z_DATA_ERROR;
-            }
-        }
-
-        if ((y -= c[i]) < 0) {
-            return Z_DATA_ERROR;
-        }
-
-        c[i] += y;
-        // Generate starting offsets into the value table for each length
-        x[1] = j = 0;
-        p = 1;  xp = 2;
-
-        while (--i != 0) {               // note that i == g from above
-            x[xp] = (j += c[p]);
-            xp++;
-            p++;
+	  // allocate new table
+          if (hn[0] + z > MANY){       // (note: doesn't matter for fixed)
+            return Z_DATA_ERROR;       // overflow of MANY
+          }
+          u[h] = q = /*hp+*/ hn[0];   // DEBUG
+          hn[0] += z;
+ 
+	  // connect to last table, if there is one
+	  if(h!=0){
+            x[h]=i;           // save pattern for backing up
+            r[0]=(byte)j;     // bits in this table
+            r[1]=(byte)l;     // bits to dump before this table
+            j=i>>>(w - l);
+            r[2] = (int)(q - u[h-1] - j);               // offset to this table
+            System.arraycopy(r, 0, hp, (u[h-1]+j)*3, 3); // connect to last table
+          }
+          else{
+            t[0] = q;               // first table is returned result
+	  }
         }
 
-        // Make a table of values in order of bit lengths
-        i = 0; p = 0;
-
-        do {
-            if ((j = b[bindex + p]) != 0) {
-                v[x[j]++] = i;
-            }
-
-            p++;
+	// set up table entry in r
+        r[1] = (byte)(k - w);
+        if (p >= n){
+          r[0] = 128 + 64;      // out of values--invalid code
+	}
+        else if (v[p] < s){
+          r[0] = (byte)(v[p] < 256 ? 0 : 32 + 64);  // 256 is end-of-block
+          r[2] = v[p++];          // simple code is just the value
         }
-        while (++i < n);
-
-        n = x[g];                     // set n to length of v
-        // Generate the Huffman codes and for each, make the table entries
-        x[0] = i = 0;                 // first Huffman code is zero
-        p = 0;                        // grab values in bit order
-        h = -1;                       // no tables yet--level -1
-        w = -l;                       // bits decoded == (l * h)
-        u[0] = 0;                     // just to keep compilers happy
-        q = 0;                        // ditto
-        z = 0;                        // ditto
-
-        // go through the bit lengths (k already is bits in shortest code)
-        for (; k <= g; k++) {
-            a = c[k];
-
-            while (a-- != 0) {
-                // here i is the Huffman code of length k bits for value *p
-                // make tables up to required level
-                while (k > w + l) {
-                    h++;
-                    w += l;                 // previous table always l bits
-                    // compute minimum size table less than or equal to l bits
-                    z = g - w;
-                    z = (z > l) ? l : z;        // table size upper limit
-
-                    if ((f = 1 << (j = k - w)) > a + 1) { // try a k-w bit table
-                        // too few codes for k-w bit table
-                        f -= a + 1;               // deduct codes from patterns left
-                        xp = k;
-
-                        if (j < z) {
-                            while (++j < z) {       // try smaller tables up to z bits
-                                if ((f <<= 1) <= c[++xp])
-                                    break;              // enough codes to use up j bits
-
-                                f -= c[xp];           // else deduct codes from patterns
-                            }
-                        }
-                    }
-
-                    z = 1 << j;                 // table entries for j-bit table
-
-                    // allocate new table
-                    if (hn[0] + z > MANY) {      // (note: doesn't matter for fixed)
-                        return Z_DATA_ERROR;       // overflow of MANY
-                    }
-
-                    u[h] = q = /*hp+*/ hn[0];   // DEBUG
-                    hn[0] += z;
-
-                    // connect to last table, if there is one
-                    if (h != 0) {
-                        x[h] = i;         // save pattern for backing up
-                        r[0] = (byte)j;   // bits in this table
-                        r[1] = (byte)l;   // bits to dump before this table
-                        j = i >>> (w - l);
-                        r[2] = (int)(q - u[h - 1] - j);             // offset to this table
-                        System.arraycopy(r, 0, hp, (u[h - 1] + j) * 3, 3);  // connect to last table
-                    }
-                    else {
-                        t[0] = q;               // first table is returned result
-                    }
-                }
-
-                // set up table entry in r
-                r[1] = (byte)(k - w);
-
-                if (p >= n) {
-                    r[0] = 128 + 64;      // out of values--invalid code
-                }
-                else if (v[p] < s) {
-                    r[0] = (byte)(v[p] < 256 ? 0 : 32 + 64);  // 256 is end-of-block
-                    r[2] = v[p++];          // simple code is just the value
-                }
-                else {
-                    r[0] = (byte)(e[v[p] - s] + 16 + 64);  // non-simple--look up in lists
-                    r[2] = d[v[p++] - s];
-                }
-
-                // fill code-like entries with r
-                f = 1 << (k - w);
-
-                for (j = i >>> w; j < z; j += f) {
-                    System.arraycopy(r, 0, hp, (q + j) * 3, 3);
-                }
-
-                // backwards increment the k-bit code i
-                for (j = 1 << (k - 1); (i & j) != 0; j >>>= 1) {
-                    i ^= j;
-                }
-
-                i ^= j;
-                // backup over finished tables
-                mask = (1 << w) - 1;      // needed on HP, cc -O bug
-
-                while ((i & mask) != x[h]) {
-                    h--;                    // don't need to update q
-                    w -= l;
-                    mask = (1 << w) - 1;
-                }
-            }
+        else{
+          r[0]=(byte)(e[v[p]-s]+16+64); // non-simple--look up in lists
+          r[2]=d[v[p++] - s];
         }
 
-        // Return Z_BUF_ERROR if we were given an incomplete table
-        return y != 0 && g != 1 ? Z_BUF_ERROR : Z_OK;
+        // fill code-like entries with r
+        f=1<<(k-w);
+        for (j=i>>>w;j<z;j+=f){
+          System.arraycopy(r, 0, hp, (q+j)*3, 3);
+	}
+
+	// backwards increment the k-bit code i
+        for (j = 1 << (k - 1); (i & j)!=0; j >>>= 1){
+          i ^= j;
+	}
+        i ^= j;
+
+	// backup over finished tables
+        mask = (1 << w) - 1;      // needed on HP, cc -O bug
+        while ((i & mask) != x[h]){
+          h--;                    // don't need to update q
+          w -= l;
+          mask = (1 << w) - 1;
+        }
+      }
     }
+    // Return Z_BUF_ERROR if we were given an incomplete table
+    return y != 0 && g != 1 ? Z_BUF_ERROR : Z_OK;
+  }
+
+  int inflate_trees_bits(int[] c,  // 19 code lengths
+                         int[] bb, // bits tree desired/actual depth
+                         int[] tb, // bits tree result
+                         int[] hp, // space for trees
+                         ZStream z // for messages
+                         ){
+    int result;
+    initWorkArea(19);
+    hn[0]=0;
+    result = huft_build(c, 0, 19, 19, null, null, tb, bb, hp, hn, v);
 
-    int inflate_trees_bits(int[] c,  // 19 code lengths
-                           int[] bb, // bits tree desired/actual depth
-                           int[] tb, // bits tree result
-                           int[] hp, // space for trees
-                           ZStream z // for messages
-                          ) {
-        int result;
-        initWorkArea(19);
-        hn[0] = 0;
-        result = huft_build(c, 0, 19, 19, null, null, tb, bb, hp, hn, v);
+    if(result == Z_DATA_ERROR){
+      z.msg = "oversubscribed dynamic bit lengths tree";
+    }
+    else if(result == Z_BUF_ERROR || bb[0] == 0){
+      z.msg = "incomplete dynamic bit lengths tree";
+      result = Z_DATA_ERROR;
+    }
+    return result;
+  }
 
-        if (result == Z_DATA_ERROR) {
-            z.msg = "oversubscribed dynamic bit lengths tree";
-        }
-        else if (result == Z_BUF_ERROR || bb[0] == 0) {
-            z.msg = "incomplete dynamic bit lengths tree";
-            result = Z_DATA_ERROR;
-        }
+  int inflate_trees_dynamic(int nl,   // number of literal/length codes
+                            int nd,   // number of distance codes
+                            int[] c,  // that many (total) code lengths
+                            int[] bl, // literal desired/actual bit depth
+                            int[] bd, // distance desired/actual bit depth 
+                            int[] tl, // literal/length tree result
+                            int[] td, // distance tree result
+                            int[] hp, // space for trees
+                            ZStream z // for messages
+                            ){
+    int result;
 
-        return result;
+    // build literal/length tree
+    initWorkArea(288);
+    hn[0]=0;
+    result = huft_build(c, 0, nl, 257, cplens, cplext, tl, bl, hp, hn, v);
+    if (result != Z_OK || bl[0] == 0){
+      if(result == Z_DATA_ERROR){
+        z.msg = "oversubscribed literal/length tree";
+      }
+      else if (result != Z_MEM_ERROR){
+        z.msg = "incomplete literal/length tree";
+        result = Z_DATA_ERROR;
+      }
+      return result;
     }
 
-    int inflate_trees_dynamic(int nl,   // number of literal/length codes
-                              int nd,   // number of distance codes
-                              int[] c,  // that many (total) code lengths
-                              int[] bl, // literal desired/actual bit depth
-                              int[] bd, // distance desired/actual bit depth
-                              int[] tl, // literal/length tree result
-                              int[] td, // distance tree result
-                              int[] hp, // space for trees
-                              ZStream z // for messages
-                             ) {
-        int result;
-        // build literal/length tree
-        initWorkArea(288);
-        hn[0] = 0;
-        result = huft_build(c, 0, nl, 257, cplens, cplext, tl, bl, hp, hn, v);
-
-        if (result != Z_OK || bl[0] == 0) {
-            if (result == Z_DATA_ERROR) {
-                z.msg = "oversubscribed literal/length tree";
-            }
-            else if (result != Z_MEM_ERROR) {
-                z.msg = "incomplete literal/length tree";
-                result = Z_DATA_ERROR;
-            }
+    // build distance tree
+    initWorkArea(288);
+    result = huft_build(c, nl, nd, 0, cpdist, cpdext, td, bd, hp, hn, v);
 
-            return result;
-        }
-
-        // build distance tree
-        initWorkArea(288);
-        result = huft_build(c, nl, nd, 0, cpdist, cpdext, td, bd, hp, hn, v);
-
-        if (result != Z_OK || (bd[0] == 0 && nl > 257)) {
-            if (result == Z_DATA_ERROR) {
-                z.msg = "oversubscribed distance tree";
-            }
-            else if (result == Z_BUF_ERROR) {
-                z.msg = "incomplete distance tree";
-                result = Z_DATA_ERROR;
-            }
-            else if (result != Z_MEM_ERROR) {
-                z.msg = "empty distance tree with lengths";
-                result = Z_DATA_ERROR;
-            }
-
-            return result;
-        }
-
-        return Z_OK;
+    if (result != Z_OK || (bd[0] == 0 && nl > 257)){
+      if (result == Z_DATA_ERROR){
+        z.msg = "oversubscribed distance tree";
+      }
+      else if (result == Z_BUF_ERROR) {
+        z.msg = "incomplete distance tree";
+        result = Z_DATA_ERROR;
+      }
+      else if (result != Z_MEM_ERROR){
+        z.msg = "empty distance tree with lengths";
+        result = Z_DATA_ERROR;
+      }
+      return result;
     }
 
-    static int inflate_trees_fixed(int[] bl,  //literal desired/actual bit depth
-                                   int[] bd,  //distance desired/actual bit depth
-                                   int[][] tl,//literal/length tree result
-                                   int[][] td,//distance tree result
-                                   ZStream z  //for memory allocation
-                                  ) {
-        bl[0] = fixed_bl;
-        bd[0] = fixed_bd;
-        tl[0] = fixed_tl;
-        td[0] = fixed_td;
-        return Z_OK;
-    }
+    return Z_OK;
+  }
+
+  static int inflate_trees_fixed(int[] bl,  //literal desired/actual bit depth
+                                 int[] bd,  //distance desired/actual bit depth
+                                 int[][] tl,//literal/length tree result
+                                 int[][] td,//distance tree result 
+                                 ZStream z  //for memory allocation
+				 ){
+    bl[0]=fixed_bl;
+    bd[0]=fixed_bd;
+    tl[0]=fixed_tl;
+    td[0]=fixed_td;
+    return Z_OK;
+  }
 
-    private void initWorkArea(int vsize) {
-        if (hn == null) {
-            hn = new int[1];
-            v = new int[vsize];
-            c = new int[BMAX + 1];
-            r = new int[3];
-            u = new int[BMAX];
-            x = new int[BMAX + 1];
-        }
-
-        if (v.length < vsize) { v = new int[vsize]; }
-
-        for (int i = 0; i < vsize; i++) {v[i] = 0;}
-
-        for (int i = 0; i < BMAX + 1; i++) {c[i] = 0;}
-
-        for (int i = 0; i < 3; i++) {r[i] = 0;}
-
-//  for(int i=0; i<BMAX; i++){u[i]=0;}
-        System.arraycopy(c, 0, u, 0, BMAX);
-//  for(int i=0; i<BMAX+1; i++){x[i]=0;}
-        System.arraycopy(c, 0, x, 0, BMAX + 1);
+  private void initWorkArea(int vsize){
+    if(hn==null){
+      hn=new int[1];
+      v=new int[vsize];
+      c=new int[BMAX+1];
+      r=new int[3];
+      u=new int[BMAX];
+      x=new int[BMAX+1];
     }
+    if(v.length<vsize){ v=new int[vsize]; }
+    for(int i=0; i<vsize; i++){v[i]=0;}
+    for(int i=0; i<BMAX+1; i++){c[i]=0;}
+    for(int i=0; i<3; i++){r[i]=0;}
+    System.arraycopy(c, 0, u, 0, BMAX);
+    System.arraycopy(c, 0, x, 0, BMAX+1);
+  }
 }
--- a/src/com/jcraft/jzlib/Inflate.java	Fri Aug 01 11:57:17 2014 -0700
+++ b/src/com/jcraft/jzlib/Inflate.java	Fri Aug 01 13:34:58 2014 -0700
@@ -1,6 +1,6 @@
 /* -*-mode:java; c-basic-offset:2; -*- */
 /*
-Copyright (c) 2000,2001,2002,2003 ymnk, JCraft,Inc. All rights reserved.
+Copyright (c) 2000-2011 ymnk, JCraft,Inc. All rights reserved.
 
 Redistribution and use in source and binary forms, with or without
 modification, are permitted provided that the following conditions are met:
@@ -8,8 +8,8 @@
   1. Redistributions of source code must retain the above copyright notice,
      this list of conditions and the following disclaimer.
 
-  2. Redistributions in binary form must reproduce the above copyright
-     notice, this list of conditions and the following disclaimer in
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
      the documentation and/or other materials provided with the distribution.
 
   3. The names of the authors may not be used to endorse or promote products
@@ -34,364 +34,718 @@
 
 package com.jcraft.jzlib;
 
-final class Inflate {
+final class Inflate{
+  
+  static final private int MAX_WBITS=15; // 32K LZ77 window
+
+  // preset dictionary flag in zlib header
+  static final private int PRESET_DICT=0x20;
 
-    static final private int MAX_WBITS = 15; // 32K LZ77 window
+  static final int Z_NO_FLUSH=0;
+  static final int Z_PARTIAL_FLUSH=1;
+  static final int Z_SYNC_FLUSH=2;
+  static final int Z_FULL_FLUSH=3;
+  static final int Z_FINISH=4;
+
+  static final private int Z_DEFLATED=8;
 
-    // preset dictionary flag in zlib header
-    static final private int PRESET_DICT = 0x20;
+  static final private int Z_OK=0;
+  static final private int Z_STREAM_END=1;
+  static final private int Z_NEED_DICT=2;
+  static final private int Z_ERRNO=-1;
+  static final private int Z_STREAM_ERROR=-2;
+  static final private int Z_DATA_ERROR=-3;
+  static final private int Z_MEM_ERROR=-4;
+  static final private int Z_BUF_ERROR=-5;
+  static final private int Z_VERSION_ERROR=-6;
 
-    static final int Z_NO_FLUSH = 0;
-    static final int Z_PARTIAL_FLUSH = 1;
-    static final int Z_SYNC_FLUSH = 2;
-    static final int Z_FULL_FLUSH = 3;
-    static final int Z_FINISH = 4;
-
-    static final private int Z_DEFLATED = 8;
+  static final private int METHOD=0;   // waiting for method byte
+  static final private int FLAG=1;     // waiting for flag byte
+  static final private int DICT4=2;    // four dictionary check bytes to go
+  static final private int DICT3=3;    // three dictionary check bytes to go
+  static final private int DICT2=4;    // two dictionary check bytes to go
+  static final private int DICT1=5;    // one dictionary check byte to go
+  static final private int DICT0=6;    // waiting for inflateSetDictionary
+  static final private int BLOCKS=7;   // decompressing blocks
+  static final private int CHECK4=8;   // four check bytes to go
+  static final private int CHECK3=9;   // three check bytes to go
+  static final private int CHECK2=10;  // two check bytes to go
+  static final private int CHECK1=11;  // one check byte to go
+  static final private int DONE=12;    // finished check, done
+  static final private int BAD=13;     // got an error--stay here
 
-    static final private int Z_OK = 0;
-    static final private int Z_STREAM_END = 1;
-    static final private int Z_NEED_DICT = 2;
-    static final private int Z_ERRNO = -1;
-    static final private int Z_STREAM_ERROR = -2;
-    static final private int Z_DATA_ERROR = -3;
-    static final private int Z_MEM_ERROR = -4;
-    static final private int Z_BUF_ERROR = -5;
-    static final private int Z_VERSION_ERROR = -6;
+  static final private int HEAD=14;
+  static final private int LENGTH=15;
+  static final private int TIME=16;
+  static final private int OS=17;
+  static final private int EXLEN=18;
+  static final private int EXTRA=19;
+  static final private int NAME=20;
+  static final private int COMMENT=21;
+  static final private int HCRC=22;
+  static final private int FLAGS=23;
+
+  static final int INFLATE_ANY=0x40000000;
+
+  int mode;                            // current inflate mode
+
+  // mode dependent information
+  int method;        // if FLAGS, method byte
+
+  // if CHECK, check values to compare
+  long was = -1;           // computed check value
+  long need;               // stream check value
+
+  // if BAD, inflateSync's marker bytes count
+  int marker;
 
-    static final private int METHOD = 0; // waiting for method byte
-    static final private int FLAG = 1;   // waiting for flag byte
-    static final private int DICT4 = 2;  // four dictionary check bytes to go
-    static final private int DICT3 = 3;  // three dictionary check bytes to go
-    static final private int DICT2 = 4;  // two dictionary check bytes to go
-    static final private int DICT1 = 5;  // one dictionary check byte to go
-    static final private int DICT0 = 6;  // waiting for inflateSetDictionary
-    static final private int BLOCKS = 7; // decompressing blocks
-    static final private int CHECK4 = 8; // four check bytes to go
-    static final private int CHECK3 = 9; // three check bytes to go
-    static final private int CHECK2 = 10; // two check bytes to go
-    static final private int CHECK1 = 11; // one check byte to go
-    static final private int DONE = 12;  // finished check, done
-    static final private int BAD = 13;   // got an error--stay here
+  // mode independent information
+  int  wrap;          // flag for no wrapper
+                      // 0: no wrapper
+                      // 1: zlib header
+                      // 2: gzip header
+                      // 4: auto detection
+
+  int wbits;            // log2(window size)  (8..15, defaults to 15)
+
+  InfBlocks blocks;     // current inflate_blocks state
+
+  private final ZStream z;
+
+  private int flags; 
 
-    int mode;                            // current inflate mode
+  private int need_bytes = -1;
+  private byte[] crcbuf=new byte[4];
+
+  GZIPHeader gheader = null;
 
-    // mode dependent information
-    int method;        // if FLAGS, method byte
+  int inflateReset(){
+    if(z == null) return Z_STREAM_ERROR;
+    
+    z.total_in = z.total_out = 0;
+    z.msg = null;
+    this.mode = HEAD;
+    this.need_bytes = -1;
+    this.blocks.reset();
+    return Z_OK;
+  }
 
-    // if CHECK, check values to compare
-    long[] was = new long[1] ; // computed check value
-    long need;               // stream check value
+  int inflateEnd(){
+    if(blocks != null){
+      blocks.free();
+    }
+    return Z_OK;
+  }
 
-    // if BAD, inflateSync's marker bytes count
-    int marker;
+  Inflate(ZStream z){
+    this.z=z;
+  }
 
-    // mode independent information
-    int  nowrap;          // flag for no wrapper
-    int wbits;            // log2(window size)  (8..15, defaults to 15)
+  int inflateInit(int w){
+    z.msg = null;
+    blocks = null;
 
-    InfBlocks blocks;     // current inflate_blocks state
-
-    int inflateReset(ZStream z) {
-        if (z == null || z.istate == null) return Z_STREAM_ERROR;
-
-        z.total_in = z.total_out = 0;
-        z.msg = null;
-        z.istate.mode = z.istate.nowrap != 0 ? BLOCKS : METHOD;
-        z.istate.blocks.reset(z, null);
-        return Z_OK;
+    // handle undocumented wrap option (no zlib header or check)
+    wrap = 0;
+    if(w < 0){
+      w = - w;
+    }
+    else if((w&INFLATE_ANY) != 0){
+      wrap = 4;
+      w &= ~INFLATE_ANY;
+      if(w < 48)
+        w &= 15;
+    }
+    else if((w & ~31) != 0) { // for example, DEF_WBITS + 32
+      wrap = 4;               // zlib and gzip wrapped data should be accepted.
+      w &= 15;
+    }
+    else {
+      wrap = (w >> 4) + 1;
+      if(w < 48)
+        w &= 15;
     }
 
-    int inflateEnd(ZStream z) {
-        if (blocks != null)
-            blocks.free(z);
-
-        blocks = null;
-        //    ZFREE(z, z->state);
-        return Z_OK;
+    if(w<8 ||w>15){
+      inflateEnd();
+      return Z_STREAM_ERROR;
+    }
+    if(blocks != null && wbits != w){
+      blocks.free();
+      blocks=null;
     }
 
-    int inflateInit(ZStream z, int w) {
-        z.msg = null;
-        blocks = null;
-        // handle undocumented nowrap option (no zlib header or check)
-        nowrap = 0;
+    // set window size
+    wbits=w;
+
+    this.blocks=new InfBlocks(z, 1<<w);
 
-        if (w < 0) {
-            w = - w;
-            nowrap = 1;
-        }
+    // reset state
+    inflateReset();
+
+    return Z_OK;
+  }
 
-        // set window size
-        if (w < 8 || w > 15) {
-            inflateEnd(z);
-            return Z_STREAM_ERROR;
-        }
+  int inflate(int f){
+    int hold = 0;
 
-        wbits = w;
-        z.istate.blocks = new InfBlocks(z,
-                                        z.istate.nowrap != 0 ? null : this,
-                                        1 << w);
-        // reset state
-        inflateReset(z);
-        return Z_OK;
+    int r;
+    int b;
+
+    if(z == null || z.next_in == null){
+      if(f == Z_FINISH && this.mode==HEAD)
+        return Z_OK; 
+      return Z_STREAM_ERROR;
     }
 
-    int inflate(ZStream z, int f) {
-        int r;
-        int b;
+    f = f == Z_FINISH ? Z_BUF_ERROR : Z_OK;
+    r = Z_BUF_ERROR;
+    while (true){
 
-        if (z == null || z.istate == null || z.next_in == null)
-            return Z_STREAM_ERROR;
+      switch (this.mode){
+      case HEAD:
+        if(wrap==0){
+	  this.mode = BLOCKS;
+          break;
+        } 
+
+        try { r=readBytes(2, r, f); }
+        catch(Return e){ return e.r; }
 
-        f = f == Z_FINISH ? Z_BUF_ERROR : Z_OK;
-        r = Z_BUF_ERROR;
+        if((wrap == 4 || (wrap&2)!=0) &&
+           this.need == 0x8b1fL) {   // gzip header
+          if(wrap == 4){
+            wrap = 2;
+          }
+	  z.adler=new CRC32();
+          checksum(2, this.need);
+
+          if(gheader==null) 
+            gheader=new GZIPHeader();
 
-        while (true) {
-//System.out.println("mode: "+z.istate.mode);
-            switch (z.istate.mode) {
-                case METHOD:
-                    if (z.avail_in == 0)return r; r = f;
+          this.mode = FLAGS;
+          break;
+        }
 
-                    z.avail_in--; z.total_in++;
+        if((wrap&2) != 0){
+          this.mode = BAD;
+          z.msg = "incorrect header check";
+          break;
+        }
+
+        flags = 0;
+
+        this.method = ((int)this.need)&0xff;
+        b=((int)(this.need>>8))&0xff;
 
-                    if (((z.istate.method = z.next_in[z.next_in_index++]) & 0xf) != Z_DEFLATED) {
-                        z.istate.mode = BAD;
-                        z.msg = "unknown compression method";
-                        z.istate.marker = 5;       // can't try inflateSync
-                        break;
-                    }
+        if(((wrap&1)==0 ||  // check if zlib header allowed
+            (((this.method << 8)+b) % 31)!=0) &&
+           (this.method&0xf)!=Z_DEFLATED){
+          if(wrap == 4){
+            z.next_in_index -= 2;
+            z.avail_in += 2;
+            z.total_in -= 2;
+            wrap = 0;
+            this.mode = BLOCKS;
+            break;
+          }  
+          this.mode = BAD;
+          z.msg = "incorrect header check";
+          // since zlib 1.2, it is allowted to inflateSync for this case.
+          /*
+          this.marker = 5;       // can't try inflateSync
+          */
+          break;
+        }
 
-                    if ((z.istate.method >> 4) + 8 > z.istate.wbits) {
-                        z.istate.mode = BAD;
-                        z.msg = "invalid window size";
-                        z.istate.marker = 5;       // can't try inflateSync
-                        break;
-                    }
+        if((this.method&0xf)!=Z_DEFLATED){
+          this.mode = BAD;
+          z.msg="unknown compression method";
+          // since zlib 1.2, it is allowted to inflateSync for this case.
+	  /*
+          this.marker = 5;       // can't try inflateSync
+	  */
+          break;
+        }
+  
+        if(wrap == 4){
+          wrap = 1;
+        }  
 
-                    z.istate.mode = FLAG;
+        if((this.method>>4)+8>this.wbits){
+          this.mode = BAD;
+          z.msg="invalid window size";
+          // since zlib 1.2, it is allowted to inflateSync for this case.
+	  /*
+          this.marker = 5;       // can't try inflateSync
+	  */
+          break;
+        }
 
-                case FLAG:
-                    if (z.avail_in == 0)return r; r = f;
+        z.adler=new Adler32();
 
-                    z.avail_in--; z.total_in++;
-                    b = (z.next_in[z.next_in_index++]) & 0xff;
+        if((b&PRESET_DICT)==0){
+          this.mode = BLOCKS;
+          break;
+        }
+        this.mode = DICT4;
+      case DICT4:
 
-                    if ((((z.istate.method << 8) + b) % 31) != 0) {
-                        z.istate.mode = BAD;
-                        z.msg = "incorrect header check";
-                        z.istate.marker = 5;       // can't try inflateSync
-                        break;
-                    }
+        if(z.avail_in==0)return r;r=f;
+
+        z.avail_in--; z.total_in++;
+        this.need=((z.next_in[z.next_in_index++]&0xff)<<24)&0xff000000L;
+        this.mode=DICT3;
+      case DICT3:
 
-                    if ((b & PRESET_DICT) == 0) {
-                        z.istate.mode = BLOCKS;
-                        break;
-                    }
+        if(z.avail_in==0)return r;r=f;
+
+        z.avail_in--; z.total_in++;
+        this.need+=((z.next_in[z.next_in_index++]&0xff)<<16)&0xff0000L;
+        this.mode=DICT2;
+      case DICT2:
 
-                    z.istate.mode = DICT4;
+        if(z.avail_in==0)return r;r=f;
 
-                case DICT4:
-                    if (z.avail_in == 0)return r; r = f;
+        z.avail_in--; z.total_in++;
+        this.need+=((z.next_in[z.next_in_index++]&0xff)<<8)&0xff00L;
+        this.mode=DICT1;
+      case DICT1:
+
+        if(z.avail_in==0)return r;r=f;
 
-                    z.avail_in--; z.total_in++;
-                    z.istate.need = ((z.next_in[z.next_in_index++] & 0xff) << 24) & 0xff000000L;
-                    z.istate.mode = DICT3;
-
-                case DICT3:
-                    if (z.avail_in == 0)return r; r = f;
+        z.avail_in--; z.total_in++;
+        this.need += (z.next_in[z.next_in_index++]&0xffL);
+        z.adler.reset(this.need);
+        this.mode = DICT0;
+        return Z_NEED_DICT;
+      case DICT0:
+        this.mode = BAD;
+        z.msg = "need dictionary";
+        this.marker = 0;       // can try inflateSync
+        return Z_STREAM_ERROR;
+      case BLOCKS:
+        r = this.blocks.proc(r);
+        if(r == Z_DATA_ERROR){
+          this.mode = BAD;
+          this.marker = 0;     // can try inflateSync
+          break;
+        }
+        if(r == Z_OK){
+          r = f;
+        }
+        if(r != Z_STREAM_END){
+          return r;
+        }
+        r = f;
+        this.was=z.adler.getValue();
+        this.blocks.reset();
+        if(this.wrap==0){
+          this.mode=DONE;
+          break;
+        }
+        this.mode=CHECK4;
+      case CHECK4:
 
-                    z.avail_in--; z.total_in++;
-                    z.istate.need += ((z.next_in[z.next_in_index++] & 0xff) << 16) & 0xff0000L;
-                    z.istate.mode = DICT2;
+        if(z.avail_in==0)return r;r=f;
+
+        z.avail_in--; z.total_in++;
+        this.need=((z.next_in[z.next_in_index++]&0xff)<<24)&0xff000000L;
+        this.mode=CHECK3;
+      case CHECK3:
 
-                case DICT2:
-                    if (z.avail_in == 0)return r; r = f;
+        if(z.avail_in==0)return r;r=f;
+
+        z.avail_in--; z.total_in++;
+        this.need+=((z.next_in[z.next_in_index++]&0xff)<<16)&0xff0000L;
+        this.mode = CHECK2;
+      case CHECK2:
+
+        if(z.avail_in==0)return r;r=f;
 
-                    z.avail_in--; z.total_in++;
-                    z.istate.need += ((z.next_in[z.next_in_index++] & 0xff) << 8) & 0xff00L;
-                    z.istate.mode = DICT1;
+        z.avail_in--; z.total_in++;
+        this.need+=((z.next_in[z.next_in_index++]&0xff)<<8)&0xff00L;
+        this.mode = CHECK1;
+      case CHECK1:
+
+        if(z.avail_in==0)return r;r=f;
 
-                case DICT1:
-                    if (z.avail_in == 0)return r; r = f;
+        z.avail_in--; z.total_in++;
+        this.need+=(z.next_in[z.next_in_index++]&0xffL);
+
+        if(flags!=0){  // gzip
+          this.need = ((this.need&0xff000000)>>24 | 
+                          (this.need&0x00ff0000)>>8 | 
+                          (this.need&0x0000ff00)<<8 | 
+                          (this.need&0x0000ffff)<<24)&0xffffffffL;
+        }
 
-                    z.avail_in--; z.total_in++;
-                    z.istate.need += (z.next_in[z.next_in_index++] & 0xffL);
-                    z.adler = z.istate.need;
-                    z.istate.mode = DICT0;
-                    return Z_NEED_DICT;
+        if(((int)(this.was)) != ((int)(this.need))){
+          z.msg = "incorrect data check";
+          // chack is delayed
+          /*
+          this.mode = BAD;
+          this.marker = 5;       // can't try inflateSync
+          break;
+	  */
+        }
+        else if(flags!=0 && gheader!=null){
+          gheader.crc = this.need; 
+        }
 
-                case DICT0:
-                    z.istate.mode = BAD;
-                    z.msg = "need dictionary";
-                    z.istate.marker = 0;       // can try inflateSync
-                    return Z_STREAM_ERROR;
+        this.mode = LENGTH;
+      case LENGTH:
+        if (wrap!=0 && flags!=0) {
 
-                case BLOCKS:
-                    r = z.istate.blocks.proc(z, r);
+          try { r=readBytes(4, r, f); }
+          catch(Return e){ return e.r; }
 
-                    if (r == Z_DATA_ERROR) {
-                        z.istate.mode = BAD;
-                        z.istate.marker = 0;     // can try inflateSync
-                        break;
-                    }
+          if(z.msg!=null && z.msg.equals("incorrect data check")){
+            this.mode = BAD;
+            this.marker = 5;       // can't try inflateSync
+            break;
+          }
 
-                    if (r == Z_OK) {
-                        r = f;
-                    }
+          if (this.need != (z.total_out & 0xffffffffL)) {
+            z.msg = "incorrect length check";
+            this.mode = BAD;
+            break;
+          }
+          z.msg = null;
+        }
+        else {
+          if(z.msg!=null && z.msg.equals("incorrect data check")){
+            this.mode = BAD;
+            this.marker = 5;       // can't try inflateSync
+            break;
+          }
+        }
 
-                    if (r != Z_STREAM_END) {
-                        return r;
-                    }
+        this.mode = DONE;
+      case DONE:
+        return Z_STREAM_END;
+      case BAD:
+        return Z_DATA_ERROR;
 
-                    r = f;
-                    z.istate.blocks.reset(z, z.istate.was);
+      case FLAGS:
+
+        try { r=readBytes(2, r, f); }
+        catch(Return e){ return e.r; }
+
+        flags = ((int)this.need)&0xffff;
 
-                    if (z.istate.nowrap != 0) {
-                        z.istate.mode = DONE;
-                        break;
-                    }
+        if ((flags & 0xff) != Z_DEFLATED) {
+          z.msg = "unknown compression method";
+          this.mode = BAD; 
+          break;
+        }
+        if ((flags & 0xe000)!=0) {
+          z.msg = "unknown header flags set";
+          this.mode = BAD; 
+          break;
+        }
 
-                    z.istate.mode = CHECK4;
+        if ((flags & 0x0200)!=0){
+          checksum(2, this.need);
+        } 
 
-                case CHECK4:
-                    if (z.avail_in == 0)return r; r = f;
+        this.mode = TIME;
 
-                    z.avail_in--; z.total_in++;
-                    z.istate.need = ((z.next_in[z.next_in_index++] & 0xff) << 24) & 0xff000000L;
-                    z.istate.mode = CHECK3;
-
-                case CHECK3:
-                    if (z.avail_in == 0)return r; r = f;
-
-                    z.avail_in--; z.total_in++;
-                    z.istate.need += ((z.next_in[z.next_in_index++] & 0xff) << 16) & 0xff0000L;
-                    z.istate.mode = CHECK2;
-
-                case CHECK2:
-                    if (z.avail_in == 0)return r; r = f;
-
-                    z.avail_in--; z.total_in++;
-                    z.istate.need += ((z.next_in[z.next_in_index++] & 0xff) << 8) & 0xff00L;
-                    z.istate.mode = CHECK1;
-
-                case CHECK1:
-                    if (z.avail_in == 0)return r; r = f;
+      case TIME:
+        try { r=readBytes(4, r, f); }
+        catch(Return e){ return e.r; }
+        if(gheader!=null)
+          gheader.time = this.need;
+        if ((flags & 0x0200)!=0){
+          checksum(4, this.need);
+        }
+        this.mode = OS;
+      case OS:
+        try { r=readBytes(2, r, f); }
+        catch(Return e){ return e.r; }
+        if(gheader!=null){
+          gheader.xflags = ((int)this.need)&0xff;
+          gheader.os = (((int)this.need)>>8)&0xff;
+        }
+        if ((flags & 0x0200)!=0){
+          checksum(2, this.need);
+        }
+        this.mode = EXLEN;
+      case EXLEN:
+        if ((flags & 0x0400)!=0) {
+          try { r=readBytes(2, r, f); }
+          catch(Return e){ return e.r; }
+          if(gheader!=null){
+            gheader.extra = new byte[((int)this.need)&0xffff];
+          }
+          if ((flags & 0x0200)!=0){
+            checksum(2, this.need);
+          }
+        }
+        else if(gheader!=null){
+          gheader.extra=null;
+        }
+        this.mode = EXTRA;
 
-                    z.avail_in--; z.total_in++;
-                    z.istate.need += (z.next_in[z.next_in_index++] & 0xffL);
-
-                    if (((int)(z.istate.was[0])) != ((int)(z.istate.need))) {
-                        z.istate.mode = BAD;
-                        z.msg = "incorrect data check";
-                        z.istate.marker = 5;       // can't try inflateSync
-                        break;
-                    }
+      case EXTRA:
+        if ((flags & 0x0400)!=0) {
+          try { 
+            r=readBytes(r, f);
+            if(gheader!=null){
+              byte[] foo = tmp_string.toByteArray();
+              tmp_string=null;
+              if(foo.length == gheader.extra.length){
+                System.arraycopy(foo, 0, gheader.extra, 0, foo.length);
+	      }
+              else{
+                z.msg = "bad extra field length";
+                this.mode = BAD; 
+                break;
+	      }
+            }
+          }
+          catch(Return e){ return e.r; }
+        }
+        else if(gheader!=null){
+          gheader.extra=null;
+	}
+	this.mode = NAME;
+      case NAME:
+	if ((flags & 0x0800)!=0) {
+          try { 
+            r=readString(r, f);
+            if(gheader!=null){
+              gheader.name=tmp_string.toByteArray();
+            }
+            tmp_string=null;
+          }
+          catch(Return e){ return e.r; }
+        }
+        else if(gheader!=null){
+          gheader.name=null;
+	}
+        this.mode = COMMENT;
+      case COMMENT:
+        if ((flags & 0x1000)!=0) {
+          try { 
+            r=readString(r, f);
+            if(gheader!=null){
+              gheader.comment=tmp_string.toByteArray();
+            }
+            tmp_string=null;
+          }
+          catch(Return e){ return e.r; }
+        }
+        else if(gheader!=null){
+          gheader.comment=null;
+	}
+        this.mode = HCRC;
+      case HCRC:
+	if ((flags & 0x0200)!=0) {
+          try { r=readBytes(2, r, f); }
+          catch(Return e){ return e.r; }
+          if(gheader!=null){
+            gheader.hcrc=(int)(this.need&0xffff);
+          }
+          if(this.need != (z.adler.getValue()&0xffffL)){
+            this.mode = BAD;
+            z.msg = "header crc mismatch";
+            this.marker = 5;       // can't try inflateSync
+            break;
+          }
+        }
+        z.adler = new CRC32();
 
-                    z.istate.mode = DONE;
-
-                case DONE:
-                    return Z_STREAM_END;
+        this.mode = BLOCKS;
+        break;
+      default:
+        return Z_STREAM_ERROR;
+      }
+    }
+  }
 
-                case BAD:
-                    return Z_DATA_ERROR;
-
-                default:
-                    return Z_STREAM_ERROR;
-            }
-        }
+  int inflateSetDictionary(byte[] dictionary, int dictLength){
+    if(z==null || (this.mode != DICT0 && this.wrap != 0)){
+      return Z_STREAM_ERROR;
     }
 
+    int index=0;
+    int length = dictLength;
 
-    int inflateSetDictionary(ZStream z, byte[] dictionary, int dictLength) {
-        int index = 0;
-        int length = dictLength;
+    if(this.mode==DICT0){
+      long adler_need=z.adler.getValue();
+      z.adler.reset();
+      z.adler.update(dictionary, 0, dictLength);
+      if(z.adler.getValue()!=adler_need){
+        return Z_DATA_ERROR;
+      }
+    }
+
+    z.adler.reset();
 
-        if (z == null || z.istate == null || z.istate.mode != DICT0)
-            return Z_STREAM_ERROR;
+    if(length >= (1<<this.wbits)){
+      length = (1<<this.wbits)-1;
+      index=dictLength - length;
+    }
+    this.blocks.set_dictionary(dictionary, index, length);
+    this.mode = BLOCKS;
+    return Z_OK;
+  }
+
+  static private byte[] mark = {(byte)0, (byte)0, (byte)0xff, (byte)0xff};
 
-        if (z._adler.adler32(1L, dictionary, 0, dictLength) != z.adler) {
-            return Z_DATA_ERROR;
-        }
-
-        z.adler = z._adler.adler32(0, null, 0, 0);
+  int inflateSync(){
+    int n;       // number of bytes to look at
+    int p;       // pointer to bytes
+    int m;       // number of marker bytes found in a row
+    long r, w;   // temporaries to save total_in and total_out
 
-        if (length >= (1 << z.istate.wbits)) {
-            length = (1 << z.istate.wbits) - 1;
-            index = dictLength - length;
-        }
+    // set up
+    if(z == null)
+      return Z_STREAM_ERROR;
+    if(this.mode != BAD){
+      this.mode = BAD;
+      this.marker = 0;
+    }
+    if((n=z.avail_in)==0)
+      return Z_BUF_ERROR;
 
-        z.istate.blocks.set_dictionary(dictionary, index, length);
-        z.istate.mode = BLOCKS;
-        return Z_OK;
+    p=z.next_in_index;
+    m=this.marker;
+    // search
+    while (n!=0 && m < 4){
+      if(z.next_in[p] == mark[m]){
+        m++;
+      }
+      else if(z.next_in[p]!=0){
+        m = 0;
+      }
+      else{
+        m = 4 - m;
+      }
+      p++; n--;
     }
 
-    static private byte[] mark = { (byte)0, (byte)0, (byte)0xff, (byte)0xff};
-
-    int inflateSync(ZStream z) {
-        int n;       // number of bytes to look at
-        int p;       // pointer to bytes
-        int m;       // number of marker bytes found in a row
-        long r, w;   // temporaries to save total_in and total_out
+    // restore
+    z.total_in += p-z.next_in_index;
+    z.next_in_index = p;
+    z.avail_in = n;
+    this.marker = m;
 
-        // set up
-        if (z == null || z.istate == null)
-            return Z_STREAM_ERROR;
+    // return no joy or set up to restart on a new block
+    if(m != 4){
+      return Z_DATA_ERROR;
+    }
+    r=z.total_in;  w=z.total_out;
+    inflateReset();
+    z.total_in=r;  z.total_out = w;
+    this.mode = BLOCKS;
 
-        if (z.istate.mode != BAD) {
-            z.istate.mode = BAD;
-            z.istate.marker = 0;
-        }
+    return Z_OK;
+  }
 
-        if ((n = z.avail_in) == 0)
-            return Z_BUF_ERROR;
-
-        p = z.next_in_index;
-        m = z.istate.marker;
+  // Returns true if inflate is currently at the end of a block generated
+  // by Z_SYNC_FLUSH or Z_FULL_FLUSH. This function is used by one PPP
+  // implementation to provide an additional safety check. PPP uses Z_SYNC_FLUSH
+  // but removes the length bytes of the resulting empty stored block. When
+  // decompressing, PPP checks that at the end of input packet, inflate is
+  // waiting for these length bytes.
+  int inflateSyncPoint(){
+    if(z == null || this.blocks == null)
+      return Z_STREAM_ERROR;
+    return this.blocks.sync_point();
+  }
 
-        // search
-        while (n != 0 && m < 4) {
-            if (z.next_in[p] == mark[m]) {
-                m++;
-            }
-            else if (z.next_in[p] != 0) {
-                m = 0;
-            }
-            else {
-                m = 4 - m;
-            }
+  private int readBytes(int n, int r, int f) throws Return{
+    if(need_bytes == -1){
+      need_bytes=n;
+      this.need=0;
+    }
+    while(need_bytes>0){
+      if(z.avail_in==0){ throw new Return(r); }; r=f;
+      z.avail_in--; z.total_in++;
+      this.need = this.need | 
+	((z.next_in[z.next_in_index++]&0xff)<<((n-need_bytes)*8));
+      need_bytes--;
+    }
+    if(n==2){
+      this.need&=0xffffL;
+    }
+    else if(n==4) {
+      this.need&=0xffffffffL;
+    }
+    need_bytes=-1;
+    return r;
+  }
+  class Return extends Exception{
+    int r;
+    Return(int r){this.r=r; }
+  }
 
-            p++; n--;
-        }
+  private java.io.ByteArrayOutputStream tmp_string = null;
+  private int readString(int r, int f) throws Return{
+    if(tmp_string == null){
+      tmp_string=new java.io.ByteArrayOutputStream();
+    }
+    int b=0; 
+    do {
+      if(z.avail_in==0){ throw new Return(r); }; r=f;
+      z.avail_in--; z.total_in++;
+      b = z.next_in[z.next_in_index];
+      if(b!=0) tmp_string.write(z.next_in, z.next_in_index, 1);
+      z.adler.update(z.next_in, z.next_in_index, 1);
+      z.next_in_index++;
+    }while(b!=0);
+    return r;
+  }
 
-        // restore
-        z.total_in += p - z.next_in_index;
-        z.next_in_index = p;
-        z.avail_in = n;
-        z.istate.marker = m;
-
-        // return no joy or set up to restart on a new block
-        if (m != 4) {
-            return Z_DATA_ERROR;
-        }
+  private int readBytes(int r, int f) throws Return{
+    if(tmp_string == null){
+      tmp_string=new java.io.ByteArrayOutputStream();
+    }
+    int b=0; 
+    while(this.need>0){
+      if(z.avail_in==0){ throw new Return(r); }; r=f;
+      z.avail_in--; z.total_in++;
+      b = z.next_in[z.next_in_index];
+      tmp_string.write(z.next_in, z.next_in_index, 1);
+      z.adler.update(z.next_in, z.next_in_index, 1);
+      z.next_in_index++;
+      this.need--;
+    }
+    return r;
+  }
 
-        r = z.total_in;  w = z.total_out;
-        inflateReset(z);
-        z.total_in = r;  z.total_out = w;
-        z.istate.mode = BLOCKS;
-        return Z_OK;
+  private void checksum(int n, long v){
+    for(int i=0; i<n; i++){
+      crcbuf[i]=(byte)(v&0xff);
+      v>>=8;
     }
+    z.adler.update(crcbuf, 0, n);
+  }
+
+  public GZIPHeader getGZIPHeader(){
+    return gheader;
+  }
 
-    // Returns true if inflate is currently at the end of a block generated
-    // by Z_SYNC_FLUSH or Z_FULL_FLUSH. This function is used by one PPP
-    // implementation to provide an additional safety check. PPP uses Z_SYNC_FLUSH
-    // but removes the length bytes of the resulting empty stored block. When
-    // decompressing, PPP checks that at the end of input packet, inflate is
-    // waiting for these length bytes.
-    int inflateSyncPoint(ZStream z) {
-        if (z == null || z.istate == null || z.istate.blocks == null)
-            return Z_STREAM_ERROR;
-
-        return z.istate.blocks.sync_point();
+  boolean inParsingHeader(){
+    switch(mode){
+      case HEAD:
+      case DICT4:
+      case DICT3:
+      case DICT2:
+      case DICT1:
+      case FLAGS:
+      case TIME:
+      case OS:
+      case EXLEN:
+      case EXTRA:
+      case NAME:
+      case COMMENT:
+      case HCRC:
+	return true;
+      default:
+	return false;
     }
+  }
 }
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/com/jcraft/jzlib/Inflater.java	Fri Aug 01 13:34:58 2014 -0700
@@ -0,0 +1,168 @@
+/* -*-mode:java; c-basic-offset:2; indent-tabs-mode:nil -*- */
+/*
+Copyright (c) 2011 ymnk, JCraft,Inc. All rights reserved.
+
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions are met:
+
+  1. Redistributions of source code must retain the above copyright notice,
+     this list of conditions and the following disclaimer.
+
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
+     the documentation and/or other materials provided with the distribution.
+
+  3. The names of the authors may not be used to endorse or promote products
+     derived from this software without specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES,
+INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
+FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JCRAFT,
+INC. OR ANY CONTRIBUTORS TO THIS SOFTWARE BE LIABLE FOR ANY DIRECT, INDIRECT,
+INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
+OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
+LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
+EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+/*
+ * This program is based on zlib-1.1.3, so all credit should go authors
+ * Jean-loup Gailly(jloup@gzip.org) and Mark Adler(madler@alumni.caltech.edu)
+ * and contributors of zlib.
+ */
+
+package com.jcraft.jzlib;
+
+final public class Inflater extends ZStream{
+
+  static final private int MAX_WBITS=15;        // 32K LZ77 window
+  static final private int DEF_WBITS=MAX_WBITS;
+
+  static final private int Z_NO_FLUSH=0;
+  static final private int Z_PARTIAL_FLUSH=1;
+  static final private int Z_SYNC_FLUSH=2;
+  static final private int Z_FULL_FLUSH=3;
+  static final private int Z_FINISH=4;
+
+  static final private int MAX_MEM_LEVEL=9;
+
+  static final private int Z_OK=0;
+  static final private int Z_STREAM_END=1;
+  static final private int Z_NEED_DICT=2;
+  static final private int Z_ERRNO=-1;
+  static final private int Z_STREAM_ERROR=-2;
+  static final private int Z_DATA_ERROR=-3;
+  static final private int Z_MEM_ERROR=-4;
+  static final private int Z_BUF_ERROR=-5;
+  static final private int Z_VERSION_ERROR=-6;
+
+  public Inflater() {
+    super();
+    init();
+  }
+
+  public Inflater(JZlib.WrapperType wrapperType) throws GZIPException {
+    this(DEF_WBITS, wrapperType);
+  }
+
+  public Inflater(int w, JZlib.WrapperType wrapperType) throws GZIPException {
+    super();
+    int ret = init(w, wrapperType);
+    if(ret!=Z_OK)
+      throw new GZIPException(ret+": "+msg);
+  }
+
+  public Inflater(int w) throws GZIPException {
+    this(w, false);
+  }
+
+  public Inflater(boolean nowrap) throws GZIPException {
+    this(DEF_WBITS, nowrap);
+  }
+
+  public Inflater(int w, boolean nowrap) throws GZIPException {
+    super();
+    int ret = init(w, nowrap);
+    if(ret!=Z_OK)
+      throw new GZIPException(ret+": "+msg);
+  }
+
+  private boolean finished = false;
+
+  public int init(){
+    return init(DEF_WBITS);
+  }
+
+  public int init(JZlib.WrapperType wrapperType){
+    return init(DEF_WBITS, wrapperType);
+  }
+
+  public int init(int w, JZlib.WrapperType wrapperType) {
+    boolean nowrap = false;
+    if(wrapperType == JZlib.W_NONE){
+      nowrap = true;
+    }
+    else if(wrapperType == JZlib.W_GZIP) {
+      w += 16;
+    }
+    else if(wrapperType == JZlib.W_ANY) {
+      w |= Inflate.INFLATE_ANY;
+    }
+    else if(wrapperType == JZlib.W_ZLIB) {
+    }
+    return init(w, nowrap);
+  }
+
+  public int init(boolean nowrap){
+    return init(DEF_WBITS, nowrap);
+  }
+
+  public int init(int w){
+    return init(w, false);
+  }
+
+  public int init(int w, boolean nowrap){
+    finished = false;
+    istate=new Inflate(this);
+    return istate.inflateInit(nowrap?-w:w);
+  }
+
+  public int inflate(int f){
+    if(istate==null) return Z_STREAM_ERROR;
+    int ret = istate.inflate(f);
+    if(ret == Z_STREAM_END) 
+      finished = true;
+    return ret;
+  }
+
+  public int end(){
+    finished = true;
+    if(istate==null) return Z_STREAM_ERROR;
+    int ret=istate.inflateEnd();
+//    istate = null;
+    return ret;
+  }
+
+  public int sync(){
+    if(istate == null)
+      return Z_STREAM_ERROR;
+    return istate.inflateSync();
+  }
+
+  public int syncPoint(){
+    if(istate == null)
+      return Z_STREAM_ERROR;
+    return istate.inflateSyncPoint();
+  }
+
+  public int setDictionary(byte[] dictionary, int dictLength){
+    if(istate == null)
+      return Z_STREAM_ERROR;
+    return istate.inflateSetDictionary(dictionary, dictLength);
+  }
+
+  public boolean finished(){
+    return istate.mode==12 /*DONE*/;
+  }
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/com/jcraft/jzlib/InflaterInputStream.java	Fri Aug 01 13:34:58 2014 -0700
@@ -0,0 +1,247 @@
+/* -*-mode:java; c-basic-offset:2; indent-tabs-mode:nil -*- */
+/*
+Copyright (c) 2011 ymnk, JCraft,Inc. All rights reserved.
+
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions are met:
+
+  1. Redistributions of source code must retain the above copyright notice,
+     this list of conditions and the following disclaimer.
+
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
+     the documentation and/or other materials provided with the distribution.
+
+  3. The names of the authors may not be used to endorse or promote products
+     derived from this software without specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES,
+INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
+FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JCRAFT,
+INC. OR ANY CONTRIBUTORS TO THIS SOFTWARE BE LIABLE FOR ANY DIRECT, INDIRECT,
+INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
+OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
+LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
+EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+package com.jcraft.jzlib;
+import java.io.*;
+
+public class InflaterInputStream extends FilterInputStream {
+  protected final Inflater inflater;
+  protected byte[] buf;
+
+  private boolean closed = false;
+
+  private boolean eof = false;
+
+  private boolean close_in = true;
+
+  protected static final int DEFAULT_BUFSIZE = 512;
+
+  public InflaterInputStream(InputStream in) throws IOException {
+    this(in, false);
+  }
+
+  public InflaterInputStream(InputStream in, boolean nowrap) throws IOException {
+    this(in, new Inflater(nowrap));
+    myinflater = true;
+  }
+
+  public InflaterInputStream(InputStream in, Inflater inflater) throws IOException {
+    this(in, inflater, DEFAULT_BUFSIZE);
+  }
+
+  public InflaterInputStream(InputStream in,
+                             Inflater inflater, int size) throws IOException {
+    this(in, inflater, size, true);
+  }
+
+  public InflaterInputStream(InputStream in,
+                             Inflater inflater,
+                             int size, boolean close_in) throws IOException {
+    super(in);
+    if (in == null || inflater == null) {
+      throw new NullPointerException();
+    }
+    else if (size <= 0) {
+      throw new IllegalArgumentException("buffer size must be greater than 0");
+    }
+    this.inflater = inflater;
+    buf = new byte[size];
+    this.close_in = close_in;
+  }
+
+  protected boolean myinflater = false;
+
+  private byte[] byte1 = new byte[1];
+
+  public int read() throws IOException {
+    if (closed) { throw new IOException("Stream closed"); }
+    return read(byte1, 0, 1) == -1 ? -1 : byte1[0] & 0xff;
+  }
+
+  public int read(byte[] b, int off, int len) throws IOException {
+    if (closed) { throw new IOException("Stream closed"); }
+    if (b == null) {
+      throw new NullPointerException();
+    }
+    else if (off < 0 || len < 0 || len > b.length - off) {
+      throw new IndexOutOfBoundsException();
+    }
+    else if (len == 0) {
+      return 0;
+    }
+    else if (eof) {
+      return -1;
+    }
+
+    int n = 0;
+    inflater.setOutput(b, off, len);
+    while(!eof) {
+      if(inflater.avail_in==0)
+        fill();
+      int err = inflater.inflate(JZlib.Z_NO_FLUSH);
+      n += inflater.next_out_index - off;
+      off = inflater.next_out_index;
+      switch(err) {
+        case JZlib.Z_DATA_ERROR:
+          throw new IOException(inflater.msg);
+        case JZlib.Z_STREAM_END:
+        case JZlib.Z_NEED_DICT:
+          eof = true;
+          if(err == JZlib.Z_NEED_DICT)
+            return -1;
+          break;
+        default:
+      } 
+      if(inflater.avail_out==0)
+        break;
+    }
+    return n;
+  }
+
+  public int available() throws IOException {
+    if (closed) { throw new IOException("Stream closed"); }
+    if (eof) {
+      return 0;
+    }
+    else {
+      return 1;
+    }
+  }
+
+  private byte[] b = new byte[512];
+
+  public long skip(long n) throws IOException {
+    if (n < 0) {
+      throw new IllegalArgumentException("negative skip length");
+    }
+
+    if (closed) { throw new IOException("Stream closed"); }
+
+    int max = (int)Math.min(n, Integer.MAX_VALUE);
+    int total = 0;
+    while (total < max) {
+      int len = max - total;
+      if (len > b.length) {
+        len = b.length;
+      }
+      len = read(b, 0, len);
+      if (len == -1) {
+        eof = true;
+        break;
+      }
+      total += len;
+    }
+    return total;
+  }
+
+  public void close() throws IOException {
+    if (!closed) {
+      if (myinflater)
+        inflater.end();
+      if(close_in)
+        in.close();
+      closed = true;
+    }
+  }
+
+  protected void fill() throws IOException {
+    if (closed) { throw new IOException("Stream closed"); }
+    int len = in.read(buf, 0, buf.length);
+    if (len == -1) {
+      if(inflater.istate.wrap == 0 &&
+         !inflater.finished()){
+        buf[0]=0;
+        len=1;
+      }
+      else if(inflater.istate.was != -1){  // in reading trailer
+        throw new IOException("footer is not found");
+      }
+      else{
+        throw new EOFException("Unexpected end of ZLIB input stream");
+      }
+    }
+    inflater.setInput(buf, 0, len, true);
+  }
+
+  public boolean markSupported() {
+    return false;
+  }
+
+  public synchronized void mark(int readlimit) {
+  }
+
+  public synchronized void reset() throws IOException {
+    throw new IOException("mark/reset not supported");
+  }
+
+  public long getTotalIn() {
+    return inflater.getTotalIn();
+  }
+
+  public long getTotalOut() {
+    return inflater.getTotalOut();
+  }
+
+  public byte[] getAvailIn() {
+    if(inflater.avail_in<=0)
+      return null;
+    byte[] tmp = new byte[inflater.avail_in];
+    System.arraycopy(inflater.next_in, inflater.next_in_index,
+                     tmp, 0, inflater.avail_in);
+    return tmp;
+  }
+
+  public void readHeader() throws IOException {
+
+    byte[] empty = "".getBytes();
+    inflater.setInput(empty, 0, 0, false);
+    inflater.setOutput(empty, 0, 0);
+
+    int err = inflater.inflate(JZlib.Z_NO_FLUSH);
+    if(!inflater.istate.inParsingHeader()){
+      return;
+    }
+
+    byte[] b1 = new byte[1];
+    do{
+      int i = in.read(b1);
+      if(i<=0)
+        throw new IOException("no input");
+      inflater.setInput(b1);
+      err = inflater.inflate(JZlib.Z_NO_FLUSH);
+      if(err!=0/*Z_OK*/)
+        throw new IOException(inflater.msg);
+    }
+    while(inflater.istate.inParsingHeader());
+  }
+
+  public Inflater getInflater(){
+    return inflater;
+  }
+}
\ No newline at end of file
--- a/src/com/jcraft/jzlib/JZlib.java	Fri Aug 01 11:57:17 2014 -0700
+++ b/src/com/jcraft/jzlib/JZlib.java	Fri Aug 01 13:34:58 2014 -0700
@@ -1,6 +1,6 @@
 /* -*-mode:java; c-basic-offset:2; -*- */
 /*
-Copyright (c) 2000,2001,2002,2003 ymnk, JCraft,Inc. All rights reserved.
+Copyright (c) 2011 ymnk, JCraft,Inc. All rights reserved.
 
 Redistribution and use in source and binary forms, with or without
 modification, are permitted provided that the following conditions are met:
@@ -8,8 +8,8 @@
   1. Redistributions of source code must retain the above copyright notice,
      this list of conditions and the following disclaimer.
 
-  2. Redistributions in binary form must reproduce the above copyright
-     notice, this list of conditions and the following disclaimer in
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
      the documentation and/or other materials provided with the distribution.
 
   3. The names of the authors may not be used to endorse or promote products
@@ -34,34 +34,59 @@
 
 package com.jcraft.jzlib;
 
-final public class JZlib {
-    private static final String version = "1.0.2";
-    public static String version() {return version;}
+final public class JZlib{
+  private static final String version="1.1.0";
+  public static String version(){return version;}
+
+  static final public int MAX_WBITS=15;        // 32K LZ77 window
+  static final public int DEF_WBITS=MAX_WBITS;
+
+  public enum WrapperType {
+    NONE, ZLIB, GZIP, ANY
+  }
 
-    // compression levels
-    static final public int Z_NO_COMPRESSION = 0;
-    static final public int Z_BEST_SPEED = 1;
-    static final public int Z_BEST_COMPRESSION = 9;
-    static final public int Z_DEFAULT_COMPRESSION = (-1);
+  public static final WrapperType W_NONE = WrapperType.NONE;
+  public static final WrapperType W_ZLIB = WrapperType.ZLIB;
+  public static final WrapperType W_GZIP = WrapperType.GZIP;
+  public static final WrapperType W_ANY = WrapperType.ANY;
 
-    // compression strategy
-    static final public int Z_FILTERED = 1;
-    static final public int Z_HUFFMAN_ONLY = 2;
-    static final public int Z_DEFAULT_STRATEGY = 0;
+  // compression levels
+  static final public int Z_NO_COMPRESSION=0;
+  static final public int Z_BEST_SPEED=1;
+  static final public int Z_BEST_COMPRESSION=9;
+  static final public int Z_DEFAULT_COMPRESSION=(-1);
+
+  // compression strategy
+  static final public int Z_FILTERED=1;
+  static final public int Z_HUFFMAN_ONLY=2;
+  static final public int Z_DEFAULT_STRATEGY=0;
 
-    static final public int Z_NO_FLUSH = 0;
-    static final public int Z_PARTIAL_FLUSH = 1;
-    static final public int Z_SYNC_FLUSH = 2;
-    static final public int Z_FULL_FLUSH = 3;
-    static final public int Z_FINISH = 4;
+  static final public int Z_NO_FLUSH=0;
+  static final public int Z_PARTIAL_FLUSH=1;
+  static final public int Z_SYNC_FLUSH=2;
+  static final public int Z_FULL_FLUSH=3;
+  static final public int Z_FINISH=4;
 
-    static final public int Z_OK = 0;
-    static final public int Z_STREAM_END = 1;
-    static final public int Z_NEED_DICT = 2;
-    static final public int Z_ERRNO = -1;
-    static final public int Z_STREAM_ERROR = -2;
-    static final public int Z_DATA_ERROR = -3;
-    static final public int Z_MEM_ERROR = -4;
-    static final public int Z_BUF_ERROR = -5;
-    static final public int Z_VERSION_ERROR = -6;
+  static final public int Z_OK=0;
+  static final public int Z_STREAM_END=1;
+  static final public int Z_NEED_DICT=2;
+  static final public int Z_ERRNO=-1;
+  static final public int Z_STREAM_ERROR=-2;
+  static final public int Z_DATA_ERROR=-3;
+  static final public int Z_MEM_ERROR=-4;
+  static final public int Z_BUF_ERROR=-5;
+  static final public int Z_VERSION_ERROR=-6;
+
+  // The three kinds of block type
+  static final public byte Z_BINARY = 0;
+  static final public byte Z_ASCII = 1;
+  static final public byte Z_UNKNOWN = 2;
+
+  public static long adler32_combine(long adler1, long adler2, long len2){
+    return Adler32.combine(adler1, adler2, len2);
+  }
+
+  public static long crc32_combine(long crc1, long crc2, long len2){
+    return CRC32.combine(crc1, crc2, len2);
+  }
 }
--- a/src/com/jcraft/jzlib/StaticTree.java	Fri Aug 01 11:57:17 2014 -0700
+++ b/src/com/jcraft/jzlib/StaticTree.java	Fri Aug 01 13:34:58 2014 -0700
@@ -8,8 +8,8 @@
   1. Redistributions of source code must retain the above copyright notice,
      this list of conditions and the following disclaimer.
 
-  2. Redistributions in binary form must reproduce the above copyright
-     notice, this list of conditions and the following disclaimer in
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
      the documentation and/or other materials provided with the distribution.
 
   3. The names of the authors may not be used to endorse or promote products
@@ -34,116 +34,115 @@
 
 package com.jcraft.jzlib;
 
-final class StaticTree {
-    static final private int MAX_BITS = 15;
+final class StaticTree{
+  static final private int MAX_BITS=15;
 
-    static final private int BL_CODES = 19;
-    static final private int D_CODES = 30;
-    static final private int LITERALS = 256;
-    static final private int LENGTH_CODES = 29;
-    static final private int L_CODES = (LITERALS + 1 + LENGTH_CODES);
+  static final private int BL_CODES=19;
+  static final private int D_CODES=30;
+  static final private int LITERALS=256;
+  static final private int LENGTH_CODES=29;
+  static final private int L_CODES=(LITERALS+1+LENGTH_CODES);
 
-    // Bit length codes must not exceed MAX_BL_BITS bits
-    static final int MAX_BL_BITS = 7;
+  // Bit length codes must not exceed MAX_BL_BITS bits
+  static final int MAX_BL_BITS=7; 
 
-    static final short[] static_ltree = {
-        12,  8, 140,  8,  76,  8, 204,  8,  44,  8,
-        172,  8, 108,  8, 236,  8,  28,  8, 156,  8,
-        92,  8, 220,  8,  60,  8, 188,  8, 124,  8,
-        252,  8,   2,  8, 130,  8,  66,  8, 194,  8,
-        34,  8, 162,  8,  98,  8, 226,  8,  18,  8,
-        146,  8,  82,  8, 210,  8,  50,  8, 178,  8,
-        114,  8, 242,  8,  10,  8, 138,  8,  74,  8,
-        202,  8,  42,  8, 170,  8, 106,  8, 234,  8,
-        26,  8, 154,  8,  90,  8, 218,  8,  58,  8,
-        186,  8, 122,  8, 250,  8,   6,  8, 134,  8,
-        70,  8, 198,  8,  38,  8, 166,  8, 102,  8,
-        230,  8,  22,  8, 150,  8,  86,  8, 214,  8,
-        54,  8, 182,  8, 118,  8, 246,  8,  14,  8,
-        142,  8,  78,  8, 206,  8,  46,  8, 174,  8,
-        110,  8, 238,  8,  30,  8, 158,  8,  94,  8,
-        222,  8,  62,  8, 190,  8, 126,  8, 254,  8,
-        1,  8, 129,  8,  65,  8, 193,  8,  33,  8,
-        161,  8,  97,  8, 225,  8,  17,  8, 145,  8,
-        81,  8, 209,  8,  49,  8, 177,  8, 113,  8,
-        241,  8,   9,  8, 137,  8,  73,  8, 201,  8,
-        41,  8, 169,  8, 105,  8, 233,  8,  25,  8,
-        153,  8,  89,  8, 217,  8,  57,  8, 185,  8,
-        121,  8, 249,  8,   5,  8, 133,  8,  69,  8,
-        197,  8,  37,  8, 165,  8, 101,  8, 229,  8,
-        21,  8, 149,  8,  85,  8, 213,  8,  53,  8,
-        181,  8, 117,  8, 245,  8,  13,  8, 141,  8,
-        77,  8, 205,  8,  45,  8, 173,  8, 109,  8,
-        237,  8,  29,  8, 157,  8,  93,  8, 221,  8,
-        61,  8, 189,  8, 125,  8, 253,  8,  19,  9,
-        275,  9, 147,  9, 403,  9,  83,  9, 339,  9,
-        211,  9, 467,  9,  51,  9, 307,  9, 179,  9,
-        435,  9, 115,  9, 371,  9, 243,  9, 499,  9,
-        11,  9, 267,  9, 139,  9, 395,  9,  75,  9,
-        331,  9, 203,  9, 459,  9,  43,  9, 299,  9,
-        171,  9, 427,  9, 107,  9, 363,  9, 235,  9,
-        491,  9,  27,  9, 283,  9, 155,  9, 411,  9,
-        91,  9, 347,  9, 219,  9, 475,  9,  59,  9,
-        315,  9, 187,  9, 443,  9, 123,  9, 379,  9,
-        251,  9, 507,  9,   7,  9, 263,  9, 135,  9,
-        391,  9,  71,  9, 327,  9, 199,  9, 455,  9,
-        39,  9, 295,  9, 167,  9, 423,  9, 103,  9,
-        359,  9, 231,  9, 487,  9,  23,  9, 279,  9,
-        151,  9, 407,  9,  87,  9, 343,  9, 215,  9,
-        471,  9,  55,  9, 311,  9, 183,  9, 439,  9,
-        119,  9, 375,  9, 247,  9, 503,  9,  15,  9,
-        271,  9, 143,  9, 399,  9,  79,  9, 335,  9,
-        207,  9, 463,  9,  47,  9, 303,  9, 175,  9,
-        431,  9, 111,  9, 367,  9, 239,  9, 495,  9,
-        31,  9, 287,  9, 159,  9, 415,  9,  95,  9,
-        351,  9, 223,  9, 479,  9,  63,  9, 319,  9,
-        191,  9, 447,  9, 127,  9, 383,  9, 255,  9,
-        511,  9,   0,  7,  64,  7,  32,  7,  96,  7,
-        16,  7,  80,  7,  48,  7, 112,  7,   8,  7,
-        72,  7,  40,  7, 104,  7,  24,  7,  88,  7,
-        56,  7, 120,  7,   4,  7,  68,  7,  36,  7,
-        100,  7,  20,  7,  84,  7,  52,  7, 116,  7,
-        3,  8, 131,  8,  67,  8, 195,  8,  35,  8,
-        163,  8,  99,  8, 227,  8
-    };
+  static final short[] static_ltree = {
+    12,  8, 140,  8,  76,  8, 204,  8,  44,  8,
+    172,  8, 108,  8, 236,  8,  28,  8, 156,  8,
+    92,  8, 220,  8,  60,  8, 188,  8, 124,  8,
+    252,  8,   2,  8, 130,  8,  66,  8, 194,  8,
+    34,  8, 162,  8,  98,  8, 226,  8,  18,  8,
+    146,  8,  82,  8, 210,  8,  50,  8, 178,  8,
+    114,  8, 242,  8,  10,  8, 138,  8,  74,  8,
+    202,  8,  42,  8, 170,  8, 106,  8, 234,  8,
+    26,  8, 154,  8,  90,  8, 218,  8,  58,  8,
+    186,  8, 122,  8, 250,  8,   6,  8, 134,  8,
+    70,  8, 198,  8,  38,  8, 166,  8, 102,  8,
+    230,  8,  22,  8, 150,  8,  86,  8, 214,  8,
+    54,  8, 182,  8, 118,  8, 246,  8,  14,  8,
+    142,  8,  78,  8, 206,  8,  46,  8, 174,  8,
+    110,  8, 238,  8,  30,  8, 158,  8,  94,  8,
+    222,  8,  62,  8, 190,  8, 126,  8, 254,  8,
+    1,  8, 129,  8,  65,  8, 193,  8,  33,  8,
+    161,  8,  97,  8, 225,  8,  17,  8, 145,  8,
+    81,  8, 209,  8,  49,  8, 177,  8, 113,  8,
+    241,  8,   9,  8, 137,  8,  73,  8, 201,  8,
+    41,  8, 169,  8, 105,  8, 233,  8,  25,  8,
+    153,  8,  89,  8, 217,  8,  57,  8, 185,  8,
+    121,  8, 249,  8,   5,  8, 133,  8,  69,  8,
+    197,  8,  37,  8, 165,  8, 101,  8, 229,  8,
+    21,  8, 149,  8,  85,  8, 213,  8,  53,  8,
+    181,  8, 117,  8, 245,  8,  13,  8, 141,  8,
+    77,  8, 205,  8,  45,  8, 173,  8, 109,  8,
+    237,  8,  29,  8, 157,  8,  93,  8, 221,  8,
+    61,  8, 189,  8, 125,  8, 253,  8,  19,  9,
+    275,  9, 147,  9, 403,  9,  83,  9, 339,  9,
+    211,  9, 467,  9,  51,  9, 307,  9, 179,  9,
+    435,  9, 115,  9, 371,  9, 243,  9, 499,  9,
+    11,  9, 267,  9, 139,  9, 395,  9,  75,  9,
+    331,  9, 203,  9, 459,  9,  43,  9, 299,  9,
+    171,  9, 427,  9, 107,  9, 363,  9, 235,  9,
+    491,  9,  27,  9, 283,  9, 155,  9, 411,  9,
+    91,  9, 347,  9, 219,  9, 475,  9,  59,  9,
+    315,  9, 187,  9, 443,  9, 123,  9, 379,  9,
+    251,  9, 507,  9,   7,  9, 263,  9, 135,  9,
+    391,  9,  71,  9, 327,  9, 199,  9, 455,  9,
+    39,  9, 295,  9, 167,  9, 423,  9, 103,  9,
+    359,  9, 231,  9, 487,  9,  23,  9, 279,  9,
+    151,  9, 407,  9,  87,  9, 343,  9, 215,  9,
+    471,  9,  55,  9, 311,  9, 183,  9, 439,  9,
+    119,  9, 375,  9, 247,  9, 503,  9,  15,  9,
+    271,  9, 143,  9, 399,  9,  79,  9, 335,  9,
+    207,  9, 463,  9,  47,  9, 303,  9, 175,  9,
+    431,  9, 111,  9, 367,  9, 239,  9, 495,  9,
+    31,  9, 287,  9, 159,  9, 415,  9,  95,  9,
+    351,  9, 223,  9, 479,  9,  63,  9, 319,  9,
+    191,  9, 447,  9, 127,  9, 383,  9, 255,  9,
+    511,  9,   0,  7,  64,  7,  32,  7,  96,  7,
+    16,  7,  80,  7,  48,  7, 112,  7,   8,  7,
+    72,  7,  40,  7, 104,  7,  24,  7,  88,  7,
+    56,  7, 120,  7,   4,  7,  68,  7,  36,  7,
+    100,  7,  20,  7,  84,  7,  52,  7, 116,  7,
+    3,  8, 131,  8,  67,  8, 195,  8,  35,  8,
+    163,  8,  99,  8, 227,  8
+  };
 
-    static final short[] static_dtree = {
-        0, 5, 16, 5,  8, 5, 24, 5,  4, 5,
-        20, 5, 12, 5, 28, 5,  2, 5, 18, 5,
-        10, 5, 26, 5,  6, 5, 22, 5, 14, 5,
-        30, 5,  1, 5, 17, 5,  9, 5, 25, 5,
-        5, 5, 21, 5, 13, 5, 29, 5,  3, 5,
-        19, 5, 11, 5, 27, 5,  7, 5, 23, 5
-    };
+  static final short[] static_dtree = {
+    0, 5, 16, 5,  8, 5, 24, 5,  4, 5,
+    20, 5, 12, 5, 28, 5,  2, 5, 18, 5,
+    10, 5, 26, 5,  6, 5, 22, 5, 14, 5,
+    30, 5,  1, 5, 17, 5,  9, 5, 25, 5,
+    5, 5, 21, 5, 13, 5, 29, 5,  3, 5,
+    19, 5, 11, 5, 27, 5,  7, 5, 23, 5
+  };
 
-    static StaticTree static_l_desc =
-        new StaticTree(static_ltree, Tree.extra_lbits,
-                       LITERALS + 1, L_CODES, MAX_BITS);
+  static StaticTree static_l_desc =
+    new StaticTree(static_ltree, Tree.extra_lbits,
+		   LITERALS+1, L_CODES, MAX_BITS);
 
-    static StaticTree static_d_desc =
-        new StaticTree(static_dtree, Tree.extra_dbits,
-                       0,  D_CODES, MAX_BITS);
+  static StaticTree static_d_desc =
+    new StaticTree(static_dtree, Tree.extra_dbits,
+		   0,  D_CODES, MAX_BITS);
 
-    static StaticTree static_bl_desc =
-        new StaticTree(null, Tree.extra_blbits,
-                       0, BL_CODES, MAX_BL_BITS);
+  static StaticTree static_bl_desc =
+    new StaticTree(null, Tree.extra_blbits,
+		   0, BL_CODES, MAX_BL_BITS);
 
-    short[] static_tree;     // static tree or null
-    int[] extra_bits;        // extra bits for each code or null
-    int extra_base;          // base index for extra_bits
-    int elems;               // max number of elements in the tree
-    int max_length;          // max bit length for the codes
+  short[] static_tree;     // static tree or null
+  int[] extra_bits;        // extra bits for each code or null
+  int extra_base;          // base index for extra_bits
+  int elems;               // max number of elements in the tree
+  int max_length;          // max bit length for the codes
 
-    StaticTree(short[] static_tree,
-               int[] extra_bits,
-               int extra_base,
-               int elems,
-               int max_length
-              ) {
-        this.static_tree = static_tree;
-        this.extra_bits = extra_bits;
-        this.extra_base = extra_base;
-        this.elems = elems;
-        this.max_length = max_length;
-    }
+  private StaticTree(short[] static_tree,
+                     int[] extra_bits,
+                     int extra_base,
+                     int elems,
+                     int max_length){
+    this.static_tree=static_tree;
+    this.extra_bits=extra_bits;
+    this.extra_base=extra_base;
+    this.elems=elems;
+    this.max_length=max_length;
+  }
 }
--- a/src/com/jcraft/jzlib/Tree.java	Fri Aug 01 11:57:17 2014 -0700
+++ b/src/com/jcraft/jzlib/Tree.java	Fri Aug 01 13:34:58 2014 -0700
@@ -8,8 +8,8 @@
   1. Redistributions of source code must retain the above copyright notice,
      this list of conditions and the following disclaimer.
 
-  2. Redistributions in binary form must reproduce the above copyright
-     notice, this list of conditions and the following disclaimer in
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
      the documentation and/or other materials provided with the distribution.
 
   3. The names of the authors may not be used to endorse or promote products
@@ -34,341 +34,334 @@
 
 package com.jcraft.jzlib;
 
-final class Tree {
-    static final private int MAX_BITS = 15;
-    static final private int BL_CODES = 19;
-    static final private int D_CODES = 30;
-    static final private int LITERALS = 256;
-    static final private int LENGTH_CODES = 29;
-    static final private int L_CODES = (LITERALS + 1 + LENGTH_CODES);
-    static final private int HEAP_SIZE = (2 * L_CODES + 1);
+final class Tree{
+  static final private int MAX_BITS=15;
+  static final private int BL_CODES=19;
+  static final private int D_CODES=30;
+  static final private int LITERALS=256;
+  static final private int LENGTH_CODES=29;
+  static final private int L_CODES=(LITERALS+1+LENGTH_CODES);
+  static final private int HEAP_SIZE=(2*L_CODES+1);
 
-    // Bit length codes must not exceed MAX_BL_BITS bits
-    static final int MAX_BL_BITS = 7;
+  // Bit length codes must not exceed MAX_BL_BITS bits
+  static final int MAX_BL_BITS=7; 
 
-    // end of block literal code
-    static final int END_BLOCK = 256;
+  // end of block literal code
+  static final int END_BLOCK=256; 
 
-    // repeat previous bit length 3-6 times (2 bits of repeat count)
-    static final int REP_3_6 = 16;
+  // repeat previous bit length 3-6 times (2 bits of repeat count)
+  static final int REP_3_6=16; 
 
-    // repeat a zero length 3-10 times  (3 bits of repeat count)
-    static final int REPZ_3_10 = 17;
+  // repeat a zero length 3-10 times  (3 bits of repeat count)
+  static final int REPZ_3_10=17; 
 
-    // repeat a zero length 11-138 times  (7 bits of repeat count)
-    static final int REPZ_11_138 = 18;
+  // repeat a zero length 11-138 times  (7 bits of repeat count)
+  static final int REPZ_11_138=18; 
 
-    // extra bits for each length code
-    static final int[] extra_lbits = {
-        0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 2, 2, 2, 2, 3, 3, 3, 3, 4, 4, 4, 4, 5, 5, 5, 5, 0
-    };
+  // extra bits for each length code
+  static final int[] extra_lbits={
+    0,0,0,0,0,0,0,0,1,1,1,1,2,2,2,2,3,3,3,3,4,4,4,4,5,5,5,5,0
+  };
 
-    // extra bits for each distance code
-    static final int[] extra_dbits = {
-        0, 0, 0, 0, 1, 1, 2, 2, 3, 3, 4, 4, 5, 5, 6, 6, 7, 7, 8, 8, 9, 9, 10, 10, 11, 11, 12, 12, 13, 13
-    };
+  // extra bits for each distance code
+  static final int[] extra_dbits={
+    0,0,0,0,1,1,2,2,3,3,4,4,5,5,6,6,7,7,8,8,9,9,10,10,11,11,12,12,13,13
+  };
 
-    // extra bits for each bit length code
-    static final int[] extra_blbits = {
-        0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 2, 3, 7
-    };
+  // extra bits for each bit length code
+  static final int[] extra_blbits={
+    0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,2,3,7
+  };
 
-    static final byte[] bl_order = {
-        16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15
-    };
+  static final byte[] bl_order={
+    16,17,18,0,8,7,9,6,10,5,11,4,12,3,13,2,14,1,15};
 
 
-    // The lengths of the bit length codes are sent in order of decreasing
-    // probability, to avoid transmitting the lengths for unused bit
-    // length codes.
+  // The lengths of the bit length codes are sent in order of decreasing
+  // probability, to avoid transmitting the lengths for unused bit
+  // length codes.
+
+  static final int Buf_size=8*2;
+
+  // see definition of array dist_code below
+  static final int DIST_CODE_LEN=512;
+
+  static final byte[] _dist_code = {
+    0,  1,  2,  3,  4,  4,  5,  5,  6,  6,  6,  6,  7,  7,  7,  7,  8,  8,  8,  8,
+    8,  8,  8,  8,  9,  9,  9,  9,  9,  9,  9,  9, 10, 10, 10, 10, 10, 10, 10, 10,
+    10, 10, 10, 10, 10, 10, 10, 10, 11, 11, 11, 11, 11, 11, 11, 11, 11, 11, 11, 11,
+    11, 11, 11, 11, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12,
+    12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 13, 13, 13, 13,
+    13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13,
+    13, 13, 13, 13, 13, 13, 13, 13, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14,
+    14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14,
+    14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14,
+    14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 15, 15, 15, 15, 15, 15, 15, 15,
+    15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15,
+    15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15,
+    15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15,  0,  0, 16, 17,
+    18, 18, 19, 19, 20, 20, 20, 20, 21, 21, 21, 21, 22, 22, 22, 22, 22, 22, 22, 22,
+    23, 23, 23, 23, 23, 23, 23, 23, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24,
+    24, 24, 24, 24, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25,
+    26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
+    26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 27, 27, 27, 27, 27, 27, 27, 27,
+    27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27,
+    27, 27, 27, 27, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28,
+    28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28,
+    28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28,
+    28, 28, 28, 28, 28, 28, 28, 28, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29,
+    29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29,
+    29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29,
+    29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29
+  };
 
-    static final int Buf_size = 8 * 2;
+  static final byte[] _length_code={
+    0,  1,  2,  3,  4,  5,  6,  7,  8,  8,  9,  9, 10, 10, 11, 11, 12, 12, 12, 12,
+    13, 13, 13, 13, 14, 14, 14, 14, 15, 15, 15, 15, 16, 16, 16, 16, 16, 16, 16, 16,
+    17, 17, 17, 17, 17, 17, 17, 17, 18, 18, 18, 18, 18, 18, 18, 18, 19, 19, 19, 19,
+    19, 19, 19, 19, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
+    21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 22, 22, 22, 22,
+    22, 22, 22, 22, 22, 22, 22, 22, 22, 22, 22, 22, 23, 23, 23, 23, 23, 23, 23, 23,
+    23, 23, 23, 23, 23, 23, 23, 23, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24,
+    24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24,
+    25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25,
+    25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 26, 26, 26, 26, 26, 26, 26, 26,
+    26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
+    26, 26, 26, 26, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27,
+    27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 28
+  };
 
-    // see definition of array dist_code below
-    static final int DIST_CODE_LEN = 512;
+  static final int[] base_length = {
+    0, 1, 2, 3, 4, 5, 6, 7, 8, 10, 12, 14, 16, 20, 24, 28, 32, 40, 48, 56,
+    64, 80, 96, 112, 128, 160, 192, 224, 0
+  };
+
+  static final int[] base_dist = {
+       0,   1,      2,     3,     4,    6,     8,    12,    16,     24,
+      32,  48,     64,    96,   128,  192,   256,   384,   512,    768,
+    1024, 1536,  2048,  3072,  4096,  6144,  8192, 12288, 16384, 24576
+  };
+
+  // Mapping from a distance to a distance code. dist is the distance - 1 and
+  // must not have side effects. _dist_code[256] and _dist_code[257] are never
+  // used.
+  static int d_code(int dist){
+    return ((dist) < 256 ? _dist_code[dist] : _dist_code[256+((dist)>>>7)]);
+  }
+
+  short[] dyn_tree;      // the dynamic tree
+  int     max_code;      // largest code with non zero frequency
+  StaticTree stat_desc;  // the corresponding static tree
 
-    static final byte[] _dist_code = {
-        0,  1,  2,  3,  4,  4,  5,  5,  6,  6,  6,  6,  7,  7,  7,  7,  8,  8,  8,  8,
-        8,  8,  8,  8,  9,  9,  9,  9,  9,  9,  9,  9, 10, 10, 10, 10, 10, 10, 10, 10,
-        10, 10, 10, 10, 10, 10, 10, 10, 11, 11, 11, 11, 11, 11, 11, 11, 11, 11, 11, 11,
-        11, 11, 11, 11, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12,
-        12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 13, 13, 13, 13,
-        13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13,
-        13, 13, 13, 13, 13, 13, 13, 13, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14,
-        14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14,
-        14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14,
-        14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14, 15, 15, 15, 15, 15, 15, 15, 15,
-        15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15,
-        15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15,
-        15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15,  0,  0, 16, 17,
-        18, 18, 19, 19, 20, 20, 20, 20, 21, 21, 21, 21, 22, 22, 22, 22, 22, 22, 22, 22,
-        23, 23, 23, 23, 23, 23, 23, 23, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24,
-        24, 24, 24, 24, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25,
-        26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
-        26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 27, 27, 27, 27, 27, 27, 27, 27,
-        27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27,
-        27, 27, 27, 27, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28,
-        28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28,
-        28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28,
-        28, 28, 28, 28, 28, 28, 28, 28, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29,
-        29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29,
-        29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29,
-        29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29, 29
-    };
+  // Compute the optimal bit lengths for a tree and update the total bit length
+  // for the current block.
+  // IN assertion: the fields freq and dad are set, heap[heap_max] and
+  //    above are the tree nodes sorted by increasing frequency.
+  // OUT assertions: the field len is set to the optimal bit length, the
+  //     array bl_count contains the frequencies for each bit length.
+  //     The length opt_len is updated; static_len is also updated if stree is
+  //     not null.
+  void gen_bitlen(Deflate s){
+    short[] tree = dyn_tree;
+    short[] stree = stat_desc.static_tree;
+    int[] extra = stat_desc.extra_bits;
+    int base = stat_desc.extra_base;
+    int max_length = stat_desc.max_length;
+    int h;              // heap index
+    int n, m;           // iterate over the tree elements
+    int bits;           // bit length
+    int xbits;          // extra bits
+    short f;            // frequency
+    int overflow = 0;   // number of elements with bit length too large
+
+    for (bits = 0; bits <= MAX_BITS; bits++) s.bl_count[bits] = 0;
+
+    // In a first pass, compute the optimal bit lengths (which may
+    // overflow in the case of the bit length tree).
+    tree[s.heap[s.heap_max]*2+1] = 0; // root of the heap
+
+    for(h=s.heap_max+1; h<HEAP_SIZE; h++){
+      n = s.heap[h];
+      bits = tree[tree[n*2+1]*2+1] + 1;
+      if (bits > max_length){ bits = max_length; overflow++; }
+      tree[n*2+1] = (short)bits;
+      // We overwrite tree[n*2+1] which is no longer needed
+
+      if (n > max_code) continue;  // not a leaf node
+
+      s.bl_count[bits]++;
+      xbits = 0;
+      if (n >= base) xbits = extra[n-base];
+      f = tree[n*2];
+      s.opt_len += f * (bits + xbits);
+      if (stree!=null) s.static_len += f * (stree[n*2+1] + xbits);
+    }
+    if (overflow == 0) return;
 
-    static final byte[] _length_code = {
-        0,  1,  2,  3,  4,  5,  6,  7,  8,  8,  9,  9, 10, 10, 11, 11, 12, 12, 12, 12,
-        13, 13, 13, 13, 14, 14, 14, 14, 15, 15, 15, 15, 16, 16, 16, 16, 16, 16, 16, 16,
-        17, 17, 17, 17, 17, 17, 17, 17, 18, 18, 18, 18, 18, 18, 18, 18, 19, 19, 19, 19,
-        19, 19, 19, 19, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20, 20,
-        21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 21, 22, 22, 22, 22,
-        22, 22, 22, 22, 22, 22, 22, 22, 22, 22, 22, 22, 23, 23, 23, 23, 23, 23, 23, 23,
-        23, 23, 23, 23, 23, 23, 23, 23, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24,
-        24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24, 24,
-        25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25,
-        25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 25, 26, 26, 26, 26, 26, 26, 26, 26,
-        26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
-        26, 26, 26, 26, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27,
-        27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 27, 28
-    };
+    // This happens for example on obj2 and pic of the Calgary corpus
+    // Find the first bit length which could increase:
+    do {
+      bits = max_length-1;
+      while(s.bl_count[bits]==0) bits--;
+      s.bl_count[bits]--;      // move one leaf down the tree
+      s.bl_count[bits+1]+=2;   // move one overflow item as its brother
+      s.bl_count[max_length]--;
+      // The brother of the overflow item also moves one step up,
+      // but this does not affect bl_count[max_length]
+      overflow -= 2;
+    }
+    while (overflow > 0);
+
+    for (bits = max_length; bits != 0; bits--) {
+      n = s.bl_count[bits];
+      while (n != 0) {
+	m = s.heap[--h];
+	if (m > max_code) continue;
+	if (tree[m*2+1] != bits) {
+	  s.opt_len += ((long)bits - (long)tree[m*2+1])*(long)tree[m*2];
+	  tree[m*2+1] = (short)bits;
+	}
+	n--;
+      }
+    }
+  }
 
-    static final int[] base_length = {
-        0, 1, 2, 3, 4, 5, 6, 7, 8, 10, 12, 14, 16, 20, 24, 28, 32, 40, 48, 56,
-        64, 80, 96, 112, 128, 160, 192, 224, 0
-    };
+  // Construct one Huffman tree and assigns the code bit strings and lengths.
+  // Update the total bit length for the current block.
+  // IN assertion: the field freq is set for all tree elements.
+  // OUT assertions: the fields len and code are set to the optimal bit length
+  //     and corresponding code. The length opt_len is updated; static_len is
+  //     also updated if stree is not null. The field max_code is set.
+  void build_tree(Deflate s){
+    short[] tree=dyn_tree;
+    short[] stree=stat_desc.static_tree;
+    int elems=stat_desc.elems;
+    int n, m;          // iterate over heap elements
+    int max_code=-1;   // largest code with non zero frequency
+    int node;          // new node being created
 
-    static final int[] base_dist = {
-        0,   1,      2,     3,     4,    6,     8,    12,    16,     24,
-        32,  48,     64,    96,   128,  192,   256,   384,   512,    768,
-        1024, 1536,  2048,  3072,  4096,  6144,  8192, 12288, 16384, 24576
-    };
+    // Construct the initial heap, with least frequent element in
+    // heap[1]. The sons of heap[n] are heap[2*n] and heap[2*n+1].
+    // heap[0] is not used.
+    s.heap_len = 0;
+    s.heap_max = HEAP_SIZE;
 
-    // Mapping from a distance to a distance code. dist is the distance - 1 and
-    // must not have side effects. _dist_code[256] and _dist_code[257] are never
-    // used.
-    static int d_code(int dist) {
-        return ((dist) < 256 ? _dist_code[dist] : _dist_code[256 + ((dist) >>> 7)]);
+    for(n=0; n<elems; n++) {
+      if(tree[n*2] != 0) {
+	s.heap[++s.heap_len] = max_code = n;
+	s.depth[n] = 0;
+      }
+      else{
+	tree[n*2+1] = 0;
+      }
     }
 
-    short[] dyn_tree;      // the dynamic tree
-    int     max_code;      // largest code with non zero frequency
-    StaticTree stat_desc;  // the corresponding static tree
+    // The pkzip format requires that at least one distance code exists,
+    // and that at least one bit should be sent even if there is only one
+    // possible code. So to avoid special checks later on we force at least
+    // two codes of non zero frequency.
+    while (s.heap_len < 2) {
+      node = s.heap[++s.heap_len] = (max_code < 2 ? ++max_code : 0);
+      tree[node*2] = 1;
+      s.depth[node] = 0;
+      s.opt_len--; if (stree!=null) s.static_len -= stree[node*2+1];
+      // node is 0 or 1 so it does not have extra bits
+    }
+    this.max_code = max_code;
+
+    // The elements heap[heap_len/2+1 .. heap_len] are leaves of the tree,
+    // establish sub-heaps of increasing lengths:
+
+    for(n=s.heap_len/2;n>=1; n--)
+      s.pqdownheap(tree, n);
 
-    // Compute the optimal bit lengths for a tree and update the total bit length
-    // for the current block.
-    // IN assertion: the fields freq and dad are set, heap[heap_max] and
-    //    above are the tree nodes sorted by increasing frequency.
-    // OUT assertions: the field len is set to the optimal bit length, the
-    //     array bl_count contains the frequencies for each bit length.
-    //     The length opt_len is updated; static_len is also updated if stree is
-    //     not null.
-    void gen_bitlen(Deflate s) {
-        short[] tree = dyn_tree;
-        short[] stree = stat_desc.static_tree;
-        int[] extra = stat_desc.extra_bits;
-        int base = stat_desc.extra_base;
-        int max_length = stat_desc.max_length;
-        int h;              // heap index
-        int n, m;           // iterate over the tree elements
-        int bits;           // bit length
-        int xbits;          // extra bits
-        short f;            // frequency
-        int overflow = 0;   // number of elements with bit length too large
+    // Construct the Huffman tree by repeatedly combining the least two
+    // frequent nodes.
 
-        for (bits = 0; bits <= MAX_BITS; bits++) s.bl_count[bits] = 0;
-
-        // In a first pass, compute the optimal bit lengths (which may
-        // overflow in the case of the bit length tree).
-        tree[s.heap[s.heap_max] * 2 + 1] = 0; // root of the heap
+    node=elems;                 // next internal node of the tree
+    do{
+      // n = node of least frequency
+      n=s.heap[1];
+      s.heap[1]=s.heap[s.heap_len--];
+      s.pqdownheap(tree, 1);
+      m=s.heap[1];                // m = node of next least frequency
 
-        for (h = s.heap_max + 1; h < HEAP_SIZE; h++) {
-            n = s.heap[h];
-            bits = tree[tree[n * 2 + 1] * 2 + 1] + 1;
+      s.heap[--s.heap_max] = n; // keep the nodes sorted by frequency
+      s.heap[--s.heap_max] = m;
 
-            if (bits > max_length) { bits = max_length; overflow++; }
-
-            tree[n * 2 + 1] = (short)bits;
-
-            // We overwrite tree[n*2+1] which is no longer needed
-            if (n > max_code) continue;  // not a leaf node
+      // Create a new node father of n and m
+      tree[node*2] = (short)(tree[n*2] + tree[m*2]);
+      s.depth[node] = (byte)(Math.max(s.depth[n],s.depth[m])+1);
+      tree[n*2+1] = tree[m*2+1] = (short)node;
 
-            s.bl_count[bits]++;
-            xbits = 0;
+      // and insert the new node in the heap
+      s.heap[1] = node++;
+      s.pqdownheap(tree, 1);
+    }
+    while(s.heap_len>=2);
 
-            if (n >= base) xbits = extra[n - base];
-
-            f = tree[n * 2];
-            s.opt_len += f * (bits + xbits);
+    s.heap[--s.heap_max] = s.heap[1];
 
-            if (stree != null) s.static_len += f * (stree[n * 2 + 1] + xbits);
-        }
-
-        if (overflow == 0) return;
+    // At this point, the fields freq and dad are set. We can now
+    // generate the bit lengths.
 
-        // This happens for example on obj2 and pic of the Calgary corpus
-        // Find the first bit length which could increase:
-        do {
-            bits = max_length - 1;
+    gen_bitlen(s);
 
-            while (s.bl_count[bits] == 0) bits--;
+    // The field len is now set, we can generate the bit codes
+    gen_codes(tree, max_code, s.bl_count, s.next_code);
+  }
 
-            s.bl_count[bits]--;      // move one leaf down the tree
-            s.bl_count[bits + 1] += 2; // move one overflow item as its brother
-            s.bl_count[max_length]--;
-            // The brother of the overflow item also moves one step up,
-            // but this does not affect bl_count[max_length]
-            overflow -= 2;
-        }
-        while (overflow > 0);
-
-        for (bits = max_length; bits != 0; bits--) {
-            n = s.bl_count[bits];
+  // Generate the codes for a given tree and bit counts (which need not be
+  // optimal).
+  // IN assertion: the array bl_count contains the bit length statistics for
+  // the given tree and the field len is set for all tree elements.
+  // OUT assertion: the field code is set for all tree elements of non
+  //     zero code length.
+  private final static void gen_codes(
+                        short[] tree, // the tree to decorate
+                        int max_code, // largest code with non zero frequency
+                        short[] bl_count, // number of codes at each bit length
+                        short[] next_code){
+    short code = 0;            // running code value
+    int bits;                  // bit index
+    int n;                     // code index
 
-            while (n != 0) {
-                m = s.heap[--h];
-
-                if (m > max_code) continue;
-
-                if (tree[m * 2 + 1] != bits) {
-                    s.opt_len += ((long)bits - (long)tree[m * 2 + 1]) * (long)tree[m * 2];
-                    tree[m * 2 + 1] = (short)bits;
-                }
-
-                n--;
-            }
-        }
+    // The distribution counts are first used to generate the code values
+    // without bit reversal.
+    next_code[0]=0;
+    for (bits = 1; bits <= MAX_BITS; bits++) {
+      next_code[bits] = code = (short)((code + bl_count[bits-1]) << 1);
     }
 
-    // Construct one Huffman tree and assigns the code bit strings and lengths.
-    // Update the total bit length for the current block.
-    // IN assertion: the field freq is set for all tree elements.
-    // OUT assertions: the fields len and code are set to the optimal bit length
-    //     and corresponding code. The length opt_len is updated; static_len is
-    //     also updated if stree is not null. The field max_code is set.
-    void build_tree(Deflate s) {
-        short[] tree = dyn_tree;
-        short[] stree = stat_desc.static_tree;
-        int elems = stat_desc.elems;
-        int n, m;          // iterate over heap elements
-        int max_code = -1; // largest code with non zero frequency
-        int node;          // new node being created
-        // Construct the initial heap, with least frequent element in
-        // heap[1]. The sons of heap[n] are heap[2*n] and heap[2*n+1].
-        // heap[0] is not used.
-        s.heap_len = 0;
-        s.heap_max = HEAP_SIZE;
+    // Check that the bit counts in bl_count are consistent. The last code
+    // must be all ones.
+    //Assert (code + bl_count[MAX_BITS]-1 == (1<<MAX_BITS)-1,
+    //        "inconsistent bit counts");
+    //Tracev((stderr,"\ngen_codes: max_code %d ", max_code));
 
-        for (n = 0; n < elems; n++) {
-            if (tree[n * 2] != 0) {
-                s.heap[++s.heap_len] = max_code = n;
-                s.depth[n] = 0;
-            }
-            else {
-                tree[n * 2 + 1] = 0;
-            }
-        }
-
-        // The pkzip format requires that at least one distance code exists,
-        // and that at least one bit should be sent even if there is only one
-        // possible code. So to avoid special checks later on we force at least
-        // two codes of non zero frequency.
-        while (s.heap_len < 2) {
-            node = s.heap[++s.heap_len] = (max_code < 2 ? ++max_code : 0);
-            tree[node * 2] = 1;
-            s.depth[node] = 0;
-            s.opt_len--; if (stree != null) s.static_len -= stree[node * 2 + 1];
-            // node is 0 or 1 so it does not have extra bits
-        }
-
-        this.max_code = max_code;
-
-        // The elements heap[heap_len/2+1 .. heap_len] are leaves of the tree,
-        // establish sub-heaps of increasing lengths:
-        for (n = s.heap_len / 2; n >= 1; n--)
-            s.pqdownheap(tree, n);
-
-        // Construct the Huffman tree by repeatedly combining the least two
-        // frequent nodes.
-        node = elems;               // next internal node of the tree
+    for (n = 0;  n <= max_code; n++) {
+      int len = tree[n*2+1];
+      if (len == 0) continue;
+      // Now reverse the bits
+      tree[n*2] = (short)(bi_reverse(next_code[len]++, len));
+    }
+  }
 
-        do {
-            // n = node of least frequency
-            n = s.heap[1];
-            s.heap[1] = s.heap[s.heap_len--];
-            s.pqdownheap(tree, 1);
-            m = s.heap[1];              // m = node of next least frequency
-            s.heap[--s.heap_max] = n; // keep the nodes sorted by frequency
-            s.heap[--s.heap_max] = m;
-            // Create a new node father of n and m
-            tree[node * 2] = (short)(tree[n * 2] + tree[m * 2]);
-            s.depth[node] = (byte)(Math.max(s.depth[n], s.depth[m]) + 1);
-            tree[n * 2 + 1] = tree[m * 2 + 1] = (short)node;
-            // and insert the new node in the heap
-            s.heap[1] = node++;
-            s.pqdownheap(tree, 1);
-        }
-        while (s.heap_len >= 2);
-
-        s.heap[--s.heap_max] = s.heap[1];
-        // At this point, the fields freq and dad are set. We can now
-        // generate the bit lengths.
-        gen_bitlen(s);
-        // The field len is now set, we can generate the bit codes
-        gen_codes(tree, max_code, s.bl_count);
-    }
-
-    // Generate the codes for a given tree and bit counts (which need not be
-    // optimal).
-    // IN assertion: the array bl_count contains the bit length statistics for
-    // the given tree and the field len is set for all tree elements.
-    // OUT assertion: the field code is set for all tree elements of non
-    //     zero code length.
-    static void gen_codes(short[] tree,  // the tree to decorate
-                          int max_code, // largest code with non zero frequency
-                          short[] bl_count // number of codes at each bit length
-                         ) {
-        short[] next_code = new short[MAX_BITS + 1]; // next code value for each bit length
-        short code = 0;            // running code value
-        int bits;                  // bit index
-        int n;                     // code index
-
-        // The distribution counts are first used to generate the code values
-        // without bit reversal.
-        for (bits = 1; bits <= MAX_BITS; bits++) {
-            next_code[bits] = code = (short)((code + bl_count[bits - 1]) << 1);
-        }
-
-        // Check that the bit counts in bl_count are consistent. The last code
-        // must be all ones.
-        //Assert (code + bl_count[MAX_BITS]-1 == (1<<MAX_BITS)-1,
-        //        "inconsistent bit counts");
-        //Tracev((stderr,"\ngen_codes: max_code %d ", max_code));
-        for (n = 0;  n <= max_code; n++) {
-            int len = tree[n * 2 + 1];
-
-            if (len == 0) continue;
-
-            // Now reverse the bits
-            tree[n * 2] = (short)(bi_reverse(next_code[len]++, len));
-        }
-    }
-
-    // Reverse the first len bits of a code, using straightforward code (a faster
-    // method would use a table)
-    // IN assertion: 1 <= len <= 15
-    static int bi_reverse(int code,  // the value to invert
-                          int len   // its bit length
-                         ) {
-        int res = 0;
-
-        do {
-            res |= code & 1;
-            code >>>= 1;
-            res <<= 1;
-        }
-        while (--len > 0);
-
-        return res >>> 1;
-    }
+  // Reverse the first len bits of a code, using straightforward code (a faster
+  // method would use a table)
+  // IN assertion: 1 <= len <= 15
+  private final static int bi_reverse(
+                        int code, // the value to invert
+			int len   // its bit length
+			){
+    int res = 0;
+    do{
+      res|=code&1;
+      code>>>=1;
+      res<<=1;
+    } 
+    while(--len>0);
+    return res>>>1;
+  }
 }
 
--- a/src/com/jcraft/jzlib/ZInputStream.java	Fri Aug 01 11:57:17 2014 -0700
+++ b/src/com/jcraft/jzlib/ZInputStream.java	Fri Aug 01 13:34:58 2014 -0700
@@ -1,6 +1,6 @@
 /* -*-mode:java; c-basic-offset:2; indent-tabs-mode:nil -*- */
 /*
-Copyright (c) 2001 Lapo Luchini.
+Copyright (c) 2011 ymnk, JCraft,Inc. All rights reserved.
 
 Redistribution and use in source and binary forms, with or without
 modification, are permitted provided that the following conditions are met:
@@ -8,8 +8,8 @@
   1. Redistributions of source code must retain the above copyright notice,
      this list of conditions and the following disclaimer.
 
-  2. Redistributions in binary form must reproduce the above copyright
-     notice, this list of conditions and the following disclaimer in
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
      the documentation and/or other materials provided with the distribution.
 
   3. The names of the authors may not be used to endorse or promote products
@@ -17,8 +17,8 @@
 
 THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES,
 INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
-FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHORS
-OR ANY CONTRIBUTORS TO THIS SOFTWARE BE LIABLE FOR ANY DIRECT, INDIRECT,
+FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JCRAFT,
+INC. OR ANY CONTRIBUTORS TO THIS SOFTWARE BE LIABLE FOR ANY DIRECT, INDIRECT,
 INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
 OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
@@ -26,135 +26,101 @@
 NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
 EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  */
-/*
- * This program is based on zlib-1.1.3, so all credit should go authors
- * Jean-loup Gailly(jloup@gzip.org) and Mark Adler(madler@alumni.caltech.edu)
- * and contributors of zlib.
- */
 
 package com.jcraft.jzlib;
 import java.io.*;
 
+/**
+ * ZInputStream
+ *
+ * @deprecated  use DeflaterOutputStream or InflaterInputStream
+ */
+@Deprecated
 public class ZInputStream extends FilterInputStream {
 
-    protected ZStream z = new ZStream();
-    protected int bufsize = 512;
-    protected int flush = JZlib.Z_NO_FLUSH;
-    protected byte[] buf = new byte[bufsize],
-    buf1 = new byte[1];
-    protected boolean compress;
+  protected int flush=JZlib.Z_NO_FLUSH;
+  protected boolean compress;
+  protected InputStream in=null;
 
-    protected InputStream in = null;
+  protected Deflater deflater;
+  protected InflaterInputStream iis;
 
-    public ZInputStream(InputStream in) {
-        this(in, false);
-    }
-    public ZInputStream(InputStream in, boolean nowrap) {
-        super(in);
-        this.in = in;
-        z.inflateInit(nowrap);
-        compress = false;
-        z.next_in = buf;
-        z.next_in_index = 0;
-        z.avail_in = 0;
-    }
+  public ZInputStream(InputStream in) throws IOException {
+    this(in, false);
+  }
+  public ZInputStream(InputStream in, boolean nowrap) throws IOException {
+    super(in);
+    iis = new InflaterInputStream(in, nowrap);
+    compress=false;
+  }
 
-    public ZInputStream(InputStream in, int level) {
-        super(in);
-        this.in = in;
-        z.deflateInit(level);
-        compress = true;
-        z.next_in = buf;
-        z.next_in_index = 0;
-        z.avail_in = 0;
-    }
-
-    /*public int available() throws IOException {
-      return inf.finished() ? 0 : 1;
-    }*/
-
-    public int read() throws IOException {
-        if (read(buf1, 0, 1) == -1)
-            return (-1);
+  public ZInputStream(InputStream in, int level) throws IOException {
+    super(in);
+    this.in=in;
+    deflater = new Deflater();
+    deflater.init(level);
+    compress=true;
+  }
 
-        return (buf1[0] & 0xFF);
-    }
-
-    private boolean nomoreinput = false;
-
-    public int read(byte[] b, int off, int len) throws IOException {
-        if (len == 0)
-            return (0);
+  private byte[] buf1 = new byte[1];
+  public int read() throws IOException {
+    if(read(buf1, 0, 1)==-1) return -1;
+    return(buf1[0]&0xFF);
+  }
 
-        int err;
-        z.next_out = b;
-        z.next_out_index = off;
-        z.avail_out = len;
-
-        do {
-            if ((z.avail_in == 0) && (!nomoreinput)) { // if buffer is empty and more input is avaiable, refill it
-                z.next_in_index = 0;
-                z.avail_in = in.read(buf, 0, bufsize);  //(bufsize<z.avail_out ? bufsize : z.avail_out));
+  private byte[] buf = new byte[512];
 
-                if (z.avail_in == -1) {
-                    z.avail_in = 0;
-                    nomoreinput = true;
-                }
-            }
-
-            if (compress)
-                err = z.deflate(flush);
-            else
-                err = z.inflate(flush);
-
-            if (nomoreinput && (err == JZlib.Z_BUF_ERROR))
-                return (-1);
-
-            if (err != JZlib.Z_OK && err != JZlib.Z_STREAM_END)
-                throw new ZStreamException((compress ? "de" : "in") + "flating: " + z.msg);
-
-            if ((nomoreinput || err == JZlib.Z_STREAM_END) && (z.avail_out == len))
-                return (-1);
+  public int read(byte[] b, int off, int len) throws IOException {
+    if(compress){
+      deflater.setOutput(b, off, len);
+      while(true){
+        int datalen = in.read(buf, 0, buf.length);
+        if(datalen == -1) return -1;
+        deflater.setInput(buf, 0, datalen, true);
+        int err = deflater.deflate(flush);
+        if(deflater.next_out_index>0)
+          return deflater.next_out_index;
+        if(err == JZlib.Z_STREAM_END)
+          return 0;
+        if(err == JZlib.Z_STREAM_ERROR ||
+           err == JZlib.Z_DATA_ERROR){
+          throw new ZStreamException("deflating: "+deflater.msg);
         }
-        while (z.avail_out == len && err == JZlib.Z_OK);
-
-        //System.err.print("("+(len-z.avail_out)+")");
-        return (len - z.avail_out);
+      }
     }
-
-    public long skip(long n) throws IOException {
-        int len = 512;
+    else{
+      return iis.read(b, off, len); 
+    }
+  }
 
-        if (n < len)
-            len = (int)n;
-
-        byte[] tmp = new byte[len];
-        return ((long)read(tmp));
-    }
+  public long skip(long n) throws IOException {
+    int len=512;
+    if(n<len)
+      len=(int)n;
+    byte[] tmp=new byte[len];
+    return((long)read(tmp));
+  }
 
-    public int getFlushMode() {
-        return (flush);
-    }
+  public int getFlushMode() {
+    return flush;
+  }
 
-    public void setFlushMode(int flush) {
-        this.flush = flush;
-    }
+  public void setFlushMode(int flush) {
+    this.flush=flush;
+  }
 
-    /**
-     * Returns the total number of bytes input so far.
-     */
-    public long getTotalIn() {
-        return z.total_in;
-    }
+  public long getTotalIn() {
+    if(compress) return deflater.total_in;
+    else return iis.getTotalIn();
+  }
 
-    /**
-     * Returns the total number of bytes output so far.
-     */
-    public long getTotalOut() {
-        return z.total_out;
-    }
+  public long getTotalOut() {
+    if(compress) return deflater.total_out;
+    else return iis.getTotalOut();
+  }
 
-    public void close() throws IOException {
-        in.close();
-    }
+  public void close() throws IOException{
+    if(compress) deflater.end();
+    else iis.close();
+  }
 }
--- a/src/com/jcraft/jzlib/ZOutputStream.java	Fri Aug 01 11:57:17 2014 -0700
+++ b/src/com/jcraft/jzlib/ZOutputStream.java	Fri Aug 01 13:34:58 2014 -0700
@@ -1,6 +1,6 @@
 /* -*-mode:java; c-basic-offset:2; indent-tabs-mode:nil -*- */
 /*
-Copyright (c) 2001 Lapo Luchini.
+Copyright (c) 2011 ymnk, JCraft,Inc. All rights reserved.
 
 Redistribution and use in source and binary forms, with or without
 modification, are permitted provided that the following conditions are met:
@@ -8,8 +8,8 @@
   1. Redistributions of source code must retain the above copyright notice,
      this list of conditions and the following disclaimer.
 
-  2. Redistributions in binary form must reproduce the above copyright
-     notice, this list of conditions and the following disclaimer in
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
      the documentation and/or other materials provided with the distribution.
 
   3. The names of the authors may not be used to endorse or promote products
@@ -17,8 +17,8 @@
 
 THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES,
 INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
-FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHORS
-OR ANY CONTRIBUTORS TO THIS SOFTWARE BE LIABLE FOR ANY DIRECT, INDIRECT,
+FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JCRAFT,
+INC. OR ANY CONTRIBUTORS TO THIS SOFTWARE BE LIABLE FOR ANY DIRECT, INDIRECT,
 INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
 OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
@@ -26,143 +26,134 @@
 NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
 EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  */
-/*
- * This program is based on zlib-1.1.3, so all credit should go authors
- * Jean-loup Gailly(jloup@gzip.org) and Mark Adler(madler@alumni.caltech.edu)
- * and contributors of zlib.
- */
 
 package com.jcraft.jzlib;
 import java.io.*;
 
-public class ZOutputStream extends OutputStream {
-
-    protected ZStream z = new ZStream();
-    protected int bufsize = 512;
-    protected int flush = JZlib.Z_NO_FLUSH;
-    protected byte[] buf = new byte[bufsize],
-    buf1 = new byte[1];
-    protected boolean compress;
-
-    protected OutputStream out;
+/**
+ * ZOutputStream
+ *
+ * @deprecated  use DeflaterOutputStream or InflaterInputStream
+ */
+@Deprecated
+public class ZOutputStream extends FilterOutputStream {
 
-    public ZOutputStream(OutputStream out) {
-        super();
-        this.out = out;
-        z.inflateInit();
-        compress = false;
-    }
+  protected int bufsize=512;
+  protected int flush=JZlib.Z_NO_FLUSH;
+  protected byte[] buf=new byte[bufsize];
+  protected boolean compress;
 
-    public ZOutputStream(OutputStream out, int level) {
-        this(out, level, false);
-    }
-    public ZOutputStream(OutputStream out, int level, boolean nowrap) {
-        super();
-        this.out = out;
-        z.deflateInit(level, nowrap);
-        compress = true;
-    }
+  protected OutputStream out;
+  private boolean end=false;
+
+  private DeflaterOutputStream dos;
+  private Inflater inflater;
 
-    public void write(int b) throws IOException {
-        buf1[0] = (byte)b;
-        write(buf1, 0, 1);
-    }
+  public ZOutputStream(OutputStream out) throws IOException {
+    super(out);
+    this.out=out;
+    inflater = new Inflater();
+    inflater.init();
+    compress=false;
+  }
 
-    public void write(byte b[], int off, int len) throws IOException {
-        if (len == 0)
-            return;
-
-        int err;
-        z.next_in = b;
-        z.next_in_index = off;
-        z.avail_in = len;
+  public ZOutputStream(OutputStream out, int level) throws IOException {
+    this(out, level, false);
+  }
 
-        do {
-            z.next_out = buf;
-            z.next_out_index = 0;
-            z.avail_out = bufsize;
-
-            if (compress)
-                err = z.deflate(flush);
-            else
-                err = z.inflate(flush);
+  public ZOutputStream(OutputStream out, int level, boolean nowrap) throws IOException {
+    super(out);
+    this.out=out;
+    Deflater deflater = new Deflater(level, nowrap);
+    dos = new DeflaterOutputStream(out, deflater);
+    compress=true;
+  }
 
-            if (err != JZlib.Z_OK)
-                throw new ZStreamException((compress ? "de" : "in") + "flating: " + z.msg);
-
-            out.write(buf, 0, bufsize - z.avail_out);
-        }
-        while (z.avail_in > 0 || z.avail_out == 0);
-    }
-
-    public int getFlushMode() {
-        return (flush);
-    }
+  private byte[] buf1 = new byte[1];
+  public void write(int b) throws IOException {
+    buf1[0]=(byte)b;
+    write(buf1, 0, 1);
+  }
 
-    public void setFlushMode(int flush) {
-        this.flush = flush;
+  public void write(byte b[], int off, int len) throws IOException {
+    if(len==0) return;
+    if(compress){
+      dos.write(b, off, len);
     }
-
-    public void finish() throws IOException {
-        int err;
-
-        do {
-            z.next_out = buf;
-            z.next_out_index = 0;
-            z.avail_out = bufsize;
-
-            if (compress) { err = z.deflate(JZlib.Z_FINISH);  }
-            else { err = z.inflate(JZlib.Z_FINISH); }
+    else {
+      inflater.setInput(b, off, len, true);
+      int err = JZlib.Z_OK;
+      while(inflater.avail_in>0){
+        inflater.setOutput(buf, 0, buf.length);
+        err = inflater.inflate(flush);
+        if(inflater.next_out_index>0)
+          out.write(buf, 0, inflater.next_out_index);
+        if(err != JZlib.Z_OK)
+          break;
+      }
+      if(err != JZlib.Z_OK)
+        throw new ZStreamException("inflating: "+inflater.msg);
+      return;
+    }
+  }
 
-            if (err != JZlib.Z_STREAM_END && err != JZlib.Z_OK)
-                throw new ZStreamException((compress ? "de" : "in") + "flating: " + z.msg);
-
-            if (bufsize - z.avail_out > 0) {
-                out.write(buf, 0, bufsize - z.avail_out);
-            }
-        }
-        while (z.avail_in > 0 || z.avail_out == 0);
+  public int getFlushMode() {
+    return flush;
+  }
 
-        flush();
-    }
-    public void end() {
-        if (z == null)
-            return;
-
-        if (compress) { z.deflateEnd(); }
-        else { z.inflateEnd(); }
+  public void setFlushMode(int flush) {
+    this.flush=flush;
+  }
 
-        z.free();
-        z = null;
+  public void finish() throws IOException {
+    int err;
+    if(compress){
+      int tmp = flush;
+      int flush = JZlib.Z_FINISH;
+      try{
+        write("".getBytes(), 0, 0);
+      }
+      finally { flush = tmp; }
+    }
+    else{
+      dos.finish();
     }
-    public void close() throws IOException {
-        try {
-            try {finish();}
-            catch (IOException ignored) {}
-        }
-        finally {
-            end();
-            out.close();
-            out = null;
-        }
+    flush();
+  }
+  public synchronized void end() {
+    if(end) return;
+    if(compress){
+      try { dos.finish(); } catch(Exception e){}
+    }
+    else{
+      inflater.end();
     }
+    end=true;
+  }
+  public void close() throws IOException {
+    try{
+      try{finish();}
+      catch (IOException ignored) {}
+    }
+    finally{
+      end();
+      out.close();
+      out=null;
+    }
+  }
 
-    /**
-     * Returns the total number of bytes input so far.
-     */
-    public long getTotalIn() {
-        return z.total_in;
-    }
+  public long getTotalIn() {
+    if(compress) return dos.getTotalIn();
+    else return inflater.total_in;
+  }
 
-    /**
-     * Returns the total number of bytes output so far.
-     */
-    public long getTotalOut() {
-        return z.total_out;
-    }
+  public long getTotalOut() {
+    if(compress) return dos.getTotalOut();
+    else return inflater.total_out;
+  }
 
-    public void flush() throws IOException {
-        out.flush();
-    }
+  public void flush() throws IOException {
+    out.flush();
+  }
 
 }
--- a/src/com/jcraft/jzlib/ZStream.java	Fri Aug 01 11:57:17 2014 -0700
+++ b/src/com/jcraft/jzlib/ZStream.java	Fri Aug 01 13:34:58 2014 -0700
@@ -1,6 +1,6 @@
 /* -*-mode:java; c-basic-offset:2; indent-tabs-mode:nil -*- */
 /*
-Copyright (c) 2000,2001,2002,2003 ymnk, JCraft,Inc. All rights reserved.
+Copyright (c) 2000-2011 ymnk, JCraft,Inc. All rights reserved.
 
 Redistribution and use in source and binary forms, with or without
 modification, are permitted provided that the following conditions are met:
@@ -8,8 +8,8 @@
   1. Redistributions of source code must retain the above copyright notice,
      this list of conditions and the following disclaimer.
 
-  2. Redistributions in binary form must reproduce the above copyright
-     notice, this list of conditions and the following disclaimer in
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
      the documentation and/or other materials provided with the distribution.
 
   3. The names of the authors may not be used to endorse or promote products
@@ -34,189 +34,344 @@
 
 package com.jcraft.jzlib;
 
-final public class ZStream {
+/**
+ * ZStream
+ *
+ * @deprecated  Not for public use in the future.
+ */
+@Deprecated
+public class ZStream{
 
-    static final private int MAX_WBITS = 15;      // 32K LZ77 window
-    static final private int DEF_WBITS = MAX_WBITS;
+  static final private int MAX_WBITS=15;        // 32K LZ77 window
+  static final private int DEF_WBITS=MAX_WBITS;
+
+  static final private int Z_NO_FLUSH=0;
+  static final private int Z_PARTIAL_FLUSH=1;
+  static final private int Z_SYNC_FLUSH=2;
+  static final private int Z_FULL_FLUSH=3;
+  static final private int Z_FINISH=4;
+
+  static final private int MAX_MEM_LEVEL=9;
 
-    static final private int Z_NO_FLUSH = 0;
-    static final private int Z_PARTIAL_FLUSH = 1;
-    static final private int Z_SYNC_FLUSH = 2;
-    static final private int Z_FULL_FLUSH = 3;
-    static final private int Z_FINISH = 4;
+  static final private int Z_OK=0;
+  static final private int Z_STREAM_END=1;
+  static final private int Z_NEED_DICT=2;
+  static final private int Z_ERRNO=-1;
+  static final private int Z_STREAM_ERROR=-2;
+  static final private int Z_DATA_ERROR=-3;
+  static final private int Z_MEM_ERROR=-4;
+  static final private int Z_BUF_ERROR=-5;
+  static final private int Z_VERSION_ERROR=-6;
 
-    static final private int MAX_MEM_LEVEL = 9;
+  public byte[] next_in;     // next input byte
+  public int next_in_index;
+  public int avail_in;       // number of bytes available at next_in
+  public long total_in;      // total nb of input bytes read so far
+
+  public byte[] next_out;    // next output byte should be put there
+  public int next_out_index;
+  public int avail_out;      // remaining free space at next_out
+  public long total_out;     // total nb of bytes output so far
+
+  public String msg;
+
+  Deflate dstate; 
+  Inflate istate; 
 
-    static final private int Z_OK = 0;
-    static final private int Z_STREAM_END = 1;
-    static final private int Z_NEED_DICT = 2;
-    static final private int Z_ERRNO = -1;
-    static final private int Z_STREAM_ERROR = -2;
-    static final private int Z_DATA_ERROR = -3;
-    static final private int Z_MEM_ERROR = -4;
-    static final private int Z_BUF_ERROR = -5;
-    static final private int Z_VERSION_ERROR = -6;
+  int data_type; // best guess about the data type: ascii or binary
+
+  Checksum adler;
+
+  public ZStream(){
+    this(new Adler32());
+  }
+
+  public ZStream(Checksum adler){
+    this.adler=adler;
+  }
 
-    public byte[] next_in;     // next input byte
-    public int next_in_index;
-    public int avail_in;       // number of bytes available at next_in
-    public long total_in;      // total nb of input bytes read so far
-
-    public byte[] next_out;    // next output byte should be put there
-    public int next_out_index;
-    public int avail_out;      // remaining free space at next_out
-    public long total_out;     // total nb of bytes output so far
-
-    public String msg;
-
-    Deflate dstate;
-    Inflate istate;
-
-    int data_type; // best guess about the data type: ascii or binary
+  public int inflateInit(){
+    return inflateInit(DEF_WBITS);
+  }
+  public int inflateInit(boolean nowrap){
+    return inflateInit(DEF_WBITS, nowrap);
+  }
+  public int inflateInit(int w){
+    return inflateInit(w, false);
+  }
+  public int inflateInit(JZlib.WrapperType wrapperType) {
+    return inflateInit(DEF_WBITS, wrapperType);
+  }
+  public int inflateInit(int w, JZlib.WrapperType wrapperType) {
+    boolean nowrap = false;
+    if(wrapperType == JZlib.W_NONE){
+      nowrap = true;
+    }
+    else if(wrapperType == JZlib.W_GZIP) {
+      w += 16;
+    }
+    else if(wrapperType == JZlib.W_ANY) {
+      w |= Inflate.INFLATE_ANY;
+    }
+    else if(wrapperType == JZlib.W_ZLIB) {
+    }
+    return inflateInit(w, nowrap);
+  }
+  public int inflateInit(int w, boolean nowrap){
+    istate=new Inflate(this);
+    return istate.inflateInit(nowrap?-w:w);
+  }
 
-    public long adler;
-    Adler32 _adler = new Adler32();
-
-    public int inflateInit() {
-        return inflateInit(DEF_WBITS);
-    }
-    public int inflateInit(boolean nowrap) {
-        return inflateInit(DEF_WBITS, nowrap);
-    }
-    public int inflateInit(int w) {
-        return inflateInit(w, false);
-    }
-
-    public int inflateInit(int w, boolean nowrap) {
-        istate = new Inflate();
-        return istate.inflateInit(this, nowrap ? -w : w);
-    }
-
-    public int inflate(int f) {
-        if (istate == null) return Z_STREAM_ERROR;
+  public int inflate(int f){
+    if(istate==null) return Z_STREAM_ERROR;
+    return istate.inflate(f);
+  }
+  public int inflateEnd(){
+    if(istate==null) return Z_STREAM_ERROR;
+    int ret=istate.inflateEnd();
+//    istate = null;
+    return ret;
+  }
+  public int inflateSync(){
+    if(istate == null)
+      return Z_STREAM_ERROR;
+    return istate.inflateSync();
+  }
+  public int inflateSyncPoint(){
+    if(istate == null)
+      return Z_STREAM_ERROR;
+    return istate.inflateSyncPoint();
+  }
+  public int inflateSetDictionary(byte[] dictionary, int dictLength){
+    if(istate == null)
+      return Z_STREAM_ERROR;
+    return istate.inflateSetDictionary(dictionary, dictLength);
+  }
+  public boolean inflateFinished(){
+    return istate.mode==12 /*DONE*/;
+  }
 
-        return istate.inflate(this, f);
+  public int deflateInit(int level){
+    return deflateInit(level, MAX_WBITS);
+  }
+  public int deflateInit(int level, boolean nowrap){
+    return deflateInit(level, MAX_WBITS, nowrap);
+  }
+  public int deflateInit(int level, int bits){
+    return deflateInit(level, bits, false);
+  }
+  public int deflateInit(int level, int bits, int memlevel, JZlib.WrapperType wrapperType){
+    if(bits < 9 || bits > 15){
+      return Z_STREAM_ERROR;
     }
-    public int inflateEnd() {
-        if (istate == null) return Z_STREAM_ERROR;
-
-        int ret = istate.inflateEnd(this);
-        istate = null;
-        return ret;
+    if(wrapperType == JZlib.W_NONE) {
+      bits *= -1;
+    }
+    else if(wrapperType == JZlib.W_GZIP) {
+        bits += 16;
+    }
+    else if(wrapperType == JZlib.W_ANY) {
+        return Z_STREAM_ERROR;
+    }
+    else if(wrapperType == JZlib.W_ZLIB) {
     }
-    public int inflateSync() {
-        if (istate == null)
-            return Z_STREAM_ERROR;
-
-        return istate.inflateSync(this);
+    return this.deflateInit(level, bits, memlevel);
+  }
+  public int deflateInit(int level, int bits, int memlevel){
+    dstate=new Deflate(this);
+    return dstate.deflateInit(level, bits, memlevel);
+  }
+  public int deflateInit(int level, int bits, boolean nowrap){
+    dstate=new Deflate(this);
+    return dstate.deflateInit(level, nowrap?-bits:bits);
+  }
+  public int deflate(int flush){
+    if(dstate==null){
+      return Z_STREAM_ERROR;
     }
-    public int inflateSetDictionary(byte[] dictionary, int dictLength) {
-        if (istate == null)
-            return Z_STREAM_ERROR;
+    return dstate.deflate(flush);
+  }
+  public int deflateEnd(){
+    if(dstate==null) return Z_STREAM_ERROR;
+    int ret=dstate.deflateEnd();
+    dstate=null;
+    return ret;
+  }
+  public int deflateParams(int level, int strategy){
+    if(dstate==null) return Z_STREAM_ERROR;
+    return dstate.deflateParams(level, strategy);
+  }
+  public int deflateSetDictionary (byte[] dictionary, int dictLength){
+    if(dstate == null)
+      return Z_STREAM_ERROR;
+    return dstate.deflateSetDictionary(dictionary, dictLength);
+  }
 
-        return istate.inflateSetDictionary(this, dictionary, dictLength);
+  // Flush as much pending output as possible. All deflate() output goes
+  // through this function so some applications may wish to modify it
+  // to avoid allocating a large strm->next_out buffer and copying into it.
+  // (See also read_buf()).
+  void flush_pending(){
+    int len=dstate.pending;
+
+    if(len>avail_out) len=avail_out;
+    if(len==0) return;
+
+    if(dstate.pending_buf.length<=dstate.pending_out ||
+       next_out.length<=next_out_index ||
+       dstate.pending_buf.length<(dstate.pending_out+len) ||
+       next_out.length<(next_out_index+len)){
+      //System.out.println(dstate.pending_buf.length+", "+dstate.pending_out+
+      //		 ", "+next_out.length+", "+next_out_index+", "+len);
+      //System.out.println("avail_out="+avail_out);
     }
 
-    public int deflateInit(int level) {
-        return deflateInit(level, MAX_WBITS);
-    }
-    public int deflateInit(int level, boolean nowrap) {
-        return deflateInit(level, MAX_WBITS, nowrap);
-    }
-    public int deflateInit(int level, int bits) {
-        return deflateInit(level, bits, false);
+    System.arraycopy(dstate.pending_buf, dstate.pending_out,
+		     next_out, next_out_index, len);
+
+    next_out_index+=len;
+    dstate.pending_out+=len;
+    total_out+=len;
+    avail_out-=len;
+    dstate.pending-=len;
+    if(dstate.pending==0){
+      dstate.pending_out=0;
     }
-    public int deflateInit(int level, int bits, boolean nowrap) {
-        dstate = new Deflate();
-        return dstate.deflateInit(this, level, nowrap ? -bits : bits);
-    }
-    public int deflate(int flush) {
-        if (dstate == null) {
-            return Z_STREAM_ERROR;
-        }
+  }
 
-        return dstate.deflate(this, flush);
-    }
-    public int deflateEnd() {
-        if (dstate == null) return Z_STREAM_ERROR;
+  // Read a new buffer from the current input stream, update the adler32
+  // and total number of bytes read.  All deflate() input goes through
+  // this function so some applications may wish to modify it to avoid
+  // allocating a large strm->next_in buffer and copying from it.
+  // (See also flush_pending()).
+  int read_buf(byte[] buf, int start, int size) {
+    int len=avail_in;
+
+    if(len>size) len=size;
+    if(len==0) return 0;
+
+    avail_in-=len;
 
-        int ret = dstate.deflateEnd();
-        dstate = null;
-        return ret;
+    if(dstate.wrap!=0) {
+      adler.update(next_in, next_in_index, len);
     }
-    public int deflateParams(int level, int strategy) {
-        if (dstate == null) return Z_STREAM_ERROR;
+    System.arraycopy(next_in, next_in_index, buf, start, len);
+    next_in_index  += len;
+    total_in += len;
+    return len;
+  }
 
-        return dstate.deflateParams(this, level, strategy);
-    }
-    public int deflateSetDictionary(byte[] dictionary, int dictLength) {
-        if (dstate == null)
-            return Z_STREAM_ERROR;
+  public long getAdler(){
+    return adler.getValue();
+  }
+
+  public void free(){
+    next_in=null;
+    next_out=null;
+    msg=null;
+  }
 
-        return dstate.deflateSetDictionary(this, dictionary, dictLength);
-    }
+  public void setOutput(byte[] buf){
+    setOutput(buf, 0, buf.length); 
+  }
+
+  public void setOutput(byte[] buf, int off, int len){
+    next_out = buf;
+    next_out_index = off;
+    avail_out = len;
+  }
 
-    // Flush as much pending output as possible. All deflate() output goes
-    // through this function so some applications may wish to modify it
-    // to avoid allocating a large strm->next_out buffer and copying into it.
-    // (See also read_buf()).
-    void flush_pending() {
-        int len = dstate.pending;
+  public void setInput(byte[] buf){
+    setInput(buf, 0, buf.length, false); 
+  }
 
-        if (len > avail_out) len = avail_out;
+  public void setInput(byte[] buf, boolean append){
+    setInput(buf, 0, buf.length, append); 
+  }
 
-        if (len == 0) return;
+  public void setInput(byte[] buf, int off, int len, boolean append){
+    if(len<=0 && append && next_in!=null) return;
 
-        if (dstate.pending_buf.length <= dstate.pending_out ||
-                next_out.length <= next_out_index ||
-                dstate.pending_buf.length < (dstate.pending_out + len) ||
-                next_out.length < (next_out_index + len)) {
-            System.out.println(dstate.pending_buf.length + ", " + dstate.pending_out +
-                               ", " + next_out.length + ", " + next_out_index + ", " + len);
-            System.out.println("avail_out=" + avail_out);
-        }
+    if(avail_in>0 && append){  
+      byte[] tmp = new byte[avail_in+len];
+      System.arraycopy(next_in, next_in_index, tmp, 0, avail_in);
+      System.arraycopy(buf, off, tmp, avail_in, len);
+      next_in=tmp;
+      next_in_index=0;
+      avail_in+=len;
+    }
+    else{
+      next_in=buf;
+      next_in_index=off;
+      avail_in=len;
+    }
+  }
+
+  public byte[] getNextIn(){
+    return next_in;
+  }
 
-        System.arraycopy(dstate.pending_buf, dstate.pending_out,
-                         next_out, next_out_index, len);
-        next_out_index += len;
-        dstate.pending_out += len;
-        total_out += len;
-        avail_out -= len;
-        dstate.pending -= len;
+  public void setNextIn(byte[] next_in){
+    this.next_in = next_in;
+  }
+
+  public int getNextInIndex(){
+    return next_in_index;
+  }
 
-        if (dstate.pending == 0) {
-            dstate.pending_out = 0;
-        }
-    }
+  public void setNextInIndex(int next_in_index){
+    this.next_in_index = next_in_index;
+  }
+
+  public int getAvailIn(){
+    return avail_in;
+  }
+
+  public void setAvailIn(int avail_in){
+    this.avail_in = avail_in;
+  }
 
-    // Read a new buffer from the current input stream, update the adler32
-    // and total number of bytes read.  All deflate() input goes through
-    // this function so some applications may wish to modify it to avoid
-    // allocating a large strm->next_in buffer and copying from it.
-    // (See also flush_pending()).
-    int read_buf(byte[] buf, int start, int size) {
-        int len = avail_in;
+  public byte[] getNextOut(){
+    return next_out;
+  }
+
+  public void setNextOut(byte[] next_out){
+    this.next_out = next_out;
+  }
 
-        if (len > size) len = size;
+  public int getNextOutIndex(){
+    return next_out_index;
+  }
 
-        if (len == 0) return 0;
+  public void setNextOutIndex(int next_out_index){
+    this.next_out_index = next_out_index;
+  }
 
-        avail_in -= len;
+  public int getAvailOut(){
+    return avail_out;
+
+  }
 
-        if (dstate.noheader == 0) {
-            adler = _adler.adler32(adler, next_in, next_in_index, len);
-        }
+  public void setAvailOut(int avail_out){
+    this.avail_out = avail_out;
+  }
+
+  public long getTotalOut(){
+    return total_out;
+  }
+
+  public long getTotalIn(){
+    return total_in;
+  }
 
-        System.arraycopy(next_in, next_in_index, buf, start, len);
-        next_in_index  += len;
-        total_in += len;
-        return len;
-    }
+  public String getMessage(){
+    return msg;
+  }
 
-    public void free() {
-        next_in = null;
-        next_out = null;
-        msg = null;
-        _adler = null;
-    }
+  /**
+   * Those methods are expected to be override by Inflater and Deflater.
+   * In the future, they will become abstract methods.
+   */ 
+  public int end(){ return Z_OK; }
+  public boolean finished(){ return false; }
 }
--- a/src/com/jcraft/jzlib/ZStreamException.java	Fri Aug 01 11:57:17 2014 -0700
+++ b/src/com/jcraft/jzlib/ZStreamException.java	Fri Aug 01 13:34:58 2014 -0700
@@ -8,8 +8,8 @@
   1. Redistributions of source code must retain the above copyright notice,
      this list of conditions and the following disclaimer.
 
-  2. Redistributions in binary form must reproduce the above copyright
-     notice, this list of conditions and the following disclaimer in
+  2. Redistributions in binary form must reproduce the above copyright 
+     notice, this list of conditions and the following disclaimer in 
      the documentation and/or other materials provided with the distribution.
 
   3. The names of the authors may not be used to endorse or promote products
@@ -35,10 +35,10 @@
 package com.jcraft.jzlib;
 
 public class ZStreamException extends java.io.IOException {
-    public ZStreamException() {
-        super();
-    }
-    public ZStreamException(String s) {
-        super(s);
-    }
+  public ZStreamException() {
+    super();
+  }
+  public ZStreamException(String s) {
+    super(s);
+  }
 }