diff options
Diffstat (limited to 'engine/src/networking/com/jme3/network/kernel/tcp/NioEndpoint.java')
-rw-r--r-- | engine/src/networking/com/jme3/network/kernel/tcp/NioEndpoint.java | 181 |
1 files changed, 181 insertions, 0 deletions
diff --git a/engine/src/networking/com/jme3/network/kernel/tcp/NioEndpoint.java b/engine/src/networking/com/jme3/network/kernel/tcp/NioEndpoint.java new file mode 100644 index 0000000..87b721f --- /dev/null +++ b/engine/src/networking/com/jme3/network/kernel/tcp/NioEndpoint.java @@ -0,0 +1,181 @@ +/* + * Copyright (c) 2011 jMonkeyEngine + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are + * met: + * + * * Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * * Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * + * * Neither the name of 'jMonkeyEngine' nor the names of its contributors + * may be used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED + * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR + * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR + * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, + * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, + * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR + * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF + * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING + * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS + * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ + +package com.jme3.network.kernel.tcp; + +import com.jme3.network.kernel.Endpoint; +import com.jme3.network.kernel.Kernel; +import com.jme3.network.kernel.KernelException; +import java.io.IOException; +import java.nio.ByteBuffer; +import java.nio.channels.SocketChannel; +import java.util.concurrent.ConcurrentLinkedQueue; + + +/** + * Endpoint implementation that encapsulates the + * channel IO based connection information and keeps + * track of the outbound data queue for the channel. + * + * @version $Revision: 8944 $ + * @author Paul Speed + */ +public class NioEndpoint implements Endpoint +{ + protected static final ByteBuffer CLOSE_MARKER = ByteBuffer.allocate(0); + + private long id; + private SocketChannel socket; + private SelectorKernel kernel; + private ConcurrentLinkedQueue<ByteBuffer> outbound = new ConcurrentLinkedQueue<ByteBuffer>(); + private boolean closing = false; + + public NioEndpoint( SelectorKernel kernel, long id, SocketChannel socket ) + { + this.id = id; + this.socket = socket; + this.kernel = kernel; + } + + public Kernel getKernel() + { + return kernel; + } + + public void close() + { + close(false); + } + + public void close( boolean flushData ) + { + if( flushData ) { + closing = true; + + // Enqueue a close marker message to let the server + // know we should close + send( CLOSE_MARKER, false, true ); + + return; + } + + try { + // Note: even though we may be disconnected from the socket.isConnected() + // standpoint, it's still safest to tell the kernel so that it can be sure + // to stop managing us gracefully. + kernel.closeEndpoint(this); + } catch( IOException e ) { + throw new KernelException( "Error closing endpoint for socket:" + socket, e ); + } + } + + public long getId() + { + return id; + } + + public String getAddress() + { + return String.valueOf(socket.socket().getRemoteSocketAddress()); + } + + public boolean isConnected() + { + return socket.isConnected(); + } + + /** + * The wakeup option is used internally when the kernel is + * broadcasting out to a bunch of endpoints and doesn't want to + * necessarily wakeup right away. + */ + protected void send( ByteBuffer data, boolean copy, boolean wakeup ) + { + // We create a ByteBuffer per endpoint since we + // use it to track the data sent to each endpoint + // separately. + ByteBuffer buffer; + if( !copy ) { + buffer = data; + } else { + // Copy the buffer + buffer = ByteBuffer.allocate(data.remaining()); + buffer.put(data); + buffer.flip(); + } + + // Queue it up + outbound.add(buffer); + + if( wakeup ) + kernel.wakeupSelector(); + } + + /** + * Called by the SelectorKernel to get the current top + * buffer for writing. + */ + protected ByteBuffer peekPending() + { + return outbound.peek(); + } + + /** + * Called by the SelectorKernel when the top buffer + * has been exhausted. + */ + protected ByteBuffer removePending() + { + return outbound.poll(); + } + + protected boolean hasPending() + { + return !outbound.isEmpty(); + } + + public void send( ByteBuffer data ) + { + if( data == null ) { + throw new IllegalArgumentException( "Data cannot be null." ); + } + if( closing ) { + throw new KernelException( "Endpoint has been closed:" + socket ); + } + send( data, true, true ); + } + + public String toString() + { + return "NioEndpoint[" + id + ", " + socket + "]"; + } +} |