jdk/src/solaris/native/java/net/PlainSocketImpl.c
author chegar
Tue, 10 Aug 2010 17:30:43 +0100
changeset 6299 a44d48dbcd50
parent 5506 202f599c92aa
child 7668 d4a77089c587
permissions -rw-r--r--
6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled. Reviewed-by: alanb
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     1
/*
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 1335
diff changeset
     2
 * Copyright (c) 1997, 2008, Oracle and/or its affiliates. All rights reserved.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
90ce3da70b43 Initial load
duke
parents:
diff changeset
     4
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
90ce3da70b43 Initial load
duke
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 1335
diff changeset
     7
 * published by the Free Software Foundation.  Oracle designates this
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 1335
diff changeset
     9
 * by Oracle in the LICENSE file that accompanied this code.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    10
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
90ce3da70b43 Initial load
duke
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
90ce3da70b43 Initial load
duke
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
90ce3da70b43 Initial load
duke
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
90ce3da70b43 Initial load
duke
parents:
diff changeset
    15
 * accompanied this code).
90ce3da70b43 Initial load
duke
parents:
diff changeset
    16
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
90ce3da70b43 Initial load
duke
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    20
 *
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 1335
diff changeset
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 1335
diff changeset
    22
 * or visit www.oracle.com if you need additional information or have any
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 1335
diff changeset
    23
 * questions.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    24
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    25
90ce3da70b43 Initial load
duke
parents:
diff changeset
    26
#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;
6299
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   184
    int type = (stream ? SOCK_STREAM : SOCK_DGRAM);
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   185
#ifdef AF_INET6
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   186
    int domain = ipv6_available() ? AF_INET6 : AF_INET;
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   187
#else
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   188
    int domain = AF_INET;
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   189
#endif
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   190
90ce3da70b43 Initial load
duke
parents:
diff changeset
   191
    if (socketExceptionCls == NULL) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   192
        jclass c = (*env)->FindClass(env, "java/net/SocketException");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   193
        CHECK_NULL(c);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   194
        socketExceptionCls = (jclass)(*env)->NewGlobalRef(env, c);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   195
        CHECK_NULL(socketExceptionCls);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   196
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   197
    fdObj = (*env)->GetObjectField(env, this, psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   198
90ce3da70b43 Initial load
duke
parents:
diff changeset
   199
    if (fdObj == NULL) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   200
        (*env)->ThrowNew(env, socketExceptionCls, "null fd object");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   201
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   202
    }
6299
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   203
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   204
    if ((fd = JVM_Socket(domain, type, 0)) == JVM_IO_ERR) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   205
        /* note: if you run out of fds, you may not be able to load
90ce3da70b43 Initial load
duke
parents:
diff changeset
   206
         * the exception class, and get a NoClassDefFoundError
90ce3da70b43 Initial load
duke
parents:
diff changeset
   207
         * instead.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   208
         */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   209
        NET_ThrowNew(env, errno, "can't create socket");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   210
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   211
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   212
6299
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   213
#ifdef AF_INET6
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   214
    /* Disable IPV6_V6ONLY to ensure dual-socket support */
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   215
    if (domain == AF_INET6) {
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   216
        int arg = 0;
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   217
        if (setsockopt(fd, IPPROTO_IPV6, IPV6_V6ONLY, (char*)&arg,
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   218
                       sizeof(int)) < 0) {
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   219
            NET_ThrowNew(env, errno, "cannot set IPPROTO_IPV6");
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   220
            close(fd);
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   221
            return;
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   222
        }
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   223
    }
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   224
#endif /* AF_INET6 */
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   225
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   226
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   227
     * If this is a server socket then enable SO_REUSEADDR
90ce3da70b43 Initial load
duke
parents:
diff changeset
   228
     * automatically and set to non blocking.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   229
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   230
    ssObj = (*env)->GetObjectField(env, this, psi_serverSocketID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   231
    if (ssObj != NULL) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   232
        int arg = 1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   233
        SET_NONBLOCKING(fd);
6299
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   234
        if (JVM_SetSockOpt(fd, SOL_SOCKET, SO_REUSEADDR, (char*)&arg,
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   235
                           sizeof(arg)) < 0) {
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   236
            NET_ThrowNew(env, errno, "cannot set SO_REUSEADDR");
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   237
            close(fd);
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   238
            return;
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   239
        }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   240
    }
6299
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   241
a44d48dbcd50 6882910: Unexplained lack of IP4 network ability when transparent IP6 to IP4 is disabled.
chegar
parents: 5506
diff changeset
   242
    (*env)->SetIntField(env, fdObj, IO_fd_fdID, fd);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   243
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   244
90ce3da70b43 Initial load
duke
parents:
diff changeset
   245
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   246
 * inetAddress is the address object passed to the socket connect
90ce3da70b43 Initial load
duke
parents:
diff changeset
   247
 * call.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   248
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   249
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   250
 * Method:    socketConnect
90ce3da70b43 Initial load
duke
parents:
diff changeset
   251
 * Signature: (Ljava/net/InetAddress;I)V
