jdk/src/share/classes/java/net/AbstractPlainSocketImpl.java
author xdono
Wed, 02 Jul 2008 12:55:45 -0700
changeset 715 f16baef3a20e
parent 82 216899854ed5
child 1335 79ed7fd4bb49
permissions -rw-r--r--
6719955: Update copyright year Summary: Update copyright year for files that have been modified in 2008 Reviewed-by: ohair, tbell
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     1
/*
715
f16baef3a20e 6719955: Update copyright year
xdono
parents: 82
diff changeset
     2
 * Copyright 1995-2008 Sun Microsystems, Inc.  All Rights Reserved.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
90ce3da70b43 Initial load
duke
parents:
diff changeset
     4
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
90ce3da70b43 Initial load
duke
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
90ce3da70b43 Initial load
duke
parents:
diff changeset
     7
 * published by the Free Software Foundation.  Sun designates this
90ce3da70b43 Initial load
duke
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
90ce3da70b43 Initial load
duke
parents:
diff changeset
     9
 * by Sun in the LICENSE file that accompanied this code.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    10
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
90ce3da70b43 Initial load
duke
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
90ce3da70b43 Initial load
duke
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
90ce3da70b43 Initial load
duke
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
90ce3da70b43 Initial load
duke
parents:
diff changeset
    15
 * accompanied this code).
90ce3da70b43 Initial load
duke
parents:
diff changeset
    16
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
90ce3da70b43 Initial load
duke
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    20
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    21
 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    22
 * CA 95054 USA or visit www.sun.com if you need additional information or
90ce3da70b43 Initial load
duke
parents:
diff changeset
    23
 * have any questions.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    24
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    25
90ce3da70b43 Initial load
duke
parents:
diff changeset
    26
package java.net;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    27
90ce3da70b43 Initial load
duke
parents:
diff changeset
    28
import java.io.IOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    29
import java.io.InputStream;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    30
import java.io.OutputStream;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    31
import java.io.InterruptedIOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    32
import java.io.FileDescriptor;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    33
import java.io.ByteArrayOutputStream;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    34
90ce3da70b43 Initial load
duke
parents:
diff changeset
    35
import sun.net.ConnectionResetException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    36
90ce3da70b43 Initial load
duke
parents:
diff changeset
    37
/**
90ce3da70b43 Initial load
duke
parents:
diff changeset
    38
 * Default Socket Implementation. This implementation does
90ce3da70b43 Initial load
duke
parents:
diff changeset
    39
 * not implement any security checks.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    40
 * Note this class should <b>NOT</b> be public.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    41
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    42
 * @author  Steven B. Byrne
90ce3da70b43 Initial load
duke
parents:
diff changeset
    43
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    44
abstract class AbstractPlainSocketImpl extends SocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
    45
{
90ce3da70b43 Initial load
duke
parents:
diff changeset
    46
    /* instance variable for SO_TIMEOUT */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    47
    int timeout;   // timeout in millisec
90ce3da70b43 Initial load
duke
parents:
diff changeset
    48
    // traffic class
90ce3da70b43 Initial load
duke
parents:
diff changeset
    49
    private int trafficClass;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    50
90ce3da70b43 Initial load
duke
parents:
diff changeset
    51
    private boolean shut_rd = false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    52
    private boolean shut_wr = false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    53
90ce3da70b43 Initial load
duke
parents:
diff changeset
    54
    private SocketInputStream socketInputStream = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    55
90ce3da70b43 Initial load
duke
parents:
diff changeset
    56
    /* number of threads using the FileDescriptor */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    57
    protected int fdUseCount = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    58
90ce3da70b43 Initial load
duke
parents:
diff changeset
    59
    /* lock when increment/decrementing fdUseCount */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    60
    protected Object fdLock = new Object();
90ce3da70b43 Initial load
duke
parents:
diff changeset
    61
