Fix race-condition when closing a NioSocketChannel or EpollSocketChannel

Motivation:

Fix a race-condition when closing NioSocketChannel or EpollSocketChannel while try to detect if a close executor should be used and the underlying socket was already closed. This could lead to an exception that then leave the channel / in an invalid state and so could lead to side-effects like heavy CPU usage.

Modifications:

Catch possible socket exception while try to get the SO_LINGER options from the underlying socket.

Result:

No more race-condition when closing the channel is possible with bad side-effects.
This commit is contained in:
Norman Maurer 2015-11-24 04:54:39 +01:00
parent 4154ea08f9
commit 61b5792340
2 changed files with 19 additions and 6 deletions

View File

@ -219,11 +219,17 @@ public final class EpollSocketChannel extends AbstractEpollStreamChannel impleme
private final class EpollSocketChannelUnsafe extends EpollStreamUnsafe { private final class EpollSocketChannelUnsafe extends EpollStreamUnsafe {
@Override @Override
protected Executor closeExecutor() { protected Executor closeExecutor() {
try {
// Check isOpen() first as otherwise it will throw a RuntimeException // Check isOpen() first as otherwise it will throw a RuntimeException
// when call getSoLinger() as the fd is not valid anymore. // when call getSoLinger() as the fd is not valid anymore.
if (isOpen() && config().getSoLinger() > 0) { if (isOpen() && config().getSoLinger() > 0) {
return GlobalEventExecutor.INSTANCE; return GlobalEventExecutor.INSTANCE;
} }
} catch (Throwable ignore) {
// Ignore the error as the underlying channel may be closed in the meantime and so
// getSoLinger() may produce an exception. In this case we just return null.
// See https://github.com/netty/netty/issues/4449
}
return null; return null;
} }
} }

View File

@ -35,6 +35,7 @@ import java.io.IOException;
import java.net.InetSocketAddress; import java.net.InetSocketAddress;
import java.net.Socket; import java.net.Socket;
import java.net.SocketAddress; import java.net.SocketAddress;
import java.net.SocketException;
import java.nio.ByteBuffer; import java.nio.ByteBuffer;
import java.nio.channels.SelectionKey; import java.nio.channels.SelectionKey;
import java.nio.channels.SocketChannel; import java.nio.channels.SocketChannel;
@ -333,9 +334,15 @@ public class NioSocketChannel extends AbstractNioByteChannel implements io.netty
private final class NioSocketChannelUnsafe extends NioByteUnsafe { private final class NioSocketChannelUnsafe extends NioByteUnsafe {
@Override @Override
protected Executor closeExecutor() { protected Executor closeExecutor() {
try {
if (javaChannel().isOpen() && config().getSoLinger() > 0) { if (javaChannel().isOpen() && config().getSoLinger() > 0) {
return GlobalEventExecutor.INSTANCE; return GlobalEventExecutor.INSTANCE;
} }
} catch (Throwable ignore) {
// Ignore the error as the underlying channel may be closed in the meantime and so
// getSoLinger() may produce an exception. In this case we just return null.
// See https://github.com/netty/netty/issues/4449
}
return null; return null;
} }
} }