Not attempt to read from fd when channel is closed during read loop. Related to [#5031]
Motivation: We need to break out of the read loop for two reasons: - If the input was shutdown in between (which may be the case when the user did it in the fireChannelRead(...) method we should not try to read again to not produce any miss-leading exceptions. - If the user closes the channel we need to ensure we not try to read from it again as the filedescriptor may be re-used already by the OS if the system is handling a lot of concurrent connections and so needs a lot of filedescriptors. If not do this we risk reading data from a filedescriptor that belongs to another socket then the socket that was "wrapped" by this Channel implementation. Modification: Break the reading loop if the input was shutdown from within the channelRead(...) method. Result: No more meaningless exceptions and no risk to read data from wrong socket after the original was closed. Conflicts: transport-native-epoll/src/main/java/io/netty/channel/epoll/AbstractEpollStreamChannel.java
This commit is contained in:
parent
ea4a8e339c
commit
900db0acff
@ -899,6 +899,21 @@ public abstract class AbstractEpollStreamChannel extends AbstractEpollChannel im
|
||||
// pending data
|
||||
break;
|
||||
}
|
||||
|
||||
if (fd().isInputShutdown()) {
|
||||
// We need to do this for two reasons:
|
||||
//
|
||||
// - If the input was shutdown in between (which may be the case when the user did it in the
|
||||
// fireChannelRead(...) method we should not try to read again to not produce any
|
||||
// miss-leading exceptions.
|
||||
//
|
||||
// - If the user closes the channel we need to ensure we not try to read from it again as
|
||||
// the filedescriptor may be re-used already by the OS if the system is handling a lot of
|
||||
// concurrent connections and so needs a lot of filedescriptors. If not do this we risk
|
||||
// reading data from a filedescriptor that belongs to another socket then the socket that
|
||||
// was "wrapped" by this Channel implementation.
|
||||
break;
|
||||
}
|
||||
} while (++ messages < maxMessagesPerRead || isRdHup());
|
||||
|
||||
pipeline.fireChannelReadComplete();
|
||||
|
@ -190,10 +190,9 @@ public class EpollSocketChannelTest {
|
||||
private static class ExceptionHandler extends ChannelInboundHandlerAdapter {
|
||||
final AtomicLong count = new AtomicLong();
|
||||
/**
|
||||
* We expect to get 2 calls to {@link #exceptionCaught(ChannelHandlerContext, Throwable)}.
|
||||
* 1 call from BuggyChannelHandler and 1 from closing the channel in this class.
|
||||
* We expect to get 1 call to {@link #exceptionCaught(ChannelHandlerContext, Throwable)}.
|
||||
*/
|
||||
final CountDownLatch latch1 = new CountDownLatch(2);
|
||||
final CountDownLatch latch1 = new CountDownLatch(1);
|
||||
final CountDownLatch latch2 = new CountDownLatch(1);
|
||||
|
||||
@Override
|
||||
@ -203,7 +202,7 @@ public class EpollSocketChannelTest {
|
||||
} else {
|
||||
latch2.countDown();
|
||||
}
|
||||
// This is expected to throw an exception!
|
||||
// This should not throw any exception.
|
||||
ctx.close();
|
||||
}
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user