2010-12-06 03:21:09 +00:00
|
|
|
/* Copyright (c) 2008-2010, Avian Contributors
|
2008-02-19 18:06:52 +00:00
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software
|
|
|
|
for any purpose with or without fee is hereby granted, provided
|
|
|
|
that the above copyright notice and this permission notice appear
|
|
|
|
in all copies.
|
|
|
|
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
|
|
details. */
|
|
|
|
|
2007-10-05 21:32:56 +00:00
|
|
|
package java.nio.channels;
|
|
|
|
|
|
|
|
import java.io.IOException;
|
2008-11-22 22:32:53 +00:00
|
|
|
import java.net.SocketException;
|
2008-03-25 17:18:17 +00:00
|
|
|
import java.net.SocketAddress;
|
2007-10-05 21:32:56 +00:00
|
|
|
import java.net.InetSocketAddress;
|
2008-11-22 22:32:53 +00:00
|
|
|
import java.net.Socket;
|
2007-10-05 21:32:56 +00:00
|
|
|
import java.nio.ByteBuffer;
|
|
|
|
|
|
|
|
public class SocketChannel extends SelectableChannel
|
2011-01-12 01:29:48 +00:00
|
|
|
implements ReadableByteChannel, GatheringByteChannel
|
2007-10-05 21:32:56 +00:00
|
|
|
{
|
|
|
|
public static final int InvalidSocket = -1;
|
|
|
|
|
2008-11-22 22:32:53 +00:00
|
|
|
int socket = InvalidSocket;
|
|
|
|
boolean connected = false;
|
2011-01-12 01:29:48 +00:00
|
|
|
boolean readyToConnect = false;
|
2009-08-03 14:58:56 +00:00
|
|
|
boolean blocking = true;
|
2007-10-05 21:32:56 +00:00
|
|
|
|
2010-06-14 22:09:56 +00:00
|
|
|
public static SocketChannel open() throws IOException {
|
|
|
|
Socket.init();
|
|
|
|
|
2007-10-05 21:32:56 +00:00
|
|
|
return new SocketChannel();
|
|
|
|
}
|
|
|
|
|
2009-08-03 14:58:56 +00:00
|
|
|
public SelectableChannel configureBlocking(boolean v) throws IOException {
|
|
|
|
blocking = v;
|
|
|
|
if (socket != InvalidSocket) {
|
|
|
|
configureBlocking(socket, v);
|
|
|
|
}
|
2008-03-25 17:18:17 +00:00
|
|
|
return this;
|
2007-10-05 21:32:56 +00:00
|
|
|
}
|
|
|
|
|
2009-08-03 14:58:56 +00:00
|
|
|
public boolean isBlocking() {
|
|
|
|
return blocking;
|
|
|
|
}
|
|
|
|
|
2009-10-29 22:19:33 +00:00
|
|
|
public boolean isConnected() {
|
|
|
|
return connected;
|
|
|
|
}
|
|
|
|
|
2008-11-22 22:32:53 +00:00
|
|
|
public Socket socket() {
|
|
|
|
return new Handle();
|
|
|
|
}
|
|
|
|
|
2009-08-03 14:58:56 +00:00
|
|
|
public boolean connect(SocketAddress address) throws IOException {
|
2008-03-25 17:18:17 +00:00
|
|
|
InetSocketAddress a;
|
|
|
|
try {
|
|
|
|
a = (InetSocketAddress) address;
|
|
|
|
} catch (ClassCastException e) {
|
|
|
|
throw new UnsupportedAddressTypeException();
|
|
|
|
}
|
|
|
|
socket = doConnect(a.getHostName(), a.getPort());
|
2009-08-03 14:58:56 +00:00
|
|
|
configureBlocking(blocking);
|
2008-03-25 17:18:17 +00:00
|
|
|
return connected;
|
2007-10-05 21:32:56 +00:00
|
|
|
}
|
|
|
|
|
2009-10-08 22:26:20 +00:00
|
|
|
public boolean finishConnect() throws IOException {
|
2009-10-29 22:19:33 +00:00
|
|
|
if (! connected) {
|
2011-01-12 01:29:48 +00:00
|
|
|
while (blocking && ! readyToConnect) {
|
|
|
|
Selector selector = Selector.open();
|
|
|
|
SelectionKey key = register(selector, SelectionKey.OP_CONNECT, null);
|
|
|
|
|
|
|
|
selector.select();
|
|
|
|
}
|
|
|
|
|
|
|
|
natFinishConnect(socket);
|
|
|
|
|
|
|
|
connected = readyToConnect;
|
2009-10-29 22:19:33 +00:00
|
|
|
}
|
2011-01-12 01:29:48 +00:00
|
|
|
|
2009-10-29 22:19:33 +00:00
|
|
|
return connected;
|
2009-10-08 22:26:20 +00:00
|
|
|
}
|
|
|
|
|
2007-10-05 21:32:56 +00:00
|
|
|
public void close() throws IOException {
|
2008-03-25 22:17:29 +00:00
|
|
|
if (isOpen()) {
|
2008-03-25 17:18:17 +00:00
|
|
|
super.close();
|
|
|
|
closeSocket();
|
|
|
|
}
|
2007-10-05 21:32:56 +00:00
|
|
|
}
|
|
|
|
|
2009-08-03 14:58:56 +00:00
|
|
|
private int doConnect(String host, int port) throws IOException {
|
2009-03-02 22:22:17 +00:00
|
|
|
if (host == null) throw new NullPointerException();
|
|
|
|
|
2007-10-05 21:32:56 +00:00
|
|
|
boolean b[] = new boolean[1];
|
2009-08-03 14:58:56 +00:00
|
|
|
int s = natDoConnect(host, port, blocking, b);
|
2007-10-05 21:32:56 +00:00
|
|
|
connected = b[0];
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int read(ByteBuffer b) throws IOException {
|
2008-03-25 17:18:17 +00:00
|
|
|
if (! isOpen()) return -1;
|
2007-10-05 21:32:56 +00:00
|
|
|
if (b.remaining() == 0) return 0;
|
2009-03-02 22:22:17 +00:00
|
|
|
|
|
|
|
byte[] array = b.array();
|
|
|
|
if (array == null) throw new NullPointerException();
|
|
|
|
|
2010-06-04 21:37:22 +00:00
|
|
|
int r = natRead(socket, array, b.arrayOffset() + b.position(), b.remaining(), blocking);
|
2007-10-07 15:53:07 +00:00
|
|
|
if (r > 0) {
|
|
|
|
b.position(b.position() + r);
|
|
|
|
}
|
2007-10-05 21:32:56 +00:00
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int write(ByteBuffer b) throws IOException {
|
|
|
|
if (! connected) {
|
|
|
|
natThrowWriteError(socket);
|
|
|
|
}
|
2007-10-11 21:41:23 +00:00
|
|
|
if (b.remaining() == 0) return 0;
|
2009-03-02 22:22:17 +00:00
|
|
|
|
|
|
|
byte[] array = b.array();
|
|
|
|
if (array == null) throw new NullPointerException();
|
|
|
|
|
2010-06-04 21:37:22 +00:00
|
|
|
int w = natWrite(socket, array, b.arrayOffset() + b.position(), b.remaining(), blocking);
|
2007-10-07 17:35:48 +00:00
|
|
|
if (w > 0) {
|
|
|
|
b.position(b.position() + w);
|
|
|
|
}
|
2007-10-05 21:32:56 +00:00
|
|
|
return w;
|
|
|
|
}
|
|
|
|
|
2011-01-12 01:29:48 +00:00
|
|
|
public long write(ByteBuffer[] srcs) throws IOException {
|
|
|
|
return write(srcs, 0, srcs.length);
|
|
|
|
}
|
|
|
|
|
|
|
|
public long write(ByteBuffer[] srcs, int offset, int length)
|
|
|
|
throws IOException
|
|
|
|
{
|
|
|
|
long total = 0;
|
|
|
|
for (int i = offset; i < offset + length; ++i) {
|
|
|
|
total += write(srcs[i]);
|
|
|
|
if (srcs[i].hasRemaining()) {
|
|
|
|
return total;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return total;
|
|
|
|
}
|
|
|
|
|
2007-10-05 21:32:56 +00:00
|
|
|
private void closeSocket() {
|
|
|
|
natCloseSocket(socket);
|
|
|
|
}
|
|
|
|
|
|
|
|
int socketFD() {
|
|
|
|
return socket;
|
|
|
|
}
|
|
|
|
|
2011-01-12 01:29:48 +00:00
|
|
|
void handleReadyOps(int ops) {
|
|
|
|
if ((ops & SelectionKey.OP_CONNECT) != 0) {
|
|
|
|
readyToConnect = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-11-22 22:32:53 +00:00
|
|
|
public class Handle extends Socket {
|
|
|
|
public void setTcpNoDelay(boolean on) throws SocketException {
|
|
|
|
natSetTcpNoDelay(socket, on);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-08-03 14:58:56 +00:00
|
|
|
private static native void configureBlocking(int socket, boolean blocking)
|
|
|
|
throws IOException;
|
|
|
|
|
2008-11-22 22:32:53 +00:00
|
|
|
private static native void natSetTcpNoDelay(int socket, boolean on)
|
|
|
|
throws SocketException;
|
|
|
|
|
2009-08-03 14:58:56 +00:00
|
|
|
private static native int natDoConnect(String host, int port, boolean blocking, boolean[] connected)
|
|
|
|
throws IOException;
|
2011-01-12 01:29:48 +00:00
|
|
|
private static native void natFinishConnect(int socket)
|
2009-10-08 22:26:20 +00:00
|
|
|
throws IOException;
|
2010-06-04 21:37:22 +00:00
|
|
|
private static native int natRead(int socket, byte[] buffer, int offset, int length, boolean blocking)
|
2007-10-05 21:32:56 +00:00
|
|
|
throws IOException;
|
2010-06-04 21:37:22 +00:00
|
|
|
private static native int natWrite(int socket, byte[] buffer, int offset, int length, boolean blocking)
|
2007-10-05 21:32:56 +00:00
|
|
|
throws IOException;
|
|
|
|
private static native void natThrowWriteError(int socket) throws IOException;
|
|
|
|
private static native void natCloseSocket(int socket);
|
|
|
|
}
|