jdk/src/solaris/native/java/net/PlainSocketImpl.c
author tbell
Fri, 27 Feb 2009 10:54:11 -0800
changeset 2091 7faffd237305
parent 1335 79ed7fd4bb49
child 5506 202f599c92aa
permissions -rw-r--r--
Merge
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: 710
diff changeset
     2
 * Copyright 1997-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
#include <errno.h>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    27
#include <string.h>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    28
#include <sys/types.h>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    29
#include <sys/socket.h>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    30
#if defined(__linux__) && !defined(USE_SELECT)
90ce3da70b43 Initial load
duke
parents:
diff changeset
    31
#include <sys/poll.h>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    32
#endif
90ce3da70b43 Initial load
duke
parents:
diff changeset
    33
#include <netinet/tcp.h>        /* Defines TCP_NODELAY, needed for 2.6 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    34
#include <netinet/in.h>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    35
#ifdef __linux__
90ce3da70b43 Initial load
duke
parents:
diff changeset
    36
#include <netinet/ip.h>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    37
#endif
90ce3da70b43 Initial load
duke
parents:
diff changeset
    38
#include <netdb.h>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    39
#include <stdlib.h>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    40
90ce3da70b43 Initial load
duke
parents:
diff changeset
    41
#ifdef __solaris__
90ce3da70b43 Initial load
duke
parents:
diff changeset
    42
#include <fcntl.h>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    43
#endif
90ce3da70b43 Initial load
duke
parents:
diff changeset
    44
#ifdef __linux__
90ce3da70b43 Initial load
duke
parents:
diff changeset
    45
#include <linux/unistd.h>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    46
#include <linux/sysctl.h>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    47
#endif
90ce3da70b43 Initial load
duke
parents:
diff changeset
    48
90ce3da70b43 Initial load
duke
parents:
diff changeset
    49
#include "jvm.h"
90ce3da70b43 Initial load
duke
parents:
diff changeset
    50
#include "jni_util.h"
90ce3da70b43 Initial load
duke
parents:
diff changeset
    51
#include "net_util.h"
90ce3da70b43 Initial load
duke
parents:
diff changeset
    52
90ce3da70b43 Initial load
duke
parents:
diff changeset
    53
#include "java_net_SocketOptions.h"
90ce3da70b43 Initial load
duke
parents:
diff changeset
    54
#include "java_net_PlainSocketImpl.h"
90ce3da70b43 Initial load
duke
parents:
diff changeset
    55
90ce3da70b43 Initial load
duke
parents:
diff changeset
    56
/************************************************************************
90ce3da70b43 Initial load
duke
parents:
diff changeset
    57
 * PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
    58
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    59
90ce3da70b43 Initial load
duke
parents:
diff changeset
    60
static jfieldID IO_fd_fdID;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    61
90ce3da70b43 Initial load
duke
parents:
diff changeset
    62
jfieldID psi_fdID;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    63
jfieldID psi_addressID;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    64
jfieldID psi_ipaddressID;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    65
jfieldID psi_portID;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    66
jfieldID psi_localportID;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    67
jfieldID psi_timeoutID;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    68
jfieldID psi_trafficClassID;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    69
jfieldID psi_serverSocketID;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    70
jfieldID psi_fdLockID;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    71
jfieldID psi_closePendingID;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    72
90ce3da70b43 Initial load
duke
parents:
diff changeset
    73
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
    74
 * file descriptor used for dup2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    75
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    76
static int marker_fd = -1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    77
90ce3da70b43 Initial load
duke
parents:
diff changeset
    78
90ce3da70b43 Initial load
duke
parents:
diff changeset
    79
#define SET_NONBLOCKING(fd) {           \
90ce3da70b43 Initial load
duke
parents:
diff changeset
    80
        int flags = fcntl(fd, F_GETFL); \
90ce3da70b43 Initial load
duke
parents:
diff changeset
    81
        flags |= O_NONBLOCK;            \
90ce3da70b43 Initial load
duke
parents:
diff changeset
    82
        fcntl(fd, F_SETFL, flags);      \
90ce3da70b43 Initial load
duke
parents:
diff changeset
    83
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
    84
90ce3da70b43 Initial load
duke
parents:
diff changeset
    85
#define SET_BLOCKING(fd) {              \
90ce3da70b43 Initial load
duke
parents:
diff changeset
    86
        int flags = fcntl(fd, F_GETFL); \
90ce3da70b43 Initial load
duke
parents:
diff changeset
    87
        flags &= ~O_NONBLOCK;           \
90ce3da70b43 Initial load
duke
parents:
diff changeset
    88
        fcntl(fd, F_SETFL, flags);      \
90ce3da70b43 Initial load
duke
parents:
diff changeset
    89
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
    90
90ce3da70b43 Initial load
duke
parents:
diff changeset
    91
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
    92
 * Create the marker file descriptor by establishing a loopback connection
90ce3da70b43 Initial load
duke
parents:
diff changeset
    93
 * which we shutdown but do not close the fd. The result is an fd that
90ce3da70b43 Initial load
duke
parents:
diff changeset
    94
 * can be used for read/write.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    95
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    96
static int getMarkerFD()
90ce3da70b43 Initial load
duke
parents:
diff changeset
    97
{
710
57367409c4a5 6483406: new ServerSocket() sometimes takes more than 3 minutes on Suse Linux
jccollet
parents: 624
diff changeset
    98
    int sv[2];
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    99
710
57367409c4a5 6483406: new ServerSocket() sometimes takes more than 3 minutes on Suse Linux
jccollet
parents: 624
diff changeset
   100
#ifdef AF_UNIX
57367409c4a5 6483406: new ServerSocket() sometimes takes more than 3 minutes on Suse Linux
jccollet
parents: 624
diff changeset
   101
    if (socketpair(AF_UNIX, SOCK_STREAM, 0, sv) == -1) {
57367409c4a5 6483406: new ServerSocket() sometimes takes more than 3 minutes on Suse Linux
jccollet
parents: 624
diff changeset
   102
        return -1;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   103
    }
710
57367409c4a5 6483406: new ServerSocket() sometimes takes more than 3 minutes on Suse Linux
jccollet
parents: 624
diff changeset
   104
#else
57367409c4a5 6483406: new ServerSocket() sometimes takes more than 3 minutes on Suse Linux
jccollet
parents: 624
diff changeset
   105
    return -1;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   106
#endif
90ce3da70b43 Initial load
duke
parents:
diff changeset
   107
90ce3da70b43 Initial load
duke
parents:
diff changeset
   108
    /*
710
57367409c4a5 6483406: new ServerSocket() sometimes takes more than 3 minutes on Suse Linux
jccollet
parents: 624
diff changeset
   109
     * Finally shutdown sv[0] (any reads to this fd will get
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   110
     * EOF; any writes will get an error).
90ce3da70b43 Initial load
duke
parents:
diff changeset
   111
     */
