GNU Classpath (0.95) | |
Frames | No Frames |
1: /* InflaterInputStream.java - Input stream filter for decompressing 2: Copyright (C) 1999, 2000, 2001, 2002, 2003, 2004 3: Free Software Foundation, Inc. 4: 5: This file is part of GNU Classpath. 6: 7: GNU Classpath is free software; you can redistribute it and/or modify 8: it under the terms of the GNU General Public License as published by 9: the Free Software Foundation; either version 2, or (at your option) 10: any later version. 11: 12: GNU Classpath is distributed in the hope that it will be useful, but 13: WITHOUT ANY WARRANTY; without even the implied warranty of 14: MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 15: General Public License for more details. 16: 17: You should have received a copy of the GNU General Public License 18: along with GNU Classpath; see the file COPYING. If not, write to the 19: Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 20: 02110-1301 USA. 21: 22: Linking this library statically or dynamically with other modules is 23: making a combined work based on this library. Thus, the terms and 24: conditions of the GNU General Public License cover the whole 25: combination. 26: 27: As a special exception, the copyright holders of this library give you 28: permission to link this library with independent modules to produce an 29: executable, regardless of the license terms of these independent 30: modules, and to copy and distribute the resulting executable under 31: terms of your choice, provided that you also meet, for each linked 32: independent module, the terms and conditions of the license of that 33: module. An independent module is a module which is not derived from 34: or based on this library. If you modify this library, you may extend 35: this exception to your version of the library, but you are not 36: obligated to do so. If you do not wish to do so, delete this 37: exception statement from your version. */ 38: 39: 40: package java.util.zip; 41: 42: import java.io.FilterInputStream; 43: import java.io.IOException; 44: import java.io.InputStream; 45: 46: /** 47: * This filter stream is used to decompress data compressed in the "deflate" 48: * format. The "deflate" format is described in RFC 1951. 49: * 50: * This stream may form the basis for other decompression filters, such 51: * as the <code>GZIPInputStream</code>. 52: * 53: * @author John Leuner 54: * @author Tom Tromey 55: * @since 1.1 56: */ 57: public class InflaterInputStream extends FilterInputStream 58: { 59: /** 60: * Decompressor for this filter 61: */ 62: protected Inflater inf; 63: 64: /** 65: * Byte array used as a buffer 66: */ 67: protected byte[] buf; 68: 69: /** 70: * Size of buffer 71: */ 72: protected int len; 73: 74: // We just use this if we are decoding one byte at a time with the 75: // read() call. 76: private byte[] onebytebuffer = new byte[1]; 77: 78: /** 79: * Create an InflaterInputStream with the default decompresseor 80: * and a default buffer size. 81: * 82: * @param in the InputStream to read bytes from 83: */ 84: public InflaterInputStream(InputStream in) 85: { 86: this(in, new Inflater(), 4096); 87: } 88: 89: /** 90: * Create an InflaterInputStream with the specified decompresseor 91: * and a default buffer size. 92: * 93: * @param in the InputStream to read bytes from 94: * @param inf the decompressor used to decompress data read from in 95: */ 96: public InflaterInputStream(InputStream in, Inflater inf) 97: { 98: this(in, inf, 4096); 99: } 100: 101: /** 102: * Create an InflaterInputStream with the specified decompresseor 103: * and a specified buffer size. 104: * 105: * @param in the InputStream to read bytes from 106: * @param inf the decompressor used to decompress data read from in 107: * @param size size of the buffer to use 108: */ 109: public InflaterInputStream(InputStream in, Inflater inf, int size) 110: { 111: super(in); 112: 113: if (in == null) 114: throw new NullPointerException("in may not be null"); 115: if (inf == null) 116: throw new NullPointerException("inf may not be null"); 117: if (size < 0) 118: throw new IllegalArgumentException("size may not be negative"); 119: 120: this.inf = inf; 121: this.buf = new byte [size]; 122: } 123: 124: /** 125: * Returns 0 once the end of the stream (EOF) has been reached. 126: * Otherwise returns 1. 127: */ 128: public int available() throws IOException 129: { 130: // According to the JDK 1.2 docs, this should only ever return 0 131: // or 1 and should not be relied upon by Java programs. 132: if (inf == null) 133: throw new IOException("stream closed"); 134: return inf.finished() ? 0 : 1; 135: } 136: 137: /** 138: * Closes the input stream 139: */ 140: public synchronized void close() throws IOException 141: { 142: if (in != null) 143: in.close(); 144: in = null; 145: } 146: 147: /** 148: * Fills the buffer with more data to decompress. 149: */ 150: protected void fill() throws IOException 151: { 152: if (in == null) 153: throw new ZipException ("InflaterInputStream is closed"); 154: 155: len = in.read(buf, 0, buf.length); 156: 157: if (len < 0) 158: throw new ZipException("Deflated stream ends early."); 159: 160: inf.setInput(buf, 0, len); 161: } 162: 163: /** 164: * Reads one byte of decompressed data. 165: * 166: * The byte is in the lower 8 bits of the int. 167: */ 168: public int read() throws IOException 169: { 170: int nread = read(onebytebuffer, 0, 1); 171: if (nread > 0) 172: return onebytebuffer[0] & 0xff; 173: return -1; 174: } 175: 176: /** 177: * Decompresses data into the byte array 178: * 179: * @param b the array to read and decompress data into 180: * @param off the offset indicating where the data should be placed 181: * @param len the number of bytes to decompress 182: */ 183: public int read(byte[] b, int off, int len) throws IOException 184: { 185: if (inf == null) 186: throw new IOException("stream closed"); 187: if (len == 0) 188: return 0; 189: 190: int count = 0; 191: for (;;) 192: { 193: 194: try 195: { 196: count = inf.inflate(b, off, len); 197: } 198: catch (DataFormatException dfe) 199: { 200: throw new ZipException(dfe.getMessage()); 201: } 202: 203: if (count > 0) 204: return count; 205: 206: if (inf.needsDictionary() 207: | inf.finished()) 208: return -1; 209: else if (inf.needsInput()) 210: fill(); 211: else 212: throw new InternalError("Don't know what to do"); 213: } 214: } 215: 216: /** 217: * Skip specified number of bytes of uncompressed data 218: * 219: * @param n number of bytes to skip 220: */ 221: public long skip(long n) throws IOException 222: { 223: if (inf == null) 224: throw new IOException("stream closed"); 225: if (n < 0) 226: throw new IllegalArgumentException(); 227: 228: if (n == 0) 229: return 0; 230: 231: int buflen = (int) Math.min(n, 2048); 232: byte[] tmpbuf = new byte[buflen]; 233: 234: long skipped = 0L; 235: while (n > 0L) 236: { 237: int numread = read(tmpbuf, 0, buflen); 238: if (numread <= 0) 239: break; 240: n -= numread; 241: skipped += numread; 242: buflen = (int) Math.min(n, 2048); 243: } 244: 245: return skipped; 246: } 247: 248: public boolean markSupported() 249: { 250: return false; 251: } 252: 253: public void mark(int readLimit) 254: { 255: } 256: 257: public void reset() throws IOException 258: { 259: throw new IOException("reset not supported"); 260: } 261: }
GNU Classpath (0.95) |