diff --git a/jdk/src/share/classes/java/util/zip/GZIPInputStream.java b/jdk/src/share/classes/java/util/zip/GZIPInputStream.java index 4db2e7ffa2f..74a5c60f1a8 100644 --- a/jdk/src/share/classes/java/util/zip/GZIPInputStream.java +++ b/jdk/src/share/classes/java/util/zip/GZIPInputStream.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1996, 2012, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1996, 2013, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -27,6 +27,7 @@ package java.util.zip; import java.io.SequenceInputStream; import java.io.ByteArrayInputStream; +import java.io.FilterInputStream; import java.io.InputStream; import java.io.IOException; import java.io.EOFException; @@ -212,7 +213,10 @@ class GZIPInputStream extends InflaterInputStream { int n = inf.getRemaining(); if (n > 0) { in = new SequenceInputStream( - new ByteArrayInputStream(buf, len - n, n), in); + new ByteArrayInputStream(buf, len - n, n), + new FilterInputStream(in) { + public void close() throws IOException {} + }); } // Uses left-to-right evaluation order if ((readUInt(in) != crc.getValue()) || diff --git a/jdk/test/java/util/zip/GZIP/GZIPInZip.java b/jdk/test/java/util/zip/GZIP/GZIPInZip.java new file mode 100644 index 00000000000..558ad962e26 --- /dev/null +++ b/jdk/test/java/util/zip/GZIP/GZIPInZip.java @@ -0,0 +1,96 @@ +/* + * Copyright (c) 2013, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +/* @test + * @bug 7021870 + * @summary Reading last gzip chain member must not close the input stream + */ + +import java.io.*; +import java.util.*; +import java.util.zip.GZIPInputStream; +import java.util.zip.GZIPOutputStream; +import java.util.zip.ZipEntry; +import java.util.zip.ZipInputStream; +import java.util.zip.ZipOutputStream; + + +public class GZIPInZip { + + private static volatile Throwable trouble; + + public static void main(String[] args) throws Throwable { + + final PipedOutputStream pos = new PipedOutputStream(); + final PipedInputStream pis = new PipedInputStream(pos); + + Thread compressor = new Thread() { + public void run() { + final byte[] xbuf = { 'x' }; + try { + ZipOutputStream zos = new ZipOutputStream(pos); + + zos.putNextEntry(new ZipEntry("a.gz")); + GZIPOutputStream gos1 = new GZIPOutputStream(zos); + gos1.write(xbuf); gos1.finish(); + zos.closeEntry(); + + zos.putNextEntry(new ZipEntry("b.gz")); + GZIPOutputStream gos2 = new GZIPOutputStream(zos); + gos2.write(xbuf); gos2.finish(); + zos.closeEntry(); + + } catch (Throwable t) { + trouble = t; + } + } + }; + + Thread uncompressor = new Thread() { + public void run() { + try { + ZipInputStream zis = new ZipInputStream(pis); + zis.getNextEntry(); + InputStream gis = new GZIPInputStream(zis); + // try to read more than the entry has + gis.skip(2); + + try { + zis.getNextEntry(); + } catch (IOException e) { + throw new AssertionError("ZIP stream was prematurely closed"); + } + + } catch (Throwable t) { + trouble = t; + } + } + }; + + compressor.start(); uncompressor.start(); + compressor.join(); uncompressor.join(); + + if (trouble != null) + throw trouble; + } +}