Make sure we always flush window update frames in AbstractHttp2StreamChannel (#10075)
Motivation: Under certain read patters the AbstractHttp2StreamChannel can fail to flush, resulting in flow window starvation. Modifications: - Ensure we flush if we exit the `doBeginRead()` method. - Account for the Http2FrameCodec always synchronously finishing writes of window update frames. Result: Fixes #10072
This commit is contained in:
parent
2559e163ca
commit
7eadca1eeb
@ -553,10 +553,7 @@ abstract class AbstractHttp2StreamChannel extends DefaultAttributeMap implements
|
|||||||
// read (unknown, reset) and the trade off is less conditionals for the hot path (headers/data) at the
|
// read (unknown, reset) and the trade off is less conditionals for the hot path (headers/data) at the
|
||||||
// cost of additional readComplete notifications on the rare path.
|
// cost of additional readComplete notifications on the rare path.
|
||||||
if (allocHandle.continueReading()) {
|
if (allocHandle.continueReading()) {
|
||||||
if (!readCompletePending) {
|
maybeAddChannelToReadCompletePendingQueue();
|
||||||
readCompletePending = true;
|
|
||||||
addChannelToReadCompletePendingQueue();
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
unsafe.notifyReadComplete(allocHandle, true);
|
unsafe.notifyReadComplete(allocHandle, true);
|
||||||
}
|
}
|
||||||
@ -797,6 +794,9 @@ abstract class AbstractHttp2StreamChannel extends DefaultAttributeMap implements
|
|||||||
if (readEOS) {
|
if (readEOS) {
|
||||||
unsafe.closeForcibly();
|
unsafe.closeForcibly();
|
||||||
}
|
}
|
||||||
|
// We need to double check that there is nothing left to flush such as a
|
||||||
|
// window update frame.
|
||||||
|
flush();
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
final RecvByteBufAllocator.Handle allocHandle = recvBufAllocHandle();
|
final RecvByteBufAllocator.Handle allocHandle = recvBufAllocHandle();
|
||||||
@ -812,10 +812,7 @@ abstract class AbstractHttp2StreamChannel extends DefaultAttributeMap implements
|
|||||||
// currently reading it is possible that more frames will be delivered to this child channel. In
|
// currently reading it is possible that more frames will be delivered to this child channel. In
|
||||||
// the case that this child channel still wants to read we delay the channelReadComplete on this
|
// the case that this child channel still wants to read we delay the channelReadComplete on this
|
||||||
// child channel until the parent is done reading.
|
// child channel until the parent is done reading.
|
||||||
if (!readCompletePending) {
|
maybeAddChannelToReadCompletePendingQueue();
|
||||||
readCompletePending = true;
|
|
||||||
addChannelToReadCompletePendingQueue();
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
notifyReadComplete(allocHandle, true);
|
notifyReadComplete(allocHandle, true);
|
||||||
}
|
}
|
||||||
@ -831,6 +828,10 @@ abstract class AbstractHttp2StreamChannel extends DefaultAttributeMap implements
|
|||||||
int bytes = flowControlledBytes;
|
int bytes = flowControlledBytes;
|
||||||
flowControlledBytes = 0;
|
flowControlledBytes = 0;
|
||||||
ChannelFuture future = write0(parentContext(), new DefaultHttp2WindowUpdateFrame(bytes).stream(stream));
|
ChannelFuture future = write0(parentContext(), new DefaultHttp2WindowUpdateFrame(bytes).stream(stream));
|
||||||
|
// window update frames are commonly swallowed by the Http2FrameCodec and the promise is synchronously
|
||||||
|
// completed but the flow controller _may_ have generated a wire level WINDOW_UPDATE. Therefore we need,
|
||||||
|
// to assume there was a write done that needs to be flushed or we risk flow control starvation.
|
||||||
|
writeDoneAndNoFlush = true;
|
||||||
// Add a listener which will notify and teardown the stream
|
// Add a listener which will notify and teardown the stream
|
||||||
// when a window update fails if needed or check the result of the future directly if it was completed
|
// when a window update fails if needed or check the result of the future directly if it was completed
|
||||||
// already.
|
// already.
|
||||||
@ -839,7 +840,6 @@ abstract class AbstractHttp2StreamChannel extends DefaultAttributeMap implements
|
|||||||
windowUpdateFrameWriteComplete(future, AbstractHttp2StreamChannel.this);
|
windowUpdateFrameWriteComplete(future, AbstractHttp2StreamChannel.this);
|
||||||
} else {
|
} else {
|
||||||
future.addListener(windowUpdateFrameWriteListener);
|
future.addListener(windowUpdateFrameWriteListener);
|
||||||
writeDoneAndNoFlush = true;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -914,55 +914,57 @@ abstract class AbstractHttp2StreamChannel extends DefaultAttributeMap implements
|
|||||||
try {
|
try {
|
||||||
if (msg instanceof Http2StreamFrame) {
|
if (msg instanceof Http2StreamFrame) {
|
||||||
Http2StreamFrame frame = validateStreamFrame((Http2StreamFrame) msg).stream(stream());
|
Http2StreamFrame frame = validateStreamFrame((Http2StreamFrame) msg).stream(stream());
|
||||||
if (!firstFrameWritten && !isStreamIdValid(stream().id())) {
|
writeHttp2StreamFrame(frame, promise);
|
||||||
if (!(frame instanceof Http2HeadersFrame)) {
|
|
||||||
ReferenceCountUtil.release(frame);
|
|
||||||
promise.setFailure(
|
|
||||||
new IllegalArgumentException("The first frame must be a headers frame. Was: "
|
|
||||||
+ frame.name()));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
firstFrameWritten = true;
|
|
||||||
ChannelFuture f = write0(parentContext(), frame);
|
|
||||||
if (f.isDone()) {
|
|
||||||
firstWriteComplete(f, promise);
|
|
||||||
} else {
|
|
||||||
final long bytes = FlowControlledFrameSizeEstimator.HANDLE_INSTANCE.size(msg);
|
|
||||||
incrementPendingOutboundBytes(bytes, false);
|
|
||||||
f.addListener((ChannelFutureListener) future -> {
|
|
||||||
firstWriteComplete(future, promise);
|
|
||||||
decrementPendingOutboundBytes(bytes, false);
|
|
||||||
});
|
|
||||||
writeDoneAndNoFlush = true;
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
String msgStr = msg.toString();
|
String msgStr = msg.toString();
|
||||||
ReferenceCountUtil.release(msg);
|
ReferenceCountUtil.release(msg);
|
||||||
promise.setFailure(new IllegalArgumentException(
|
promise.setFailure(new IllegalArgumentException(
|
||||||
"Message must be an " + StringUtil.simpleClassName(Http2StreamFrame.class) +
|
"Message must be an " + StringUtil.simpleClassName(Http2StreamFrame.class) +
|
||||||
": " + msgStr));
|
": " + msgStr));
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
ChannelFuture f = write0(parentContext(), msg);
|
|
||||||
if (f.isDone()) {
|
|
||||||
writeComplete(f, promise);
|
|
||||||
} else {
|
|
||||||
final long bytes = FlowControlledFrameSizeEstimator.HANDLE_INSTANCE.size(msg);
|
|
||||||
incrementPendingOutboundBytes(bytes, false);
|
|
||||||
f.addListener((ChannelFutureListener) future -> {
|
|
||||||
writeComplete(future, promise);
|
|
||||||
decrementPendingOutboundBytes(bytes, false);
|
|
||||||
});
|
|
||||||
writeDoneAndNoFlush = true;
|
|
||||||
}
|
}
|
||||||
} catch (Throwable t) {
|
} catch (Throwable t) {
|
||||||
promise.tryFailure(t);
|
promise.tryFailure(t);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void writeHttp2StreamFrame(Http2StreamFrame frame, final ChannelPromise promise) {
|
||||||
|
if (!firstFrameWritten && !isStreamIdValid(stream().id()) && !(frame instanceof Http2HeadersFrame)) {
|
||||||
|
ReferenceCountUtil.release(frame);
|
||||||
|
promise.setFailure(
|
||||||
|
new IllegalArgumentException("The first frame must be a headers frame. Was: "
|
||||||
|
+ frame.name()));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
final boolean firstWrite;
|
||||||
|
if (firstFrameWritten) {
|
||||||
|
firstWrite = false;
|
||||||
|
} else {
|
||||||
|
firstWrite = firstFrameWritten = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
ChannelFuture f = write0(parentContext(), frame);
|
||||||
|
if (f.isDone()) {
|
||||||
|
if (firstWrite) {
|
||||||
|
firstWriteComplete(f, promise);
|
||||||
|
} else {
|
||||||
|
writeComplete(f, promise);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
final long bytes = FlowControlledFrameSizeEstimator.HANDLE_INSTANCE.size(frame);
|
||||||
|
incrementPendingOutboundBytes(bytes, false);
|
||||||
|
f.addListener((ChannelFutureListener) future -> {
|
||||||
|
if (firstWrite) {
|
||||||
|
firstWriteComplete(future, promise);
|
||||||
|
} else {
|
||||||
|
writeComplete(future, promise);
|
||||||
|
}
|
||||||
|
decrementPendingOutboundBytes(bytes, false);
|
||||||
|
});
|
||||||
|
writeDoneAndNoFlush = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private void firstWriteComplete(ChannelFuture future, ChannelPromise promise) {
|
private void firstWriteComplete(ChannelFuture future, ChannelPromise promise) {
|
||||||
Throwable cause = future.cause();
|
Throwable cause = future.cause();
|
||||||
if (cause == null) {
|
if (cause == null) {
|
||||||
@ -1072,6 +1074,13 @@ abstract class AbstractHttp2StreamChannel extends DefaultAttributeMap implements
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void maybeAddChannelToReadCompletePendingQueue() {
|
||||||
|
if (!readCompletePending) {
|
||||||
|
readCompletePending = true;
|
||||||
|
addChannelToReadCompletePendingQueue();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
protected void flush0(ChannelHandlerContext ctx) {
|
protected void flush0(ChannelHandlerContext ctx) {
|
||||||
ctx.flush();
|
ctx.flush();
|
||||||
}
|
}
|
||||||
|
@ -1115,6 +1115,70 @@ public abstract class Http2MultiplexTest<C extends Http2FrameCodec> {
|
|||||||
verifyFramesMultiplexedToCorrectChannel(childChannel, inboundHandler, 3);
|
verifyFramesMultiplexedToCorrectChannel(childChannel, inboundHandler, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static final class FlushSniffer implements ChannelHandler {
|
||||||
|
|
||||||
|
private boolean didFlush;
|
||||||
|
|
||||||
|
public boolean checkFlush() {
|
||||||
|
boolean r = didFlush;
|
||||||
|
didFlush = false;
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void flush(ChannelHandlerContext ctx) throws Exception {
|
||||||
|
didFlush = true;
|
||||||
|
ctx.flush();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void windowUpdatesAreFlushed() {
|
||||||
|
LastInboundHandler inboundHandler = new LastInboundHandler();
|
||||||
|
FlushSniffer flushSniffer = new FlushSniffer();
|
||||||
|
parentChannel.pipeline().addFirst(flushSniffer);
|
||||||
|
|
||||||
|
Http2StreamChannel childChannel = newInboundStream(3, false, inboundHandler);
|
||||||
|
assertTrue(childChannel.config().isAutoRead());
|
||||||
|
childChannel.config().setAutoRead(false);
|
||||||
|
assertFalse(childChannel.config().isAutoRead());
|
||||||
|
|
||||||
|
Http2HeadersFrame headersFrame = inboundHandler.readInbound();
|
||||||
|
assertNotNull(headersFrame);
|
||||||
|
|
||||||
|
assertTrue(flushSniffer.checkFlush());
|
||||||
|
|
||||||
|
// Write some bytes to get the channel into the idle state with buffered data and also verify we
|
||||||
|
// do not dispatch it until we receive a read() call.
|
||||||
|
frameInboundWriter.writeInboundData(childChannel.stream().id(), bb(16 * 1024), 0, false);
|
||||||
|
frameInboundWriter.writeInboundData(childChannel.stream().id(), bb(16 * 1024), 0, false);
|
||||||
|
assertTrue(flushSniffer.checkFlush());
|
||||||
|
|
||||||
|
verify(frameWriter, never())
|
||||||
|
.writeWindowUpdate(any(ChannelHandlerContext.class), anyInt(), anyInt(), anyChannelPromise());
|
||||||
|
// only the first one was read because it was legacy auto-read behavior.
|
||||||
|
verifyFramesMultiplexedToCorrectChannel(childChannel, inboundHandler, 1);
|
||||||
|
assertFalse(flushSniffer.checkFlush());
|
||||||
|
|
||||||
|
// Trigger a read of the second frame.
|
||||||
|
childChannel.read();
|
||||||
|
verifyFramesMultiplexedToCorrectChannel(childChannel, inboundHandler, 1);
|
||||||
|
// We expect a flush here because the StreamChannel will flush the smaller increment but the
|
||||||
|
// connection will collect the bytes and decide not to send a wire level frame until more are consumed.
|
||||||
|
assertTrue(flushSniffer.checkFlush());
|
||||||
|
verify(frameWriter, never())
|
||||||
|
.writeWindowUpdate(any(ChannelHandlerContext.class), anyInt(), anyInt(), anyChannelPromise());
|
||||||
|
|
||||||
|
// Call read one more time which should trigger the writing of the flow control update.
|
||||||
|
childChannel.read();
|
||||||
|
verify(frameWriter)
|
||||||
|
.writeWindowUpdate(any(ChannelHandlerContext.class), eq(0), eq(32 * 1024), anyChannelPromise());
|
||||||
|
verify(frameWriter)
|
||||||
|
.writeWindowUpdate(any(ChannelHandlerContext.class), eq(childChannel.stream().id()),
|
||||||
|
eq(32 * 1024), anyChannelPromise());
|
||||||
|
assertTrue(flushSniffer.checkFlush());
|
||||||
|
}
|
||||||
|
|
||||||
private static void verifyFramesMultiplexedToCorrectChannel(Http2StreamChannel streamChannel,
|
private static void verifyFramesMultiplexedToCorrectChannel(Http2StreamChannel streamChannel,
|
||||||
LastInboundHandler inboundHandler,
|
LastInboundHandler inboundHandler,
|
||||||
int numFrames) {
|
int numFrames) {
|
||||||
|
@ -640,6 +640,10 @@ public final class Http2TestUtil {
|
|||||||
return ByteBufUtil.writeUtf8(UnpooledByteBufAllocator.DEFAULT, s);
|
return ByteBufUtil.writeUtf8(UnpooledByteBufAllocator.DEFAULT, s);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static ByteBuf bb(int size) {
|
||||||
|
return UnpooledByteBufAllocator.DEFAULT.buffer().writeZero(size);
|
||||||
|
}
|
||||||
|
|
||||||
static void assertEqualsAndRelease(Http2Frame expected, Http2Frame actual) {
|
static void assertEqualsAndRelease(Http2Frame expected, Http2Frame actual) {
|
||||||
try {
|
try {
|
||||||
assertEquals(expected, actual);
|
assertEquals(expected, actual);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user