Merge pull request #210 from netty/threading_fix
Merge in fix for threading (related to #140 and #187). This also includes the new feature that allow to submit a Runnable that gets executed later in the io thread.
This commit is contained in:
commit
8579f09c59
@ -22,9 +22,11 @@ import java.util.ConcurrentModificationException;
|
|||||||
import java.util.LinkedList;
|
import java.util.LinkedList;
|
||||||
import java.util.Queue;
|
import java.util.Queue;
|
||||||
|
|
||||||
|
import io.netty.channel.Channels;
|
||||||
import io.netty.buffer.ChannelBufferFactory;
|
import io.netty.buffer.ChannelBufferFactory;
|
||||||
import io.netty.channel.Channel;
|
import io.netty.channel.Channel;
|
||||||
import io.netty.channel.ChannelEvent;
|
import io.netty.channel.ChannelEvent;
|
||||||
|
import io.netty.channel.ChannelFuture;
|
||||||
import io.netty.channel.ChannelHandler;
|
import io.netty.channel.ChannelHandler;
|
||||||
import io.netty.channel.ChannelHandlerContext;
|
import io.netty.channel.ChannelHandlerContext;
|
||||||
import io.netty.channel.ChannelPipeline;
|
import io.netty.channel.ChannelPipeline;
|
||||||
@ -224,6 +226,16 @@ abstract class AbstractCodecEmbedder<E> implements CodecEmbedder<E> {
|
|||||||
|
|
||||||
throw new CodecEmbedderException(actualCause);
|
throw new CodecEmbedderException(actualCause);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ChannelFuture execute(ChannelPipeline pipeline, Runnable task) {
|
||||||
|
try {
|
||||||
|
task.run();
|
||||||
|
return Channels.succeededFuture(pipeline.getChannel());
|
||||||
|
} catch (Throwable t) {
|
||||||
|
return Channels.failedFuture(pipeline.getChannel(), t);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static final class EmbeddedChannelPipeline extends DefaultChannelPipeline {
|
private static final class EmbeddedChannelPipeline extends DefaultChannelPipeline {
|
||||||
|
@ -90,7 +90,7 @@ public class ChunkedWriteHandler implements ChannelUpstreamHandler, ChannelDowns
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
flush(ctx);
|
flush(ctx, false);
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
if (logger.isWarnEnabled()) {
|
if (logger.isWarnEnabled()) {
|
||||||
logger.warn("Unexpected exception while sending chunks.", e);
|
logger.warn("Unexpected exception while sending chunks.", e);
|
||||||
@ -112,10 +112,10 @@ public class ChunkedWriteHandler implements ChannelUpstreamHandler, ChannelDowns
|
|||||||
final Channel channel = ctx.getChannel();
|
final Channel channel = ctx.getChannel();
|
||||||
if (channel.isWritable()) {
|
if (channel.isWritable()) {
|
||||||
this.ctx = ctx;
|
this.ctx = ctx;
|
||||||
flush(ctx);
|
flush(ctx, false);
|
||||||
} else if (!channel.isConnected()) {
|
} else if (!channel.isConnected()) {
|
||||||
this.ctx = ctx;
|
this.ctx = ctx;
|
||||||
discard(ctx);
|
discard(ctx, false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -127,12 +127,12 @@ public class ChunkedWriteHandler implements ChannelUpstreamHandler, ChannelDowns
|
|||||||
switch (cse.getState()) {
|
switch (cse.getState()) {
|
||||||
case INTEREST_OPS:
|
case INTEREST_OPS:
|
||||||
// Continue writing when the channel becomes writable.
|
// Continue writing when the channel becomes writable.
|
||||||
flush(ctx);
|
flush(ctx, true);
|
||||||
break;
|
break;
|
||||||
case OPEN:
|
case OPEN:
|
||||||
if (!Boolean.TRUE.equals(cse.getValue())) {
|
if (!Boolean.TRUE.equals(cse.getValue())) {
|
||||||
// Fail all pending writes
|
// Fail all pending writes
|
||||||
discard(ctx);
|
discard(ctx, true);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -140,7 +140,7 @@ public class ChunkedWriteHandler implements ChannelUpstreamHandler, ChannelDowns
|
|||||||
ctx.sendUpstream(e);
|
ctx.sendUpstream(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void discard(ChannelHandlerContext ctx) {
|
private void discard(ChannelHandlerContext ctx, boolean fireNow) {
|
||||||
ClosedChannelException cause = null;
|
ClosedChannelException cause = null;
|
||||||
boolean fireExceptionCaught = false;
|
boolean fireExceptionCaught = false;
|
||||||
|
|
||||||
@ -175,14 +175,18 @@ public class ChunkedWriteHandler implements ChannelUpstreamHandler, ChannelDowns
|
|||||||
|
|
||||||
|
|
||||||
if (fireExceptionCaught) {
|
if (fireExceptionCaught) {
|
||||||
Channels.fireExceptionCaught(ctx.getChannel(), cause);
|
if (fireNow) {
|
||||||
|
fireExceptionCaught(ctx.getChannel(), cause);
|
||||||
|
} else {
|
||||||
|
fireExceptionCaughtLater(ctx.getChannel(), cause);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private synchronized void flush(ChannelHandlerContext ctx) throws Exception {
|
private synchronized void flush(ChannelHandlerContext ctx, boolean fireNow) throws Exception {
|
||||||
final Channel channel = ctx.getChannel();
|
final Channel channel = ctx.getChannel();
|
||||||
if (!channel.isConnected()) {
|
if (!channel.isConnected()) {
|
||||||
discard(ctx);
|
discard(ctx, fireNow);
|
||||||
}
|
}
|
||||||
|
|
||||||
while (channel.isWritable()) {
|
while (channel.isWritable()) {
|
||||||
@ -220,7 +224,11 @@ public class ChunkedWriteHandler implements ChannelUpstreamHandler, ChannelDowns
|
|||||||
this.currentEvent = null;
|
this.currentEvent = null;
|
||||||
|
|
||||||
currentEvent.getFuture().setFailure(t);
|
currentEvent.getFuture().setFailure(t);
|
||||||
fireExceptionCaught(ctx, t);
|
if (fireNow) {
|
||||||
|
fireExceptionCaught(ctx, t);
|
||||||
|
} else {
|
||||||
|
fireExceptionCaughtLater(ctx.getChannel(), t);
|
||||||
|
}
|
||||||
|
|
||||||
closeInput(chunks);
|
closeInput(chunks);
|
||||||
break;
|
break;
|
||||||
@ -262,7 +270,7 @@ public class ChunkedWriteHandler implements ChannelUpstreamHandler, ChannelDowns
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!channel.isConnected()) {
|
if (!channel.isConnected()) {
|
||||||
discard(ctx);
|
discard(ctx, fireNow);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -154,7 +154,7 @@ public class WriteTimeoutHandler extends SimpleChannelDownstreamHandler
|
|||||||
}
|
}
|
||||||
|
|
||||||
protected void writeTimedOut(ChannelHandlerContext ctx) throws Exception {
|
protected void writeTimedOut(ChannelHandlerContext ctx) throws Exception {
|
||||||
Channels.fireExceptionCaught(ctx, EXCEPTION);
|
Channels.fireExceptionCaughtLater(ctx.getChannel(), EXCEPTION);
|
||||||
}
|
}
|
||||||
|
|
||||||
private final class WriteTimeoutTask implements TimerTask {
|
private final class WriteTimeoutTask implements TimerTask {
|
||||||
|
@ -0,0 +1,38 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2011 The Netty Project
|
||||||
|
*
|
||||||
|
* The Netty Project licenses this file to you under the Apache License,
|
||||||
|
* version 2.0 (the "License"); you may not use this file except in compliance
|
||||||
|
* with the License. You may obtain a copy of the License at:
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
* License for the specific language governing permissions and limitations
|
||||||
|
* under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package io.netty.channel.sctp;
|
||||||
|
|
||||||
|
import io.netty.channel.AbstractChannelSink;
|
||||||
|
import io.netty.channel.Channel;
|
||||||
|
import io.netty.channel.ChannelFuture;
|
||||||
|
import io.netty.channel.ChannelPipeline;
|
||||||
|
|
||||||
|
public abstract class AbstractSctpChannelSink extends AbstractChannelSink {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ChannelFuture execute(ChannelPipeline pipeline, final Runnable task) {
|
||||||
|
Channel ch = pipeline.getChannel();
|
||||||
|
if (ch instanceof SctpChannelImpl) {
|
||||||
|
SctpChannelImpl channel = (SctpChannelImpl) ch;
|
||||||
|
return channel.worker.executeInIoThread(channel, task);
|
||||||
|
|
||||||
|
} else {
|
||||||
|
return super.execute(pipeline, task);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
@ -32,7 +32,6 @@ import java.util.concurrent.ExecutorService;
|
|||||||
import java.util.concurrent.atomic.AtomicBoolean;
|
import java.util.concurrent.atomic.AtomicBoolean;
|
||||||
import java.util.concurrent.atomic.AtomicInteger;
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
|
||||||
import io.netty.channel.AbstractChannelSink;
|
|
||||||
import io.netty.channel.ChannelEvent;
|
import io.netty.channel.ChannelEvent;
|
||||||
import io.netty.channel.ChannelException;
|
import io.netty.channel.ChannelException;
|
||||||
import io.netty.channel.ChannelFuture;
|
import io.netty.channel.ChannelFuture;
|
||||||
@ -48,7 +47,7 @@ import io.netty.util.internal.QueueFactory;
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
*/
|
*/
|
||||||
class SctpClientPipelineSink extends AbstractChannelSink {
|
class SctpClientPipelineSink extends AbstractSctpChannelSink {
|
||||||
|
|
||||||
static final InternalLogger logger =
|
static final InternalLogger logger =
|
||||||
InternalLoggerFactory.getInstance(SctpClientPipelineSink.class);
|
InternalLoggerFactory.getInstance(SctpClientPipelineSink.class);
|
||||||
|
@ -31,7 +31,6 @@ import java.util.concurrent.atomic.AtomicInteger;
|
|||||||
|
|
||||||
import com.sun.nio.sctp.SctpChannel;
|
import com.sun.nio.sctp.SctpChannel;
|
||||||
|
|
||||||
import io.netty.channel.AbstractChannelSink;
|
|
||||||
import io.netty.channel.Channel;
|
import io.netty.channel.Channel;
|
||||||
import io.netty.channel.ChannelEvent;
|
import io.netty.channel.ChannelEvent;
|
||||||
import io.netty.channel.ChannelFuture;
|
import io.netty.channel.ChannelFuture;
|
||||||
@ -45,7 +44,7 @@ import io.netty.util.internal.DeadLockProofWorker;
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
*/
|
*/
|
||||||
class SctpServerPipelineSink extends AbstractChannelSink {
|
class SctpServerPipelineSink extends AbstractSctpChannelSink {
|
||||||
|
|
||||||
static final InternalLogger logger =
|
static final InternalLogger logger =
|
||||||
InternalLoggerFactory.getInstance(SctpServerPipelineSink.class);
|
InternalLoggerFactory.getInstance(SctpServerPipelineSink.class);
|
||||||
|
@ -31,6 +31,7 @@ import java.util.Queue;
|
|||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.concurrent.Executor;
|
import java.util.concurrent.Executor;
|
||||||
import java.util.concurrent.ExecutorService;
|
import java.util.concurrent.ExecutorService;
|
||||||
|
import java.util.concurrent.RejectedExecutionException;
|
||||||
import java.util.concurrent.atomic.AtomicBoolean;
|
import java.util.concurrent.atomic.AtomicBoolean;
|
||||||
import java.util.concurrent.locks.ReadWriteLock;
|
import java.util.concurrent.locks.ReadWriteLock;
|
||||||
import java.util.concurrent.locks.ReentrantReadWriteLock;
|
import java.util.concurrent.locks.ReentrantReadWriteLock;
|
||||||
@ -45,6 +46,8 @@ import io.netty.channel.ChannelFuture;
|
|||||||
import io.netty.channel.MessageEvent;
|
import io.netty.channel.MessageEvent;
|
||||||
import io.netty.channel.ReceiveBufferSizePredictor;
|
import io.netty.channel.ReceiveBufferSizePredictor;
|
||||||
import io.netty.channel.sctp.SctpSendBufferPool.SendBuffer;
|
import io.netty.channel.sctp.SctpSendBufferPool.SendBuffer;
|
||||||
|
import io.netty.channel.socket.ChannelRunnableWrapper;
|
||||||
|
import io.netty.channel.socket.Worker;
|
||||||
import io.netty.logging.InternalLogger;
|
import io.netty.logging.InternalLogger;
|
||||||
import io.netty.logging.InternalLoggerFactory;
|
import io.netty.logging.InternalLoggerFactory;
|
||||||
import io.netty.util.internal.DeadLockProofWorker;
|
import io.netty.util.internal.DeadLockProofWorker;
|
||||||
@ -53,7 +56,7 @@ import io.netty.util.internal.QueueFactory;
|
|||||||
/**
|
/**
|
||||||
*/
|
*/
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
class SctpWorker implements Runnable {
|
class SctpWorker implements Worker {
|
||||||
|
|
||||||
private static final InternalLogger logger =
|
private static final InternalLogger logger =
|
||||||
InternalLoggerFactory.getInstance(SctpWorker.class);
|
InternalLoggerFactory.getInstance(SctpWorker.class);
|
||||||
@ -64,13 +67,15 @@ class SctpWorker implements Runnable {
|
|||||||
|
|
||||||
private final Executor executor;
|
private final Executor executor;
|
||||||
private boolean started;
|
private boolean started;
|
||||||
private volatile Thread thread;
|
volatile Thread thread;
|
||||||
volatile Selector selector;
|
volatile Selector selector;
|
||||||
private final AtomicBoolean wakenUp = new AtomicBoolean();
|
private final AtomicBoolean wakenUp = new AtomicBoolean();
|
||||||
private final ReadWriteLock selectorGuard = new ReentrantReadWriteLock();
|
private final ReadWriteLock selectorGuard = new ReentrantReadWriteLock();
|
||||||
private final Object startStopLock = new Object();
|
private final Object startStopLock = new Object();
|
||||||
private final Queue<Runnable> registerTaskQueue = QueueFactory.createQueue(Runnable.class);
|
private final Queue<Runnable> registerTaskQueue = QueueFactory.createQueue(Runnable.class);
|
||||||
private final Queue<Runnable> writeTaskQueue = QueueFactory.createQueue(Runnable.class);
|
private final Queue<Runnable> writeTaskQueue = QueueFactory.createQueue(Runnable.class);
|
||||||
|
private final Queue<Runnable> eventQueue = QueueFactory.createQueue(Runnable.class);
|
||||||
|
|
||||||
private volatile int cancelledKeys; // should use AtomicInteger but we just need approximation
|
private volatile int cancelledKeys; // should use AtomicInteger but we just need approximation
|
||||||
|
|
||||||
private final SctpReceiveBufferPool recvBufferPool = new SctpReceiveBufferPool();
|
private final SctpReceiveBufferPool recvBufferPool = new SctpReceiveBufferPool();
|
||||||
@ -188,6 +193,7 @@ class SctpWorker implements Runnable {
|
|||||||
|
|
||||||
cancelledKeys = 0;
|
cancelledKeys = 0;
|
||||||
processRegisterTaskQueue();
|
processRegisterTaskQueue();
|
||||||
|
processEventQueue();
|
||||||
processWriteTaskQueue();
|
processWriteTaskQueue();
|
||||||
processSelectedKeys(selector.selectedKeys());
|
processSelectedKeys(selector.selectedKeys());
|
||||||
|
|
||||||
@ -240,7 +246,35 @@ class SctpWorker implements Runnable {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ChannelFuture executeInIoThread(Channel channel, Runnable task) {
|
||||||
|
if (channel instanceof SctpChannelImpl && isIoThread((SctpChannelImpl) channel)) {
|
||||||
|
try {
|
||||||
|
task.run();
|
||||||
|
return succeededFuture(channel);
|
||||||
|
} catch (Throwable t) {
|
||||||
|
return failedFuture(channel, t);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
ChannelRunnableWrapper channelRunnable = new ChannelRunnableWrapper(channel, task);
|
||||||
|
boolean added = eventQueue.offer(channelRunnable);
|
||||||
|
|
||||||
|
if (added) {
|
||||||
|
// wake up the selector to speed things
|
||||||
|
selector.wakeup();
|
||||||
|
} else {
|
||||||
|
channelRunnable.setFailure(new RejectedExecutionException("Unable to queue task " + task));
|
||||||
|
}
|
||||||
|
return channelRunnable;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
static boolean isIoThread(SctpChannelImpl channel) {
|
||||||
|
return Thread.currentThread() == channel.worker.thread;
|
||||||
|
}
|
||||||
|
|
||||||
private void processRegisterTaskQueue() throws IOException {
|
private void processRegisterTaskQueue() throws IOException {
|
||||||
for (; ;) {
|
for (; ;) {
|
||||||
final Runnable task = registerTaskQueue.poll();
|
final Runnable task = registerTaskQueue.poll();
|
||||||
@ -264,7 +298,19 @@ class SctpWorker implements Runnable {
|
|||||||
cleanUpCancelledKeys();
|
cleanUpCancelledKeys();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void processEventQueue() throws IOException {
|
||||||
|
for (;;) {
|
||||||
|
final Runnable task = eventQueue.poll();
|
||||||
|
if (task == null) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
task.run();
|
||||||
|
cleanUpCancelledKeys();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private void processSelectedKeys(final Set<SelectionKey> selectedKeys) throws IOException {
|
private void processSelectedKeys(final Set<SelectionKey> selectedKeys) throws IOException {
|
||||||
for (Iterator<SelectionKey> i = selectedKeys.iterator(); i.hasNext();) {
|
for (Iterator<SelectionKey> i = selectedKeys.iterator(); i.hasNext();) {
|
||||||
SelectionKey k = i.next();
|
SelectionKey k = i.next();
|
||||||
|
@ -43,7 +43,29 @@ public abstract class AbstractChannelSink implements ChannelSink {
|
|||||||
if (actualCause == null) {
|
if (actualCause == null) {
|
||||||
actualCause = cause;
|
actualCause = cause;
|
||||||
}
|
}
|
||||||
|
if (isFireExceptionCaughtLater(event, actualCause)) {
|
||||||
fireExceptionCaught(event.getChannel(), actualCause);
|
fireExceptionCaughtLater(event.getChannel(), actualCause);
|
||||||
|
} else {
|
||||||
|
fireExceptionCaught(event.getChannel(), actualCause);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected boolean isFireExceptionCaughtLater(ChannelEvent event, Throwable actualCause) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This implementation just directly call {@link Runnable#run()}. Sub-classes should override this if they can handle it
|
||||||
|
* in a better way
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public ChannelFuture execute(ChannelPipeline pipeline, Runnable task) {
|
||||||
|
try {
|
||||||
|
task.run();
|
||||||
|
return Channels.succeededFuture(pipeline.getChannel());
|
||||||
|
} catch (Throwable t) {
|
||||||
|
return Channels.failedFuture(pipeline.getChannel(), t);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -442,6 +442,9 @@ public interface ChannelPipeline {
|
|||||||
*/
|
*/
|
||||||
void sendUpstream(ChannelEvent e);
|
void sendUpstream(ChannelEvent e);
|
||||||
|
|
||||||
|
|
||||||
|
ChannelFuture execute(Runnable task);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sends the specified {@link ChannelEvent} to the last
|
* Sends the specified {@link ChannelEvent} to the last
|
||||||
* {@link ChannelDownstreamHandler} in this pipeline.
|
* {@link ChannelDownstreamHandler} in this pipeline.
|
||||||
|
@ -37,4 +37,9 @@ public interface ChannelSink {
|
|||||||
* one of its {@link ChannelHandler}s process a {@link ChannelEvent}.
|
* one of its {@link ChannelHandler}s process a {@link ChannelEvent}.
|
||||||
*/
|
*/
|
||||||
void exceptionCaught(ChannelPipeline pipeline, ChannelEvent e, ChannelPipelineException cause) throws Exception;
|
void exceptionCaught(ChannelPipeline pipeline, ChannelEvent e, ChannelPipelineException cause) throws Exception;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Execute the given {@link Runnable} later in the io-thread. Some implementation may not support his and just execute it directly
|
||||||
|
*/
|
||||||
|
ChannelFuture execute(ChannelPipeline pipeline, Runnable task);
|
||||||
}
|
}
|
||||||
|
@ -298,6 +298,22 @@ public final class Channels {
|
|||||||
ctx.getChannel(), message, remoteAddress));
|
ctx.getChannel(), message, remoteAddress));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sends a {@code "writeComplete"} event to the first
|
||||||
|
* {@link ChannelUpstreamHandler} in the {@link ChannelPipeline} of
|
||||||
|
* the specified {@link Channel} in the next io-thread.
|
||||||
|
*/
|
||||||
|
public static ChannelFuture fireWriteCompleteLater(final Channel channel, final long amount) {
|
||||||
|
return channel.getPipeline().execute(new Runnable() {
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
fireWriteComplete(channel, amount);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sends a {@code "writeComplete"} event to the first
|
* Sends a {@code "writeComplete"} event to the first
|
||||||
* {@link ChannelUpstreamHandler} in the {@link ChannelPipeline} of
|
* {@link ChannelUpstreamHandler} in the {@link ChannelPipeline} of
|
||||||
@ -321,6 +337,25 @@ public final class Channels {
|
|||||||
public static void fireWriteComplete(ChannelHandlerContext ctx, long amount) {
|
public static void fireWriteComplete(ChannelHandlerContext ctx, long amount) {
|
||||||
ctx.sendUpstream(new DefaultWriteCompletionEvent(ctx.getChannel(), amount));
|
ctx.sendUpstream(new DefaultWriteCompletionEvent(ctx.getChannel(), amount));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sends a {@code "channelInterestChanged"} event to the first
|
||||||
|
* {@link ChannelUpstreamHandler} in the {@link ChannelPipeline} of
|
||||||
|
* the specified {@link Channel} once the io-thread runs again.
|
||||||
|
*/
|
||||||
|
public static ChannelFuture fireChannelInterestChangedLater(final Channel channel) {
|
||||||
|
return channel.getPipeline().execute(new Runnable() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
fireChannelInterestChanged(channel);
|
||||||
|
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sends a {@code "channelInterestChanged"} event to the first
|
* Sends a {@code "channelInterestChanged"} event to the first
|
||||||
* {@link ChannelUpstreamHandler} in the {@link ChannelPipeline} of
|
* {@link ChannelUpstreamHandler} in the {@link ChannelPipeline} of
|
||||||
@ -346,6 +381,21 @@ public final class Channels {
|
|||||||
ctx.getChannel(), ChannelState.INTEREST_OPS, Channel.OP_READ));
|
ctx.getChannel(), ChannelState.INTEREST_OPS, Channel.OP_READ));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sends a {@code "channelDisconnected"} event to the first
|
||||||
|
* {@link ChannelUpstreamHandler} in the {@link ChannelPipeline} of
|
||||||
|
* the specified {@link Channel} once the io-thread runs again.
|
||||||
|
*/
|
||||||
|
public static ChannelFuture fireChannelDisconnectedLater(final Channel channel) {
|
||||||
|
return channel.getPipeline().execute(new Runnable() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
fireChannelDisconnected(channel);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sends a {@code "channelDisconnected"} event to the first
|
* Sends a {@code "channelDisconnected"} event to the first
|
||||||
* {@link ChannelUpstreamHandler} in the {@link ChannelPipeline} of
|
* {@link ChannelUpstreamHandler} in the {@link ChannelPipeline} of
|
||||||
@ -368,6 +418,23 @@ public final class Channels {
|
|||||||
ctx.getChannel(), ChannelState.CONNECTED, null));
|
ctx.getChannel(), ChannelState.CONNECTED, null));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sends a {@code "channelUnbound"} event to the first
|
||||||
|
* {@link ChannelUpstreamHandler} in the {@link ChannelPipeline} of
|
||||||
|
* the specified {@link Channel} once the io-thread runs again.
|
||||||
|
*/
|
||||||
|
public static ChannelFuture fireChannelUnboundLater(final Channel channel) {
|
||||||
|
return channel.getPipeline().execute(new Runnable() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
fireChannelUnbound(channel);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sends a {@code "channelUnbound"} event to the first
|
* Sends a {@code "channelUnbound"} event to the first
|
||||||
* {@link ChannelUpstreamHandler} in the {@link ChannelPipeline} of
|
* {@link ChannelUpstreamHandler} in the {@link ChannelPipeline} of
|
||||||
@ -390,6 +457,24 @@ public final class Channels {
|
|||||||
ctx.getChannel(), ChannelState.BOUND, null));
|
ctx.getChannel(), ChannelState.BOUND, null));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sends a {@code "channelClosed"} event to the first
|
||||||
|
* {@link ChannelUpstreamHandler} in the {@link ChannelPipeline} of
|
||||||
|
* the specified {@link Channel} once the io-thread runs again.
|
||||||
|
*/
|
||||||
|
public static ChannelFuture fireChannelClosedLater(final Channel channel) {
|
||||||
|
return channel.getPipeline().execute(new Runnable() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
fireChannelClosed(channel);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sends a {@code "channelClosed"} event to the first
|
* Sends a {@code "channelClosed"} event to the first
|
||||||
* {@link ChannelUpstreamHandler} in the {@link ChannelPipeline} of
|
* {@link ChannelUpstreamHandler} in the {@link ChannelPipeline} of
|
||||||
@ -418,6 +503,24 @@ public final class Channels {
|
|||||||
ctx.getChannel(), ChannelState.OPEN, Boolean.FALSE));
|
ctx.getChannel(), ChannelState.OPEN, Boolean.FALSE));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sends a {@code "exceptionCaught"} event to the first
|
||||||
|
* {@link ChannelUpstreamHandler} in the {@link ChannelPipeline} of
|
||||||
|
* the specified {@link Channel} once the io-thread runs again.
|
||||||
|
*/
|
||||||
|
public static ChannelFuture fireExceptionCaughtLater(final Channel channel, final Throwable cause) {
|
||||||
|
return channel.getPipeline().execute(new Runnable() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
fireExceptionCaught(channel, cause);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sends a {@code "exceptionCaught"} event to the first
|
* Sends a {@code "exceptionCaught"} event to the first
|
||||||
* {@link ChannelUpstreamHandler} in the {@link ChannelPipeline} of
|
* {@link ChannelUpstreamHandler} in the {@link ChannelPipeline} of
|
||||||
@ -444,6 +547,7 @@ public final class Channels {
|
|||||||
new DefaultChildChannelStateEvent(channel, childChannel));
|
new DefaultChildChannelStateEvent(channel, childChannel));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sends a {@code "bind"} request to the last
|
* Sends a {@code "bind"} request to the last
|
||||||
* {@link ChannelDownstreamHandler} in the {@link ChannelPipeline} of
|
* {@link ChannelDownstreamHandler} in the {@link ChannelPipeline} of
|
||||||
|
@ -21,6 +21,7 @@ import java.util.LinkedHashMap;
|
|||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.NoSuchElementException;
|
import java.util.NoSuchElementException;
|
||||||
|
import java.util.concurrent.RejectedExecutionException;
|
||||||
|
|
||||||
import io.netty.logging.InternalLogger;
|
import io.netty.logging.InternalLogger;
|
||||||
import io.netty.logging.InternalLoggerFactory;
|
import io.netty.logging.InternalLoggerFactory;
|
||||||
@ -583,6 +584,11 @@ public class DefaultChannelPipeline implements ChannelPipeline {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ChannelFuture execute(Runnable task) {
|
||||||
|
return getSink().execute(this, task);
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void sendDownstream(ChannelEvent e) {
|
public void sendDownstream(ChannelEvent e) {
|
||||||
DefaultChannelHandlerContext tail = getActualDownstreamContext(this.tail);
|
DefaultChannelHandlerContext tail = getActualDownstreamContext(this.tail);
|
||||||
@ -832,5 +838,14 @@ public class DefaultChannelPipeline implements ChannelPipeline {
|
|||||||
ChannelEvent e, ChannelPipelineException cause) throws Exception {
|
ChannelEvent e, ChannelPipelineException cause) throws Exception {
|
||||||
throw cause;
|
throw cause;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ChannelFuture execute(ChannelPipeline pipeline, Runnable task) {
|
||||||
|
if (logger.isWarnEnabled()) {
|
||||||
|
logger.warn("Not attached yet; rejecting: " + task);
|
||||||
|
}
|
||||||
|
return Channels.failedFuture(pipeline.getChannel(), new RejectedExecutionException("Not attached yet"));
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,42 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2011 The Netty Project
|
||||||
|
*
|
||||||
|
* The Netty Project licenses this file to you under the Apache License,
|
||||||
|
* version 2.0 (the "License"); you may not use this file except in compliance
|
||||||
|
* with the License. You may obtain a copy of the License at:
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
* License for the specific language governing permissions and limitations
|
||||||
|
* under the License.
|
||||||
|
*/
|
||||||
|
package io.netty.channel.socket;
|
||||||
|
|
||||||
|
import io.netty.channel.Channel;
|
||||||
|
import io.netty.channel.DefaultChannelFuture;
|
||||||
|
|
||||||
|
public class ChannelRunnableWrapper extends DefaultChannelFuture implements Runnable {
|
||||||
|
|
||||||
|
private final Runnable task;
|
||||||
|
|
||||||
|
public ChannelRunnableWrapper(Channel channel, Runnable task) {
|
||||||
|
super(channel, true);
|
||||||
|
this.task = task;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
try {
|
||||||
|
task.run();
|
||||||
|
setSuccess();
|
||||||
|
} catch (Throwable t) {
|
||||||
|
setFailure(t);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
}
|
34
transport/src/main/java/io/netty/channel/socket/Worker.java
Normal file
34
transport/src/main/java/io/netty/channel/socket/Worker.java
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2011 The Netty Project
|
||||||
|
*
|
||||||
|
* The Netty Project licenses this file to you under the Apache License,
|
||||||
|
* version 2.0 (the "License"); you may not use this file except in compliance
|
||||||
|
* with the License. You may obtain a copy of the License at:
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
* License for the specific language governing permissions and limitations
|
||||||
|
* under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package io.netty.channel.socket;
|
||||||
|
|
||||||
|
import io.netty.channel.Channel;
|
||||||
|
import io.netty.channel.ChannelFuture;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A {@link Worker} is responsible to dispatch IO operations
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public interface Worker extends Runnable {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Execute the given {@link Runnable} in the IO-Thread. This may be now or later once the IO-Thread do some other work.
|
||||||
|
*
|
||||||
|
* @param task the {@link Runnable} to execute
|
||||||
|
*/
|
||||||
|
ChannelFuture executeInIoThread(Channel channel, Runnable task);
|
||||||
|
}
|
@ -0,0 +1,50 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2011 The Netty Project
|
||||||
|
*
|
||||||
|
* The Netty Project licenses this file to you under the Apache License,
|
||||||
|
* version 2.0 (the "License"); you may not use this file except in compliance
|
||||||
|
* with the License. You may obtain a copy of the License at:
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
* License for the specific language governing permissions and limitations
|
||||||
|
* under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package io.netty.channel.socket.nio;
|
||||||
|
|
||||||
|
import io.netty.channel.AbstractChannelSink;
|
||||||
|
import io.netty.channel.Channel;
|
||||||
|
import io.netty.channel.ChannelEvent;
|
||||||
|
import io.netty.channel.ChannelFuture;
|
||||||
|
import io.netty.channel.ChannelPipeline;
|
||||||
|
|
||||||
|
public abstract class AbstractNioChannelSink extends AbstractChannelSink {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ChannelFuture execute(ChannelPipeline pipeline, final Runnable task) {
|
||||||
|
Channel ch = pipeline.getChannel();
|
||||||
|
if (ch instanceof AbstractNioChannel<?>) {
|
||||||
|
AbstractNioChannel<?> channel = (AbstractNioChannel<?>) ch;
|
||||||
|
|
||||||
|
return channel.worker.executeInIoThread(ch, task);
|
||||||
|
}
|
||||||
|
return super.execute(pipeline, task);
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected boolean isFireExceptionCaughtLater(ChannelEvent event, Throwable actualCause) {
|
||||||
|
Channel channel = event.getChannel();
|
||||||
|
boolean fireLater = false;
|
||||||
|
if (channel instanceof AbstractNioChannel<?>) {
|
||||||
|
fireLater = !AbstractNioWorker.isIoThread((AbstractNioChannel<?>) channel);
|
||||||
|
}
|
||||||
|
return fireLater;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -21,6 +21,8 @@ import io.netty.channel.Channel;
|
|||||||
import io.netty.channel.ChannelException;
|
import io.netty.channel.ChannelException;
|
||||||
import io.netty.channel.ChannelFuture;
|
import io.netty.channel.ChannelFuture;
|
||||||
import io.netty.channel.MessageEvent;
|
import io.netty.channel.MessageEvent;
|
||||||
|
import io.netty.channel.socket.ChannelRunnableWrapper;
|
||||||
|
import io.netty.channel.socket.Worker;
|
||||||
import io.netty.channel.socket.nio.SocketSendBufferPool.SendBuffer;
|
import io.netty.channel.socket.nio.SocketSendBufferPool.SendBuffer;
|
||||||
import io.netty.logging.InternalLogger;
|
import io.netty.logging.InternalLogger;
|
||||||
import io.netty.logging.InternalLoggerFactory;
|
import io.netty.logging.InternalLoggerFactory;
|
||||||
@ -40,11 +42,12 @@ import java.util.Queue;
|
|||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.concurrent.Executor;
|
import java.util.concurrent.Executor;
|
||||||
import java.util.concurrent.ExecutorService;
|
import java.util.concurrent.ExecutorService;
|
||||||
|
import java.util.concurrent.RejectedExecutionException;
|
||||||
import java.util.concurrent.atomic.AtomicBoolean;
|
import java.util.concurrent.atomic.AtomicBoolean;
|
||||||
import java.util.concurrent.locks.ReadWriteLock;
|
import java.util.concurrent.locks.ReadWriteLock;
|
||||||
import java.util.concurrent.locks.ReentrantReadWriteLock;
|
import java.util.concurrent.locks.ReentrantReadWriteLock;
|
||||||
|
|
||||||
abstract class AbstractNioWorker implements Runnable {
|
abstract class AbstractNioWorker implements Worker {
|
||||||
/**
|
/**
|
||||||
* Internal Netty logger.
|
* Internal Netty logger.
|
||||||
*/
|
*/
|
||||||
@ -106,6 +109,9 @@ abstract class AbstractNioWorker implements Runnable {
|
|||||||
*/
|
*/
|
||||||
protected final Queue<Runnable> writeTaskQueue = QueueFactory.createQueue(Runnable.class);
|
protected final Queue<Runnable> writeTaskQueue = QueueFactory.createQueue(Runnable.class);
|
||||||
|
|
||||||
|
private final Queue<Runnable> eventQueue = QueueFactory.createQueue(Runnable.class);
|
||||||
|
|
||||||
|
|
||||||
private volatile int cancelledKeys; // should use AtomicInteger but we just need approximation
|
private volatile int cancelledKeys; // should use AtomicInteger but we just need approximation
|
||||||
|
|
||||||
private final SocketSendBufferPool sendBufferPool = new SocketSendBufferPool();
|
private final SocketSendBufferPool sendBufferPool = new SocketSendBufferPool();
|
||||||
@ -216,6 +222,7 @@ abstract class AbstractNioWorker implements Runnable {
|
|||||||
|
|
||||||
cancelledKeys = 0;
|
cancelledKeys = 0;
|
||||||
processRegisterTaskQueue();
|
processRegisterTaskQueue();
|
||||||
|
processEventQueue();
|
||||||
processWriteTaskQueue();
|
processWriteTaskQueue();
|
||||||
processSelectedKeys(selector.selectedKeys());
|
processSelectedKeys(selector.selectedKeys());
|
||||||
|
|
||||||
@ -266,7 +273,31 @@ abstract class AbstractNioWorker implements Runnable {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ChannelFuture executeInIoThread(Channel channel, Runnable task) {
|
||||||
|
if (channel instanceof AbstractNioChannel<?> && isIoThread((AbstractNioChannel<?>) channel)) {
|
||||||
|
try {
|
||||||
|
task.run();
|
||||||
|
return succeededFuture(channel);
|
||||||
|
} catch (Throwable t) {
|
||||||
|
return failedFuture(channel, t);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
ChannelRunnableWrapper channelRunnable = new ChannelRunnableWrapper(channel, task);
|
||||||
|
boolean added = eventQueue.offer(channelRunnable);
|
||||||
|
|
||||||
|
if (added) {
|
||||||
|
// wake up the selector to speed things
|
||||||
|
selector.wakeup();
|
||||||
|
} else {
|
||||||
|
channelRunnable.setFailure(new RejectedExecutionException("Unable to queue task " + task));
|
||||||
|
}
|
||||||
|
return channelRunnable;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
private void processRegisterTaskQueue() throws IOException {
|
private void processRegisterTaskQueue() throws IOException {
|
||||||
for (;;) {
|
for (;;) {
|
||||||
final Runnable task = registerTaskQueue.poll();
|
final Runnable task = registerTaskQueue.poll();
|
||||||
@ -291,6 +322,18 @@ abstract class AbstractNioWorker implements Runnable {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void processEventQueue() throws IOException {
|
||||||
|
for (;;) {
|
||||||
|
final Runnable task = eventQueue.poll();
|
||||||
|
if (task == null) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
task.run();
|
||||||
|
cleanUpCancelledKeys();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private void processSelectedKeys(Set<SelectionKey> selectedKeys) throws IOException {
|
private void processSelectedKeys(Set<SelectionKey> selectedKeys) throws IOException {
|
||||||
for (Iterator<SelectionKey> i = selectedKeys.iterator(); i.hasNext();) {
|
for (Iterator<SelectionKey> i = selectedKeys.iterator(); i.hasNext();) {
|
||||||
SelectionKey k = i.next();
|
SelectionKey k = i.next();
|
||||||
@ -374,6 +417,7 @@ abstract class AbstractNioWorker implements Runnable {
|
|||||||
boolean open = true;
|
boolean open = true;
|
||||||
boolean addOpWrite = false;
|
boolean addOpWrite = false;
|
||||||
boolean removeOpWrite = false;
|
boolean removeOpWrite = false;
|
||||||
|
boolean iothread = isIoThread(channel);
|
||||||
|
|
||||||
long writtenBytes = 0;
|
long writtenBytes = 0;
|
||||||
|
|
||||||
@ -444,7 +488,11 @@ abstract class AbstractNioWorker implements Runnable {
|
|||||||
buf = null;
|
buf = null;
|
||||||
evt = null;
|
evt = null;
|
||||||
future.setFailure(t);
|
future.setFailure(t);
|
||||||
fireExceptionCaught(channel, t);
|
if (iothread) {
|
||||||
|
fireExceptionCaught(channel, t);
|
||||||
|
} else {
|
||||||
|
fireExceptionCaughtLater(channel, t);
|
||||||
|
}
|
||||||
if (t instanceof IOException) {
|
if (t instanceof IOException) {
|
||||||
open = false;
|
open = false;
|
||||||
close(channel, succeededFuture(channel));
|
close(channel, succeededFuture(channel));
|
||||||
@ -467,10 +515,17 @@ abstract class AbstractNioWorker implements Runnable {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (iothread) {
|
||||||
fireWriteComplete(channel, writtenBytes);
|
fireWriteComplete(channel, writtenBytes);
|
||||||
|
} else {
|
||||||
|
fireWriteCompleteLater(channel, writtenBytes);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static boolean isIoThread(AbstractNioChannel<?> channel) {
|
||||||
|
return Thread.currentThread() == channel.worker.thread;
|
||||||
|
}
|
||||||
|
|
||||||
private void setOpWrite(AbstractNioChannel<?> channel) {
|
private void setOpWrite(AbstractNioChannel<?> channel) {
|
||||||
Selector selector = this.selector;
|
Selector selector = this.selector;
|
||||||
SelectionKey key = channel.channel.keyFor(selector);
|
SelectionKey key = channel.channel.keyFor(selector);
|
||||||
@ -521,6 +576,8 @@ abstract class AbstractNioWorker implements Runnable {
|
|||||||
void close(AbstractNioChannel<?> channel, ChannelFuture future) {
|
void close(AbstractNioChannel<?> channel, ChannelFuture future) {
|
||||||
boolean connected = channel.isConnected();
|
boolean connected = channel.isConnected();
|
||||||
boolean bound = channel.isBound();
|
boolean bound = channel.isBound();
|
||||||
|
boolean iothread = isIoThread(channel);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
channel.channel.close();
|
channel.channel.close();
|
||||||
cancelledKeys ++;
|
cancelledKeys ++;
|
||||||
@ -528,20 +585,36 @@ abstract class AbstractNioWorker implements Runnable {
|
|||||||
if (channel.setClosed()) {
|
if (channel.setClosed()) {
|
||||||
future.setSuccess();
|
future.setSuccess();
|
||||||
if (connected) {
|
if (connected) {
|
||||||
fireChannelDisconnected(channel);
|
if (iothread) {
|
||||||
|
fireChannelDisconnected(channel);
|
||||||
|
} else {
|
||||||
|
fireChannelDisconnectedLater(channel);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (bound) {
|
if (bound) {
|
||||||
fireChannelUnbound(channel);
|
if (iothread) {
|
||||||
|
fireChannelUnbound(channel);
|
||||||
|
} else {
|
||||||
|
fireChannelUnboundLater(channel);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
cleanUpWriteBuffer(channel);
|
cleanUpWriteBuffer(channel);
|
||||||
fireChannelClosed(channel);
|
if (iothread) {
|
||||||
|
fireChannelClosed(channel);
|
||||||
|
} else {
|
||||||
|
fireChannelClosedLater(channel);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
future.setSuccess();
|
future.setSuccess();
|
||||||
}
|
}
|
||||||
} catch (Throwable t) {
|
} catch (Throwable t) {
|
||||||
future.setFailure(t);
|
future.setFailure(t);
|
||||||
fireExceptionCaught(channel, t);
|
if (iothread) {
|
||||||
|
fireExceptionCaught(channel, t);
|
||||||
|
} else {
|
||||||
|
fireExceptionCaughtLater(channel, t);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -594,12 +667,17 @@ abstract class AbstractNioWorker implements Runnable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (fireExceptionCaught) {
|
if (fireExceptionCaught) {
|
||||||
fireExceptionCaught(channel, cause);
|
if (isIoThread(channel)) {
|
||||||
|
fireExceptionCaught(channel, cause);
|
||||||
|
} else {
|
||||||
|
fireExceptionCaughtLater(channel, cause);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void setInterestOps(AbstractNioChannel<?> channel, ChannelFuture future, int interestOps) {
|
void setInterestOps(AbstractNioChannel<?> channel, ChannelFuture future, int interestOps) {
|
||||||
boolean changed = false;
|
boolean changed = false;
|
||||||
|
boolean iothread = isIoThread(channel);
|
||||||
try {
|
try {
|
||||||
// interestOps can change at any time and at any thread.
|
// interestOps can change at any time and at any thread.
|
||||||
// Acquire a lock to avoid possible race condition.
|
// Acquire a lock to avoid possible race condition.
|
||||||
@ -660,16 +738,28 @@ abstract class AbstractNioWorker implements Runnable {
|
|||||||
|
|
||||||
future.setSuccess();
|
future.setSuccess();
|
||||||
if (changed) {
|
if (changed) {
|
||||||
fireChannelInterestChanged(channel);
|
if (iothread) {
|
||||||
|
fireChannelInterestChanged(channel);
|
||||||
|
} else {
|
||||||
|
fireChannelInterestChangedLater(channel);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} catch (CancelledKeyException e) {
|
} catch (CancelledKeyException e) {
|
||||||
// setInterestOps() was called on a closed channel.
|
// setInterestOps() was called on a closed channel.
|
||||||
ClosedChannelException cce = new ClosedChannelException();
|
ClosedChannelException cce = new ClosedChannelException();
|
||||||
future.setFailure(cce);
|
future.setFailure(cce);
|
||||||
fireExceptionCaught(channel, cce);
|
if (iothread) {
|
||||||
|
fireExceptionCaught(channel, cce);
|
||||||
|
} else {
|
||||||
|
fireExceptionCaughtLater(channel, cce);
|
||||||
|
}
|
||||||
} catch (Throwable t) {
|
} catch (Throwable t) {
|
||||||
future.setFailure(t);
|
future.setFailure(t);
|
||||||
fireExceptionCaught(channel, t);
|
if (iothread) {
|
||||||
|
fireExceptionCaught(channel, t);
|
||||||
|
} else {
|
||||||
|
fireExceptionCaughtLater(channel, t);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -31,7 +31,6 @@ import java.util.concurrent.ExecutorService;
|
|||||||
import java.util.concurrent.atomic.AtomicBoolean;
|
import java.util.concurrent.atomic.AtomicBoolean;
|
||||||
import java.util.concurrent.atomic.AtomicInteger;
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
|
||||||
import io.netty.channel.AbstractChannelSink;
|
|
||||||
import io.netty.channel.ChannelEvent;
|
import io.netty.channel.ChannelEvent;
|
||||||
import io.netty.channel.ChannelException;
|
import io.netty.channel.ChannelException;
|
||||||
import io.netty.channel.ChannelFuture;
|
import io.netty.channel.ChannelFuture;
|
||||||
@ -45,7 +44,7 @@ import io.netty.logging.InternalLoggerFactory;
|
|||||||
import io.netty.util.internal.DeadLockProofWorker;
|
import io.netty.util.internal.DeadLockProofWorker;
|
||||||
import io.netty.util.internal.QueueFactory;
|
import io.netty.util.internal.QueueFactory;
|
||||||
|
|
||||||
class NioClientSocketPipelineSink extends AbstractChannelSink {
|
class NioClientSocketPipelineSink extends AbstractNioChannelSink {
|
||||||
|
|
||||||
static final InternalLogger logger =
|
static final InternalLogger logger =
|
||||||
InternalLoggerFactory.getInstance(NioClientSocketPipelineSink.class);
|
InternalLoggerFactory.getInstance(NioClientSocketPipelineSink.class);
|
||||||
|
@ -22,7 +22,6 @@ import java.net.SocketAddress;
|
|||||||
import java.util.concurrent.Executor;
|
import java.util.concurrent.Executor;
|
||||||
import java.util.concurrent.atomic.AtomicInteger;
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
|
||||||
import io.netty.channel.AbstractChannelSink;
|
|
||||||
import io.netty.channel.ChannelEvent;
|
import io.netty.channel.ChannelEvent;
|
||||||
import io.netty.channel.ChannelFuture;
|
import io.netty.channel.ChannelFuture;
|
||||||
import io.netty.channel.ChannelFutureListener;
|
import io.netty.channel.ChannelFutureListener;
|
||||||
@ -35,7 +34,7 @@ import io.netty.channel.MessageEvent;
|
|||||||
* Receives downstream events from a {@link ChannelPipeline}. It contains
|
* Receives downstream events from a {@link ChannelPipeline}. It contains
|
||||||
* an array of I/O workers.
|
* an array of I/O workers.
|
||||||
*/
|
*/
|
||||||
class NioDatagramPipelineSink extends AbstractChannelSink {
|
class NioDatagramPipelineSink extends AbstractNioChannelSink {
|
||||||
|
|
||||||
private final NioDatagramWorker[] workers;
|
private final NioDatagramWorker[] workers;
|
||||||
private final AtomicInteger workerIndex = new AtomicInteger();
|
private final AtomicInteger workerIndex = new AtomicInteger();
|
||||||
|
@ -16,7 +16,9 @@
|
|||||||
package io.netty.channel.socket.nio;
|
package io.netty.channel.socket.nio;
|
||||||
|
|
||||||
import static io.netty.channel.Channels.fireChannelDisconnected;
|
import static io.netty.channel.Channels.fireChannelDisconnected;
|
||||||
|
import static io.netty.channel.Channels.fireChannelDisconnectedLater;
|
||||||
import static io.netty.channel.Channels.fireExceptionCaught;
|
import static io.netty.channel.Channels.fireExceptionCaught;
|
||||||
|
import static io.netty.channel.Channels.fireExceptionCaughtLater;
|
||||||
import static io.netty.channel.Channels.fireMessageReceived;
|
import static io.netty.channel.Channels.fireMessageReceived;
|
||||||
import static io.netty.channel.Channels.succeededFuture;
|
import static io.netty.channel.Channels.succeededFuture;
|
||||||
import io.netty.buffer.ChannelBufferFactory;
|
import io.netty.buffer.ChannelBufferFactory;
|
||||||
@ -126,15 +128,24 @@ class NioDatagramWorker extends AbstractNioWorker {
|
|||||||
|
|
||||||
static void disconnect(NioDatagramChannel channel, ChannelFuture future) {
|
static void disconnect(NioDatagramChannel channel, ChannelFuture future) {
|
||||||
boolean connected = channel.isConnected();
|
boolean connected = channel.isConnected();
|
||||||
|
boolean iothread = isIoThread(channel);
|
||||||
try {
|
try {
|
||||||
channel.getDatagramChannel().disconnect();
|
channel.getDatagramChannel().disconnect();
|
||||||
future.setSuccess();
|
future.setSuccess();
|
||||||
if (connected) {
|
if (connected) {
|
||||||
fireChannelDisconnected(channel);
|
if (iothread) {
|
||||||
|
fireChannelDisconnected(channel);
|
||||||
|
} else {
|
||||||
|
fireChannelDisconnectedLater(channel);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} catch (Throwable t) {
|
} catch (Throwable t) {
|
||||||
future.setFailure(t);
|
future.setFailure(t);
|
||||||
fireExceptionCaught(channel, t);
|
if (iothread) {
|
||||||
|
fireExceptionCaught(channel, t);
|
||||||
|
} else {
|
||||||
|
fireExceptionCaughtLater(channel, t);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -29,7 +29,6 @@ import java.nio.channels.SocketChannel;
|
|||||||
import java.util.concurrent.Executor;
|
import java.util.concurrent.Executor;
|
||||||
import java.util.concurrent.atomic.AtomicInteger;
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
|
||||||
import io.netty.channel.AbstractChannelSink;
|
|
||||||
import io.netty.channel.Channel;
|
import io.netty.channel.Channel;
|
||||||
import io.netty.channel.ChannelEvent;
|
import io.netty.channel.ChannelEvent;
|
||||||
import io.netty.channel.ChannelFuture;
|
import io.netty.channel.ChannelFuture;
|
||||||
@ -41,7 +40,7 @@ import io.netty.logging.InternalLogger;
|
|||||||
import io.netty.logging.InternalLoggerFactory;
|
import io.netty.logging.InternalLoggerFactory;
|
||||||
import io.netty.util.internal.DeadLockProofWorker;
|
import io.netty.util.internal.DeadLockProofWorker;
|
||||||
|
|
||||||
class NioServerSocketPipelineSink extends AbstractChannelSink {
|
class NioServerSocketPipelineSink extends AbstractNioChannelSink {
|
||||||
|
|
||||||
static final InternalLogger logger =
|
static final InternalLogger logger =
|
||||||
InternalLoggerFactory.getInstance(NioServerSocketPipelineSink.class);
|
InternalLoggerFactory.getInstance(NioServerSocketPipelineSink.class);
|
||||||
|
@ -25,11 +25,14 @@ import io.netty.channel.ChannelFactory;
|
|||||||
import io.netty.channel.ChannelFuture;
|
import io.netty.channel.ChannelFuture;
|
||||||
import io.netty.channel.ChannelPipeline;
|
import io.netty.channel.ChannelPipeline;
|
||||||
import io.netty.channel.ChannelSink;
|
import io.netty.channel.ChannelSink;
|
||||||
|
import io.netty.channel.socket.Worker;
|
||||||
|
|
||||||
abstract class AbstractOioChannel extends AbstractChannel {
|
abstract class AbstractOioChannel extends AbstractChannel {
|
||||||
private volatile InetSocketAddress localAddress;
|
private volatile InetSocketAddress localAddress;
|
||||||
volatile InetSocketAddress remoteAddress;
|
volatile InetSocketAddress remoteAddress;
|
||||||
volatile Thread workerThread;
|
volatile Thread workerThread;
|
||||||
|
volatile Worker worker;
|
||||||
|
|
||||||
final Object interestOpsLock = new Object();
|
final Object interestOpsLock = new Object();
|
||||||
|
|
||||||
AbstractOioChannel(
|
AbstractOioChannel(
|
||||||
|
@ -0,0 +1,54 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2011 The Netty Project
|
||||||
|
*
|
||||||
|
* The Netty Project licenses this file to you under the Apache License,
|
||||||
|
* version 2.0 (the "License"); you may not use this file except in compliance
|
||||||
|
* with the License. You may obtain a copy of the License at:
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
* License for the specific language governing permissions and limitations
|
||||||
|
* under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package io.netty.channel.socket.oio;
|
||||||
|
|
||||||
|
import io.netty.channel.AbstractChannelSink;
|
||||||
|
import io.netty.channel.Channel;
|
||||||
|
import io.netty.channel.ChannelEvent;
|
||||||
|
import io.netty.channel.ChannelFuture;
|
||||||
|
import io.netty.channel.ChannelPipeline;
|
||||||
|
import io.netty.channel.socket.Worker;
|
||||||
|
|
||||||
|
public abstract class AbstractOioChannelSink extends AbstractChannelSink {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ChannelFuture execute(final ChannelPipeline pipeline, final Runnable task) {
|
||||||
|
Channel ch = pipeline.getChannel();
|
||||||
|
if (ch instanceof AbstractOioChannel) {
|
||||||
|
AbstractOioChannel channel = (AbstractOioChannel) ch;
|
||||||
|
Worker worker = channel.worker;
|
||||||
|
if (worker != null) {
|
||||||
|
return channel.worker.executeInIoThread(ch, task);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return super.execute(pipeline, task);
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected boolean isFireExceptionCaughtLater(ChannelEvent event, Throwable actualCause) {
|
||||||
|
Channel channel = event.getChannel();
|
||||||
|
boolean fireLater = false;
|
||||||
|
if (channel instanceof AbstractOioChannel) {
|
||||||
|
fireLater = !AbstractOioWorker.isIoThread((AbstractOioChannel) channel);
|
||||||
|
}
|
||||||
|
return fireLater;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -15,29 +15,32 @@
|
|||||||
*/
|
*/
|
||||||
package io.netty.channel.socket.oio;
|
package io.netty.channel.socket.oio;
|
||||||
|
|
||||||
import static io.netty.channel.Channels.fireChannelClosed;
|
import static io.netty.channel.Channels.*;
|
||||||
import static io.netty.channel.Channels.fireChannelDisconnected;
|
|
||||||
import static io.netty.channel.Channels.fireChannelInterestChanged;
|
|
||||||
import static io.netty.channel.Channels.fireChannelUnbound;
|
|
||||||
import static io.netty.channel.Channels.fireExceptionCaught;
|
|
||||||
import static io.netty.channel.Channels.succeededFuture;
|
|
||||||
import io.netty.channel.Channel;
|
import io.netty.channel.Channel;
|
||||||
import io.netty.channel.ChannelFuture;
|
import io.netty.channel.ChannelFuture;
|
||||||
import io.netty.channel.Channels;
|
import io.netty.channel.Channels;
|
||||||
|
import io.netty.channel.socket.ChannelRunnableWrapper;
|
||||||
|
import io.netty.channel.socket.Worker;
|
||||||
|
import io.netty.util.internal.QueueFactory;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
import java.util.Queue;
|
||||||
|
import java.util.concurrent.RejectedExecutionException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Abstract base class for Oio-Worker implementations
|
* Abstract base class for Oio-Worker implementations
|
||||||
*
|
*
|
||||||
* @param <C> {@link AbstractOioChannel}
|
* @param <C> {@link AbstractOioChannel}
|
||||||
*/
|
*/
|
||||||
abstract class AbstractOioWorker<C extends AbstractOioChannel> implements Runnable {
|
abstract class AbstractOioWorker<C extends AbstractOioChannel> implements Worker {
|
||||||
|
|
||||||
|
private final Queue<Runnable> eventQueue = QueueFactory.createQueue(Runnable.class);
|
||||||
|
|
||||||
protected final C channel;
|
protected final C channel;
|
||||||
|
|
||||||
public AbstractOioWorker(C channel) {
|
public AbstractOioWorker(C channel) {
|
||||||
this.channel = channel;
|
this.channel = channel;
|
||||||
|
channel.worker = this;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -60,9 +63,13 @@ abstract class AbstractOioWorker<C extends AbstractOioChannel> implements Runnab
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
if (!process()) {
|
boolean cont = process();
|
||||||
break;
|
|
||||||
}
|
processEventQueue();
|
||||||
|
|
||||||
|
if (!cont) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
} catch (Throwable t) {
|
} catch (Throwable t) {
|
||||||
if (!channel.isSocketClosed()) {
|
if (!channel.isSocketClosed()) {
|
||||||
fireExceptionCaught(channel, t);
|
fireExceptionCaught(channel, t);
|
||||||
@ -76,9 +83,48 @@ abstract class AbstractOioWorker<C extends AbstractOioChannel> implements Runnab
|
|||||||
channel.workerThread = null;
|
channel.workerThread = null;
|
||||||
|
|
||||||
// Clean up.
|
// Clean up.
|
||||||
close(channel, succeededFuture(channel));
|
close(channel, succeededFuture(channel), true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static boolean isIoThread(AbstractOioChannel channel) {
|
||||||
|
return Thread.currentThread() == channel.workerThread;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ChannelFuture executeInIoThread(Channel channel, Runnable task) {
|
||||||
|
if (channel instanceof AbstractOioChannel && isIoThread((AbstractOioChannel) channel)) {
|
||||||
|
try {
|
||||||
|
task.run();
|
||||||
|
return succeededFuture(channel);
|
||||||
|
} catch (Throwable t) {
|
||||||
|
return failedFuture(channel, t);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
ChannelRunnableWrapper channelRunnable = new ChannelRunnableWrapper(channel, task);
|
||||||
|
boolean added = eventQueue.offer(channelRunnable);
|
||||||
|
|
||||||
|
if (added) {
|
||||||
|
// as we set the SO_TIMEOUT to 1 second this task will get picked up in 1 second at latest
|
||||||
|
|
||||||
|
} else {
|
||||||
|
channelRunnable.setFailure(new RejectedExecutionException("Unable to queue task " + task));
|
||||||
|
}
|
||||||
|
return channelRunnable;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void processEventQueue() throws IOException {
|
||||||
|
for (;;) {
|
||||||
|
final Runnable task = eventQueue.poll();
|
||||||
|
if (task == null) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
task.run();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Process the incoming messages and also is responsible for call {@link Channels#fireMessageReceived(Channel, Object)} once a message
|
* Process the incoming messages and also is responsible for call {@link Channels#fireMessageReceived(Channel, Object)} once a message
|
||||||
* was processed without errors.
|
* was processed without errors.
|
||||||
@ -90,7 +136,8 @@ abstract class AbstractOioWorker<C extends AbstractOioChannel> implements Runnab
|
|||||||
|
|
||||||
static void setInterestOps(
|
static void setInterestOps(
|
||||||
AbstractOioChannel channel, ChannelFuture future, int interestOps) {
|
AbstractOioChannel channel, ChannelFuture future, int interestOps) {
|
||||||
|
boolean iothread = isIoThread(channel);
|
||||||
|
|
||||||
// Override OP_WRITE flag - a user cannot change this flag.
|
// Override OP_WRITE flag - a user cannot change this flag.
|
||||||
interestOps &= ~Channel.OP_WRITE;
|
interestOps &= ~Channel.OP_WRITE;
|
||||||
interestOps |= channel.getInterestOps() & Channel.OP_WRITE;
|
interestOps |= channel.getInterestOps() & Channel.OP_WRITE;
|
||||||
@ -118,18 +165,30 @@ abstract class AbstractOioWorker<C extends AbstractOioChannel> implements Runnab
|
|||||||
workerThread.interrupt();
|
workerThread.interrupt();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (iothread) {
|
||||||
fireChannelInterestChanged(channel);
|
fireChannelInterestChanged(channel);
|
||||||
|
} else {
|
||||||
|
fireChannelInterestChangedLater(channel);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} catch (Throwable t) {
|
} catch (Throwable t) {
|
||||||
future.setFailure(t);
|
future.setFailure(t);
|
||||||
fireExceptionCaught(channel, t);
|
if (iothread) {
|
||||||
|
fireExceptionCaught(channel, t);
|
||||||
|
} else {
|
||||||
|
fireExceptionCaughtLater(channel, t);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void close(AbstractOioChannel channel, ChannelFuture future) {
|
static void close(AbstractOioChannel channel, ChannelFuture future) {
|
||||||
|
close(channel, future, isIoThread(channel));
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void close(AbstractOioChannel channel, ChannelFuture future, boolean iothread) {
|
||||||
boolean connected = channel.isConnected();
|
boolean connected = channel.isConnected();
|
||||||
boolean bound = channel.isBound();
|
boolean bound = channel.isBound();
|
||||||
|
|
||||||
try {
|
try {
|
||||||
channel.closeSocket();
|
channel.closeSocket();
|
||||||
if (channel.setClosed()) {
|
if (channel.setClosed()) {
|
||||||
@ -141,18 +200,34 @@ abstract class AbstractOioWorker<C extends AbstractOioChannel> implements Runnab
|
|||||||
if (workerThread != null && currentThread != workerThread) {
|
if (workerThread != null && currentThread != workerThread) {
|
||||||
workerThread.interrupt();
|
workerThread.interrupt();
|
||||||
}
|
}
|
||||||
fireChannelDisconnected(channel);
|
if (iothread) {
|
||||||
|
fireChannelDisconnected(channel);
|
||||||
|
} else {
|
||||||
|
fireChannelDisconnectedLater(channel);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (bound) {
|
if (bound) {
|
||||||
fireChannelUnbound(channel);
|
if (iothread) {
|
||||||
|
fireChannelUnbound(channel);
|
||||||
|
} else {
|
||||||
|
fireChannelUnboundLater(channel);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (iothread) {
|
||||||
|
fireChannelClosed(channel);
|
||||||
|
} else {
|
||||||
|
fireChannelClosedLater(channel);
|
||||||
}
|
}
|
||||||
fireChannelClosed(channel);
|
|
||||||
} else {
|
} else {
|
||||||
future.setSuccess();
|
future.setSuccess();
|
||||||
}
|
}
|
||||||
} catch (Throwable t) {
|
} catch (Throwable t) {
|
||||||
future.setFailure(t);
|
future.setFailure(t);
|
||||||
fireExceptionCaught(channel, t);
|
if (iothread) {
|
||||||
|
fireExceptionCaught(channel, t);
|
||||||
|
} else {
|
||||||
|
fireExceptionCaughtLater(channel, t);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -21,7 +21,6 @@ import java.io.PushbackInputStream;
|
|||||||
import java.net.SocketAddress;
|
import java.net.SocketAddress;
|
||||||
import java.util.concurrent.Executor;
|
import java.util.concurrent.Executor;
|
||||||
|
|
||||||
import io.netty.channel.AbstractChannelSink;
|
|
||||||
import io.netty.channel.ChannelEvent;
|
import io.netty.channel.ChannelEvent;
|
||||||
import io.netty.channel.ChannelFuture;
|
import io.netty.channel.ChannelFuture;
|
||||||
import io.netty.channel.ChannelFutureListener;
|
import io.netty.channel.ChannelFutureListener;
|
||||||
@ -31,7 +30,7 @@ import io.netty.channel.ChannelStateEvent;
|
|||||||
import io.netty.channel.MessageEvent;
|
import io.netty.channel.MessageEvent;
|
||||||
import io.netty.util.internal.DeadLockProofWorker;
|
import io.netty.util.internal.DeadLockProofWorker;
|
||||||
|
|
||||||
class OioClientSocketPipelineSink extends AbstractChannelSink {
|
class OioClientSocketPipelineSink extends AbstractOioChannelSink {
|
||||||
|
|
||||||
private final Executor workerExecutor;
|
private final Executor workerExecutor;
|
||||||
|
|
||||||
|
@ -20,7 +20,6 @@ import static io.netty.channel.Channels.*;
|
|||||||
import java.net.SocketAddress;
|
import java.net.SocketAddress;
|
||||||
import java.util.concurrent.Executor;
|
import java.util.concurrent.Executor;
|
||||||
|
|
||||||
import io.netty.channel.AbstractChannelSink;
|
|
||||||
import io.netty.channel.ChannelEvent;
|
import io.netty.channel.ChannelEvent;
|
||||||
import io.netty.channel.ChannelFuture;
|
import io.netty.channel.ChannelFuture;
|
||||||
import io.netty.channel.ChannelFutureListener;
|
import io.netty.channel.ChannelFutureListener;
|
||||||
@ -30,7 +29,7 @@ import io.netty.channel.ChannelStateEvent;
|
|||||||
import io.netty.channel.MessageEvent;
|
import io.netty.channel.MessageEvent;
|
||||||
import io.netty.util.internal.DeadLockProofWorker;
|
import io.netty.util.internal.DeadLockProofWorker;
|
||||||
|
|
||||||
class OioDatagramPipelineSink extends AbstractChannelSink {
|
class OioDatagramPipelineSink extends AbstractOioChannelSink {
|
||||||
|
|
||||||
private final Executor workerExecutor;
|
private final Executor workerExecutor;
|
||||||
|
|
||||||
|
@ -63,6 +63,8 @@ class OioDatagramWorker extends AbstractOioWorker<OioDatagramChannel> {
|
|||||||
static void write(
|
static void write(
|
||||||
OioDatagramChannel channel, ChannelFuture future,
|
OioDatagramChannel channel, ChannelFuture future,
|
||||||
Object message, SocketAddress remoteAddress) {
|
Object message, SocketAddress remoteAddress) {
|
||||||
|
boolean iothread = isIoThread(channel);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
ChannelBuffer buf = (ChannelBuffer) message;
|
ChannelBuffer buf = (ChannelBuffer) message;
|
||||||
int offset = buf.readerIndex();
|
int offset = buf.readerIndex();
|
||||||
@ -84,27 +86,45 @@ class OioDatagramWorker extends AbstractOioWorker<OioDatagramChannel> {
|
|||||||
packet.setSocketAddress(remoteAddress);
|
packet.setSocketAddress(remoteAddress);
|
||||||
}
|
}
|
||||||
channel.socket.send(packet);
|
channel.socket.send(packet);
|
||||||
fireWriteComplete(channel, length);
|
if (iothread) {
|
||||||
|
fireWriteComplete(channel, length);
|
||||||
|
} else {
|
||||||
|
fireWriteCompleteLater(channel, length);
|
||||||
|
}
|
||||||
future.setSuccess();
|
future.setSuccess();
|
||||||
} catch (Throwable t) {
|
} catch (Throwable t) {
|
||||||
future.setFailure(t);
|
future.setFailure(t);
|
||||||
fireExceptionCaught(channel, t);
|
if (iothread) {
|
||||||
|
fireExceptionCaught(channel, t);
|
||||||
|
} else {
|
||||||
|
fireExceptionCaughtLater(channel, t);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void disconnect(OioDatagramChannel channel, ChannelFuture future) {
|
static void disconnect(OioDatagramChannel channel, ChannelFuture future) {
|
||||||
boolean connected = channel.isConnected();
|
boolean connected = channel.isConnected();
|
||||||
|
boolean iothread = isIoThread(channel);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
channel.socket.disconnect();
|
channel.socket.disconnect();
|
||||||
future.setSuccess();
|
future.setSuccess();
|
||||||
if (connected) {
|
if (connected) {
|
||||||
// Notify.
|
// Notify.
|
||||||
fireChannelDisconnected(channel);
|
if (iothread) {
|
||||||
|
fireChannelDisconnected(channel);
|
||||||
|
} else {
|
||||||
|
fireChannelDisconnectedLater(channel);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} catch (Throwable t) {
|
} catch (Throwable t) {
|
||||||
future.setFailure(t);
|
future.setFailure(t);
|
||||||
fireExceptionCaught(channel, t);
|
if (iothread) {
|
||||||
|
fireExceptionCaught(channel, t);
|
||||||
|
} else {
|
||||||
|
fireExceptionCaughtLater(channel, t);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -24,7 +24,6 @@ import java.net.SocketTimeoutException;
|
|||||||
import java.util.concurrent.Executor;
|
import java.util.concurrent.Executor;
|
||||||
import java.util.concurrent.atomic.AtomicInteger;
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
|
||||||
import io.netty.channel.AbstractChannelSink;
|
|
||||||
import io.netty.channel.Channel;
|
import io.netty.channel.Channel;
|
||||||
import io.netty.channel.ChannelEvent;
|
import io.netty.channel.ChannelEvent;
|
||||||
import io.netty.channel.ChannelFuture;
|
import io.netty.channel.ChannelFuture;
|
||||||
@ -36,7 +35,7 @@ import io.netty.logging.InternalLogger;
|
|||||||
import io.netty.logging.InternalLoggerFactory;
|
import io.netty.logging.InternalLoggerFactory;
|
||||||
import io.netty.util.internal.DeadLockProofWorker;
|
import io.netty.util.internal.DeadLockProofWorker;
|
||||||
|
|
||||||
class OioServerSocketPipelineSink extends AbstractChannelSink {
|
class OioServerSocketPipelineSink extends AbstractOioChannelSink {
|
||||||
|
|
||||||
static final InternalLogger logger =
|
static final InternalLogger logger =
|
||||||
InternalLoggerFactory.getInstance(OioServerSocketPipelineSink.class);
|
InternalLoggerFactory.getInstance(OioServerSocketPipelineSink.class);
|
||||||
|
@ -65,11 +65,16 @@ class OioWorker extends AbstractOioWorker<OioSocketChannel> {
|
|||||||
OioSocketChannel channel, ChannelFuture future,
|
OioSocketChannel channel, ChannelFuture future,
|
||||||
Object message) {
|
Object message) {
|
||||||
|
|
||||||
|
boolean iothread = isIoThread(channel);
|
||||||
OutputStream out = channel.getOutputStream();
|
OutputStream out = channel.getOutputStream();
|
||||||
if (out == null) {
|
if (out == null) {
|
||||||
Exception e = new ClosedChannelException();
|
Exception e = new ClosedChannelException();
|
||||||
future.setFailure(e);
|
future.setFailure(e);
|
||||||
fireExceptionCaught(channel, e);
|
if (iothread) {
|
||||||
|
fireExceptionCaught(channel, e);
|
||||||
|
} else {
|
||||||
|
fireExceptionCaughtLater(channel, e);
|
||||||
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -106,7 +111,11 @@ class OioWorker extends AbstractOioWorker<OioSocketChannel> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fireWriteComplete(channel, length);
|
if (iothread) {
|
||||||
|
fireWriteComplete(channel, length);
|
||||||
|
} else {
|
||||||
|
fireWriteCompleteLater(channel, length);
|
||||||
|
}
|
||||||
future.setSuccess();
|
future.setSuccess();
|
||||||
|
|
||||||
} catch (Throwable t) {
|
} catch (Throwable t) {
|
||||||
@ -118,7 +127,11 @@ class OioWorker extends AbstractOioWorker<OioSocketChannel> {
|
|||||||
t = new ClosedChannelException();
|
t = new ClosedChannelException();
|
||||||
}
|
}
|
||||||
future.setFailure(t);
|
future.setFailure(t);
|
||||||
fireExceptionCaught(channel, t);
|
if (iothread) {
|
||||||
|
fireExceptionCaught(channel, t);
|
||||||
|
} else {
|
||||||
|
fireExceptionCaughtLater(channel, t);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user