source: trunk/src/gcc/libjava/java/util/zip/ZipInputStream.java@ 2

Last change on this file since 2 was 2, checked in by bird, 22 years ago

Initial revision

  • Property cvs2svn:cvs-rev set to 1.1
  • Property svn:eol-style set to native
  • Property svn:executable set to *
File size: 8.1 KB
Line 
1/* ZipInputStream.java - Input filter for reading zip file
2 Copyright (C) 1999, 2000 Free Software Foundation, Inc.
3
4This file is part of GNU Classpath.
5
6GNU Classpath is free software; you can redistribute it and/or modify
7it under the terms of the GNU General Public License as published by
8the Free Software Foundation; either version 2, or (at your option)
9any later version.
10
11GNU Classpath is distributed in the hope that it will be useful, but
12WITHOUT ANY WARRANTY; without even the implied warranty of
13MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14General Public License for more details.
15
16You should have received a copy of the GNU General Public License
17along with GNU Classpath; see the file COPYING. If not, write to the
18Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
1902111-1307 USA.
20
21Linking this library statically or dynamically with other modules is
22making a combined work based on this library. Thus, the terms and
23conditions of the GNU General Public License cover the whole
24combination.
25
26As a special exception, the copyright holders of this library give you
27permission to link this library with independent modules to produce an
28executable, regardless of the license terms of these independent
29modules, and to copy and distribute the resulting executable under
30terms of your choice, provided that you also meet, for each linked
31independent module, the terms and conditions of the license of that
32module. An independent module is a module which is not derived from
33or based on this library. If you modify this library, you may extend
34this exception to your version of the library, but you are not
35obligated to do so. If you do not wish to do so, delete this
36exception statement from your version. */
37
38package java.util.zip;
39import java.io.*;
40
41/**
42 * @author Per Bothner
43 * @date May 1999.
44 */
45
46/*
47 * Written using on-line Java Platform 1.2 API Specification, as well
48 * as "The Java Class Libraries", 2nd edition (Addison-Wesley, 1998).
49 * Status: Quite incomplete, but can read uncompressed .zip archives.
50 */
51
52// We do not calculate the CRC and compare it with the specified value;
53// we probably should. FIXME.
54
55
56public class ZipInputStream extends InflaterInputStream implements ZipConstants
57{
58 public ZipInputStream (InputStream in)
59 {
60 super (in, new Inflater (true));
61 }
62
63 public ZipEntry getNextEntry () throws IOException
64 {
65 if (closed)
66 throw new IOException ("stream closed");
67 if (current != null)
68 closeEntry();
69 if (in.read() != 'P'
70 || in.read() != 'K')
71 return null;
72 int code = in.read();
73 while (code == '\001')
74 {
75 code = in.read();
76 if (code != '\002')
77 return null;
78 in.skip(16);
79 int size = read4();
80 in.skip(4);
81 int fname_length = readu2();
82 int extra_length = readu2();
83 int fcomment_length = readu2();
84 // `12' is the number of bytes between the comment length
85 // field and the end of the fixed part of the header:
86 // 2 bytes for `disk number start'
87 // 2 bytes for `internal file attributes'
88 // 4 bytes for `external file attributes'
89 // 4 bytes for `relative offset of local header'
90 in.skip(12 + fname_length + extra_length + fcomment_length);
91 if (in.read() != 'P' || in.read() != 'K')
92 return null;
93 code = in.read();
94 }
95 if (code == '\005')
96 {
97 if (in.read() != '\006')
98 return null;
99 in.skip(16);
100 int comment_size = readu2();
101 in.skip(comment_size);
102 if (in.read() != 'P' || in.read() != 'K')
103 return null;
104 code = in.read();
105 }
106 if (code != '\003'
107 || in.read() != '\004')
108 return null;
109 int ex_version = readu2();
110 current_flags = readu2();
111 int method = readu2();
112 int modtime = readu2();
113 int moddate = readu2();
114 int crc = read4();
115 int compressedSize = read4();
116 int uncompressedSize = read4();
117 int filenameLength = readu2();
118 int extraLength = readu2();
119 byte[] bname = new byte[filenameLength];
120 readFully(bname);
121 ZipEntry entry = createZipEntry(new String(bname, "8859_1"));
122 if (extraLength > 0)
123 {
124 byte[] bextra = new byte[extraLength];
125 readFully(bextra);
126 entry.extra = bextra;
127 }
128 entry.compressedSize = compressedSize;
129 entry.size = uncompressedSize;
130 entry.crc = (long) crc & 0xffffffffL;
131 entry.method = method;
132 entry.time = ZipEntry.timeFromDOS(moddate, modtime);
133 current = entry;
134 avail = uncompressedSize;
135 compressed_bytes = compressedSize;
136 return entry;
137 }
138
139 // We override fill to let us control how much data gets read from
140 // the underlying input stream. This lets us avoid having to push
141 // back data.
142 protected void fill () throws IOException
143 {
144 if (closed)
145 throw new IOException ("stream closed");
146 int count = buf.length;
147 if (count > compressed_bytes)
148 count = compressed_bytes;
149 len = in.read(buf, 0, count);
150 if (len != -1)
151 {
152 compressed_bytes -= len;
153 inf.setInput(buf, 0, len);
154 }
155 }
156
157 /**
158 * Creates a new ZipEntry with the given name.
159 * Used by ZipInputStream when normally <code>new ZipEntry (name)</code>
160 * would be called. This gives subclasses such as JarInputStream a change
161 * to override this method and add aditional information to the ZipEntry
162 * (subclass).
163 */
164 protected ZipEntry createZipEntry (String name)
165 {
166 return new ZipEntry (name);
167 }
168
169 public int read (byte[] b, int off, int len) throws IOException
170 {
171 if (closed)
172 throw new IOException ("stream closed");
173 if (len > avail)
174 len = avail;
175 int count;
176 if (current.method == Deflater.DEFLATED)
177 count = super.read(b, off, len);
178 else
179 count = in.read(b, off, len);
180 if (count == -1 || avail == 0)
181 {
182 inf.reset();
183 count = -1;
184 }
185 else
186 avail -= count;
187 return count;
188 }
189
190 public long skip (long n) throws IOException
191 {
192 if (closed)
193 throw new IOException ("stream closed");
194 if (n > avail)
195 n = avail;
196 long count;
197 if (current.method == Deflater.DEFLATED)
198 count = super.skip(n);
199 else
200 count = in.skip(n);
201 avail = avail - (int) count;
202 return count;
203 }
204
205 /**
206 * Returns 0 if the ZipInputStream is closed and 1 otherwise.
207 *
208 * @since 1.2
209 */
210 public int available()
211 {
212 return closed ? 0 : 1;
213 }
214
215 private void readFully (byte[] b) throws IOException
216 {
217 int off = 0;
218 int len = b.length;
219 while (len > 0)
220 {
221 int count = in.read(b, off, len);
222 if (count <= 0)
223 throw new EOFException(".zip archive ended prematurely");
224 off += count;
225 len -= count;
226 }
227 }
228
229 private int readu2 () throws IOException
230 {
231 int byte0 = in.read();
232 int byte1 = in.read();
233 if (byte0 < 0 || byte1 < 0)
234 throw new EOFException(".zip archive ended prematurely");
235 return ((byte1 & 0xFF) << 8) | (byte0 & 0xFF);
236 }
237
238 private int read4 () throws IOException
239 {
240 int byte0 = in.read();
241 int byte1 = in.read();
242 int byte2 = in.read();
243 int byte3 = in.read();
244 if (byte3 < 0)
245 throw new EOFException(".zip archive ended prematurely");
246 return ((byte3 & 0xFF) << 24) + ((byte2 & 0xFF) << 16)
247 + ((byte1 & 0xFF) << 8) + (byte0 & 0xFF);
248 }
249
250 public void closeEntry () throws IOException
251 {
252 if (current != null)
253 {
254 if (avail > 0)
255 skip (avail);
256 if ((current_flags & 8) != 0)
257 {
258 int sig = read4();
259 if (sig != 0x04034b50)
260 throw new ZipException("bad/missing magic number at end of .zip entry");
261 int crc = read4();
262 int compressedSize = read4();
263 int uncompressedSize = read4();
264 if (current.compressedSize != compressedSize
265 || current.size != uncompressedSize
266 || current.crc != crc)
267 throw new ZipException("bad data descriptor at end of .zip entry");
268 }
269 current = null;
270 avail = 0;
271 }
272 }
273
274 /**
275 * Closes this InflaterInputStream.
276 *
277 * @since 1.2
278 */
279 public void close () throws IOException
280 {
281 current = null;
282 closed = true;
283 super.close();
284 }
285
286 private ZipEntry current;
287 private int current_flags;
288 // Number of uncompressed bytes to be read.
289 private int avail;
290 // Number of bytes we can read from underlying stream.
291 private int compressed_bytes;
292 // Is this ZipInputStream closed? Set by the close() method.
293 private boolean closed = false;
294}
Note: See TracBrowser for help on using the repository browser.