710
57367409c4a5 6483406: new ServerSocket() sometimes takes more than 3 minutes on Suse Linux
jccollet
parents: 624
diff changeset
   112
    JVM_SocketShutdown(sv[0], 2);
57367409c4a5 6483406: new ServerSocket() sometimes takes more than 3 minutes on Suse Linux
jccollet
parents: 624
diff changeset
   113
    JVM_SocketClose(sv[1]);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   114
710
57367409c4a5 6483406: new ServerSocket() sometimes takes more than 3 minutes on Suse Linux
jccollet
parents: 624
diff changeset
   115
    return sv[0];
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   116
}
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
 * Return the file descriptor given a PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   121
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   122
static int getFD(JNIEnv *env, jobject this) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   123
    jobject fdObj = (*env)->GetObjectField(env, this, psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   124
    CHECK_NULL_RETURN(fdObj, -1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   125
    return (*env)->GetIntField(env, fdObj, IO_fd_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   126
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   127
90ce3da70b43 Initial load
duke
parents:
diff changeset
   128
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   129
 * The initroto function is called whenever PlainSocketImpl is
90ce3da70b43 Initial load
duke
parents:
diff changeset
   130
 * loaded, to cache fieldIds for efficiency. This is called everytime
90ce3da70b43 Initial load
duke
parents:
diff changeset
   131
 * the Java class is loaded.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   132
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   133
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   134
 * Method:    initProto
90ce3da70b43 Initial load
duke
parents:
diff changeset
   135
 * Signature: ()V
90ce3da70b43 Initial load
duke
parents:
diff changeset
   136
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   137
JNIEXPORT void JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   138
Java_java_net_PlainSocketImpl_initProto(JNIEnv *env, jclass cls) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   139
    psi_fdID = (*env)->GetFieldID(env, cls , "fd",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   140
                                  "Ljava/io/FileDescriptor;");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   141
    CHECK_NULL(psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   142
    psi_addressID = (*env)->GetFieldID(env, cls, "address",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   143
                                          "Ljava/net/InetAddress;");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   144
    CHECK_NULL(psi_addressID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   145
    psi_portID = (*env)->GetFieldID(env, cls, "port", "I");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   146
    CHECK_NULL(psi_portID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   147
    psi_localportID = (*env)->GetFieldID(env, cls, "localport", "I");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   148
    CHECK_NULL(psi_localportID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   149
    psi_timeoutID = (*env)->GetFieldID(env, cls, "timeout", "I");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   150
    CHECK_NULL(psi_timeoutID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   151
    psi_trafficClassID = (*env)->GetFieldID(env, cls, "trafficClass", "I");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   152
    CHECK_NULL(psi_trafficClassID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   153
    psi_serverSocketID = (*env)->GetFieldID(env, cls, "serverSocket",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   154
                        "Ljava/net/ServerSocket;");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   155
    CHECK_NULL(psi_serverSocketID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   156
    psi_fdLockID = (*env)->GetFieldID(env, cls, "fdLock",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   157
                                      "Ljava/lang/Object;");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   158
    CHECK_NULL(psi_fdLockID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   159
    psi_closePendingID = (*env)->GetFieldID(env, cls, "closePending", "Z");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   160
    CHECK_NULL(psi_closePendingID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   161
    IO_fd_fdID = NET_GetFileDescriptorID(env);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   162
    CHECK_NULL(IO_fd_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   163
90ce3da70b43 Initial load
duke
parents:
diff changeset
   164
    /* Create the marker fd used for dup2 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   165
    marker_fd = getMarkerFD();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   166
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   167
90ce3da70b43 Initial load
duke
parents:
diff changeset
   168
/* a global reference to the java.net.SocketException class. In
90ce3da70b43 Initial load
duke
parents:
diff changeset
   169
 * socketCreate, we ensure that this is initialized. This is to
90ce3da70b43 Initial load
duke
parents:
diff changeset
   170
 * prevent the problem where socketCreate runs out of file
90ce3da70b43 Initial load
duke
parents:
diff changeset
   171
 * descriptors, and is then unable to load the exception class.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   172
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   173
static jclass socketExceptionCls;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   174
90ce3da70b43 Initial load
duke
parents:
diff changeset
   175
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   176
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   177
 * Method:    socketCreate
90ce3da70b43 Initial load
duke
parents:
diff changeset
   178
 * Signature: (Z)V */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   179
JNIEXPORT void JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   180
Java_java_net_PlainSocketImpl_socketCreate(JNIEnv *env, jobject this,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   181
                                           jboolean stream) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   182
    jobject fdObj, ssObj;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   183
    int fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   184
90ce3da70b43 Initial load
duke
parents:
diff changeset
   185
    if (socketExceptionCls == NULL) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   186
        jclass c = (*env)->FindClass(env, "java/net/SocketException");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   187
        CHECK_NULL(c);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   188
        socketExceptionCls = (jclass)(*env)->NewGlobalRef(env, c);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   189
        CHECK_NULL(socketExceptionCls);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   190
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   191
    fdObj = (*env)->GetObjectField(env, this, psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   192
90ce3da70b43 Initial load
duke
parents:
diff changeset
   193
    if (fdObj == NULL) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   194
        (*env)->ThrowNew(env, socketExceptionCls, "null fd object");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   195
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   196
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   197
#ifdef AF_INET6
90ce3da70b43 Initial load
duke
parents:
diff changeset
   198
    if (ipv6_available()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   199
        fd = JVM_Socket(AF_INET6, (stream ? SOCK_STREAM: SOCK_DGRAM), 0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   200
    } else
90ce3da70b43 Initial load
duke
parents:
diff changeset
   201
#endif /* AF_INET6 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   202
        {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   203
            fd = JVM_Socket(AF_INET, (stream ? SOCK_STREAM: SOCK_DGRAM), 0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   204
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   205
    if (fd == JVM_IO_ERR) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   206
        /* note: if you run out of fds, you may not be able to load
90ce3da70b43 Initial load
duke
parents:
diff changeset
   207
         * the exception class, and get a NoClassDefFoundError
90ce3da70b43 Initial load
duke
parents:
diff changeset
   208
         * instead.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   209
         */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   210
        NET_ThrowNew(env, errno, "can't create socket");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   211
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   212
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   213
        (*env)->SetIntField(env, fdObj, IO_fd_fdID, fd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   214
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   215
90ce3da70b43 Initial load
duke
parents:
diff changeset
   216
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   217
     * If this is a server socket then enable SO_REUSEADDR
90ce3da70b43 Initial load
duke
parents:
diff changeset
   218
     * automatically and set to non blocking.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   219
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   220
    ssObj = (*env)->GetObjectField(env, this, psi_serverSocketID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   221
    if (ssObj != NULL) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   222
        int arg = 1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   223
        SET_NONBLOCKING(fd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   224
        JVM_SetSockOpt(fd, SOL_SOCKET, SO_REUSEADDR, (char*)&arg,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   225
            sizeof(arg));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   226
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   227
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   228
90ce3da70b43 Initial load
duke
parents:
diff changeset
   229
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   230
 * inetAddress is the address object passed to the socket connect
90ce3da70b43 Initial load
duke
parents:
diff changeset
   231
 * call.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   232
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   233
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   234
 * Method:    socketConnect
90ce3da70b43 Initial load
duke
parents:
diff changeset
   235
 * Signature: (Ljava/net/InetAddress;I)V
90ce3da70b43 Initial load
duke
parents:
diff changeset
   236
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   237
JNIEXPORT void JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   238
Java_java_net_PlainSocketImpl_socketConnect(JNIEnv *env, jobject this,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   239
                                            jobject iaObj, jint port,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   240
                                            jint timeout)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   241
{
90ce3da70b43 Initial load
duke
parents:
diff changeset
   242
    jint localport = (*env)->GetIntField(env, this, psi_localportID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   243
    int len = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   244
90ce3da70b43 Initial load
duke
parents:
diff changeset
   245
    /* fdObj is the FileDescriptor field on this */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   246
    jobject fdObj = (*env)->GetObjectField(env, this, psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   247
    jobject fdLock;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   248
90ce3da70b43 Initial load
duke
parents:
diff changeset
   249
    jint trafficClass = (*env)->GetIntField(env, this, psi_trafficClassID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   250
90ce3da70b43 Initial load
duke
parents:
diff changeset
   251
    /* fd is an int field on iaObj */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   252
    jint fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   253
90ce3da70b43 Initial load
duke
parents:
diff changeset
   254
    SOCKADDR him;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   255
    /* The result of the connection */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   256
    int connect_rv = -1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   257
90ce3da70b43 Initial load
duke
parents:
diff changeset
   258
    if (IS_NULL(fdObj)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   259
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException", "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   260
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   261
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   262
        fd = (*env)->GetIntField(env, fdObj, IO_fd_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   263
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   264
    if (IS_NULL(iaObj)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   265
        JNU_ThrowNullPointerException(env, "inet address argument null.");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   266
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   267
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   268
90ce3da70b43 Initial load
duke
parents:
diff changeset
   269
    /* connect */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   270
    if (NET_InetAddressToSockaddr(env, iaObj, port, (struct sockaddr *)&him, &len, JNI_TRUE) != 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   271
      return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   272
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   273
90ce3da70b43 Initial load
duke
parents:
diff changeset
   274
#ifdef AF_INET6
90ce3da70b43 Initial load
duke
parents:
diff changeset
   275
    if (trafficClass != 0 && ipv6_available()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   276
        NET_SetTrafficClass((struct sockaddr *)&him, trafficClass);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   277
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   278
#endif /* AF_INET6 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   279
    if (timeout <= 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   280
        connect_rv = NET_Connect(fd, (struct sockaddr *)&him, len);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   281
#ifdef __solaris__
90ce3da70b43 Initial load
duke
parents:
diff changeset
   282
        if (connect_rv == JVM_IO_ERR && errno == EINPROGRESS ) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   283
90ce3da70b43 Initial load
duke
parents:
diff changeset
   284
            /* This can happen if a blocking connect is interrupted by a signal.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   285
             * See 6343810.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   286
             */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   287
            while (1) {
624
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   288
#ifndef USE_SELECT
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   289
                {
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   290
                    struct pollfd pfd;
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   291
                    pfd.fd = fd;
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   292
                    pfd.events = POLLOUT;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   293
624
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   294
                    connect_rv = NET_Poll(&pfd, 1, -1);
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   295
                }
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   296
#else
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   297
                {
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   298
                    fd_set wr, ex;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   299
624
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   300
                    FD_ZERO(&wr);
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   301
                    FD_SET(fd, &wr);
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   302
                    FD_ZERO(&ex);
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   303
                    FD_SET(fd, &ex);
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   304
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   305
                    connect_rv = NET_Select(fd+1, 0, &wr, &ex, 0);
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   306
                }
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   307
#endif
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   308
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   309
                if (connect_rv == JVM_IO_ERR) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   310
                    if (errno == EINTR) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   311
                        continue;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   312
                    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   313
                        break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   314
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   315
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   316
                if (connect_rv > 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   317
                    int optlen;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   318
                    /* has connection been established */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   319
                    optlen = sizeof(connect_rv);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   320
                    if (JVM_GetSockOpt(fd, SOL_SOCKET, SO_ERROR,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   321
                                        (void*)&connect_rv, &optlen) <0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   322
                        connect_rv = errno;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   323
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   324
90ce3da70b43 Initial load
duke
parents:
diff changeset
   325
                    if (connect_rv != 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   326
                        /* restore errno */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   327
                        errno = connect_rv;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   328
                        connect_rv = JVM_IO_ERR;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   329
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   330
                    break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   331
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   332
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   333
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   334
#endif
90ce3da70b43 Initial load
duke
parents:
diff changeset
   335
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   336
        /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   337
         * A timeout was specified. We put the socket into non-blocking
90ce3da70b43 Initial load
duke
parents:
diff changeset
   338
         * mode, connect, and then wait for the connection to be
90ce3da70b43 Initial load
duke
parents:
diff changeset
   339
         * established, fail, or timeout.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   340
         */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   341
        SET_NONBLOCKING(fd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   342
90ce3da70b43 Initial load
duke
parents:
diff changeset
   343
        /* no need to use NET_Connect as non-blocking */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   344
        connect_rv = connect(fd, (struct sockaddr *)&him, len);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   345
90ce3da70b43 Initial load
duke
parents:
diff changeset
   346
        /* connection not established immediately */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   347
        if (connect_rv != 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   348
            int optlen;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   349
            jlong prevTime = JVM_CurrentTimeMillis(env, 0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   350
90ce3da70b43 Initial load
duke
parents:
diff changeset
   351
            if (errno != EINPROGRESS) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   352
                NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "ConnectException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   353
                             "connect failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   354
                SET_BLOCKING(fd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   355
                return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   356
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   357
90ce3da70b43 Initial load
duke
parents:
diff changeset
   358
            /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   359
             * Wait for the connection to be established or a
90ce3da70b43 Initial load
duke
parents:
diff changeset
   360
             * timeout occurs. poll/select needs to handle EINTR in
90ce3da70b43 Initial load
duke
parents:
diff changeset
   361
             * case lwp sig handler redirects any process signals to
90ce3da70b43 Initial load
duke
parents:
diff changeset
   362
             * this thread.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   363
             */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   364
            while (1) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   365
                jlong newTime;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   366
#ifndef USE_SELECT
90ce3da70b43 Initial load
duke
parents:
diff changeset
   367
                {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   368
                    struct pollfd pfd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   369
                    pfd.fd = fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   370
                    pfd.events = POLLOUT;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   371
90ce3da70b43 Initial load
duke
parents:
diff changeset
   372
                    errno = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   373
                    connect_rv = NET_Poll(&pfd, 1, timeout);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   374
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   375
#else
90ce3da70b43 Initial load
duke
parents:
diff changeset
   376
                {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   377
                    fd_set wr, ex;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   378
                    struct timeval t;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   379
90ce3da70b43 Initial load
duke
parents:
diff changeset
   380
                    t.tv_sec = timeout / 1000;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   381
                    t.tv_usec = (timeout % 1000) * 1000;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   382
90ce3da70b43 Initial load
duke
parents:
diff changeset
   383
                    FD_ZERO(&wr);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   384
                    FD_SET(fd, &wr);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   385
                    FD_ZERO(&ex);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   386
                    FD_SET(fd, &ex);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   387
90ce3da70b43 Initial load
duke
parents:
diff changeset
   388
                    errno = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   389
                    connect_rv = NET_Select(fd+1, 0, &wr, &ex, &t);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   390
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   391
#endif
90ce3da70b43 Initial load
duke
parents:
diff changeset
   392
90ce3da70b43 Initial load
duke
parents:
diff changeset
   393
                if (connect_rv >= 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   394
                    break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   395
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   396
                if (errno != EINTR) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   397
                    break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   398
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   399
90ce3da70b43 Initial load
duke
parents:
diff changeset
   400
                /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   401
                 * The poll was interrupted so adjust timeout and
90ce3da70b43 Initial load
duke
parents:
diff changeset
   402
                 * restart
90ce3da70b43 Initial load
duke
parents:
diff changeset
   403
                 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   404
                newTime = JVM_CurrentTimeMillis(env, 0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   405
                timeout -= (newTime - prevTime);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   406
                if (timeout <= 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   407
                    connect_rv = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   408
                    break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   409
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   410
                prevTime = newTime;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   411
90ce3da70b43 Initial load
duke
parents:
diff changeset
   412
            } /* while */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   413
90ce3da70b43 Initial load
duke
parents:
diff changeset
   414
            if (connect_rv == 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   415
                JNU_ThrowByName(env, JNU_JAVANETPKG "SocketTimeoutException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   416
                            "connect timed out");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   417
90ce3da70b43 Initial load
duke
parents:
diff changeset
   418
                /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   419
                 * Timeout out but connection may still be established.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   420
                 * At the high level it should be closed immediately but
90ce3da70b43 Initial load
duke
parents:
diff changeset
   421
                 * just in case we make the socket blocking again and
90ce3da70b43 Initial load
duke
parents:
diff changeset
   422
                 * shutdown input & output.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   423
                 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   424
                SET_BLOCKING(fd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   425
                JVM_SocketShutdown(fd, 2);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   426
                return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   427
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   428
90ce3da70b43 Initial load
duke
parents:
diff changeset
   429
            /* has connection been established */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   430
            optlen = sizeof(connect_rv);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   431
            if (JVM_GetSockOpt(fd, SOL_SOCKET, SO_ERROR, (void*)&connect_rv,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   432
                               &optlen) <0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   433
                connect_rv = errno;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   434
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   435
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   436
90ce3da70b43 Initial load
duke
parents:
diff changeset
   437
        /* make socket blocking again */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   438
        SET_BLOCKING(fd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   439
90ce3da70b43 Initial load
duke
parents:
diff changeset
   440
        /* restore errno */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   441
        if (connect_rv != 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   442
            errno = connect_rv;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   443
            connect_rv = JVM_IO_ERR;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   444
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   445
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   446
90ce3da70b43 Initial load
duke
parents:
diff changeset
   447
    /* report the appropriate exception */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   448
    if (connect_rv < 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   449
90ce3da70b43 Initial load
duke
parents:
diff changeset
   450
#ifdef __linux__
90ce3da70b43 Initial load
duke
parents:
diff changeset
   451
        /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   452
         * Linux/GNU distribution setup /etc/hosts so that
90ce3da70b43 Initial load
duke
parents:
diff changeset
   453
         * InetAddress.getLocalHost gets back the loopback address
90ce3da70b43 Initial load
duke
parents:
diff changeset
   454
         * rather than the host address. Thus a socket can be
90ce3da70b43 Initial load
duke
parents:
diff changeset
   455
         * bound to the loopback address and the connect will
90ce3da70b43 Initial load
duke
parents:
diff changeset
   456
         * fail with EADDRNOTAVAIL. In addition the Linux kernel
90ce3da70b43 Initial load
duke
parents:
diff changeset
   457
         * returns the wrong error in this case - it returns EINVAL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   458
         * instead of EADDRNOTAVAIL. We handle this here so that
90ce3da70b43 Initial load
duke
parents:
diff changeset
   459
         * a more descriptive exception text is used.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   460
         */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   461
        if (connect_rv == JVM_IO_ERR && errno == EINVAL) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   462
            JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   463
                "Invalid argument or cannot assign requested address");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   464
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   465
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   466
#endif
90ce3da70b43 Initial load
duke
parents:
diff changeset
   467
        if (connect_rv == JVM_IO_INTR) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   468
            JNU_ThrowByName(env, JNU_JAVAIOPKG "InterruptedIOException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   469
                            "operation interrupted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   470
        } else if (errno == EPROTO) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   471
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "ProtocolException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   472
                           "Protocol error");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   473
        } else if (errno == ECONNREFUSED) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   474
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "ConnectException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   475
                           "Connection refused");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   476
        } else if (errno == ETIMEDOUT) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   477
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "ConnectException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   478
                           "Connection timed out");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   479
        } else if (errno == EHOSTUNREACH) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   480
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "NoRouteToHostException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   481
                           "Host unreachable");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   482
        } else if (errno == EADDRNOTAVAIL) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   483
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "NoRouteToHostException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   484
                             "Address not available");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   485
        } else if ((errno == EISCONN) || (errno == EBADF)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   486
            JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   487
                            "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   488
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   489
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException", "connect failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   490
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   491
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   492
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   493
90ce3da70b43 Initial load
duke
parents:
diff changeset
   494
    (*env)->SetIntField(env, fdObj, IO_fd_fdID, fd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   495
90ce3da70b43 Initial load
duke
parents:
diff changeset
   496
    /* set the remote peer address and port */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   497
    (*env)->SetObjectField(env, this, psi_addressID, iaObj);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   498
    (*env)->SetIntField(env, this, psi_portID, port);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   499
90ce3da70b43 Initial load
duke
parents:
diff changeset
   500
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   501
     * we need to initialize the local port field if bind was called
90ce3da70b43 Initial load
duke
parents:
diff changeset
   502
     * previously to the connect (by the client) then localport field
90ce3da70b43 Initial load
duke
parents:
diff changeset
   503
     * will already be initialized
90ce3da70b43 Initial load
duke
parents:
diff changeset
   504
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   505
    if (localport == 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   506
        /* Now that we're a connected socket, let's extract the port number
90ce3da70b43 Initial load
duke
parents:
diff changeset
   507
         * that the system chose for us and store it in the Socket object.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   508
         */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   509
        len = SOCKADDR_LEN;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   510
        if (JVM_GetSockName(fd, (struct sockaddr *)&him, &len) == -1) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   511
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   512
                           "Error getting socket name");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   513
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   514
            localport = NET_GetPortFromSockaddr((struct sockaddr *)&him);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   515
            (*env)->SetIntField(env, this, psi_localportID, localport);
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
90ce3da70b43 Initial load
duke
parents:
diff changeset
   520
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   521
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   522
 * Method:    socketBind
90ce3da70b43 Initial load
duke
parents:
diff changeset
   523
 * Signature: (Ljava/net/InetAddress;I)V
90ce3da70b43 Initial load
duke
parents:
diff changeset
   524
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   525
JNIEXPORT void JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   526
Java_java_net_PlainSocketImpl_socketBind(JNIEnv *env, jobject this,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   527
                                         jobject iaObj, jint localport) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   528
90ce3da70b43 Initial load
duke
parents:
diff changeset
   529
    /* fdObj is the FileDescriptor field on this */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   530
    jobject fdObj = (*env)->GetObjectField(env, this, psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   531
    /* fd is an int field on fdObj */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   532
    int fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   533
    int len;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   534
    SOCKADDR him;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   535
90ce3da70b43 Initial load
duke
parents:
diff changeset
   536
    if (IS_NULL(fdObj)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   537
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   538
                        "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   539
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   540
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   541
        fd = (*env)->GetIntField(env, fdObj, IO_fd_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   542
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   543
    if (IS_NULL(iaObj)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   544
        JNU_ThrowNullPointerException(env, "iaObj is null.");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   545
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   546
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   547
90ce3da70b43 Initial load
duke
parents:
diff changeset
   548
    /* bind */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   549
    if (NET_InetAddressToSockaddr(env, iaObj, localport, (struct sockaddr *)&him, &len, JNI_TRUE) != 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   550
      return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   551
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   552
90ce3da70b43 Initial load
duke
parents:
diff changeset
   553
    if (NET_Bind(fd, (struct sockaddr *)&him, len) < 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   554
        if (errno == EADDRINUSE || errno == EADDRNOTAVAIL ||
90ce3da70b43 Initial load
duke
parents:
diff changeset
   555
            errno == EPERM || errno == EACCES) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   556
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "BindException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   557
                           "Bind failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   558
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   559
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   560
                           "Bind failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   561
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   562
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   563
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   564
90ce3da70b43 Initial load
duke
parents:
diff changeset
   565
    /* set the address */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   566
    (*env)->SetObjectField(env, this, psi_addressID, iaObj);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   567
90ce3da70b43 Initial load
duke
parents:
diff changeset
   568
    /* intialize the local port */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   569
    if (localport == 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   570
        /* Now that we're a connected socket, let's extract the port number
90ce3da70b43 Initial load
duke
parents:
diff changeset
   571
         * that the system chose for us and store it in the Socket object.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   572
         */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   573
        if (JVM_GetSockName(fd, (struct sockaddr *)&him, &len) == -1) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   574
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   575
                           "Error getting socket name");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   576
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   577
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   578
        localport = NET_GetPortFromSockaddr((struct sockaddr *)&him);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   579
        (*env)->SetIntField(env, this, psi_localportID, localport);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   580
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   581
        (*env)->SetIntField(env, this, psi_localportID, localport);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   582
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   583
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   584
90ce3da70b43 Initial load
duke
parents:
diff changeset
   585
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   586
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   587
 * Method:    socketListen
90ce3da70b43 Initial load
duke
parents:
diff changeset
   588
 * Signature: (I)V
90ce3da70b43 Initial load
duke
parents:
diff changeset
   589
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   590
JNIEXPORT void JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   591
Java_java_net_PlainSocketImpl_socketListen (JNIEnv *env, jobject this,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   592
                                            jint count)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   593
{
90ce3da70b43 Initial load
duke
parents:
diff changeset
   594
    /* this FileDescriptor fd field */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   595
    jobject fdObj = (*env)->GetObjectField(env, this, psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   596
    /* fdObj's int fd field */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   597
    int fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   598
90ce3da70b43 Initial load
duke
parents:
diff changeset
   599
    if (IS_NULL(fdObj)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   600
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   601
                        "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   602
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   603
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   604
        fd = (*env)->GetIntField(env, fdObj, IO_fd_fdID);
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
     * Workaround for bugid 4101691 in Solaris 2.6. See 4106600.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   609
     * If listen backlog is Integer.MAX_VALUE then subtract 1.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   610
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   611
    if (count == 0x7fffffff)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   612
        count -= 1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   613
90ce3da70b43 Initial load
duke
parents:
diff changeset
   614
    if (JVM_Listen(fd, count) == JVM_IO_ERR) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   615
        NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   616
                       "Listen failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   617
    }
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
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   622
 * Method:    socketAccept
90ce3da70b43 Initial load
duke
parents:
diff changeset
   623
 * Signature: (Ljava/net/SocketImpl;)V
90ce3da70b43 Initial load
duke
parents:
diff changeset
   624
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   625
JNIEXPORT void JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   626
Java_java_net_PlainSocketImpl_socketAccept(JNIEnv *env, jobject this,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   627
                                           jobject socket)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   628
{
90ce3da70b43 Initial load
duke
parents:
diff changeset
   629
    /* fields on this */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   630
    int port;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   631
    jint timeout = (*env)->GetIntField(env, this, psi_timeoutID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   632
    jlong prevTime = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   633
    jobject fdObj = (*env)->GetObjectField(env, this, psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   634
90ce3da70b43 Initial load
duke
parents:
diff changeset
   635
    /* the FileDescriptor field on socket */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   636
    jobject socketFdObj;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   637
    /* the InetAddress field on socket */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   638
    jobject socketAddressObj;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   639
90ce3da70b43 Initial load
duke
parents:
diff changeset
   640
    /* the ServerSocket fd int field on fdObj */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   641
    jint fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   642
90ce3da70b43 Initial load
duke
parents:
diff changeset
   643
    /* accepted fd */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   644
    jint newfd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   645
90ce3da70b43 Initial load
duke
parents:
diff changeset
   646
    SOCKADDR him;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   647
    int len;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   648
90ce3da70b43 Initial load
duke
parents:
diff changeset
   649
    len = SOCKADDR_LEN;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   650
90ce3da70b43 Initial load
duke
parents:
diff changeset
   651
    if (IS_NULL(fdObj)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   652
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   653
                        "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   654
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   655
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   656
        fd = (*env)->GetIntField(env, fdObj, IO_fd_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   657
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   658
    if (IS_NULL(socket)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   659
        JNU_ThrowNullPointerException(env, "socket is null");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   660
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   661
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   662
90ce3da70b43 Initial load
duke
parents:
diff changeset
   663
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   664
     * accept connection but ignore ECONNABORTED indicating that
90ce3da70b43 Initial load
duke
parents:
diff changeset
   665
     * connection was eagerly accepted by the OS but was reset
90ce3da70b43 Initial load
duke
parents:
diff changeset
   666
     * before accept() was called.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   667
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   668
     * If accept timeout in place and timeout is adjusted with
90ce3da70b43 Initial load
duke
parents:
diff changeset
   669
     * each ECONNABORTED or EWOULDBLOCK to ensure that semantics
90ce3da70b43 Initial load
duke
parents:
diff changeset
   670
     * of timeout are preserved.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   671
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   672
    for (;;) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   673
        int ret;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   674
90ce3da70b43 Initial load
duke
parents:
diff changeset
   675
        /* first usage pick up current time */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   676
        if (prevTime == 0 && timeout > 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   677
            prevTime = JVM_CurrentTimeMillis(env, 0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   678
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   679
90ce3da70b43 Initial load
duke
parents:
diff changeset
   680
        /* passing a timeout of 0 to poll will return immediately,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   681
           but in the case of ServerSocket 0 means infinite. */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   682
        if (timeout <= 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   683
            ret = NET_Timeout(fd, -1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   684
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   685
            ret = NET_Timeout(fd, timeout);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   686
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   687
90ce3da70b43 Initial load
duke
parents:
diff changeset
   688
        if (ret == 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   689
            JNU_ThrowByName(env, JNU_JAVANETPKG "SocketTimeoutException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   690
                            "Accept timed out");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   691
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   692
        } else if (ret == JVM_IO_ERR) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   693
            if (errno == EBADF) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   694
               JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException", "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   695
            } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   696
               NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException", "Accept failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   697
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   698
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   699
        } else if (ret == JVM_IO_INTR) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   700
            JNU_ThrowByName(env, JNU_JAVAIOPKG "InterruptedIOException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   701
                            "operation interrupted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   702
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   703
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   704
90ce3da70b43 Initial load
duke
parents:
diff changeset
   705
        newfd = NET_Accept(fd, (struct sockaddr *)&him, (jint*)&len);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   706
90ce3da70b43 Initial load
duke
parents:
diff changeset
   707
        /* connection accepted */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   708
        if (newfd >= 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   709
            SET_BLOCKING(newfd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   710
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   711
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   712
90ce3da70b43 Initial load
duke
parents:
diff changeset
   713
        /* non (ECONNABORTED or EWOULDBLOCK) error */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   714
        if (!(errno == ECONNABORTED || errno == EWOULDBLOCK)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   715
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   716
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   717
90ce3da70b43 Initial load
duke
parents:
diff changeset
   718
        /* ECONNABORTED or EWOULDBLOCK error so adjust timeout if there is one. */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   719
        if (timeout) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   720
            jlong currTime = JVM_CurrentTimeMillis(env, 0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   721
            timeout -= (currTime - prevTime);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   722
90ce3da70b43 Initial load
duke
parents:
diff changeset
   723
            if (timeout <= 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   724
                JNU_ThrowByName(env, JNU_JAVANETPKG "SocketTimeoutException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   725
                                "Accept timed out");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   726
                return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   727
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   728
            prevTime = currTime;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   729
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   730
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   731
90ce3da70b43 Initial load
duke
parents:
diff changeset
   732
    if (newfd < 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   733
        if (newfd == -2) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   734
            JNU_ThrowByName(env, JNU_JAVAIOPKG "InterruptedIOException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   735
                            "operation interrupted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   736
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   737
            if (errno == EINVAL) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   738
                errno = EBADF;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   739
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   740
            if (errno == EBADF) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   741
                JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException", "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   742
            } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   743
                NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException", "Accept failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   744
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   745
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   746
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   747
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   748
90ce3da70b43 Initial load
duke
parents:
diff changeset
   749
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   750
     * fill up the remote peer port and address in the new socket structure.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   751
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   752
    socketAddressObj = NET_SockaddrToInetAddress(env, (struct sockaddr *)&him, &port);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   753
    if (socketAddressObj == NULL) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   754
        /* should be pending exception */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   755
        close(newfd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   756
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   757
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   758
90ce3da70b43 Initial load
duke
parents:
diff changeset
   759
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   760
     * Populate SocketImpl.fd.fd
90ce3da70b43 Initial load
duke
parents:
diff changeset
   761
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   762
    socketFdObj = (*env)->GetObjectField(env, socket, psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   763
    (*env)->SetIntField(env, socketFdObj, IO_fd_fdID, newfd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   764
90ce3da70b43 Initial load
duke
parents:
diff changeset
   765
    (*env)->SetObjectField(env, socket, psi_addressID, socketAddressObj);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   766
    (*env)->SetIntField(env, socket, psi_portID, port);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   767
    /* also fill up the local port information */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   768
     port = (*env)->GetIntField(env, this, psi_localportID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   769
    (*env)->SetIntField(env, socket, psi_localportID, port);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   770
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   771
90ce3da70b43 Initial load
duke
parents:
diff changeset
   772
90ce3da70b43 Initial load
duke
parents:
diff changeset
   773
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   774
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   775
 * Method:    socketAvailable
90ce3da70b43 Initial load
duke
parents:
diff changeset
   776
 * Signature: ()I
90ce3da70b43 Initial load
duke
parents:
diff changeset
   777
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   778
JNIEXPORT jint JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   779
Java_java_net_PlainSocketImpl_socketAvailable(JNIEnv *env, jobject this) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   780
90ce3da70b43 Initial load
duke
parents:
diff changeset
   781
    jint ret = -1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   782
    jobject fdObj = (*env)->GetObjectField(env, this, psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   783
    jint fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   784
90ce3da70b43 Initial load
duke
parents:
diff changeset
   785
    if (IS_NULL(fdObj)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   786
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   787
                        "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   788
        return -1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   789
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   790
        fd = (*env)->GetIntField(env, fdObj, IO_fd_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   791
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   792
    /* JVM_SocketAvailable returns 0 for failure, 1 for success */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   793
    if (!JVM_SocketAvailable(fd, &ret)){
90ce3da70b43 Initial load
duke
parents:
diff changeset
   794
        if (errno == ECONNRESET) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   795
            JNU_ThrowByName(env, "sun/net/ConnectionResetException", "");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   796
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   797
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   798
                                         "ioctl FIONREAD failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   799
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   800
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   801
    return ret;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   802
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   803
90ce3da70b43 Initial load
duke
parents:
diff changeset
   804
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   805
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   806
 * Method:    socketClose0
90ce3da70b43 Initial load
duke
parents:
diff changeset
   807
 * Signature: (Z)V
90ce3da70b43 Initial load
duke
parents:
diff changeset
   808
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   809
JNIEXPORT void JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   810
Java_java_net_PlainSocketImpl_socketClose0(JNIEnv *env, jobject this,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   811
                                          jboolean useDeferredClose) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   812
90ce3da70b43 Initial load
duke
parents:
diff changeset
   813
    jobject fdObj = (*env)->GetObjectField(env, this, psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   814
    jint fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   815
90ce3da70b43 Initial load
duke
parents:
diff changeset
   816
    if (IS_NULL(fdObj)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   817
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   818
                        "socket already closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   819
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   820
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   821
        fd = (*env)->GetIntField(env, fdObj, IO_fd_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   822
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   823
    if (fd != -1) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   824
        if (useDeferredClose && marker_fd >= 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   825
            NET_Dup2(marker_fd, fd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   826
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   827
            (*env)->SetIntField(env, fdObj, IO_fd_fdID, -1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   828
            NET_SocketClose(fd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   829
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   830
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   831
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   832
90ce3da70b43 Initial load
duke
parents:
diff changeset
   833
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   834
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   835
 * Method:    socketShutdown
90ce3da70b43 Initial load
duke
parents:
diff changeset
   836
 * Signature: (I)V
90ce3da70b43 Initial load
duke
parents:
diff changeset
   837
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   838
JNIEXPORT void JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   839
Java_java_net_PlainSocketImpl_socketShutdown(JNIEnv *env, jobject this,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   840
                                             jint howto)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   841
{
90ce3da70b43 Initial load
duke
parents:
diff changeset
   842
90ce3da70b43 Initial load
duke
parents:
diff changeset
   843
    jobject fdObj = (*env)->GetObjectField(env, this, psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   844
    jint fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   845
90ce3da70b43 Initial load
duke
parents:
diff changeset
   846
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   847
     * WARNING: THIS NEEDS LOCKING. ALSO: SHOULD WE CHECK for fd being
90ce3da70b43 Initial load
duke
parents:
diff changeset
   848
     * -1 already?
90ce3da70b43 Initial load
duke
parents:
diff changeset
   849
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   850
    if (IS_NULL(fdObj)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   851
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   852
                        "socket already closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   853
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   854
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   855
        fd = (*env)->GetIntField(env, fdObj, IO_fd_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   856
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   857
    JVM_SocketShutdown(fd, howto);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   858
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   859
90ce3da70b43 Initial load
duke
parents:
diff changeset
   860
90ce3da70b43 Initial load
duke
parents:
diff changeset
   861
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   862
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   863
 * Method:    socketSetOption
90ce3da70b43 Initial load
duke
parents:
diff changeset
   864
 * Signature: (IZLjava/lang/Object;)V
90ce3da70b43 Initial load
duke
parents:
diff changeset
   865
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   866
JNIEXPORT void JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   867
Java_java_net_PlainSocketImpl_socketSetOption(JNIEnv *env, jobject this,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   868
                                              jint cmd, jboolean on,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   869
                                              jobject value) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   870
    int fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   871
    int level, optname, optlen;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   872
    union {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   873
        int i;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   874
        struct linger ling;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   875
    } optval;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   876
90ce3da70b43 Initial load
duke
parents:
diff changeset
   877
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   878
     * Check that socket hasn't been closed
90ce3da70b43 Initial load
duke
parents:
diff changeset
   879
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   880
    fd = getFD(env, this);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   881
    if (fd < 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   882
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   883
                        "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   884
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   885
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   886
90ce3da70b43 Initial load
duke
parents:
diff changeset
   887
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   888
     * SO_TIMEOUT is a no-op on Solaris/Linux
90ce3da70b43 Initial load
duke
parents:
diff changeset
   889
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   890
    if (cmd == java_net_SocketOptions_SO_TIMEOUT) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   891
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   892
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   893
90ce3da70b43 Initial load
duke
parents:
diff changeset
   894
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   895
     * Map the Java level socket option to the platform specific
90ce3da70b43 Initial load
duke
parents:
diff changeset
   896
     * level and option name.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   897
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   898
    if (NET_MapSocketOption(cmd, &level, &optname)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   899
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException", "Invalid option");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   900
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   901
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   902
90ce3da70b43 Initial load
duke
parents:
diff changeset
   903
    switch (cmd) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   904
        case java_net_SocketOptions_SO_SNDBUF :
90ce3da70b43 Initial load
duke
parents:
diff changeset
   905
        case java_net_SocketOptions_SO_RCVBUF :
90ce3da70b43 Initial load
duke
parents:
diff changeset
   906
        case java_net_SocketOptions_SO_LINGER :
90ce3da70b43 Initial load
duke
parents:
diff changeset
   907
        case java_net_SocketOptions_IP_TOS :
90ce3da70b43 Initial load
duke
parents:
diff changeset
   908
            {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   909
                jclass cls;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   910
                jfieldID fid;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   911
90ce3da70b43 Initial load
duke
parents:
diff changeset
   912
                cls = (*env)->FindClass(env, "java/lang/Integer");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   913
                CHECK_NULL(cls);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   914
                fid = (*env)->GetFieldID(env, cls, "value", "I");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   915
                CHECK_NULL(fid);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   916
90ce3da70b43 Initial load
duke
parents:
diff changeset
   917
                if (cmd == java_net_SocketOptions_SO_LINGER) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   918
                    if (on) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   919
                        optval.ling.l_onoff = 1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   920
                        optval.ling.l_linger = (*env)->GetIntField(env, value, fid);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   921
                    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   922
                        optval.ling.l_onoff = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   923
                        optval.ling.l_linger = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   924
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   925
                    optlen = sizeof(optval.ling);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   926
                } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   927
                    optval.i = (*env)->GetIntField(env, value, fid);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   928
                    optlen = sizeof(optval.i);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   929
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   930
90ce3da70b43 Initial load
duke
parents:
diff changeset
   931
                break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   932
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   933
90ce3da70b43 Initial load
duke
parents:
diff changeset
   934
        /* Boolean -> int */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   935
        default :
90ce3da70b43 Initial load
duke
parents:
diff changeset
   936
            optval.i = (on ? 1 : 0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   937
            optlen = sizeof(optval.i);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   938
90ce3da70b43 Initial load
duke
parents:
diff changeset
   939
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   940
90ce3da70b43 Initial load
duke
parents:
diff changeset
   941
    if (NET_SetSockOpt(fd, level, optname, (const void *)&optval, optlen) < 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   942
#ifdef __solaris__
90ce3da70b43 Initial load
duke
parents:
diff changeset
   943
        if (errno == EINVAL) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   944
            // On Solaris setsockopt will set errno to EINVAL if the socket
90ce3da70b43 Initial load
duke
parents:
diff changeset
   945
            // is closed. The default error message is then confusing
90ce3da70b43 Initial load
duke
parents:
diff changeset
   946
            char fullMsg[128];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   947
            jio_snprintf(fullMsg, sizeof(fullMsg), "Invalid option or socket reset by remote peer");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   948
            JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException", fullMsg);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   949
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   950
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   951
#endif /* __solaris__ */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   952
        NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   953
                                      "Error setting socket option");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   954
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   955
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   956
90ce3da70b43 Initial load
duke
parents:
diff changeset
   957
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   958
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   959
 * Method:    socketGetOption
90ce3da70b43 Initial load
duke
parents:
diff changeset
   960
 * Signature: (I)I
90ce3da70b43 Initial load
duke
parents:
diff changeset
   961
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   962
JNIEXPORT jint JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   963
Java_java_net_PlainSocketImpl_socketGetOption(JNIEnv *env, jobject this,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   964
                                              jint cmd, jobject iaContainerObj) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   965
90ce3da70b43 Initial load
duke
parents:
diff changeset
   966
    int fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   967
    int level, optname, optlen;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   968
    union {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   969
        int i;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   970
        struct linger ling;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   971
    } optval;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   972
90ce3da70b43 Initial load
duke
parents:
diff changeset
   973
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   974
     * Check that socket hasn't been closed
90ce3da70b43 Initial load
duke
parents:
diff changeset
   975
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   976
    fd = getFD(env, this);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   977
    if (fd < 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   978
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   979
                        "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   980
        return -1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   981
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   982
90ce3da70b43 Initial load
duke
parents:
diff changeset
   983
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   984
     * SO_BINDADDR isn't a socket option
90ce3da70b43 Initial load
duke
parents:
diff changeset
   985
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   986
    if (cmd == java_net_SocketOptions_SO_BINDADDR) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   987
        SOCKADDR him;
710
57367409c4a5 6483406: new ServerSocket() sometimes takes more than 3 minutes on Suse Linux
jccollet
parents: 624
diff changeset
   988
        socklen_t len = 0;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   989
        int port;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   990
        jobject iaObj;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   991
        jclass iaCntrClass;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   992
        jfieldID iaFieldID;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   993
90ce3da70b43 Initial load
duke
parents:
diff changeset
   994
        len = SOCKADDR_LEN;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   995
90ce3da70b43 Initial load
duke
parents:
diff changeset
   996
        if (getsockname(fd, (struct sockaddr *)&him, &len) < 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   997
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   998
                             "Error getting socket name");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   999
            return -1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1000
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1001
        iaObj = NET_SockaddrToInetAddress(env, (struct sockaddr *)&him, &port);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1002
        CHECK_NULL_RETURN(iaObj, -1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1003
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1004
        iaCntrClass = (*env)->GetObjectClass(env, iaContainerObj);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1005
        iaFieldID = (*env)->GetFieldID(env, iaCntrClass, "addr", "Ljava/net/InetAddress;");
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1006
        CHECK_NULL_RETURN(iaFieldID, -1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1007
        (*env)->SetObjectField(env, iaContainerObj, iaFieldID, iaObj);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1008
        return 0; /* notice change from before */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1009
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1010
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1011
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1012
     * Map the Java level socket option to the platform specific
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1013
     * level and option name.
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1014
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1015
    if (NET_MapSocketOption(cmd, &level, &optname)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1016
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException", "Invalid option");
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1017
        return -1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1018
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1019
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1020
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1021
     * Args are int except for SO_LINGER
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1022
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1023
    if (cmd == java_net_SocketOptions_SO_LINGER) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1024
        optlen = sizeof(optval.ling);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1025
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1026
        optlen = sizeof(optval.i);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1027
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1028
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1029
    if (NET_GetSockOpt(fd, level, optname, (void *)&optval, &optlen) < 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1030
        NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1031
                                      "Error getting socket option");
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1032
        return -1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1033
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1034
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1035
    switch (cmd) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1036
        case java_net_SocketOptions_SO_LINGER:
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1037
            return (optval.ling.l_onoff ? optval.ling.l_linger: -1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1038
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1039
        case java_net_SocketOptions_SO_SNDBUF:
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1040
        case java_net_SocketOptions_SO_RCVBUF:
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1041
        case java_net_SocketOptions_IP_TOS:
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1042
            return optval.i;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1043
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1044
        default :
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1045
            return (optval.i == 0) ? -1 : 1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1046
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1047
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1048
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1049
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1050
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1051
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1052
 * Method:    socketSendUrgentData
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1053
 * Signature: (B)V
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1054
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1055
JNIEXPORT void JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1056
Java_java_net_PlainSocketImpl_socketSendUrgentData(JNIEnv *env, jobject this,
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1057
                                             jint data) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1058
    /* The fd field */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1059
    jobject fdObj = (*env)->GetObjectField(env, this, psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1060
    int n, fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1061
    unsigned char d = data & 0xFF;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1062
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1063
    if (IS_NULL(fdObj)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1064
        JNU_ThrowByName(env, "java/net/SocketException", "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1065
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1066
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1067
        fd = (*env)->GetIntField(env, fdObj, IO_fd_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1068
        /* Bug 4086704 - If the Socket associated with this file descriptor
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1069
         * was closed (sysCloseFD), the the file descriptor is set to -1.
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1070
         */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1071
        if (fd == -1) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1072
            JNU_ThrowByName(env, "java/net/SocketException", "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1073
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1074
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1075
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1076
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1077
    n = JVM_Send(fd, (char *)&d, 1, MSG_OOB);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1078
    if (n == JVM_IO_ERR) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1079
        NET_ThrowByNameWithLastError(env, "java/io/IOException", "Write failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1080
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1081
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1082
    if (n == JVM_IO_INTR) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1083
        JNU_ThrowByName(env, "java/io/InterruptedIOException", 0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1084
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1085
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1086
}