Ensure we don't leak the ClassLoader in the backtrace (#10691)
Motivation: We have a few classes in which we store and reuse static instances of various exceptions. When doing so it is important to also override fillInStacktrace() and so prevent the leak of the ClassLoader in the internal backtrace field. Modifications: - Add overrides of fillInStracktrace when needed - Move ThrowableUtil usage in the static methods Result: Fixes https://github.com/netty/netty/pull/10686
This commit is contained in:
parent
e3b3cf27da
commit
5b00058fa7
@ -49,35 +49,34 @@ import static io.netty.handler.codec.http2.Http2Headers.PseudoHeaderName.getPseu
|
|||||||
import static io.netty.handler.codec.http2.Http2Headers.PseudoHeaderName.hasPseudoHeaderFormat;
|
import static io.netty.handler.codec.http2.Http2Headers.PseudoHeaderName.hasPseudoHeaderFormat;
|
||||||
import static io.netty.util.AsciiString.EMPTY_STRING;
|
import static io.netty.util.AsciiString.EMPTY_STRING;
|
||||||
import static io.netty.util.internal.ObjectUtil.checkPositive;
|
import static io.netty.util.internal.ObjectUtil.checkPositive;
|
||||||
import static io.netty.util.internal.ThrowableUtil.unknownStackTrace;
|
|
||||||
|
|
||||||
final class HpackDecoder {
|
final class HpackDecoder {
|
||||||
private static final Http2Exception DECODE_ULE_128_DECOMPRESSION_EXCEPTION = unknownStackTrace(
|
private static final Http2Exception DECODE_ULE_128_DECOMPRESSION_EXCEPTION =
|
||||||
Http2Exception.newStatic(COMPRESSION_ERROR, "HPACK - decompression failure",
|
Http2Exception.newStatic(COMPRESSION_ERROR, "HPACK - decompression failure",
|
||||||
Http2Exception.ShutdownHint.HARD_SHUTDOWN), HpackDecoder.class,
|
Http2Exception.ShutdownHint.HARD_SHUTDOWN, HpackDecoder.class,
|
||||||
"decodeULE128(..)");
|
"decodeULE128(..)");
|
||||||
private static final Http2Exception DECODE_ULE_128_TO_LONG_DECOMPRESSION_EXCEPTION = unknownStackTrace(
|
private static final Http2Exception DECODE_ULE_128_TO_LONG_DECOMPRESSION_EXCEPTION =
|
||||||
Http2Exception.newStatic(COMPRESSION_ERROR, "HPACK - long overflow",
|
Http2Exception.newStatic(COMPRESSION_ERROR, "HPACK - long overflow",
|
||||||
Http2Exception.ShutdownHint.HARD_SHUTDOWN), HpackDecoder.class, "decodeULE128(..)");
|
Http2Exception.ShutdownHint.HARD_SHUTDOWN, HpackDecoder.class, "decodeULE128(..)");
|
||||||
private static final Http2Exception DECODE_ULE_128_TO_INT_DECOMPRESSION_EXCEPTION = unknownStackTrace(
|
private static final Http2Exception DECODE_ULE_128_TO_INT_DECOMPRESSION_EXCEPTION =
|
||||||
Http2Exception.newStatic(COMPRESSION_ERROR, "HPACK - int overflow",
|
Http2Exception.newStatic(COMPRESSION_ERROR, "HPACK - int overflow",
|
||||||
Http2Exception.ShutdownHint.HARD_SHUTDOWN), HpackDecoder.class, "decodeULE128ToInt(..)");
|
Http2Exception.ShutdownHint.HARD_SHUTDOWN, HpackDecoder.class, "decodeULE128ToInt(..)");
|
||||||
private static final Http2Exception DECODE_ILLEGAL_INDEX_VALUE = unknownStackTrace(
|
private static final Http2Exception DECODE_ILLEGAL_INDEX_VALUE =
|
||||||
Http2Exception.newStatic(COMPRESSION_ERROR, "HPACK - illegal index value",
|
Http2Exception.newStatic(COMPRESSION_ERROR, "HPACK - illegal index value",
|
||||||
Http2Exception.ShutdownHint.HARD_SHUTDOWN), HpackDecoder.class, "decode(..)");
|
Http2Exception.ShutdownHint.HARD_SHUTDOWN, HpackDecoder.class, "decode(..)");
|
||||||
private static final Http2Exception INDEX_HEADER_ILLEGAL_INDEX_VALUE = unknownStackTrace(
|
private static final Http2Exception INDEX_HEADER_ILLEGAL_INDEX_VALUE =
|
||||||
Http2Exception.newStatic(COMPRESSION_ERROR, "HPACK - illegal index value",
|
Http2Exception.newStatic(COMPRESSION_ERROR, "HPACK - illegal index value",
|
||||||
Http2Exception.ShutdownHint.HARD_SHUTDOWN), HpackDecoder.class, "indexHeader(..)");
|
Http2Exception.ShutdownHint.HARD_SHUTDOWN, HpackDecoder.class, "indexHeader(..)");
|
||||||
private static final Http2Exception READ_NAME_ILLEGAL_INDEX_VALUE = unknownStackTrace(
|
private static final Http2Exception READ_NAME_ILLEGAL_INDEX_VALUE =
|
||||||
Http2Exception.newStatic(COMPRESSION_ERROR, "HPACK - illegal index value",
|
Http2Exception.newStatic(COMPRESSION_ERROR, "HPACK - illegal index value",
|
||||||
Http2Exception.ShutdownHint.HARD_SHUTDOWN), HpackDecoder.class, "readName(..)");
|
Http2Exception.ShutdownHint.HARD_SHUTDOWN, HpackDecoder.class, "readName(..)");
|
||||||
private static final Http2Exception INVALID_MAX_DYNAMIC_TABLE_SIZE = unknownStackTrace(
|
private static final Http2Exception INVALID_MAX_DYNAMIC_TABLE_SIZE =
|
||||||
Http2Exception.newStatic(COMPRESSION_ERROR, "HPACK - invalid max dynamic table size",
|
Http2Exception.newStatic(COMPRESSION_ERROR, "HPACK - invalid max dynamic table size",
|
||||||
Http2Exception.ShutdownHint.HARD_SHUTDOWN), HpackDecoder.class,
|
Http2Exception.ShutdownHint.HARD_SHUTDOWN, HpackDecoder.class,
|
||||||
"setDynamicTableSize(..)");
|
"setDynamicTableSize(..)");
|
||||||
private static final Http2Exception MAX_DYNAMIC_TABLE_SIZE_CHANGE_REQUIRED = unknownStackTrace(
|
private static final Http2Exception MAX_DYNAMIC_TABLE_SIZE_CHANGE_REQUIRED =
|
||||||
Http2Exception.newStatic(COMPRESSION_ERROR, "HPACK - max dynamic table size change required",
|
Http2Exception.newStatic(COMPRESSION_ERROR, "HPACK - max dynamic table size change required",
|
||||||
Http2Exception.ShutdownHint.HARD_SHUTDOWN), HpackDecoder.class, "decode(..)");
|
Http2Exception.ShutdownHint.HARD_SHUTDOWN, HpackDecoder.class, "decode(..)");
|
||||||
private static final byte READ_HEADER_REPRESENTATION = 0;
|
private static final byte READ_HEADER_REPRESENTATION = 0;
|
||||||
private static final byte READ_MAX_DYNAMIC_TABLE_SIZE = 1;
|
private static final byte READ_MAX_DYNAMIC_TABLE_SIZE = 1;
|
||||||
private static final byte READ_INDEXED_HEADER = 2;
|
private static final byte READ_INDEXED_HEADER = 2;
|
||||||
|
@ -34,7 +34,6 @@ package io.netty.handler.codec.http2;
|
|||||||
import io.netty.buffer.ByteBuf;
|
import io.netty.buffer.ByteBuf;
|
||||||
import io.netty.util.AsciiString;
|
import io.netty.util.AsciiString;
|
||||||
import io.netty.util.ByteProcessor;
|
import io.netty.util.ByteProcessor;
|
||||||
import io.netty.util.internal.ThrowableUtil;
|
|
||||||
|
|
||||||
import static io.netty.handler.codec.http2.Http2Error.COMPRESSION_ERROR;
|
import static io.netty.handler.codec.http2.Http2Error.COMPRESSION_ERROR;
|
||||||
|
|
||||||
@ -4665,9 +4664,9 @@ final class HpackHuffmanDecoder implements ByteProcessor {
|
|||||||
HUFFMAN_FAIL << 8,
|
HUFFMAN_FAIL << 8,
|
||||||
};
|
};
|
||||||
|
|
||||||
private static final Http2Exception BAD_ENCODING = ThrowableUtil.unknownStackTrace(
|
private static final Http2Exception BAD_ENCODING =
|
||||||
Http2Exception.newStatic(COMPRESSION_ERROR, "HPACK - Bad Encoding",
|
Http2Exception.newStatic(COMPRESSION_ERROR, "HPACK - Bad Encoding",
|
||||||
Http2Exception.ShutdownHint.HARD_SHUTDOWN), HpackHuffmanDecoder.class, "decode(..)");
|
Http2Exception.ShutdownHint.HARD_SHUTDOWN, HpackHuffmanDecoder.class, "decode(..)");
|
||||||
|
|
||||||
private byte[] dest;
|
private byte[] dest;
|
||||||
private int k;
|
private int k;
|
||||||
|
@ -15,7 +15,7 @@
|
|||||||
|
|
||||||
package io.netty.handler.codec.http2;
|
package io.netty.handler.codec.http2;
|
||||||
|
|
||||||
import io.netty.util.internal.PlatformDependent;
|
import io.netty.util.internal.ThrowableUtil;
|
||||||
import io.netty.util.internal.UnstableApi;
|
import io.netty.util.internal.UnstableApi;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
@ -63,11 +63,10 @@ public class Http2Exception extends Exception {
|
|||||||
this.shutdownHint = requireNonNull(shutdownHint, "shutdownHint");
|
this.shutdownHint = requireNonNull(shutdownHint, "shutdownHint");
|
||||||
}
|
}
|
||||||
|
|
||||||
static Http2Exception newStatic(Http2Error error, String message, ShutdownHint shutdownHint) {
|
static Http2Exception newStatic(Http2Error error, String message, ShutdownHint shutdownHint,
|
||||||
if (PlatformDependent.javaVersion() >= 7) {
|
Class<?> clazz, String method) {
|
||||||
return new Http2Exception(error, message, shutdownHint, true);
|
return ThrowableUtil.unknownStackTrace(
|
||||||
}
|
new StacklessHttp2Exception(error, message, shutdownHint), clazz, method);
|
||||||
return new Http2Exception(error, message, shutdownHint);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private Http2Exception(Http2Error error, String message, ShutdownHint shutdownHint, boolean shared) {
|
private Http2Exception(Http2Error error, String message, ShutdownHint shutdownHint, boolean shared) {
|
||||||
@ -302,4 +301,20 @@ public class Http2Exception extends Exception {
|
|||||||
return exceptions.iterator();
|
return exceptions.iterator();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static final class StacklessHttp2Exception extends Http2Exception {
|
||||||
|
|
||||||
|
private static final long serialVersionUID = 1077888485687219443L;
|
||||||
|
|
||||||
|
StacklessHttp2Exception(Http2Error error, String message, ShutdownHint shutdownHint) {
|
||||||
|
super(error, message, shutdownHint, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Override fillInStackTrace() so we not populate the backtrace via a native call and so leak the
|
||||||
|
// Classloader.
|
||||||
|
@Override
|
||||||
|
public Throwable fillInStackTrace() {
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -39,8 +39,8 @@ public class DefaultPromise<V> implements Promise<V> {
|
|||||||
AtomicReferenceFieldUpdater.newUpdater(DefaultPromise.class, Object.class, "result");
|
AtomicReferenceFieldUpdater.newUpdater(DefaultPromise.class, Object.class, "result");
|
||||||
private static final Object SUCCESS = new Object();
|
private static final Object SUCCESS = new Object();
|
||||||
private static final Object UNCANCELLABLE = new Object();
|
private static final Object UNCANCELLABLE = new Object();
|
||||||
private static final CauseHolder CANCELLATION_CAUSE_HOLDER = new CauseHolder(ThrowableUtil.unknownStackTrace(
|
private static final CauseHolder CANCELLATION_CAUSE_HOLDER = new CauseHolder(
|
||||||
new CancellationException(), DefaultPromise.class, "cancel(...)"));
|
StacklessCancellationException.newInstance(DefaultPromise.class, "cancel(...)"));
|
||||||
private static final StackTraceElement[] CANCELLATION_STACK = CANCELLATION_CAUSE_HOLDER.cause.getStackTrace();
|
private static final StackTraceElement[] CANCELLATION_STACK = CANCELLATION_CAUSE_HOLDER.cause.getStackTrace();
|
||||||
|
|
||||||
private volatile Object result;
|
private volatile Object result;
|
||||||
@ -754,4 +754,22 @@ public class DefaultPromise<V> implements Promise<V> {
|
|||||||
}
|
}
|
||||||
return stageAdapter;
|
return stageAdapter;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static final class StacklessCancellationException extends CancellationException {
|
||||||
|
|
||||||
|
private static final long serialVersionUID = -2974906711413716191L;
|
||||||
|
|
||||||
|
private StacklessCancellationException() { }
|
||||||
|
|
||||||
|
// Override fillInStackTrace() so we not populate the backtrace via a native call and so leak the
|
||||||
|
// Classloader.
|
||||||
|
@Override
|
||||||
|
public Throwable fillInStackTrace() {
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
static StacklessCancellationException newInstance(Class<?> clazz, String method) {
|
||||||
|
return ThrowableUtil.unknownStackTrace(new StacklessCancellationException(), clazz, method);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -61,26 +61,21 @@ import static java.util.Objects.requireNonNull;
|
|||||||
|
|
||||||
abstract class DnsResolveContext<T> {
|
abstract class DnsResolveContext<T> {
|
||||||
|
|
||||||
private static final RuntimeException NXDOMAIN_QUERY_FAILED_EXCEPTION = ThrowableUtil.unknownStackTrace(
|
private static final RuntimeException NXDOMAIN_QUERY_FAILED_EXCEPTION =
|
||||||
DnsResolveContextException.newStatic("No answer found and NXDOMAIN response code returned"),
|
DnsResolveContextException.newStatic("No answer found and NXDOMAIN response code returned",
|
||||||
DnsResolveContext.class,
|
DnsResolveContext.class, "onResponse(..)");
|
||||||
"onResponse(..)");
|
private static final RuntimeException CNAME_NOT_FOUND_QUERY_FAILED_EXCEPTION =
|
||||||
private static final RuntimeException CNAME_NOT_FOUND_QUERY_FAILED_EXCEPTION = ThrowableUtil.unknownStackTrace(
|
DnsResolveContextException.newStatic("No matching CNAME record found",
|
||||||
DnsResolveContextException.newStatic("No matching CNAME record found"),
|
DnsResolveContext.class, "onResponseCNAME(..)");
|
||||||
DnsResolveContext.class,
|
private static final RuntimeException NO_MATCHING_RECORD_QUERY_FAILED_EXCEPTION =
|
||||||
"onResponseCNAME(..)");
|
DnsResolveContextException.newStatic("No matching record type found",
|
||||||
private static final RuntimeException NO_MATCHING_RECORD_QUERY_FAILED_EXCEPTION = ThrowableUtil.unknownStackTrace(
|
DnsResolveContext.class, "onResponseAorAAAA(..)");
|
||||||
DnsResolveContextException.newStatic("No matching record type found"),
|
private static final RuntimeException UNRECOGNIZED_TYPE_QUERY_FAILED_EXCEPTION =
|
||||||
DnsResolveContext.class,
|
DnsResolveContextException.newStatic("Response type was unrecognized",
|
||||||
"onResponseAorAAAA(..)");
|
DnsResolveContext.class, "onResponse(..)");
|
||||||
private static final RuntimeException UNRECOGNIZED_TYPE_QUERY_FAILED_EXCEPTION = ThrowableUtil.unknownStackTrace(
|
private static final RuntimeException NAME_SERVERS_EXHAUSTED_EXCEPTION =
|
||||||
new RuntimeException("Response type was unrecognized"),
|
DnsResolveContextException.newStatic("No name servers returned an answer",
|
||||||
DnsResolveContext.class,
|
DnsResolveContext.class, "tryToFinishResolve(..)");
|
||||||
"onResponse(..)");
|
|
||||||
private static final RuntimeException NAME_SERVERS_EXHAUSTED_EXCEPTION = ThrowableUtil.unknownStackTrace(
|
|
||||||
DnsResolveContextException.newStatic("No name servers returned an answer"),
|
|
||||||
DnsResolveContext.class,
|
|
||||||
"tryToFinishResolve(..)");
|
|
||||||
|
|
||||||
final DnsNameResolver parent;
|
final DnsNameResolver parent;
|
||||||
private final Promise<?> originalPromise;
|
private final Promise<?> originalPromise;
|
||||||
@ -117,12 +112,21 @@ abstract class DnsResolveContext<T> {
|
|||||||
|
|
||||||
static final class DnsResolveContextException extends RuntimeException {
|
static final class DnsResolveContextException extends RuntimeException {
|
||||||
|
|
||||||
|
private static final long serialVersionUID = 1209303419266433003L;
|
||||||
|
|
||||||
private DnsResolveContextException(String message) {
|
private DnsResolveContextException(String message) {
|
||||||
super(message, null, false, true);
|
super(message, null, false, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
static DnsResolveContextException newStatic(String message) {
|
// Override fillInStackTrace() so we not populate the backtrace via a native call and so leak the
|
||||||
return new DnsResolveContextException(message);
|
// Classloader.
|
||||||
|
@Override
|
||||||
|
public Throwable fillInStackTrace() {
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
static DnsResolveContextException newStatic(String message, Class<?> clazz, String method) {
|
||||||
|
return ThrowableUtil.unknownStackTrace(new DnsResolveContextException(message), clazz, method);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user