[#4435] Always invoke the actual deregisteration later in the EventLoop.
Motivation: As a user may call deregister() from within any method while doing processing in the ChannelPipeline, we need to ensure we do the actual deregister operation later. This is needed as for example, we may be in the ByteToMessageDecoder.callDecode(...) method and so still try to do processing in the old EventLoop while the user already registered the Channel to a new EventLoop. Without delay, the deregister operation this could lead to have a handler invoked by different EventLoop and so threads. Modifications: Ensure the actual deregister will be done later on and not directly when invoked. Result: Calling deregister() within ByteToMessageDecoder.decode(..) is safe.
This commit is contained in:
parent
9b62af67f9
commit
15162202fb
@ -618,16 +618,7 @@ public abstract class AbstractChannel extends DefaultAttributeMap implements Cha
|
||||
}
|
||||
|
||||
private void fireChannelInactiveAndDeregister(final boolean wasActive) {
|
||||
if (wasActive && !isActive()) {
|
||||
invokeLater(new OneTimeTask() {
|
||||
@Override
|
||||
public void run() {
|
||||
pipeline.fireChannelInactive();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
deregister(voidPromise());
|
||||
deregister(voidPromise(), wasActive && !isActive());
|
||||
}
|
||||
|
||||
@Override
|
||||
@ -641,6 +632,10 @@ public abstract class AbstractChannel extends DefaultAttributeMap implements Cha
|
||||
|
||||
@Override
|
||||
public final void deregister(final ChannelPromise promise) {
|
||||
deregister(promise, false);
|
||||
}
|
||||
|
||||
private void deregister(final ChannelPromise promise, final boolean fireChannelInactive) {
|
||||
if (!promise.setUncancellable()) {
|
||||
return;
|
||||
}
|
||||
@ -650,27 +645,38 @@ public abstract class AbstractChannel extends DefaultAttributeMap implements Cha
|
||||
return;
|
||||
}
|
||||
|
||||
// As a user may call deregister() from within any method while doing processing in the ChannelPipeline,
|
||||
// we need to ensure we do the actual deregister operation later. This is needed as for example,
|
||||
// we may be in the ByteToMessageDecoder.callDecode(...) method and so still try to do processing in
|
||||
// the old EventLoop while the user already registered the Channel to a new EventLoop. Without delay,
|
||||
// the deregister operation this could lead to have a handler invoked by different EventLoop and so
|
||||
// threads.
|
||||
//
|
||||
// See:
|
||||
// https://github.com/netty/netty/issues/4435
|
||||
invokeLater(new OneTimeTask() {
|
||||
@Override
|
||||
public void run() {
|
||||
try {
|
||||
doDeregister();
|
||||
} catch (Throwable t) {
|
||||
logger.warn("Unexpected exception occurred while deregistering a channel.", t);
|
||||
} finally {
|
||||
if (registered) {
|
||||
registered = false;
|
||||
invokeLater(new OneTimeTask() {
|
||||
@Override
|
||||
public void run() {
|
||||
pipeline.fireChannelUnregistered();
|
||||
if (fireChannelInactive) {
|
||||
pipeline.fireChannelInactive();
|
||||
}
|
||||
});
|
||||
safeSetSuccess(promise);
|
||||
} else {
|
||||
// Some transports like local and AIO does not allow the deregistration of
|
||||
// an open channel. Their doDeregister() calls close(). Consequently,
|
||||
// close() calls deregister() again - no need to fire channelUnregistered.
|
||||
// close() calls deregister() again - no need to fire channelUnregistered, so check
|
||||
// if it was registered.
|
||||
if (registered) {
|
||||
registered = false;
|
||||
pipeline.fireChannelUnregistered();
|
||||
}
|
||||
safeSetSuccess(promise);
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@Override
|
||||
|
Loading…
x
Reference in New Issue
Block a user