90ce3da70b43 Initial load
duke
parents:
diff changeset
   252
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   253
JNIEXPORT void JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   254
Java_java_net_PlainSocketImpl_socketConnect(JNIEnv *env, jobject this,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   255
                                            jobject iaObj, jint port,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   256
                                            jint timeout)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   257
{
90ce3da70b43 Initial load
duke
parents:
diff changeset
   258
    jint localport = (*env)->GetIntField(env, this, psi_localportID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   259
    int len = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   260
90ce3da70b43 Initial load
duke
parents:
diff changeset
   261
    /* fdObj is the FileDescriptor field on this */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   262
    jobject fdObj = (*env)->GetObjectField(env, this, psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   263
    jobject fdLock;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   264
90ce3da70b43 Initial load
duke
parents:
diff changeset
   265
    jint trafficClass = (*env)->GetIntField(env, this, psi_trafficClassID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   266
90ce3da70b43 Initial load
duke
parents:
diff changeset
   267
    /* fd is an int field on iaObj */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   268
    jint fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   269
90ce3da70b43 Initial load
duke
parents:
diff changeset
   270
    SOCKADDR him;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   271
    /* The result of the connection */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   272
    int connect_rv = -1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   273
90ce3da70b43 Initial load
duke
parents:
diff changeset
   274
    if (IS_NULL(fdObj)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   275
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException", "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   276
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   277
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   278
        fd = (*env)->GetIntField(env, fdObj, IO_fd_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   279
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   280
    if (IS_NULL(iaObj)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   281
        JNU_ThrowNullPointerException(env, "inet address argument null.");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   282
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   283
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   284
90ce3da70b43 Initial load
duke
parents:
diff changeset
   285
    /* connect */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   286
    if (NET_InetAddressToSockaddr(env, iaObj, port, (struct sockaddr *)&him, &len, JNI_TRUE) != 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   287
      return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   288
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   289
90ce3da70b43 Initial load
duke
parents:
diff changeset
   290
#ifdef AF_INET6
90ce3da70b43 Initial load
duke
parents:
diff changeset
   291
    if (trafficClass != 0 && ipv6_available()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   292
        NET_SetTrafficClass((struct sockaddr *)&him, trafficClass);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   293
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   294
#endif /* AF_INET6 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   295
    if (timeout <= 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   296
        connect_rv = NET_Connect(fd, (struct sockaddr *)&him, len);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   297
#ifdef __solaris__
90ce3da70b43 Initial load
duke
parents:
diff changeset
   298
        if (connect_rv == JVM_IO_ERR && errno == EINPROGRESS ) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   299
90ce3da70b43 Initial load
duke
parents:
diff changeset
   300
            /* This can happen if a blocking connect is interrupted by a signal.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   301
             * See 6343810.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   302
             */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   303
            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
   304
#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
   305
                {
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
                    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
   307
                    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
   308
                    pfd.events = POLLOUT;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   309
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
   310
                    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
   311
                }
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   312
#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
   313
                {
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   314
                    fd_set wr, ex;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   315
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
   316
                    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
   317
                    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
   318
                    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
   319
                    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
   320
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   321
                    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
   322
                }
e2bffc6b2d97 6670408: testcase panics 1.5.0_12&_14 JVM when java.net.PlainSocketImpl trying to throw an exception
chegar
parents: 2
diff changeset
   323
#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
   324
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   325
                if (connect_rv == JVM_IO_ERR) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   326
                    if (errno == EINTR) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   327
                        continue;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   328
                    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   329
                        break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   330
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   331
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   332
                if (connect_rv > 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   333
                    int optlen;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   334
                    /* has connection been established */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   335
                    optlen = sizeof(connect_rv);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   336
                    if (JVM_GetSockOpt(fd, SOL_SOCKET, SO_ERROR,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   337
                                        (void*)&connect_rv, &optlen) <0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   338
                        connect_rv = errno;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   339
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   340
90ce3da70b43 Initial load
duke
parents:
diff changeset
   341
                    if (connect_rv != 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   342
                        /* restore errno */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   343
                        errno = connect_rv;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   344
                        connect_rv = JVM_IO_ERR;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   345
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   346
                    break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   347
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   348
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   349
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   350
#endif
90ce3da70b43 Initial load
duke
parents:
diff changeset
   351
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   352
        /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   353
         * A timeout was specified. We put the socket into non-blocking
90ce3da70b43 Initial load
duke
parents:
diff changeset
   354
         * mode, connect, and then wait for the connection to be
90ce3da70b43 Initial load
duke
parents:
diff changeset
   355
         * established, fail, or timeout.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   356
         */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   357
        SET_NONBLOCKING(fd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   358
90ce3da70b43 Initial load
duke
parents:
diff changeset
   359
        /* no need to use NET_Connect as non-blocking */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   360
        connect_rv = connect(fd, (struct sockaddr *)&him, len);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   361
90ce3da70b43 Initial load
duke
parents:
diff changeset
   362
        /* connection not established immediately */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   363
        if (connect_rv != 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   364
            int optlen;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   365
            jlong prevTime = JVM_CurrentTimeMillis(env, 0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   366
90ce3da70b43 Initial load
duke
parents:
diff changeset
   367
            if (errno != EINPROGRESS) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   368
                NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "ConnectException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   369
                             "connect failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   370
                SET_BLOCKING(fd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   371
                return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   372
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   373
90ce3da70b43 Initial load
duke
parents:
diff changeset
   374
            /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   375
             * Wait for the connection to be established or a
90ce3da70b43 Initial load
duke
parents:
diff changeset
   376
             * timeout occurs. poll/select needs to handle EINTR in
90ce3da70b43 Initial load
duke
parents:
diff changeset
   377
             * case lwp sig handler redirects any process signals to
90ce3da70b43 Initial load
duke
parents:
diff changeset
   378
             * this thread.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   379
             */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   380
            while (1) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   381
                jlong newTime;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   382
#ifndef USE_SELECT
90ce3da70b43 Initial load
duke
parents:
diff changeset
   383
                {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   384
                    struct pollfd pfd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   385
                    pfd.fd = fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   386
                    pfd.events = POLLOUT;
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_Poll(&pfd, 1, timeout);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   390
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   391
#else
90ce3da70b43 Initial load
duke
parents:
diff changeset
   392
                {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   393
                    fd_set wr, ex;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   394
                    struct timeval t;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   395
90ce3da70b43 Initial load
duke
parents:
diff changeset
   396
                    t.tv_sec = timeout / 1000;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   397
                    t.tv_usec = (timeout % 1000) * 1000;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   398
90ce3da70b43 Initial load
duke
parents:
diff changeset
   399
                    FD_ZERO(&wr);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   400
                    FD_SET(fd, &wr);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   401
                    FD_ZERO(&ex);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   402
                    FD_SET(fd, &ex);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   403
90ce3da70b43 Initial load
duke
parents:
diff changeset
   404
                    errno = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   405
                    connect_rv = NET_Select(fd+1, 0, &wr, &ex, &t);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   406
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   407
#endif
90ce3da70b43 Initial load
duke
parents:
diff changeset
   408
90ce3da70b43 Initial load
duke
parents:
diff changeset
   409
                if (connect_rv >= 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   410
                    break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   411
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   412
                if (errno != EINTR) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   413
                    break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   414
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   415
90ce3da70b43 Initial load
duke
parents:
diff changeset
   416
                /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   417
                 * The poll was interrupted so adjust timeout and
90ce3da70b43 Initial load
duke
parents:
diff changeset
   418
                 * restart
90ce3da70b43 Initial load
duke
parents:
diff changeset
   419
                 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   420
                newTime = JVM_CurrentTimeMillis(env, 0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   421
                timeout -= (newTime - prevTime);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   422
                if (timeout <= 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   423
                    connect_rv = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   424
                    break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   425
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   426
                prevTime = newTime;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   427
90ce3da70b43 Initial load
duke
parents:
diff changeset
   428
            } /* while */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   429
90ce3da70b43 Initial load
duke
parents:
diff changeset
   430
            if (connect_rv == 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   431
                JNU_ThrowByName(env, JNU_JAVANETPKG "SocketTimeoutException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   432
                            "connect timed out");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   433
90ce3da70b43 Initial load
duke
parents:
diff changeset
   434
                /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   435
                 * Timeout out but connection may still be established.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   436
                 * At the high level it should be closed immediately but
90ce3da70b43 Initial load
duke
parents:
diff changeset
   437
                 * just in case we make the socket blocking again and
90ce3da70b43 Initial load
duke
parents:
diff changeset
   438
                 * shutdown input & output.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   439
                 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   440
                SET_BLOCKING(fd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   441
                JVM_SocketShutdown(fd, 2);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   442
                return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   443
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   444
90ce3da70b43 Initial load
duke
parents:
diff changeset
   445
            /* has connection been established */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   446
            optlen = sizeof(connect_rv);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   447
            if (JVM_GetSockOpt(fd, SOL_SOCKET, SO_ERROR, (void*)&connect_rv,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   448
                               &optlen) <0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   449
                connect_rv = errno;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   450
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   451
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   452
90ce3da70b43 Initial load
duke
parents:
diff changeset
   453
        /* make socket blocking again */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   454
        SET_BLOCKING(fd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   455
90ce3da70b43 Initial load
duke
parents:
diff changeset
   456
        /* restore errno */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   457
        if (connect_rv != 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   458
            errno = connect_rv;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   459
            connect_rv = JVM_IO_ERR;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   460
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   461
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   462
90ce3da70b43 Initial load
duke
parents:
diff changeset
   463
    /* report the appropriate exception */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   464
    if (connect_rv < 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   465
90ce3da70b43 Initial load
duke
parents:
diff changeset
   466
#ifdef __linux__
90ce3da70b43 Initial load
duke
parents:
diff changeset
   467
        /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   468
         * Linux/GNU distribution setup /etc/hosts so that
90ce3da70b43 Initial load
duke
parents:
diff changeset
   469
         * InetAddress.getLocalHost gets back the loopback address
90ce3da70b43 Initial load
duke
parents:
diff changeset
   470
         * rather than the host address. Thus a socket can be
90ce3da70b43 Initial load
duke
parents:
diff changeset
   471
         * bound to the loopback address and the connect will
90ce3da70b43 Initial load
duke
parents:
diff changeset
   472
         * fail with EADDRNOTAVAIL. In addition the Linux kernel
90ce3da70b43 Initial load
duke
parents:
diff changeset
   473
         * returns the wrong error in this case - it returns EINVAL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   474
         * instead of EADDRNOTAVAIL. We handle this here so that
90ce3da70b43 Initial load
duke
parents:
diff changeset
   475
         * a more descriptive exception text is used.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   476
         */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   477
        if (connect_rv == JVM_IO_ERR && errno == EINVAL) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   478
            JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   479
                "Invalid argument or cannot assign requested address");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   480
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   481
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   482
#endif
90ce3da70b43 Initial load
duke
parents:
diff changeset
   483
        if (connect_rv == JVM_IO_INTR) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   484
            JNU_ThrowByName(env, JNU_JAVAIOPKG "InterruptedIOException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   485
                            "operation interrupted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   486
        } else if (errno == EPROTO) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   487
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "ProtocolException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   488
                           "Protocol error");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   489
        } else if (errno == ECONNREFUSED) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   490
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "ConnectException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   491
                           "Connection refused");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   492
        } else if (errno == ETIMEDOUT) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   493
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "ConnectException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   494
                           "Connection timed out");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   495
        } else if (errno == EHOSTUNREACH) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   496
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "NoRouteToHostException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   497
                           "Host unreachable");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   498
        } else if (errno == EADDRNOTAVAIL) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   499
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "NoRouteToHostException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   500
                             "Address not available");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   501
        } else if ((errno == EISCONN) || (errno == EBADF)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   502
            JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   503
                            "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   504
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   505
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException", "connect failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   506
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   507
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   508
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   509
90ce3da70b43 Initial load
duke
parents:
diff changeset
   510
    (*env)->SetIntField(env, fdObj, IO_fd_fdID, fd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   511
90ce3da70b43 Initial load
duke
parents:
diff changeset
   512
    /* set the remote peer address and port */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   513
    (*env)->SetObjectField(env, this, psi_addressID, iaObj);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   514
    (*env)->SetIntField(env, this, psi_portID, port);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   515
90ce3da70b43 Initial load
duke
parents:
diff changeset
   516
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   517
     * we need to initialize the local port field if bind was called
90ce3da70b43 Initial load
duke
parents:
diff changeset
   518
     * previously to the connect (by the client) then localport field
90ce3da70b43 Initial load
duke
parents:
diff changeset
   519
     * will already be initialized
90ce3da70b43 Initial load
duke
parents:
diff changeset
   520
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   521
    if (localport == 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   522
        /* Now that we're a connected socket, let's extract the port number
90ce3da70b43 Initial load
duke
parents:
diff changeset
   523
         * that the system chose for us and store it in the Socket object.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   524
         */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   525
        len = SOCKADDR_LEN;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   526
        if (JVM_GetSockName(fd, (struct sockaddr *)&him, &len) == -1) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   527
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   528
                           "Error getting socket name");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   529
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   530
            localport = NET_GetPortFromSockaddr((struct sockaddr *)&him);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   531
            (*env)->SetIntField(env, this, psi_localportID, localport);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   532
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   533
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   534
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   535
90ce3da70b43 Initial load
duke
parents:
diff changeset
   536
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   537
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   538
 * Method:    socketBind
90ce3da70b43 Initial load
duke
parents:
diff changeset
   539
 * Signature: (Ljava/net/InetAddress;I)V
90ce3da70b43 Initial load
duke
parents:
diff changeset
   540
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   541
JNIEXPORT void JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   542
Java_java_net_PlainSocketImpl_socketBind(JNIEnv *env, jobject this,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   543
                                         jobject iaObj, jint localport) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   544
90ce3da70b43 Initial load
duke
parents:
diff changeset
   545
    /* fdObj is the FileDescriptor field on this */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   546
    jobject fdObj = (*env)->GetObjectField(env, this, psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   547
    /* fd is an int field on fdObj */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   548
    int fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   549
    int len;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   550
    SOCKADDR him;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   551
90ce3da70b43 Initial load
duke
parents:
diff changeset
   552
    if (IS_NULL(fdObj)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   553
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   554
                        "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   555
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   556
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   557
        fd = (*env)->GetIntField(env, fdObj, IO_fd_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   558
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   559
    if (IS_NULL(iaObj)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   560
        JNU_ThrowNullPointerException(env, "iaObj is null.");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   561
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   562
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   563
90ce3da70b43 Initial load
duke
parents:
diff changeset
   564
    /* bind */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   565
    if (NET_InetAddressToSockaddr(env, iaObj, localport, (struct sockaddr *)&him, &len, JNI_TRUE) != 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   566
      return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   567
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   568
90ce3da70b43 Initial load
duke
parents:
diff changeset
   569
    if (NET_Bind(fd, (struct sockaddr *)&him, len) < 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   570
        if (errno == EADDRINUSE || errno == EADDRNOTAVAIL ||
90ce3da70b43 Initial load
duke
parents:
diff changeset
   571
            errno == EPERM || errno == EACCES) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   572
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "BindException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   573
                           "Bind failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   574
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   575
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   576
                           "Bind failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   577
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   578
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   579
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   580
90ce3da70b43 Initial load
duke
parents:
diff changeset
   581
    /* set the address */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   582
    (*env)->SetObjectField(env, this, psi_addressID, iaObj);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   583
90ce3da70b43 Initial load
duke
parents:
diff changeset
   584
    /* intialize the local port */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   585
    if (localport == 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   586
        /* Now that we're a connected socket, let's extract the port number
90ce3da70b43 Initial load
duke
parents:
diff changeset
   587
         * that the system chose for us and store it in the Socket object.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   588
         */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   589
        if (JVM_GetSockName(fd, (struct sockaddr *)&him, &len) == -1) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   590
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   591
                           "Error getting socket name");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   592
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   593
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   594
        localport = NET_GetPortFromSockaddr((struct sockaddr *)&him);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   595
        (*env)->SetIntField(env, this, psi_localportID, localport);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   596
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   597
        (*env)->SetIntField(env, this, psi_localportID, localport);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   598
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   599
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   600
90ce3da70b43 Initial load
duke
parents:
diff changeset
   601
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   602
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   603
 * Method:    socketListen
90ce3da70b43 Initial load
duke
parents:
diff changeset
   604
 * Signature: (I)V
90ce3da70b43 Initial load
duke
parents:
diff changeset
   605
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   606
JNIEXPORT void JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   607
Java_java_net_PlainSocketImpl_socketListen (JNIEnv *env, jobject this,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   608
                                            jint count)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   609
{
90ce3da70b43 Initial load
duke
parents:
diff changeset
   610
    /* this FileDescriptor fd field */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   611
    jobject fdObj = (*env)->GetObjectField(env, this, psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   612
    /* fdObj's int fd field */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   613
    int fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   614
90ce3da70b43 Initial load
duke
parents:
diff changeset
   615
    if (IS_NULL(fdObj)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   616
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   617
                        "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   618
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   619
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   620
        fd = (*env)->GetIntField(env, fdObj, IO_fd_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   621
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   622
90ce3da70b43 Initial load
duke
parents:
diff changeset
   623
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   624
     * Workaround for bugid 4101691 in Solaris 2.6. See 4106600.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   625
     * If listen backlog is Integer.MAX_VALUE then subtract 1.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   626
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   627
    if (count == 0x7fffffff)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   628
        count -= 1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   629
90ce3da70b43 Initial load
duke
parents:
diff changeset
   630
    if (JVM_Listen(fd, count) == JVM_IO_ERR) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   631
        NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   632
                       "Listen failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   633
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   634
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   635
90ce3da70b43 Initial load
duke
parents:
diff changeset
   636
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   637
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   638
 * Method:    socketAccept
90ce3da70b43 Initial load
duke
parents:
diff changeset
   639
 * Signature: (Ljava/net/SocketImpl;)V
90ce3da70b43 Initial load
duke
parents:
diff changeset
   640
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   641
JNIEXPORT void JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   642
Java_java_net_PlainSocketImpl_socketAccept(JNIEnv *env, jobject this,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   643
                                           jobject socket)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   644
{
90ce3da70b43 Initial load
duke
parents:
diff changeset
   645
    /* fields on this */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   646
    int port;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   647
    jint timeout = (*env)->GetIntField(env, this, psi_timeoutID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   648
    jlong prevTime = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   649
    jobject fdObj = (*env)->GetObjectField(env, this, psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   650
90ce3da70b43 Initial load
duke
parents:
diff changeset
   651
    /* the FileDescriptor field on socket */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   652
    jobject socketFdObj;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   653
    /* the InetAddress field on socket */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   654
    jobject socketAddressObj;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   655
90ce3da70b43 Initial load
duke
parents:
diff changeset
   656
    /* the ServerSocket fd int field on fdObj */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   657
    jint fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   658
90ce3da70b43 Initial load
duke
parents:
diff changeset
   659
    /* accepted fd */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   660
    jint newfd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   661
90ce3da70b43 Initial load
duke
parents:
diff changeset
   662
    SOCKADDR him;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   663
    int len;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   664
90ce3da70b43 Initial load
duke
parents:
diff changeset
   665
    len = SOCKADDR_LEN;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   666
90ce3da70b43 Initial load
duke
parents:
diff changeset
   667
    if (IS_NULL(fdObj)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   668
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   669
                        "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   670
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   671
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   672
        fd = (*env)->GetIntField(env, fdObj, IO_fd_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   673
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   674
    if (IS_NULL(socket)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   675
        JNU_ThrowNullPointerException(env, "socket is null");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   676
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   677
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   678
90ce3da70b43 Initial load
duke
parents:
diff changeset
   679
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   680
     * accept connection but ignore ECONNABORTED indicating that
90ce3da70b43 Initial load
duke
parents:
diff changeset
   681
     * connection was eagerly accepted by the OS but was reset
90ce3da70b43 Initial load
duke
parents:
diff changeset
   682
     * before accept() was called.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   683
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   684
     * If accept timeout in place and timeout is adjusted with
90ce3da70b43 Initial load
duke
parents:
diff changeset
   685
     * each ECONNABORTED or EWOULDBLOCK to ensure that semantics
90ce3da70b43 Initial load
duke
parents:
diff changeset
   686
     * of timeout are preserved.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   687
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   688
    for (;;) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   689
        int ret;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   690
90ce3da70b43 Initial load
duke
parents:
diff changeset
   691
        /* first usage pick up current time */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   692
        if (prevTime == 0 && timeout > 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   693
            prevTime = JVM_CurrentTimeMillis(env, 0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   694
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   695
90ce3da70b43 Initial load
duke
parents:
diff changeset
   696
        /* passing a timeout of 0 to poll will return immediately,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   697
           but in the case of ServerSocket 0 means infinite. */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   698
        if (timeout <= 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   699
            ret = NET_Timeout(fd, -1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   700
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   701
            ret = NET_Timeout(fd, timeout);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   702
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   703
90ce3da70b43 Initial load
duke
parents:
diff changeset
   704
        if (ret == 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   705
            JNU_ThrowByName(env, JNU_JAVANETPKG "SocketTimeoutException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   706
                            "Accept timed out");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   707
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   708
        } else if (ret == JVM_IO_ERR) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   709
            if (errno == EBADF) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   710
               JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException", "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   711
            } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   712
               NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException", "Accept failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   713
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   714
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   715
        } else if (ret == JVM_IO_INTR) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   716
            JNU_ThrowByName(env, JNU_JAVAIOPKG "InterruptedIOException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   717
                            "operation interrupted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   718
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   719
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   720
90ce3da70b43 Initial load
duke
parents:
diff changeset
   721
        newfd = NET_Accept(fd, (struct sockaddr *)&him, (jint*)&len);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   722
90ce3da70b43 Initial load
duke
parents:
diff changeset
   723
        /* connection accepted */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   724
        if (newfd >= 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   725
            SET_BLOCKING(newfd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   726
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   727
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   728
90ce3da70b43 Initial load
duke
parents:
diff changeset
   729
        /* non (ECONNABORTED or EWOULDBLOCK) error */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   730
        if (!(errno == ECONNABORTED || errno == EWOULDBLOCK)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   731
            break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   732
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   733
90ce3da70b43 Initial load
duke
parents:
diff changeset
   734
        /* ECONNABORTED or EWOULDBLOCK error so adjust timeout if there is one. */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   735
        if (timeout) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   736
            jlong currTime = JVM_CurrentTimeMillis(env, 0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   737
            timeout -= (currTime - prevTime);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   738
90ce3da70b43 Initial load
duke
parents:
diff changeset
   739
            if (timeout <= 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   740
                JNU_ThrowByName(env, JNU_JAVANETPKG "SocketTimeoutException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   741
                                "Accept timed out");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   742
                return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   743
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   744
            prevTime = currTime;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   745
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   746
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   747
90ce3da70b43 Initial load
duke
parents:
diff changeset
   748
    if (newfd < 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   749
        if (newfd == -2) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   750
            JNU_ThrowByName(env, JNU_JAVAIOPKG "InterruptedIOException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   751
                            "operation interrupted");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   752
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   753
            if (errno == EINVAL) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   754
                errno = EBADF;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   755
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   756
            if (errno == EBADF) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   757
                JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException", "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   758
            } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   759
                NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException", "Accept failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   760
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   761
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   762
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   763
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   764
90ce3da70b43 Initial load
duke
parents:
diff changeset
   765
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   766
     * fill up the remote peer port and address in the new socket structure.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   767
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   768
    socketAddressObj = NET_SockaddrToInetAddress(env, (struct sockaddr *)&him, &port);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   769
    if (socketAddressObj == NULL) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   770
        /* should be pending exception */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   771
        close(newfd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   772
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   773
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   774
90ce3da70b43 Initial load
duke
parents:
diff changeset
   775
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   776
     * Populate SocketImpl.fd.fd
90ce3da70b43 Initial load
duke
parents:
diff changeset
   777
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   778
    socketFdObj = (*env)->GetObjectField(env, socket, psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   779
    (*env)->SetIntField(env, socketFdObj, IO_fd_fdID, newfd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   780
90ce3da70b43 Initial load
duke
parents:
diff changeset
   781
    (*env)->SetObjectField(env, socket, psi_addressID, socketAddressObj);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   782
    (*env)->SetIntField(env, socket, psi_portID, port);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   783
    /* also fill up the local port information */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   784
     port = (*env)->GetIntField(env, this, psi_localportID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   785
    (*env)->SetIntField(env, socket, psi_localportID, port);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   786
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   787
90ce3da70b43 Initial load
duke
parents:
diff changeset
   788
90ce3da70b43 Initial load
duke
parents:
diff changeset
   789
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   790
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   791
 * Method:    socketAvailable
90ce3da70b43 Initial load
duke
parents:
diff changeset
   792
 * Signature: ()I
90ce3da70b43 Initial load
duke
parents:
diff changeset
   793
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   794
JNIEXPORT jint JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   795
Java_java_net_PlainSocketImpl_socketAvailable(JNIEnv *env, jobject this) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   796
90ce3da70b43 Initial load
duke
parents:
diff changeset
   797
    jint ret = -1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   798
    jobject fdObj = (*env)->GetObjectField(env, this, psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   799
    jint fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   800
90ce3da70b43 Initial load
duke
parents:
diff changeset
   801
    if (IS_NULL(fdObj)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   802
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   803
                        "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   804
        return -1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   805
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   806
        fd = (*env)->GetIntField(env, fdObj, IO_fd_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   807
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   808
    /* JVM_SocketAvailable returns 0 for failure, 1 for success */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   809
    if (!JVM_SocketAvailable(fd, &ret)){
90ce3da70b43 Initial load
duke
parents:
diff changeset
   810
        if (errno == ECONNRESET) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   811
            JNU_ThrowByName(env, "sun/net/ConnectionResetException", "");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   812
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   813
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   814
                                         "ioctl FIONREAD failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   815
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   816
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   817
    return ret;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   818
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   819
90ce3da70b43 Initial load
duke
parents:
diff changeset
   820
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   821
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   822
 * Method:    socketClose0
90ce3da70b43 Initial load
duke
parents:
diff changeset
   823
 * Signature: (Z)V
90ce3da70b43 Initial load
duke
parents:
diff changeset
   824
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   825
JNIEXPORT void JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   826
Java_java_net_PlainSocketImpl_socketClose0(JNIEnv *env, jobject this,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   827
                                          jboolean useDeferredClose) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   828
90ce3da70b43 Initial load
duke
parents:
diff changeset
   829
    jobject fdObj = (*env)->GetObjectField(env, this, psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   830
    jint fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   831
90ce3da70b43 Initial load
duke
parents:
diff changeset
   832
    if (IS_NULL(fdObj)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   833
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   834
                        "socket already closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   835
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   836
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   837
        fd = (*env)->GetIntField(env, fdObj, IO_fd_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   838
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   839
    if (fd != -1) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   840
        if (useDeferredClose && marker_fd >= 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   841
            NET_Dup2(marker_fd, fd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   842
        } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   843
            (*env)->SetIntField(env, fdObj, IO_fd_fdID, -1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   844
            NET_SocketClose(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
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   848
90ce3da70b43 Initial load
duke
parents:
diff changeset
   849
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   850
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   851
 * Method:    socketShutdown
90ce3da70b43 Initial load
duke
parents:
diff changeset
   852
 * Signature: (I)V
90ce3da70b43 Initial load
duke
parents:
diff changeset
   853
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   854
JNIEXPORT void JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   855
Java_java_net_PlainSocketImpl_socketShutdown(JNIEnv *env, jobject this,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   856
                                             jint howto)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   857
{
90ce3da70b43 Initial load
duke
parents:
diff changeset
   858
90ce3da70b43 Initial load
duke
parents:
diff changeset
   859
    jobject fdObj = (*env)->GetObjectField(env, this, psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   860
    jint fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   861
90ce3da70b43 Initial load
duke
parents:
diff changeset
   862
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   863
     * WARNING: THIS NEEDS LOCKING. ALSO: SHOULD WE CHECK for fd being
90ce3da70b43 Initial load
duke
parents:
diff changeset
   864
     * -1 already?
90ce3da70b43 Initial load
duke
parents:
diff changeset
   865
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   866
    if (IS_NULL(fdObj)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   867
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   868
                        "socket already closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   869
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   870
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   871
        fd = (*env)->GetIntField(env, fdObj, IO_fd_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   872
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   873
    JVM_SocketShutdown(fd, howto);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   874
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   875
90ce3da70b43 Initial load
duke
parents:
diff changeset
   876
90ce3da70b43 Initial load
duke
parents:
diff changeset
   877
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   878
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   879
 * Method:    socketSetOption
90ce3da70b43 Initial load
duke
parents:
diff changeset
   880
 * Signature: (IZLjava/lang/Object;)V
90ce3da70b43 Initial load
duke
parents:
diff changeset
   881
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   882
JNIEXPORT void JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   883
Java_java_net_PlainSocketImpl_socketSetOption(JNIEnv *env, jobject this,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   884
                                              jint cmd, jboolean on,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   885
                                              jobject value) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   886
    int fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   887
    int level, optname, optlen;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   888
    union {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   889
        int i;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   890
        struct linger ling;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   891
    } optval;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   892
90ce3da70b43 Initial load
duke
parents:
diff changeset
   893
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   894
     * Check that socket hasn't been closed
90ce3da70b43 Initial load
duke
parents:
diff changeset
   895
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   896
    fd = getFD(env, this);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   897
    if (fd < 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   898
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   899
                        "Socket closed");
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
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   904
     * SO_TIMEOUT is a no-op on Solaris/Linux
90ce3da70b43 Initial load
duke
parents:
diff changeset
   905
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   906
    if (cmd == java_net_SocketOptions_SO_TIMEOUT) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   907
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   908
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   909
90ce3da70b43 Initial load
duke
parents:
diff changeset
   910
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   911
     * Map the Java level socket option to the platform specific
90ce3da70b43 Initial load
duke
parents:
diff changeset
   912
     * level and option name.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   913
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   914
    if (NET_MapSocketOption(cmd, &level, &optname)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   915
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException", "Invalid option");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   916
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   917
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   918
90ce3da70b43 Initial load
duke
parents:
diff changeset
   919
    switch (cmd) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   920
        case java_net_SocketOptions_SO_SNDBUF :
90ce3da70b43 Initial load
duke
parents:
diff changeset
   921
        case java_net_SocketOptions_SO_RCVBUF :
90ce3da70b43 Initial load
duke
parents:
diff changeset
   922
        case java_net_SocketOptions_SO_LINGER :
90ce3da70b43 Initial load
duke
parents:
diff changeset
   923
        case java_net_SocketOptions_IP_TOS :
90ce3da70b43 Initial load
duke
parents:
diff changeset
   924
            {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   925
                jclass cls;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   926
                jfieldID fid;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   927
90ce3da70b43 Initial load
duke
parents:
diff changeset
   928
                cls = (*env)->FindClass(env, "java/lang/Integer");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   929
                CHECK_NULL(cls);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   930
                fid = (*env)->GetFieldID(env, cls, "value", "I");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   931
                CHECK_NULL(fid);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   932
90ce3da70b43 Initial load
duke
parents:
diff changeset
   933
                if (cmd == java_net_SocketOptions_SO_LINGER) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   934
                    if (on) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   935
                        optval.ling.l_onoff = 1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   936
                        optval.ling.l_linger = (*env)->GetIntField(env, value, fid);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   937
                    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   938
                        optval.ling.l_onoff = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   939
                        optval.ling.l_linger = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   940
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   941
                    optlen = sizeof(optval.ling);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   942
                } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   943
                    optval.i = (*env)->GetIntField(env, value, fid);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   944
                    optlen = sizeof(optval.i);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   945
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   946
90ce3da70b43 Initial load
duke
parents:
diff changeset
   947
                break;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   948
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   949
90ce3da70b43 Initial load
duke
parents:
diff changeset
   950
        /* Boolean -> int */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   951
        default :
90ce3da70b43 Initial load
duke
parents:
diff changeset
   952
            optval.i = (on ? 1 : 0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   953
            optlen = sizeof(optval.i);
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
    if (NET_SetSockOpt(fd, level, optname, (const void *)&optval, optlen) < 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   958
#ifdef __solaris__
90ce3da70b43 Initial load
duke
parents:
diff changeset
   959
        if (errno == EINVAL) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   960
            // On Solaris setsockopt will set errno to EINVAL if the socket
90ce3da70b43 Initial load
duke
parents:
diff changeset
   961
            // is closed. The default error message is then confusing
90ce3da70b43 Initial load
duke
parents:
diff changeset
   962
            char fullMsg[128];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   963
            jio_snprintf(fullMsg, sizeof(fullMsg), "Invalid option or socket reset by remote peer");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   964
            JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException", fullMsg);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   965
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   966
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   967
#endif /* __solaris__ */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   968
        NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   969
                                      "Error setting socket option");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   970
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   971
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   972
90ce3da70b43 Initial load
duke
parents:
diff changeset
   973
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   974
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
   975
 * Method:    socketGetOption
90ce3da70b43 Initial load
duke
parents:
diff changeset
   976
 * Signature: (I)I
90ce3da70b43 Initial load
duke
parents:
diff changeset
   977
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   978
JNIEXPORT jint JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
   979
Java_java_net_PlainSocketImpl_socketGetOption(JNIEnv *env, jobject this,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   980
                                              jint cmd, jobject iaContainerObj) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   981
90ce3da70b43 Initial load
duke
parents:
diff changeset
   982
    int fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   983
    int level, optname, optlen;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   984
    union {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   985
        int i;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   986
        struct linger ling;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   987
    } optval;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   988
90ce3da70b43 Initial load
duke
parents:
diff changeset
   989
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   990
     * Check that socket hasn't been closed
90ce3da70b43 Initial load
duke
parents:
diff changeset
   991
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   992
    fd = getFD(env, this);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   993
    if (fd < 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   994
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
   995
                        "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   996
        return -1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   997
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   998
90ce3da70b43 Initial load
duke
parents:
diff changeset
   999
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1000
     * SO_BINDADDR isn't a socket option
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1001
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1002
    if (cmd == java_net_SocketOptions_SO_BINDADDR) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1003
        SOCKADDR him;
710
57367409c4a5 6483406: new ServerSocket() sometimes takes more than 3 minutes on Suse Linux
jccollet
parents: 624
diff changeset
  1004
        socklen_t len = 0;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1005
        int port;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1006
        jobject iaObj;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1007
        jclass iaCntrClass;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1008
        jfieldID iaFieldID;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1009
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1010
        len = SOCKADDR_LEN;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1011
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1012
        if (getsockname(fd, (struct sockaddr *)&him, &len) < 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1013
            NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1014
                             "Error getting socket name");
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1015
            return -1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1016
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1017
        iaObj = NET_SockaddrToInetAddress(env, (struct sockaddr *)&him, &port);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1018
        CHECK_NULL_RETURN(iaObj, -1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1019
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1020
        iaCntrClass = (*env)->GetObjectClass(env, iaContainerObj);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1021
        iaFieldID = (*env)->GetFieldID(env, iaCntrClass, "addr", "Ljava/net/InetAddress;");
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1022
        CHECK_NULL_RETURN(iaFieldID, -1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1023
        (*env)->SetObjectField(env, iaContainerObj, iaFieldID, iaObj);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1024
        return 0; /* notice change from before */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1025
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1026
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1027
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1028
     * Map the Java level socket option to the platform specific
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1029
     * level and option name.
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1030
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1031
    if (NET_MapSocketOption(cmd, &level, &optname)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1032
        JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException", "Invalid option");
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1033
        return -1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1034
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1035
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1036
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1037
     * Args are int except for SO_LINGER
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1038
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1039
    if (cmd == java_net_SocketOptions_SO_LINGER) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1040
        optlen = sizeof(optval.ling);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1041
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1042
        optlen = sizeof(optval.i);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1043
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1044
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1045
    if (NET_GetSockOpt(fd, level, optname, (void *)&optval, &optlen) < 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1046
        NET_ThrowByNameWithLastError(env, JNU_JAVANETPKG "SocketException",
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1047
                                      "Error getting socket option");
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1048
        return -1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1049
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1050
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1051
    switch (cmd) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1052
        case java_net_SocketOptions_SO_LINGER:
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1053
            return (optval.ling.l_onoff ? optval.ling.l_linger: -1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1054
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1055
        case java_net_SocketOptions_SO_SNDBUF:
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1056
        case java_net_SocketOptions_SO_RCVBUF:
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1057
        case java_net_SocketOptions_IP_TOS:
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1058
            return optval.i;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1059
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1060
        default :
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1061
            return (optval.i == 0) ? -1 : 1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1062
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1063
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1064
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1065
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1066
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1067
 * Class:     java_net_PlainSocketImpl
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1068
 * Method:    socketSendUrgentData
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1069
 * Signature: (B)V
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1070
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1071
JNIEXPORT void JNICALL
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1072
Java_java_net_PlainSocketImpl_socketSendUrgentData(JNIEnv *env, jobject this,
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1073
                                             jint data) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1074
    /* The fd field */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1075
    jobject fdObj = (*env)->GetObjectField(env, this, psi_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1076
    int n, fd;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1077
    unsigned char d = data & 0xFF;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1078
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1079
    if (IS_NULL(fdObj)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1080
        JNU_ThrowByName(env, "java/net/SocketException", "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1081
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1082
    } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1083
        fd = (*env)->GetIntField(env, fdObj, IO_fd_fdID);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1084
        /* Bug 4086704 - If the Socket associated with this file descriptor
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1085
         * was closed (sysCloseFD), the the file descriptor is set to -1.
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1086
         */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1087
        if (fd == -1) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1088
            JNU_ThrowByName(env, "java/net/SocketException", "Socket closed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1089
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1090
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1091
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1092
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1093
    n = JVM_Send(fd, (char *)&d, 1, MSG_OOB);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1094
    if (n == JVM_IO_ERR) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1095
        NET_ThrowByNameWithLastError(env, "java/io/IOException", "Write failed");
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1096
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1097
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1098
    if (n == JVM_IO_INTR) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1099
        JNU_ThrowByName(env, "java/io/InterruptedIOException", 0);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1100
        return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1101
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1102
}