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:
Norman Maurer 2016-03-24 19:37:43 +01:00
parent ea4a8e339c
commit 900db0acff
2 changed files with 18 additions and 4 deletions

View File

@ -899,6 +899,21 @@ public abstract class AbstractEpollStreamChannel extends AbstractEpollChannel im
// pending data // pending data
break; 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()); } while (++ messages < maxMessagesPerRead || isRdHup());
pipeline.fireChannelReadComplete(); pipeline.fireChannelReadComplete();

View File

@ -190,10 +190,9 @@ public class EpollSocketChannelTest {
private static class ExceptionHandler extends ChannelInboundHandlerAdapter { private static class ExceptionHandler extends ChannelInboundHandlerAdapter {
final AtomicLong count = new AtomicLong(); final AtomicLong count = new AtomicLong();
/** /**
* We expect to get 2 calls to {@link #exceptionCaught(ChannelHandlerContext, Throwable)}. * We expect to get 1 call to {@link #exceptionCaught(ChannelHandlerContext, Throwable)}.
* 1 call from BuggyChannelHandler and 1 from closing the channel in this class.
*/ */
final CountDownLatch latch1 = new CountDownLatch(2); final CountDownLatch latch1 = new CountDownLatch(1);
final CountDownLatch latch2 = new CountDownLatch(1); final CountDownLatch latch2 = new CountDownLatch(1);
@Override @Override
@ -203,7 +202,7 @@ public class EpollSocketChannelTest {
} else { } else {
latch2.countDown(); latch2.countDown();
} }
// This is expected to throw an exception! // This should not throw any exception.
ctx.close(); ctx.close();
} }
} }