8193682: Infinite loop in ZipOutputStream.close()
Reviewed-by: lancea, coffeys
This commit is contained in:
parent
abaa073bcb
commit
1e9ed54d36
@ -234,9 +234,12 @@ public class DeflaterOutputStream extends FilterOutputStream {
|
|||||||
*/
|
*/
|
||||||
public void close() throws IOException {
|
public void close() throws IOException {
|
||||||
if (!closed) {
|
if (!closed) {
|
||||||
finish();
|
try {
|
||||||
if (usesDefaultDeflater)
|
finish();
|
||||||
def.end();
|
} finally {
|
||||||
|
if (usesDefaultDeflater)
|
||||||
|
def.end();
|
||||||
|
}
|
||||||
out.close();
|
out.close();
|
||||||
closed = true;
|
closed = true;
|
||||||
}
|
}
|
||||||
|
@ -157,24 +157,30 @@ public class GZIPOutputStream extends DeflaterOutputStream {
|
|||||||
*/
|
*/
|
||||||
public void finish() throws IOException {
|
public void finish() throws IOException {
|
||||||
if (!def.finished()) {
|
if (!def.finished()) {
|
||||||
def.finish();
|
try {
|
||||||
while (!def.finished()) {
|
def.finish();
|
||||||
int len = def.deflate(buf, 0, buf.length);
|
while (!def.finished()) {
|
||||||
if (def.finished() && len <= buf.length - TRAILER_SIZE) {
|
int len = def.deflate(buf, 0, buf.length);
|
||||||
// last deflater buffer. Fit trailer at the end
|
if (def.finished() && len <= buf.length - TRAILER_SIZE) {
|
||||||
writeTrailer(buf, len);
|
// last deflater buffer. Fit trailer at the end
|
||||||
len = len + TRAILER_SIZE;
|
writeTrailer(buf, len);
|
||||||
out.write(buf, 0, len);
|
len = len + TRAILER_SIZE;
|
||||||
return;
|
out.write(buf, 0, len);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (len > 0)
|
||||||
|
out.write(buf, 0, len);
|
||||||
}
|
}
|
||||||
if (len > 0)
|
// if we can't fit the trailer at the end of the last
|
||||||
out.write(buf, 0, len);
|
// deflater buffer, we write it separately
|
||||||
|
byte[] trailer = new byte[TRAILER_SIZE];
|
||||||
|
writeTrailer(trailer, 0);
|
||||||
|
out.write(trailer);
|
||||||
|
} catch (IOException e) {
|
||||||
|
if (usesDefaultDeflater)
|
||||||
|
def.end();
|
||||||
|
throw e;
|
||||||
}
|
}
|
||||||
// if we can't fit the trailer at the end of the last
|
|
||||||
// deflater buffer, we write it separately
|
|
||||||
byte[] trailer = new byte[TRAILER_SIZE];
|
|
||||||
writeTrailer(trailer, 0);
|
|
||||||
out.write(trailer);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -259,58 +259,64 @@ public class ZipOutputStream extends DeflaterOutputStream implements ZipConstant
|
|||||||
public void closeEntry() throws IOException {
|
public void closeEntry() throws IOException {
|
||||||
ensureOpen();
|
ensureOpen();
|
||||||
if (current != null) {
|
if (current != null) {
|
||||||
ZipEntry e = current.entry;
|
try {
|
||||||
switch (e.method) {
|
ZipEntry e = current.entry;
|
||||||
case DEFLATED -> {
|
switch (e.method) {
|
||||||
def.finish();
|
case DEFLATED -> {
|
||||||
while (!def.finished()) {
|
def.finish();
|
||||||
deflate();
|
while (!def.finished()) {
|
||||||
}
|
deflate();
|
||||||
if ((e.flag & 8) == 0) {
|
}
|
||||||
// verify size, compressed size, and crc-32 settings
|
if ((e.flag & 8) == 0) {
|
||||||
if (e.size != def.getBytesRead()) {
|
// verify size, compressed size, and crc-32 settings
|
||||||
throw new ZipException(
|
if (e.size != def.getBytesRead()) {
|
||||||
"invalid entry size (expected " + e.size +
|
throw new ZipException(
|
||||||
" but got " + def.getBytesRead() + " bytes)");
|
"invalid entry size (expected " + e.size +
|
||||||
|
" but got " + def.getBytesRead() + " bytes)");
|
||||||
|
}
|
||||||
|
if (e.csize != def.getBytesWritten()) {
|
||||||
|
throw new ZipException(
|
||||||
|
"invalid entry compressed size (expected " +
|
||||||
|
e.csize + " but got " + def.getBytesWritten() + " bytes)");
|
||||||
|
}
|
||||||
|
if (e.crc != crc.getValue()) {
|
||||||
|
throw new ZipException(
|
||||||
|
"invalid entry CRC-32 (expected 0x" +
|
||||||
|
Long.toHexString(e.crc) + " but got 0x" +
|
||||||
|
Long.toHexString(crc.getValue()) + ")");
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
e.size = def.getBytesRead();
|
||||||
|
e.csize = def.getBytesWritten();
|
||||||
|
e.crc = crc.getValue();
|
||||||
|
writeEXT(e);
|
||||||
|
}
|
||||||
|
def.reset();
|
||||||
|
written += e.csize;
|
||||||
}
|
}
|
||||||
if (e.csize != def.getBytesWritten()) {
|
case STORED -> {
|
||||||
throw new ZipException(
|
// we already know that both e.size and e.csize are the same
|
||||||
"invalid entry compressed size (expected " +
|
if (e.size != written - locoff) {
|
||||||
e.csize + " but got " + def.getBytesWritten() + " bytes)");
|
throw new ZipException(
|
||||||
|
"invalid entry size (expected " + e.size +
|
||||||
|
" but got " + (written - locoff) + " bytes)");
|
||||||
|
}
|
||||||
|
if (e.crc != crc.getValue()) {
|
||||||
|
throw new ZipException(
|
||||||
|
"invalid entry crc-32 (expected 0x" +
|
||||||
|
Long.toHexString(e.crc) + " but got 0x" +
|
||||||
|
Long.toHexString(crc.getValue()) + ")");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (e.crc != crc.getValue()) {
|
default -> throw new ZipException("invalid compression method");
|
||||||
throw new ZipException(
|
|
||||||
"invalid entry CRC-32 (expected 0x" +
|
|
||||||
Long.toHexString(e.crc) + " but got 0x" +
|
|
||||||
Long.toHexString(crc.getValue()) + ")");
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
e.size = def.getBytesRead();
|
|
||||||
e.csize = def.getBytesWritten();
|
|
||||||
e.crc = crc.getValue();
|
|
||||||
writeEXT(e);
|
|
||||||
}
|
}
|
||||||
def.reset();
|
crc.reset();
|
||||||
written += e.csize;
|
current = null;
|
||||||
|
} catch (IOException e) {
|
||||||
|
if (usesDefaultDeflater && !(e instanceof ZipException))
|
||||||
|
def.end();
|
||||||
|
throw e;
|
||||||
}
|
}
|
||||||
case STORED -> {
|
|
||||||
// we already know that both e.size and e.csize are the same
|
|
||||||
if (e.size != written - locoff) {
|
|
||||||
throw new ZipException(
|
|
||||||
"invalid entry size (expected " + e.size +
|
|
||||||
" but got " + (written - locoff) + " bytes)");
|
|
||||||
}
|
|
||||||
if (e.crc != crc.getValue()) {
|
|
||||||
throw new ZipException(
|
|
||||||
"invalid entry crc-32 (expected 0x" +
|
|
||||||
Long.toHexString(e.crc) + " but got 0x" +
|
|
||||||
Long.toHexString(crc.getValue()) + ")");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
default -> throw new ZipException("invalid compression method");
|
|
||||||
}
|
|
||||||
crc.reset();
|
|
||||||
current = null;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
147
test/jdk/java/util/zip/CloseDeflaterTest.java
Normal file
147
test/jdk/java/util/zip/CloseDeflaterTest.java
Normal file
@ -0,0 +1,147 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2021, 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 8193682
|
||||||
|
* @summary Test Infinite loop while writing on closed GZipOutputStream , ZipOutputStream and JarOutputStream.
|
||||||
|
* @run testng CloseDeflaterTest
|
||||||
|
*/
|
||||||
|
import java.io.*;
|
||||||
|
import java.util.Random;
|
||||||
|
import java.util.jar.JarOutputStream;
|
||||||
|
import java.util.zip.GZIPOutputStream;
|
||||||
|
import java.util.zip.ZipOutputStream;
|
||||||
|
import java.util.zip.ZipEntry;
|
||||||
|
|
||||||
|
import org.testng.annotations.BeforeTest;
|
||||||
|
import org.testng.annotations.DataProvider;
|
||||||
|
import org.testng.annotations.Test;
|
||||||
|
import static org.testng.Assert.fail;
|
||||||
|
|
||||||
|
|
||||||
|
public class CloseDeflaterTest {
|
||||||
|
|
||||||
|
//number of bytes to write
|
||||||
|
private static final int INPUT_LENGTH= 512;
|
||||||
|
//OutputStream that will throw an exception during a write operation
|
||||||
|
private static OutputStream outStream = new OutputStream() {
|
||||||
|
@Override
|
||||||
|
public void write(byte[] b, int off, int len) throws IOException {
|
||||||
|
//throw exception during write
|
||||||
|
throw new IOException();
|
||||||
|
}
|
||||||
|
@Override
|
||||||
|
public void write(byte b[]) throws IOException {}
|
||||||
|
@Override
|
||||||
|
public void write(int b) throws IOException {}
|
||||||
|
};
|
||||||
|
private static byte[] inputBytes = new byte[INPUT_LENGTH];
|
||||||
|
private static Random rand = new Random();
|
||||||
|
|
||||||
|
@DataProvider(name = "testgzipinput")
|
||||||
|
public Object[][] testGZipInput() {
|
||||||
|
//testGZip will close the GZipOutputStream using close() method when the boolean
|
||||||
|
//useCloseMethod is set to true and finish() method if the value is set to false
|
||||||
|
return new Object[][] {
|
||||||
|
{ GZIPOutputStream.class, true },
|
||||||
|
{ GZIPOutputStream.class, false },
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
@DataProvider(name = "testzipjarinput")
|
||||||
|
public Object[][] testZipAndJarInput() {
|
||||||
|
//testZipAndJarInput will perfrom write/closeEntry operations on JarOutputStream when the boolean
|
||||||
|
//useJar is set to true and on ZipOutputStream if the value is set to false
|
||||||
|
return new Object[][] {
|
||||||
|
{ JarOutputStream.class, true },
|
||||||
|
{ ZipOutputStream.class, false },
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
@BeforeTest
|
||||||
|
public void before_test()
|
||||||
|
{
|
||||||
|
//add inputBytes array with random bytes to write into Zip
|
||||||
|
rand.nextBytes(inputBytes);
|
||||||
|
}
|
||||||
|
|
||||||
|
//Test for infinite loop by writing bytes to closed GZIPOutputStream
|
||||||
|
@Test(dataProvider = "testgzipinput")
|
||||||
|
public void testGZip(Class<?> type, boolean useCloseMethod) throws IOException {
|
||||||
|
GZIPOutputStream zip = new GZIPOutputStream(outStream);
|
||||||
|
try {
|
||||||
|
zip.write(inputBytes, 0, INPUT_LENGTH);
|
||||||
|
//close zip
|
||||||
|
if(useCloseMethod) {
|
||||||
|
zip.close();
|
||||||
|
} else {
|
||||||
|
zip.finish();
|
||||||
|
}
|
||||||
|
} catch (IOException e) {
|
||||||
|
//expected
|
||||||
|
}
|
||||||
|
for (int i = 0; i < 3; i++) {
|
||||||
|
try {
|
||||||
|
//write on a closed GZIPOutputStream
|
||||||
|
zip.write(inputBytes, 0, INPUT_LENGTH);
|
||||||
|
fail("Deflater closed exception not thrown");
|
||||||
|
} catch (NullPointerException e) {
|
||||||
|
//expected , Deflater has been closed exception
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//Test for infinite loop by writing bytes to closed ZipOutputStream/JarOutputStream
|
||||||
|
@Test(dataProvider = "testzipjarinput")
|
||||||
|
public void testZipCloseEntry(Class<?> type,boolean useJar) throws IOException {
|
||||||
|
ZipOutputStream zip = null;
|
||||||
|
if(useJar) {
|
||||||
|
zip = new JarOutputStream(outStream);
|
||||||
|
} else {
|
||||||
|
zip = new ZipOutputStream(outStream);
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
zip.putNextEntry(new ZipEntry(""));
|
||||||
|
} catch (IOException e) {
|
||||||
|
//expected to throw IOException since putNextEntry calls write method
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
zip.write(inputBytes, 0, INPUT_LENGTH);
|
||||||
|
//close zip entry
|
||||||
|
zip.closeEntry();
|
||||||
|
} catch (IOException e) {
|
||||||
|
//expected
|
||||||
|
}
|
||||||
|
for (int i = 0; i < 3; i++) {
|
||||||
|
try {
|
||||||
|
//write on a closed ZipOutputStream
|
||||||
|
zip.write(inputBytes, 0, INPUT_LENGTH);
|
||||||
|
fail("Deflater closed exception not thrown");
|
||||||
|
} catch (NullPointerException e) {
|
||||||
|
//expected , Deflater has been closed exception
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user