aboutsummaryrefslogtreecommitdiff
path: root/engine/src/networking/com/jme3/network/kernel/tcp/NioEndpoint.java
diff options
context:
space:
mode:
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.java181
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 + "]";
+ }
+}