mirror of
https://github.com/openjdk/jdk.git
synced 2025-08-27 23:04:50 +02:00
8191918: tomcat gzip-compressed response bodies appear to be broken in update 151
Reviewed-by: psandoz
This commit is contained in:
parent
3e26f1114d
commit
fad5094503
4 changed files with 119 additions and 15 deletions
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 1997, 2017, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
|
@ -164,17 +164,14 @@ Java_java_util_zip_Deflater_deflateBytes(JNIEnv *env, jobject this, jlong addr,
|
||||||
res = deflateParams(strm, level, strategy);
|
res = deflateParams(strm, level, strategy);
|
||||||
(*env)->ReleasePrimitiveArrayCritical(env, b, out_buf, 0);
|
(*env)->ReleasePrimitiveArrayCritical(env, b, out_buf, 0);
|
||||||
(*env)->ReleasePrimitiveArrayCritical(env, this_buf, in_buf, 0);
|
(*env)->ReleasePrimitiveArrayCritical(env, this_buf, in_buf, 0);
|
||||||
|
|
||||||
switch (res) {
|
switch (res) {
|
||||||
case Z_OK:
|
case Z_OK:
|
||||||
(*env)->SetBooleanField(env, this, setParamsID, JNI_FALSE);
|
(*env)->SetBooleanField(env, this, setParamsID, JNI_FALSE);
|
||||||
|
case Z_BUF_ERROR:
|
||||||
this_off += this_len - strm->avail_in;
|
this_off += this_len - strm->avail_in;
|
||||||
(*env)->SetIntField(env, this, offID, this_off);
|
(*env)->SetIntField(env, this, offID, this_off);
|
||||||
(*env)->SetIntField(env, this, lenID, strm->avail_in);
|
(*env)->SetIntField(env, this, lenID, strm->avail_in);
|
||||||
return (jint) (len - strm->avail_out);
|
return (jint) (len - strm->avail_out);
|
||||||
case Z_BUF_ERROR:
|
|
||||||
(*env)->SetBooleanField(env, this, setParamsID, JNI_FALSE);
|
|
||||||
return 0;
|
|
||||||
default:
|
default:
|
||||||
JNU_ThrowInternalError(env, strm->msg);
|
JNU_ThrowInternalError(env, strm->msg);
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -203,19 +200,17 @@ Java_java_util_zip_Deflater_deflateBytes(JNIEnv *env, jobject this, jlong addr,
|
||||||
res = deflate(strm, finish ? Z_FINISH : flush);
|
res = deflate(strm, finish ? Z_FINISH : flush);
|
||||||
(*env)->ReleasePrimitiveArrayCritical(env, b, out_buf, 0);
|
(*env)->ReleasePrimitiveArrayCritical(env, b, out_buf, 0);
|
||||||
(*env)->ReleasePrimitiveArrayCritical(env, this_buf, in_buf, 0);
|
(*env)->ReleasePrimitiveArrayCritical(env, this_buf, in_buf, 0);
|
||||||
|
|
||||||
switch (res) {
|
switch (res) {
|
||||||
case Z_STREAM_END:
|
case Z_STREAM_END:
|
||||||
(*env)->SetBooleanField(env, this, finishedID, JNI_TRUE);
|
(*env)->SetBooleanField(env, this, finishedID, JNI_TRUE);
|
||||||
/* fall through */
|
/* fall through */
|
||||||
case Z_OK:
|
case Z_OK:
|
||||||
|
case Z_BUF_ERROR:
|
||||||
this_off += this_len - strm->avail_in;
|
this_off += this_len - strm->avail_in;
|
||||||
(*env)->SetIntField(env, this, offID, this_off);
|
(*env)->SetIntField(env, this, offID, this_off);
|
||||||
(*env)->SetIntField(env, this, lenID, strm->avail_in);
|
(*env)->SetIntField(env, this, lenID, strm->avail_in);
|
||||||
return len - strm->avail_out;
|
return len - strm->avail_out;
|
||||||
case Z_BUF_ERROR:
|
default:
|
||||||
return 0;
|
|
||||||
default:
|
|
||||||
JNU_ThrowInternalError(env, strm->msg);
|
JNU_ThrowInternalError(env, strm->msg);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -505,8 +505,6 @@ int ZEXPORT deflateResetKeep (strm)
|
||||||
s->pending = 0;
|
s->pending = 0;
|
||||||
s->pending_out = s->pending_buf;
|
s->pending_out = s->pending_buf;
|
||||||
|
|
||||||
s->high_water = 0; /* reset to its inital value 0 */
|
|
||||||
|
|
||||||
if (s->wrap < 0) {
|
if (s->wrap < 0) {
|
||||||
s->wrap = -s->wrap; /* was made negative by deflate(..., Z_FINISH); */
|
s->wrap = -s->wrap; /* was made negative by deflate(..., Z_FINISH); */
|
||||||
}
|
}
|
||||||
|
@ -520,7 +518,7 @@ int ZEXPORT deflateResetKeep (strm)
|
||||||
s->wrap == 2 ? crc32(0L, Z_NULL, 0) :
|
s->wrap == 2 ? crc32(0L, Z_NULL, 0) :
|
||||||
#endif
|
#endif
|
||||||
adler32(0L, Z_NULL, 0);
|
adler32(0L, Z_NULL, 0);
|
||||||
s->last_flush = Z_NO_FLUSH;
|
s->last_flush = -2;
|
||||||
|
|
||||||
_tr_init(s);
|
_tr_init(s);
|
||||||
|
|
||||||
|
@ -613,7 +611,7 @@ int ZEXPORT deflateParams(strm, level, strategy)
|
||||||
func = configuration_table[s->level].func;
|
func = configuration_table[s->level].func;
|
||||||
|
|
||||||
if ((strategy != s->strategy || func != configuration_table[level].func) &&
|
if ((strategy != s->strategy || func != configuration_table[level].func) &&
|
||||||
s->high_water) {
|
s->last_flush != -2) {
|
||||||
/* Flush the last buffer: */
|
/* Flush the last buffer: */
|
||||||
int err = deflate(strm, Z_BLOCK);
|
int err = deflate(strm, Z_BLOCK);
|
||||||
if (err == Z_STREAM_ERROR)
|
if (err == Z_STREAM_ERROR)
|
||||||
|
|
|
@ -93,4 +93,6 @@
|
||||||
s->status =
|
s->status =
|
||||||
#ifdef GZIP
|
#ifdef GZIP
|
||||||
|
|
||||||
|
(7) deflate.c undo (6), replaced withe the official zlib repo fix see#305/#f969409
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -23,8 +23,9 @@
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @test
|
* @test
|
||||||
* @bug 4206909 4813885
|
* @bug 4206909 4813885 8191918
|
||||||
* @summary Test basic functionality of DeflaterOutputStream/InflaterInputStream and GZIPOutputStream/GZIPInputStream, including flush
|
* @summary Test basic functionality of DeflaterOutputStream/InflaterInputStream
|
||||||
|
* and GZIPOutputStream/GZIPInputStream, including flush
|
||||||
* @key randomness
|
* @key randomness
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
@ -147,6 +148,36 @@ public class InflateIn_DeflateOut {
|
||||||
check(Arrays.equals(data, buf));
|
check(Arrays.equals(data, buf));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static void TestFlushableGZIPOutputStream() throws Throwable {
|
||||||
|
var random = new Random(new Date().getTime());
|
||||||
|
|
||||||
|
var byteOutStream = new ByteArrayOutputStream();
|
||||||
|
var output = new FlushableGZIPOutputStream(byteOutStream);
|
||||||
|
|
||||||
|
var data = new byte[random.nextInt(1024 * 1024)];
|
||||||
|
var buf = new byte[data.length];
|
||||||
|
random.nextBytes(data);
|
||||||
|
|
||||||
|
output.write(data);
|
||||||
|
for (int i=0; i<data.length; i++) {
|
||||||
|
output.write(data[i]);
|
||||||
|
}
|
||||||
|
output.flush();
|
||||||
|
for (int i=0; i<data.length; i++) {
|
||||||
|
output.write(data[i]);
|
||||||
|
}
|
||||||
|
output.write(data);
|
||||||
|
output.close();
|
||||||
|
|
||||||
|
var baos = new ByteArrayOutputStream();
|
||||||
|
try (var gzis = new GZIPInputStream(new
|
||||||
|
ByteArrayInputStream(byteOutStream.toByteArray()))) {
|
||||||
|
gzis.transferTo(baos);
|
||||||
|
}
|
||||||
|
var decompressedBytes = baos.toByteArray();
|
||||||
|
check(decompressedBytes.length == data.length * 4);
|
||||||
|
}
|
||||||
|
|
||||||
private static void check(InputStream is, OutputStream os)
|
private static void check(InputStream is, OutputStream os)
|
||||||
throws Throwable
|
throws Throwable
|
||||||
{
|
{
|
||||||
|
@ -268,6 +299,7 @@ public class InflateIn_DeflateOut {
|
||||||
LineOrientedProtocol();
|
LineOrientedProtocol();
|
||||||
GZWriteFlushRead();
|
GZWriteFlushRead();
|
||||||
GZLineOrientedProtocol();
|
GZLineOrientedProtocol();
|
||||||
|
TestFlushableGZIPOutputStream();
|
||||||
}
|
}
|
||||||
|
|
||||||
//--------------------- Infrastructure ---------------------------
|
//--------------------- Infrastructure ---------------------------
|
||||||
|
@ -285,3 +317,80 @@ public class InflateIn_DeflateOut {
|
||||||
System.out.println("\nPassed = " + passed + " failed = " + failed);
|
System.out.println("\nPassed = " + passed + " failed = " + failed);
|
||||||
if (failed > 0) throw new AssertionError("Some tests failed");}
|
if (failed > 0) throw new AssertionError("Some tests failed");}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class FlushableGZIPOutputStream extends GZIPOutputStream {
|
||||||
|
public FlushableGZIPOutputStream(OutputStream os) throws IOException {
|
||||||
|
super(os);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static final byte[] EMPTYBYTEARRAY = new byte[0];
|
||||||
|
private boolean hasData = false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Here we make sure we have received data, so that the header has been for
|
||||||
|
* sure written to the output stream already.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public synchronized void write(byte[] bytes, int i, int i1)
|
||||||
|
throws IOException {
|
||||||
|
super.write(bytes, i, i1);
|
||||||
|
hasData = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized void write(int i) throws IOException {
|
||||||
|
super.write(i);
|
||||||
|
hasData = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized void write(byte[] bytes) throws IOException {
|
||||||
|
super.write(bytes);
|
||||||
|
hasData = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized void flush() throws IOException {
|
||||||
|
if (!hasData) {
|
||||||
|
return; // do not allow the gzip header to be flushed on its own
|
||||||
|
}
|
||||||
|
|
||||||
|
// trick the deflater to flush
|
||||||
|
/**
|
||||||
|
* Now this is tricky: We force the Deflater to flush its data by
|
||||||
|
* switching compression level. As yet, a perplexingly simple workaround
|
||||||
|
* for
|
||||||
|
* http://developer.java.sun.com/developer/bugParade/bugs/4255743.html
|
||||||
|
*/
|
||||||
|
if (!def.finished()) {
|
||||||
|
def.setInput(EMPTYBYTEARRAY, 0, 0);
|
||||||
|
|
||||||
|
def.setLevel(Deflater.NO_COMPRESSION);
|
||||||
|
deflate();
|
||||||
|
|
||||||
|
def.setLevel(Deflater.DEFAULT_COMPRESSION);
|
||||||
|
deflate();
|
||||||
|
|
||||||
|
out.flush();
|
||||||
|
}
|
||||||
|
|
||||||
|
hasData = false; // no more data to flush
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Keep on calling deflate until it runs dry. The default implementation
|
||||||
|
* only does it once and can therefore hold onto data when they need to be
|
||||||
|
* flushed out.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
protected void deflate() throws IOException {
|
||||||
|
int len;
|
||||||
|
do {
|
||||||
|
len = def.deflate(buf, 0, buf.length);
|
||||||
|
if (len > 0) {
|
||||||
|
out.write(buf, 0, len);
|
||||||
|
}
|
||||||
|
} while (len != 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue