Makes LengthFieldBasedFrameDecoder::decode inlineable
Motivation: The decode method is too large to be inlined with default compiler settings, hence the uncommon paths need to be packed and moved away form the common one. Modifications: The uncommon paths of the decode call (eg failures with thrown exceptions) are packed and moved in private methods in order to reduce the size of the common one and let it being inlined. Result: The decode method is being inlined if the stack depth allows it.
This commit is contained in:
parent
0b0de76adc
commit
6780183a89
@ -15,13 +15,13 @@
|
|||||||
*/
|
*/
|
||||||
package io.netty.handler.codec;
|
package io.netty.handler.codec;
|
||||||
|
|
||||||
|
import java.nio.ByteOrder;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
import io.netty.buffer.ByteBuf;
|
import io.netty.buffer.ByteBuf;
|
||||||
import io.netty.channel.ChannelHandlerContext;
|
import io.netty.channel.ChannelHandlerContext;
|
||||||
import io.netty.handler.codec.serialization.ObjectDecoder;
|
import io.netty.handler.codec.serialization.ObjectDecoder;
|
||||||
|
|
||||||
import java.nio.ByteOrder;
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A decoder that splits the received {@link ByteBuf}s dynamically by the
|
* A decoder that splits the received {@link ByteBuf}s dynamically by the
|
||||||
* value of the length field in the message. It is particularly useful when you
|
* value of the length field in the message. It is particularly useful when you
|
||||||
@ -346,6 +346,56 @@ public class LengthFieldBasedFrameDecoder extends ByteToMessageDecoder {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void discardingTooLongFrame(ByteBuf in) {
|
||||||
|
long bytesToDiscard = this.bytesToDiscard;
|
||||||
|
int localBytesToDiscard = (int) Math.min(bytesToDiscard, in.readableBytes());
|
||||||
|
in.skipBytes(localBytesToDiscard);
|
||||||
|
bytesToDiscard -= localBytesToDiscard;
|
||||||
|
this.bytesToDiscard = bytesToDiscard;
|
||||||
|
|
||||||
|
failIfNecessary(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void failOnNegativeLengthField(ByteBuf in, long frameLength, int lengthFieldEndOffset) {
|
||||||
|
in.skipBytes(lengthFieldEndOffset);
|
||||||
|
throw new CorruptedFrameException(
|
||||||
|
"negative pre-adjustment length field: " + frameLength);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void failOnFrameLengthLessThanLengthFieldEndOffset(ByteBuf in,
|
||||||
|
long frameLength,
|
||||||
|
int lengthFieldEndOffset) {
|
||||||
|
in.skipBytes(lengthFieldEndOffset);
|
||||||
|
throw new CorruptedFrameException(
|
||||||
|
"Adjusted frame length (" + frameLength + ") is less " +
|
||||||
|
"than lengthFieldEndOffset: " + lengthFieldEndOffset);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void exceededFrameLength(ByteBuf in, long frameLength) {
|
||||||
|
long discard = frameLength - in.readableBytes();
|
||||||
|
tooLongFrameLength = frameLength;
|
||||||
|
|
||||||
|
if (discard < 0) {
|
||||||
|
// buffer contains more bytes then the frameLength so we can discard all now
|
||||||
|
in.skipBytes((int) frameLength);
|
||||||
|
} else {
|
||||||
|
// Enter the discard mode and discard everything received so far.
|
||||||
|
discardingTooLongFrame = true;
|
||||||
|
bytesToDiscard = discard;
|
||||||
|
in.skipBytes(in.readableBytes());
|
||||||
|
}
|
||||||
|
failIfNecessary(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void failOnFrameLengthLessThanInitialBytesToStrip(ByteBuf in,
|
||||||
|
long frameLength,
|
||||||
|
int initialBytesToStrip) {
|
||||||
|
in.skipBytes((int) frameLength);
|
||||||
|
throw new CorruptedFrameException(
|
||||||
|
"Adjusted frame length (" + frameLength + ") is less " +
|
||||||
|
"than initialBytesToStrip: " + initialBytesToStrip);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a frame out of the {@link ByteBuf} and return it.
|
* Create a frame out of the {@link ByteBuf} and return it.
|
||||||
*
|
*
|
||||||
@ -356,13 +406,7 @@ public class LengthFieldBasedFrameDecoder extends ByteToMessageDecoder {
|
|||||||
*/
|
*/
|
||||||
protected Object decode(ChannelHandlerContext ctx, ByteBuf in) throws Exception {
|
protected Object decode(ChannelHandlerContext ctx, ByteBuf in) throws Exception {
|
||||||
if (discardingTooLongFrame) {
|
if (discardingTooLongFrame) {
|
||||||
long bytesToDiscard = this.bytesToDiscard;
|
discardingTooLongFrame(in);
|
||||||
int localBytesToDiscard = (int) Math.min(bytesToDiscard, in.readableBytes());
|
|
||||||
in.skipBytes(localBytesToDiscard);
|
|
||||||
bytesToDiscard -= localBytesToDiscard;
|
|
||||||
this.bytesToDiscard = bytesToDiscard;
|
|
||||||
|
|
||||||
failIfNecessary(false);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (in.readableBytes() < lengthFieldEndOffset) {
|
if (in.readableBytes() < lengthFieldEndOffset) {
|
||||||
@ -373,34 +417,17 @@ public class LengthFieldBasedFrameDecoder extends ByteToMessageDecoder {
|
|||||||
long frameLength = getUnadjustedFrameLength(in, actualLengthFieldOffset, lengthFieldLength, byteOrder);
|
long frameLength = getUnadjustedFrameLength(in, actualLengthFieldOffset, lengthFieldLength, byteOrder);
|
||||||
|
|
||||||
if (frameLength < 0) {
|
if (frameLength < 0) {
|
||||||
in.skipBytes(lengthFieldEndOffset);
|
failOnNegativeLengthField(in, frameLength, lengthFieldEndOffset);
|
||||||
throw new CorruptedFrameException(
|
|
||||||
"negative pre-adjustment length field: " + frameLength);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
frameLength += lengthAdjustment + lengthFieldEndOffset;
|
frameLength += lengthAdjustment + lengthFieldEndOffset;
|
||||||
|
|
||||||
if (frameLength < lengthFieldEndOffset) {
|
if (frameLength < lengthFieldEndOffset) {
|
||||||
in.skipBytes(lengthFieldEndOffset);
|
failOnFrameLengthLessThanLengthFieldEndOffset(in, frameLength, lengthFieldEndOffset);
|
||||||
throw new CorruptedFrameException(
|
|
||||||
"Adjusted frame length (" + frameLength + ") is less " +
|
|
||||||
"than lengthFieldEndOffset: " + lengthFieldEndOffset);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (frameLength > maxFrameLength) {
|
if (frameLength > maxFrameLength) {
|
||||||
long discard = frameLength - in.readableBytes();
|
exceededFrameLength(in, frameLength);
|
||||||
tooLongFrameLength = frameLength;
|
|
||||||
|
|
||||||
if (discard < 0) {
|
|
||||||
// buffer contains more bytes then the frameLength so we can discard all now
|
|
||||||
in.skipBytes((int) frameLength);
|
|
||||||
} else {
|
|
||||||
// Enter the discard mode and discard everything received so far.
|
|
||||||
discardingTooLongFrame = true;
|
|
||||||
bytesToDiscard = discard;
|
|
||||||
in.skipBytes(in.readableBytes());
|
|
||||||
}
|
|
||||||
failIfNecessary(true);
|
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -411,10 +438,7 @@ public class LengthFieldBasedFrameDecoder extends ByteToMessageDecoder {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (initialBytesToStrip > frameLengthInt) {
|
if (initialBytesToStrip > frameLengthInt) {
|
||||||
in.skipBytes(frameLengthInt);
|
failOnFrameLengthLessThanInitialBytesToStrip(in, frameLength, initialBytesToStrip);
|
||||||
throw new CorruptedFrameException(
|
|
||||||
"Adjusted frame length (" + frameLength + ") is less " +
|
|
||||||
"than initialBytesToStrip: " + initialBytesToStrip);
|
|
||||||
}
|
}
|
||||||
in.skipBytes(initialBytesToStrip);
|
in.skipBytes(initialBytesToStrip);
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user