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