90ce3da70b43 Initial load
duke
parents:
diff changeset
    62
    /* indicates a close is pending on the file descriptor */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    63
    protected boolean closePending = false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    64
90ce3da70b43 Initial load
duke
parents:
diff changeset
    65
    /* indicates connection reset state */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    66
    private int CONNECTION_NOT_RESET = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    67
    private int CONNECTION_RESET_PENDING = 1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    68
    private int CONNECTION_RESET = 2;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    69
    private int resetState;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    70
    private Object resetLock = new Object();
90ce3da70b43 Initial load
duke
parents:
diff changeset
    71
90ce3da70b43 Initial load
duke
parents:
diff changeset
    72
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
    73
     * Load net library into runtime.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    74
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    75
    static {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    76
        java.security.AccessController.doPrivileged(
90ce3da70b43 Initial load
duke
parents:
diff changeset
    77
                  new sun.security.action.LoadLibraryAction("net"));
90ce3da70b43 Initial load
duke
parents:
diff changeset
    78
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    79
90ce3da70b43 Initial load
duke
parents:
diff changeset
    80
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
    81
     * Creates a socket with a boolean that specifies whether this
90ce3da70b43 Initial load
duke
parents:
diff changeset
    82
     * is a stream socket (true) or an unconnected UDP socket (false).
90ce3da70b43 Initial load
duke
parents:
diff changeset
    83
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    84
    protected synchronized void create(boolean stream) throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    85
        fd = new FileDescriptor();
90ce3da70b43 Initial load
duke
parents:
diff changeset
    86
        socketCreate(stream);
90ce3da70b43 Initial load
duke
parents:
diff changeset
    87
        if (socket != null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
    88
            socket.setCreated();
90ce3da70b43 Initial load
duke
parents:
diff changeset
    89
        if (serverSocket != null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
    90
            serverSocket.setCreated();
90ce3da70b43 Initial load
duke
parents:
diff changeset
    91
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    92
90ce3da70b43 Initial load
duke
parents:
diff changeset
    93
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
    94
     * Creates a socket and connects it to the specified port on
90ce3da70b43 Initial load
duke
parents:
diff changeset
    95
     * the specified host.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    96
     * @param host the specified host
90ce3da70b43 Initial load
duke
parents:
diff changeset
    97
     * @param port the specified port
90ce3da70b43 Initial load
duke
parents:
diff changeset
    98
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    99
    protected void connect(String host, int port)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   100
        throws UnknownHostException, IOException
90ce3da70b43 Initial load
duke
parents:
diff changeset
   101
    {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   102
        IOException pending = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   103
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   104
            InetAddress address = InetAddress.getByName(host);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   105
            this.port = port;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   106
            this.address = address;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   107
90ce3da70b43 Initial load
duke
parents:
diff changeset
   108
            try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   109
                connectToAddress(address, port, timeout);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   110
                return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   111
            } catch (IOException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   112
                pending = e;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   113
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   114
        } catch (UnknownHostException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   115
            pending = e;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   116
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   117
90ce3da70b43 Initial load
duke
parents:
diff changeset
   118
        // everything failed
90ce3da70b43 Initial load
duke
parents:
diff changeset
   119
        close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   120
        throw pending;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   121
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   122
90ce3da70b43 Initial load
duke
parents:
diff changeset
   123
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   124
     * Creates a socket and connects it to the specified address on
90ce3da70b43 Initial load
duke
parents:
diff changeset
   125
     * the specified port.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   126
     * @param address the address
90ce3da70b43 Initial load
duke
parents:
diff changeset
   127
     * @param port the specified port
90ce3da70b43 Initial load
duke
parents:
diff changeset
   128
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   129
    protected void connect(InetAddress address, int port) throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   130
        this.port = port;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   131
        this.address = address;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   132
90ce3da70b43 Initial load
duke
parents:
diff changeset
   133
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   134
            connectToAddress(address, port, timeout);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   135
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   136
        } catch (IOException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   137
            // everything failed
90ce3da70b43 Initial load
duke
parents:
diff changeset
   138
            close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   139
            throw e;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   140
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   141
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   142
90ce3da70b43 Initial load
duke
parents:
diff changeset
   143
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   144
     * Creates a socket and connects it to the specified address on
90ce3da70b43 Initial load
duke
parents:
diff changeset
   145
     * the specified port.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   146
     * @param address the address
90ce3da70b43 Initial load
duke
parents:
diff changeset
   147
     * @param timeout the timeout value in milliseconds, or zero for no timeout.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   148
     * @throws IOException if connection fails
90ce3da70b43 Initial load
duke
parents:
diff changeset
   149
     * @throws  IllegalArgumentException if address is null or is a
90ce3da70b43 Initial load
duke
parents:
diff changeset
   150
     *          SocketAddress subclass not supported by this socket
90ce3da70b43 Initial load
duke
parents:
diff changeset
   151
     * @since 1.4
90ce3da70b43 Initial load
duke
parents:
diff changeset
   152
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   153
    protected void connect(SocketAddress address, int timeout) throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   154
        if (address == null || !(address instanceof InetSocketAddress))
90ce3da70b43 Initial load
duke
parents:
diff changeset
   155
            throw new IllegalArgumentException("unsupported address type");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   156
        InetSocketAddress addr = (InetSocketAddress) address;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   157
        if (addr.isUnresolved())
90ce3da70b43 Initial load
duke
parents:
diff changeset
   158
            throw new UnknownHostException(addr.getHostName());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   159
        this.port = addr.getPort();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   160
        this.address = addr.getAddress();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   161
90ce3da70b43 Initial load
duke
parents:
diff changeset
   162
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   163
            connectToAddress(this.address, port, timeout);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   164
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   165
        } catch (IOException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   166
            // everything failed
90ce3da70b43 Initial load
duke
parents:
diff changeset
   167
            close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   168
            throw e;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   169
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   170
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   171
90ce3da70b43 Initial load
duke
parents:
diff changeset
   172
    private void connectToAddress(InetAddress address, int port, int timeout) throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   173
        if (address.isAnyLocalAddress()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   174
            doConnect(InetAddress.getLocalHost(), port, timeout);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   175
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   176
            doConnect(address, port, timeout);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   177
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   178
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   179
90ce3da70b43 Initial load
duke
parents:
diff changeset
   180
    public void setOption(int opt, Object val) throws SocketException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   181
        if (isClosedOrPending()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   182
            throw new SocketException("Socket Closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   183
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   184
        boolean on = true;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   185
        switch (opt) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   186
            /* check type safety b4 going native.  These should never
90ce3da70b43 Initial load
duke
parents:
diff changeset
   187
             * fail, since only java.Socket* has access to
90ce3da70b43 Initial load
duke
parents:
diff changeset
   188
             * PlainSocketImpl.setOption().
90ce3da70b43 Initial load
duke
parents:
diff changeset
   189
             */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   190
        case SO_LINGER:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   191
            if (val == null || (!(val instanceof Integer) && !(val instanceof Boolean)))
90ce3da70b43 Initial load
duke
parents:
diff changeset
   192
                throw new SocketException("Bad parameter for option");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   193
            if (val instanceof Boolean) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   194
                /* true only if disabling - enabling should be Integer */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   195
                on = false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   196
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   197
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   198
        case SO_TIMEOUT:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   199
            if (val == null || (!(val instanceof Integer)))
90ce3da70b43 Initial load
duke
parents:
diff changeset
   200
                throw new SocketException("Bad parameter for SO_TIMEOUT");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   201
            int tmp = ((Integer) val).intValue();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   202
            if (tmp < 0)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   203
                throw new IllegalArgumentException("timeout < 0");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   204
            timeout = tmp;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   205
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   206
        case IP_TOS:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   207
             if (val == null || !(val instanceof Integer)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   208
                 throw new SocketException("bad argument for IP_TOS");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   209
             }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   210
             trafficClass = ((Integer)val).intValue();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   211
             break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   212
        case SO_BINDADDR:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   213
            throw new SocketException("Cannot re-bind socket");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   214
        case TCP_NODELAY:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   215
            if (val == null || !(val instanceof Boolean))
90ce3da70b43 Initial load
duke
parents:
diff changeset
   216
                throw new SocketException("bad parameter for TCP_NODELAY");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   217
            on = ((Boolean)val).booleanValue();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   218
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   219
        case SO_SNDBUF:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   220
        case SO_RCVBUF:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   221
            if (val == null || !(val instanceof Integer) ||
90ce3da70b43 Initial load
duke
parents:
diff changeset
   222
                !(((Integer)val).intValue() > 0)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   223
                throw new SocketException("bad parameter for SO_SNDBUF " +
90ce3da70b43 Initial load
duke
parents:
diff changeset
   224
                                          "or SO_RCVBUF");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   225
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   226
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   227
        case SO_KEEPALIVE:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   228
            if (val == null || !(val instanceof Boolean))
90ce3da70b43 Initial load
duke
parents:
diff changeset
   229
                throw new SocketException("bad parameter for SO_KEEPALIVE");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   230
            on = ((Boolean)val).booleanValue();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   231
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   232
        case SO_OOBINLINE:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   233
            if (val == null || !(val instanceof Boolean))
90ce3da70b43 Initial load
duke
parents:
diff changeset
   234
                throw new SocketException("bad parameter for SO_OOBINLINE");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   235
            on = ((Boolean)val).booleanValue();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   236
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   237
        case SO_REUSEADDR:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   238
            if (val == null || !(val instanceof Boolean))
90ce3da70b43 Initial load
duke
parents:
diff changeset
   239
                throw new SocketException("bad parameter for SO_REUSEADDR");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   240
            on = ((Boolean)val).booleanValue();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   241
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   242
        default:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   243
            throw new SocketException("unrecognized TCP option: " + opt);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   244
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   245
        socketSetOption(opt, on, val);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   246
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   247
    public Object getOption(int opt) throws SocketException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   248
        if (isClosedOrPending()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   249
            throw new SocketException("Socket Closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   250
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   251
        if (opt == SO_TIMEOUT) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   252
            return new Integer(timeout);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   253
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   254
        int ret = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   255
        /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   256
         * The native socketGetOption() knows about 3 options.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   257
         * The 32 bit value it returns will be interpreted according
90ce3da70b43 Initial load
duke
parents:
diff changeset
   258
         * to what we're asking.  A return of -1 means it understands
90ce3da70b43 Initial load
duke
parents:
diff changeset
   259
         * the option but its turned off.  It will raise a SocketException
90ce3da70b43 Initial load
duke
parents:
diff changeset
   260
         * if "opt" isn't one it understands.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   261
         */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   262
90ce3da70b43 Initial load
duke
parents:
diff changeset
   263
        switch (opt) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   264
        case TCP_NODELAY:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   265
            ret = socketGetOption(opt, null);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   266
            return Boolean.valueOf(ret != -1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   267
        case SO_OOBINLINE:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   268
            ret = socketGetOption(opt, null);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   269
            return Boolean.valueOf(ret != -1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   270
        case SO_LINGER:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   271
            ret = socketGetOption(opt, null);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   272
            return (ret == -1) ? Boolean.FALSE: (Object)(new Integer(ret));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   273
        case SO_REUSEADDR:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   274
            ret = socketGetOption(opt, null);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   275
            return Boolean.valueOf(ret != -1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   276
        case SO_BINDADDR:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   277
            InetAddressContainer in = new InetAddressContainer();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   278
            ret = socketGetOption(opt, in);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   279
            return in.addr;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   280
        case SO_SNDBUF:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   281
        case SO_RCVBUF:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   282
            ret = socketGetOption(opt, null);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   283
            return new Integer(ret);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   284
        case IP_TOS:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   285
            ret = socketGetOption(opt, null);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   286
            if (ret == -1) { // ipv6 tos
90ce3da70b43 Initial load
duke
parents:
diff changeset
   287
                return new Integer(trafficClass);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   288
            } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   289
                return new Integer(ret);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   290
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   291
        case SO_KEEPALIVE:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   292
            ret = socketGetOption(opt, null);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   293
            return Boolean.valueOf(ret != -1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   294
        // should never get here
90ce3da70b43 Initial load
duke
parents:
diff changeset
   295
        default:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   296
            return null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   297
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   298
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   299
90ce3da70b43 Initial load
duke
parents:
diff changeset
   300
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   301
     * The workhorse of the connection operation.  Tries several times to
90ce3da70b43 Initial load
duke
parents:
diff changeset
   302
     * establish a connection to the given <host, port>.  If unsuccessful,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   303
     * throws an IOException indicating what went wrong.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   304
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   305
90ce3da70b43 Initial load
duke
parents:
diff changeset
   306
    synchronized void doConnect(InetAddress address, int port, int timeout) throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   307
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   308
            FileDescriptor fd = acquireFD();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   309
            try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   310
                socketConnect(address, port, timeout);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   311
                // If we have a ref. to the Socket, then sets the flags
90ce3da70b43 Initial load
duke
parents:
diff changeset
   312
                // created, bound & connected to true.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   313
                // This is normally done in Socket.connect() but some
90ce3da70b43 Initial load
duke
parents:
diff changeset
   314
                // subclasses of Socket may call impl.connect() directly!
90ce3da70b43 Initial load
duke
parents:
diff changeset
   315
                if (socket != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   316
                    socket.setBound();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   317
                    socket.setConnected();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   318
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   319
            } finally {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   320
                releaseFD();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   321
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   322
        } catch (IOException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   323
            close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   324
            throw e;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   325
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   326
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   327
90ce3da70b43 Initial load
duke
parents:
diff changeset
   328
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   329
     * Binds the socket to the specified address of the specified local port.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   330
     * @param address the address
90ce3da70b43 Initial load
duke
parents:
diff changeset
   331
     * @param port the port
90ce3da70b43 Initial load
duke
parents:
diff changeset
   332
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   333
    protected synchronized void bind(InetAddress address, int lport)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   334
        throws IOException
90ce3da70b43 Initial load
duke
parents:
diff changeset
   335
    {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   336
        socketBind(address, lport);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   337
        if (socket != null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   338
            socket.setBound();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   339
        if (serverSocket != null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   340
            serverSocket.setBound();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   341
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   342
90ce3da70b43 Initial load
duke
parents:
diff changeset
   343
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   344
     * Listens, for a specified amount of time, for connections.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   345
     * @param count the amount of time to listen for connections
90ce3da70b43 Initial load
duke
parents:
diff changeset
   346
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   347
    protected synchronized void listen(int count) throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   348
        socketListen(count);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   349
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   350
90ce3da70b43 Initial load
duke
parents:
diff changeset
   351
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   352
     * Accepts connections.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   353
     * @param s the connection
90ce3da70b43 Initial load
duke
parents:
diff changeset
   354
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   355
    protected void accept(SocketImpl s) throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   356
        FileDescriptor fd = acquireFD();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   357
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   358
            socketAccept(s);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   359
        } finally {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   360
            releaseFD();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   361
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   362
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   363
90ce3da70b43 Initial load
duke
parents:
diff changeset
   364
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   365
     * Gets an InputStream for this socket.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   366
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   367
    protected synchronized InputStream getInputStream() throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   368
        if (isClosedOrPending()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   369
            throw new IOException("Socket Closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   370
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   371
        if (shut_rd) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   372
            throw new IOException("Socket input is shutdown");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   373
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   374
        if (socketInputStream == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   375
            socketInputStream = new SocketInputStream(this);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   376
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   377
        return socketInputStream;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   378
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   379
90ce3da70b43 Initial load
duke
parents:
diff changeset
   380
    void setInputStream(SocketInputStream in) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   381
        socketInputStream = in;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   382
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   383
90ce3da70b43 Initial load
duke
parents:
diff changeset
   384
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   385
     * Gets an OutputStream for this socket.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   386
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   387
    protected synchronized OutputStream getOutputStream() throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   388
        if (isClosedOrPending()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   389
            throw new IOException("Socket Closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   390
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   391
        if (shut_wr) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   392
            throw new IOException("Socket output is shutdown");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   393
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   394
        return new SocketOutputStream(this);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   395
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   396
90ce3da70b43 Initial load
duke
parents:
diff changeset
   397
    void setFileDescriptor(FileDescriptor fd) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   398
        this.fd = fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   399
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   400
90ce3da70b43 Initial load
duke
parents:
diff changeset
   401
    void setAddress(InetAddress address) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   402
        this.address = address;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   403
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   404
90ce3da70b43 Initial load
duke
parents:
diff changeset
   405
    void setPort(int port) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   406
        this.port = port;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   407
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   408
90ce3da70b43 Initial load
duke
parents:
diff changeset
   409
    void setLocalPort(int localport) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   410
        this.localport = localport;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   411
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   412
90ce3da70b43 Initial load
duke
parents:
diff changeset
   413
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   414
     * Returns the number of bytes that can be read without blocking.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   415
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   416
    protected synchronized int available() throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   417
        if (isClosedOrPending()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   418
            throw new IOException("Stream closed.");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   419
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   420
90ce3da70b43 Initial load
duke
parents:
diff changeset
   421
        /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   422
         * If connection has been reset then return 0 to indicate
90ce3da70b43 Initial load
duke
parents:
diff changeset
   423
         * there are no buffered bytes.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   424
         */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   425
        if (isConnectionReset()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   426
            return 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   427
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   428
90ce3da70b43 Initial load
duke
parents:
diff changeset
   429
        /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   430
         * If no bytes available and we were previously notified
90ce3da70b43 Initial load
duke
parents:
diff changeset
   431
         * of a connection reset then we move to the reset state.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   432
         *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   433
         * If are notified of a connection reset then check
90ce3da70b43 Initial load
duke
parents:
diff changeset
   434
         * again if there are bytes buffered on the socket.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   435
         */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   436
        int n = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   437
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   438
            n = socketAvailable();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   439
            if (n == 0 && isConnectionResetPending()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   440
                setConnectionReset();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   441
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   442
        } catch (ConnectionResetException exc1) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   443
            setConnectionResetPending();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   444
            try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   445
                n = socketAvailable();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   446
                if (n == 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   447
                    setConnectionReset();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   448
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   449
            } catch (ConnectionResetException exc2) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   450
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   451
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   452
        return n;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   453
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   454
90ce3da70b43 Initial load
duke
parents:
diff changeset
   455
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   456
     * Closes the socket.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   457
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   458
    protected void close() throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   459
        synchronized(fdLock) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   460
            if (fd != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   461
                if (fdUseCount == 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   462
                    if (closePending) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   463
                        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   464
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   465
                    closePending = true;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   466
                    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   467
                     * We close the FileDescriptor in two-steps - first the
90ce3da70b43 Initial load
duke
parents:
diff changeset
   468
                     * "pre-close" which closes the socket but doesn't
90ce3da70b43 Initial load
duke
parents:
diff changeset
   469
                     * release the underlying file descriptor. This operation
90ce3da70b43 Initial load
duke
parents:
diff changeset
   470
                     * may be lengthy due to untransmitted data and a long
90ce3da70b43 Initial load
duke
parents:
diff changeset
   471
                     * linger interval. Once the pre-close is done we do the
90ce3da70b43 Initial load
duke
parents:
diff changeset
   472
                     * actual socket to release the fd.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   473
                     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   474
                    try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   475
                        socketPreClose();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   476
                    } finally {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   477
                        socketClose();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   478
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   479
                    fd = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   480
                    return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   481
                } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   482
                    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   483
                     * If a thread has acquired the fd and a close
90ce3da70b43 Initial load
duke
parents:
diff changeset
   484
                     * isn't pending then use a deferred close.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   485
                     * Also decrement fdUseCount to signal the last
90ce3da70b43 Initial load
duke
parents:
diff changeset
   486
                     * thread that releases the fd to close it.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   487
                     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   488
                    if (!closePending) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   489
                        closePending = true;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   490
                        fdUseCount--;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   491
                        socketPreClose();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   492
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   493
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   494
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   495
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   496
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   497
90ce3da70b43 Initial load
duke
parents:
diff changeset
   498
    void reset() throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   499
        if (fd != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   500
            socketClose();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   501
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   502
        fd = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   503
        super.reset();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   504
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   505
90ce3da70b43 Initial load
duke
parents:
diff changeset
   506
90ce3da70b43 Initial load
duke
parents:
diff changeset
   507
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   508
     * Shutdown read-half of the socket connection;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   509
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   510
    protected void shutdownInput() throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   511
      if (fd != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   512
          socketShutdown(SHUT_RD);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   513
          if (socketInputStream != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   514
              socketInputStream.setEOF(true);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   515
          }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   516
          shut_rd = true;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   517
      }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   518
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   519
90ce3da70b43 Initial load
duke
parents:
diff changeset
   520
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   521
     * Shutdown write-half of the socket connection;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   522
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   523
    protected void shutdownOutput() throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   524
      if (fd != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   525
          socketShutdown(SHUT_WR);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   526
          shut_wr = true;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   527
      }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   528
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   529
90ce3da70b43 Initial load
duke
parents:
diff changeset
   530
    protected boolean supportsUrgentData () {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   531
        return true;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   532
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   533
90ce3da70b43 Initial load
duke
parents:
diff changeset
   534
    protected void sendUrgentData (int data) throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   535
        if (fd == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   536
            throw new IOException("Socket Closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   537
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   538
        socketSendUrgentData (data);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   539
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   540
90ce3da70b43 Initial load
duke
parents:
diff changeset
   541
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   542
     * Cleans up if the user forgets to close it.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   543
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   544
    protected void finalize() throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   545
        close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   546
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   547
90ce3da70b43 Initial load
duke
parents:
diff changeset
   548
90ce3da70b43 Initial load
duke
parents:
diff changeset
   549
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   550
     * "Acquires" and returns the FileDescriptor for this impl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   551
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   552
     * A corresponding releaseFD is required to "release" the
90ce3da70b43 Initial load
duke
parents:
diff changeset
   553
     * FileDescriptor.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   554
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   555
    FileDescriptor acquireFD() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   556
        synchronized (fdLock) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   557
            fdUseCount++;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   558
            return fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   559
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   560
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   561
90ce3da70b43 Initial load
duke
parents:
diff changeset
   562
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   563
     * "Release" the FileDescriptor for this impl.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   564
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   565
     * If the use count goes to -1 then the socket is closed.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   566
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   567
    void releaseFD() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   568
        synchronized (fdLock) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   569
            fdUseCount--;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   570
            if (fdUseCount == -1) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   571
                if (fd != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   572
                    try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   573
                        socketClose();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   574
                    } catch (IOException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   575
                    } finally {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   576
                        fd = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   577
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   578
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   579
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   580
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   581
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   582
90ce3da70b43 Initial load
duke
parents:
diff changeset
   583
    public boolean isConnectionReset() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   584
        synchronized (resetLock) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   585
            return (resetState == CONNECTION_RESET);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   586
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   587
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   588
90ce3da70b43 Initial load
duke
parents:
diff changeset
   589
    public boolean isConnectionResetPending() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   590
        synchronized (resetLock) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   591
            return (resetState == CONNECTION_RESET_PENDING);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   592
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   593
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   594
90ce3da70b43 Initial load
duke
parents:
diff changeset
   595
    public void setConnectionReset() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   596
        synchronized (resetLock) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   597
            resetState = CONNECTION_RESET;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   598
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   599
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   600
90ce3da70b43 Initial load
duke
parents:
diff changeset
   601
    public void setConnectionResetPending() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   602
        synchronized (resetLock) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   603
            if (resetState == CONNECTION_NOT_RESET) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   604
                resetState = CONNECTION_RESET_PENDING;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   605
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   606
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   607
90ce3da70b43 Initial load
duke
parents:
diff changeset
   608
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   609
90ce3da70b43 Initial load
duke
parents:
diff changeset
   610
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   611
     * Return true if already closed or close is pending
90ce3da70b43 Initial load
duke
parents:
diff changeset
   612
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   613
    public boolean isClosedOrPending() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   614
        /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   615
         * Lock on fdLock to ensure that we wait if a
90ce3da70b43 Initial load
duke
parents:
diff changeset
   616
         * close is in progress.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   617
         */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   618
        synchronized (fdLock) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   619
            if (closePending || (fd == null)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   620
                return true;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   621
            } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   622
                return false;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   623
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   624
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   625
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   626
90ce3da70b43 Initial load
duke
parents:
diff changeset
   627
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   628
     * Return the current value of SO_TIMEOUT
90ce3da70b43 Initial load
duke
parents:
diff changeset
   629
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   630
    public int getTimeout() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   631
        return timeout;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   632
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   633
90ce3da70b43 Initial load
duke
parents:
diff changeset
   634
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   635
     * "Pre-close" a socket by dup'ing the file descriptor - this enables
90ce3da70b43 Initial load
duke
parents:
diff changeset
   636
     * the socket to be closed without releasing the file descriptor.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   637
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   638
    private void socketPreClose() throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   639
        socketClose0(true);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   640
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   641
90ce3da70b43 Initial load
duke
parents:
diff changeset
   642
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   643
     * Close the socket (and release the file descriptor).
90ce3da70b43 Initial load
duke
parents:
diff changeset
   644
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   645
    protected void socketClose() throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   646
        socketClose0(false);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   647
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   648
90ce3da70b43 Initial load
duke
parents:
diff changeset
   649
    abstract void socketCreate(boolean isServer) throws IOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   650
    abstract void socketConnect(InetAddress address, int port, int timeout)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   651
        throws IOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   652
    abstract void socketBind(InetAddress address, int port)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   653
        throws IOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   654
    abstract void socketListen(int count)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   655
        throws IOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   656
    abstract void socketAccept(SocketImpl s)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   657
        throws IOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   658
    abstract int socketAvailable()
90ce3da70b43 Initial load
duke
parents:
diff changeset
   659
        throws IOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   660
    abstract void socketClose0(boolean useDeferredClose)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   661
        throws IOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   662
    abstract void socketShutdown(int howto)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   663
        throws IOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   664
    abstract void socketSetOption(int cmd, boolean on, Object value)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   665
        throws SocketException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   666
    abstract int socketGetOption(int opt, Object iaContainerObj) throws SocketException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   667
    abstract void socketSendUrgentData(int data)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   668
        throws IOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   669
90ce3da70b43 Initial load
duke
parents:
diff changeset
   670
    public final static int SHUT_RD = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   671
    public final static int SHUT_WR = 1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   672
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   673
90ce3da70b43 Initial load
duke
parents:
diff changeset
   674
class InetAddressContainer {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   675
    InetAddress addr;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   676
}