/* * Copyright (c) 2002-2011 by XMLVM.org * * Project Info: http://www.xmlvm.org * * This program is free software; you can redistribute it and/or modify it * under the terms of the GNU Lesser General Public License as published by * the Free Software Foundation; either version 2.1 of the License, or * (at your option) any later version. * * This library is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public * License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, * USA. */ #include #include #include //#include //#include //#include "java_io_FileDescriptor.h" //#include "java_net_SocketException.h" //#include "java_lang_Thread.h" #include "xmlvm.h" #include "xmlvm-hy.h" #include "xmlvm-sock.h" #include "java_lang_String.h" #include "java_io_FileDescriptor.h" #include "java_net_InetAddress.h" #include "java_net_Inet6Address.h" #define MAX_RETRIES 50 #define INVALID_SOCKET (hysocket_t) -1 #if NO_R /*use a mutex if the gethostbyaddr, gethostbyname calls are not threadsafe*/ MUTEX hostentLock = PTHREAD_MUTEX_INITIALIZER; #endif /*NO_R */ const char* hysock_error_message () { return hyerror_last_error_message (); } char* netLookupErrorString (I_32 anErrorNum) { switch (anErrorNum) { case HYPORT_ERROR_SOCKET_BADSOCKET: return "Bad socket"; case HYPORT_ERROR_SOCKET_NOTINITIALIZED: return "Socket library uninitialized"; case HYPORT_ERROR_SOCKET_BADAF: return "Bad address family"; case HYPORT_ERROR_SOCKET_BADPROTO: return "Bad protocol"; case HYPORT_ERROR_SOCKET_BADTYPE: return "Bad type"; case HYPORT_ERROR_SOCKET_SYSTEMBUSY: return "System busy handling requests"; case HYPORT_ERROR_SOCKET_SYSTEMFULL: return "Too many sockets allocated"; case HYPORT_ERROR_SOCKET_NOTCONNECTED: return "Socket is not connected"; case HYPORT_ERROR_SOCKET_INTERRUPTED: return "The call was cancelled"; case HYPORT_ERROR_SOCKET_TIMEOUT: return "The operation timed out"; case HYPORT_ERROR_SOCKET_CONNRESET: return "The connection was reset"; case HYPORT_ERROR_SOCKET_WOULDBLOCK: return "The socket is marked as nonblocking operation would block"; case HYPORT_ERROR_SOCKET_ADDRNOTAVAIL: return "The address is not available"; case HYPORT_ERROR_SOCKET_ADDRINUSE: return "The address is already in use"; case HYPORT_ERROR_SOCKET_NOTBOUND: return "The socket is not bound"; case HYPORT_ERROR_SOCKET_UNKNOWNSOCKET: return "Resolution of the FileDescriptor to socket failed"; case HYPORT_ERROR_SOCKET_INVALIDTIMEOUT: return "The specified timeout is invalid"; case HYPORT_ERROR_SOCKET_FDSETFULL: return "Unable to create an FDSET"; case HYPORT_ERROR_SOCKET_TIMEVALFULL: return "Unable to create a TIMEVAL"; case HYPORT_ERROR_SOCKET_REMSOCKSHUTDOWN: return "The remote socket has shutdown gracefully"; case HYPORT_ERROR_SOCKET_NOTLISTENING: return "Listen() was not invoked prior to accept()"; case HYPORT_ERROR_SOCKET_NOTSTREAMSOCK: return "The socket does not support connection-oriented service"; case HYPORT_ERROR_SOCKET_ALREADYBOUND: return "The socket is already bound to an address"; case HYPORT_ERROR_SOCKET_NBWITHLINGER: return "The socket is marked non-blocking & SO_LINGER is non-zero"; case HYPORT_ERROR_SOCKET_ISCONNECTED: return "The socket is already connected"; case HYPORT_ERROR_SOCKET_NOBUFFERS: return "No buffer space is available"; case HYPORT_ERROR_SOCKET_HOSTNOTFOUND: return "Authoritative Answer Host not found"; case HYPORT_ERROR_SOCKET_NODATA: return "Valid name, no data record of requested type"; case HYPORT_ERROR_SOCKET_BOUNDORCONN: return "The socket has not been bound or is already connected"; case HYPORT_ERROR_SOCKET_OPNOTSUPP: return "The socket does not support the operation"; case HYPORT_ERROR_SOCKET_OPTUNSUPP: return "The socket option is not supported"; case HYPORT_ERROR_SOCKET_OPTARGSINVALID: return "The socket option arguments are invalid"; case HYPORT_ERROR_SOCKET_SOCKLEVELINVALID: return "The socket level is invalid"; case HYPORT_ERROR_SOCKET_TIMEOUTFAILURE: return "The timeout operation failed"; case HYPORT_ERROR_SOCKET_SOCKADDRALLOCFAIL: return "Failed to allocate address structure"; case HYPORT_ERROR_SOCKET_FDSET_SIZEBAD: return "The calculated maximum size of the file descriptor set is bad"; case HYPORT_ERROR_SOCKET_UNKNOWNFLAG: return "The flag is unknown"; case HYPORT_ERROR_SOCKET_MSGSIZE: return "The datagram was too big to fit the specified buffer, so truncated"; case HYPORT_ERROR_SOCKET_NORECOVERY: return "The operation failed with no recovery possible"; case HYPORT_ERROR_SOCKET_ARGSINVALID: return "The arguments are invalid"; case HYPORT_ERROR_SOCKET_BADDESC: return "The socket argument is not a valid file descriptor"; case HYPORT_ERROR_SOCKET_NOTSOCK: return "The socket argument is not a socket"; case HYPORT_ERROR_SOCKET_HOSTENTALLOCFAIL: return "Unable to allocate the hostent structure"; case HYPORT_ERROR_SOCKET_TIMEVALALLOCFAIL: return "Unable to allocate the timeval structure"; case HYPORT_ERROR_SOCKET_LINGERALLOCFAIL: return "Unable to allocate the linger structure"; case HYPORT_ERROR_SOCKET_IPMREQALLOCFAIL: return "Unable to allocate the ipmreq structure"; case HYPORT_ERROR_SOCKET_FDSETALLOCFAIL: return "Unable to allocate the fdset structure"; case HYPORT_ERROR_SOCKET_CONNECTION_REFUSED: return "Connection refused"; case HYPORT_ERROR_FILE_LOCKED: return "File has been locked by another process"; default: return (char *) hysock_error_message (); } } static I_32 findError (I_32 errorCode) { switch (errorCode) { case HYPORT_ERROR_SOCKET_UNIX_EBADF: return HYPORT_ERROR_SOCKET_BADDESC; case HYPORT_ERROR_SOCKET_UNIX_ENOBUFS: return HYPORT_ERROR_SOCKET_NOBUFFERS; case HYPORT_ERROR_SOCKET_UNIX_EOPNOTSUP: return HYPORT_ERROR_SOCKET_OPNOTSUPP; case HYPORT_ERROR_SOCKET_UNIX_ENOPROTOOPT: return HYPORT_ERROR_SOCKET_OPTUNSUPP; case HYPORT_ERROR_SOCKET_UNIX_EINVAL: return HYPORT_ERROR_SOCKET_SOCKLEVELINVALID; case HYPORT_ERROR_SOCKET_UNIX_ENOTSOCK: return HYPORT_ERROR_SOCKET_NOTSOCK; case HYPORT_ERROR_SOCKET_UNIX_EINTR: return HYPORT_ERROR_SOCKET_INTERRUPTED; case HYPORT_ERROR_SOCKET_UNIX_ENOTCONN: return HYPORT_ERROR_SOCKET_NOTCONNECTED; case HYPORT_ERROR_SOCKET_UNIX_EAFNOSUPPORT: return HYPORT_ERROR_SOCKET_BADAF; /* note: HYPORT_ERROR_SOCKET_UNIX_ECONNRESET not included because it has the same * value as HYPORT_ERROR_SOCKET_UNIX_CONNRESET and they both map to HYPORT_ERROR_SOCKET_CONNRESET */ case HYPORT_ERROR_SOCKET_UNIX_CONNRESET: return HYPORT_ERROR_SOCKET_CONNRESET; case HYPORT_ERROR_SOCKET_UNIX_EAGAIN: return HYPORT_ERROR_SOCKET_WOULDBLOCK; case HYPORT_ERROR_SOCKET_UNIX_EPROTONOSUPPORT: return HYPORT_ERROR_SOCKET_BADPROTO; case HYPORT_ERROR_SOCKET_UNIX_EFAULT: return HYPORT_ERROR_SOCKET_ARGSINVALID; case HYPORT_ERROR_SOCKET_UNIX_ETIMEDOUT: return HYPORT_ERROR_SOCKET_TIMEOUT; case HYPORT_ERROR_SOCKET_UNIX_CONNREFUSED: return HYPORT_ERROR_SOCKET_CONNECTION_REFUSED; case HYPORT_ERROR_SOCKET_UNIX_ENETUNREACH: return HYPORT_ERROR_SOCKET_ENETUNREACH; case HYPORT_ERROR_SOCKET_UNIX_EACCES: return HYPORT_ERROR_SOCKET_EACCES; default: return HYPORT_ERROR_SOCKET_OPFAILED; } } int harmony_supports_ipv6() { return 0; } int preferIPv4Stack() { return 1; } int preferIPv6Addresses() { return 0; } static socklen_t getAddrLength(hysockaddr_t addr) { return #if defined(IPv6_FUNCTION_SUPPORT) ((OSSOCKADDR *) & addr->addr)->sin_family == OS_AF_INET6 ? sizeof(OSSOCKADDR_IN6) : #endif sizeof(OSSOCKADDR); } U_16 hysock_htons(U_16 port) { return htons(port); } U_16 hysock_ntohs (U_16 val) { return ntohs (val); } I_32 hysock_socketIsValid (hysocket_t handle) { return ((handle != NULL) && (handle != INVALID_SOCKET)); } U_16 hysock_sockaddr_port (hysockaddr_t handle) { if (((OSSOCKADDR *) & handle->addr)->sin_family == OS_AF_INET4) { return ((OSSOCKADDR *) & handle->addr)->sin_port; } #if defined(IPv6_FUNCTION_SUPPORT) else { return ((OSSOCKADDR_IN6 *) & handle->addr)->sin6_port; } #endif } void setJavaIoFileDescriptorContents (JAVA_OBJECT fd, void *value) { ((java_io_FileDescriptor*) fd)->fields.java_io_FileDescriptor.descriptor_ = (JAVA_ULONG) value; } void* getJavaIoFileDescriptorContentsAsAPointer (JAVA_OBJECT fd) { return (void*) ((java_io_FileDescriptor*) fd)->fields.java_io_FileDescriptor.descriptor_; } void netGetJavaNetInetAddressValue (JAVA_OBJECT anInetAddress, U_8* buffer, U_32* length) { org_xmlvm_runtime_XMLVMArray* byte_array = ((java_net_InetAddress*) anInetAddress)->fields.java_net_InetAddress.ipaddress_; *length = byte_array->fields.org_xmlvm_runtime_XMLVMArray.length_; XMLVM_MEMCPY(buffer, byte_array->fields.org_xmlvm_runtime_XMLVMArray.array_, *length); } void netGetJavaNetInetAddressScopeId (JAVA_OBJECT anInetAddress, U_32* scope_id) { if (XMLVM_ISA(anInetAddress, __CLASS_java_net_Inet6Address)) { java_net_Inet6Address* addr = (java_net_Inet6Address*) anInetAddress; *scope_id = addr->fields.java_net_Inet6Address.scope_id_; } else { *scope_id = 0; } } I_32 netGetSockAddr (JAVA_OBJECT fileDescriptor, hysockaddr_t sockaddrP, JAVA_BOOLEAN preferIPv6Addresses) { I_32 result = 0; hysocket_t socketP; U_8 ipAddr[HYSOCK_INADDR6_LEN]; memset (ipAddr, 0, HYSOCK_INADDR6_LEN); socketP = getJavaIoFileDescriptorContentsAsAPointer (fileDescriptor); if (!hysock_socketIsValid (socketP)) { return HYPORT_ERROR_SOCKET_UNKNOWNSOCKET; } else { if (preferIPv6Addresses) { hysock_sockaddr_init6 (sockaddrP, ipAddr, HYSOCK_INADDR6_LEN, HYADDR_FAMILY_UNSPEC, 0, 0, 0, socketP); result = hysock_getsockname (socketP, sockaddrP); } else { hysock_sockaddr_init6 (sockaddrP, ipAddr, HYSOCK_INADDR_LEN, HYADDR_FAMILY_AFINET4, 0, 0, 0, socketP); result = hysock_getsockname (socketP, sockaddrP); } return result; } } I_32 map_addr_family_Hy_to_OS (I_32 addr_family) { switch (addr_family) { case HYADDR_FAMILY_AFINET4: return OS_AF_INET4; case HYADDR_FAMILY_AFINET6: return OS_AF_INET6; } return OS_AF_UNSPEC; } static I_32 copy_hostent (OSHOSTENT * source, PortlibPTBuffers_t * ptBuffers) { int h_len = strlen (source->h_name); int total = 0; int i = 0; char *buffer; OSHOSTENT *dest = &(*ptBuffers)->hostent; /* add trailing NULL, and round off to nearest pointer size */ h_len = (h_len + 1 + sizeof (void *)) & ~(sizeof (void *) - 1); while (source->h_addr_list[i]) i++; total = h_len + ((i + 2) * sizeof (void *)) + (i * sizeof (U_32)); if (!(*ptBuffers)->gethostBuffer || (*ptBuffers)->gethostBufferSize < total) { (*ptBuffers)->gethostBuffer = XMLVM_ATOMIC_MALLOC(total); if (!(*ptBuffers)->gethostBuffer) { return HYPORT_ERROR_SOCKET_SYSTEMFULL; } (*ptBuffers)->gethostBufferSize = total; } buffer = (*ptBuffers)->gethostBuffer; dest->h_name = buffer; strcpy (buffer, source->h_name); buffer += h_len; dest->h_aliases = (void *) buffer; *((void **) buffer) = NULL; buffer += sizeof (void *); dest->h_addrtype = source->h_addrtype; dest->h_length = source->h_length; dest->h_addr_list = (void *) buffer; buffer += (i + 1) * sizeof (void *); i = 0; while (source->h_addr_list[i]) { dest->h_addr_list[i] = (void *) buffer; *((U_32 *) buffer) = *((U_32 *) source->h_addr_list[i]); buffer += sizeof (U_32); i++; } dest->h_addr_list[i] = NULL; return 0; } JAVA_OBJECT newJavaByteArray (JAVA_ARRAY_BYTE* bytes, JAVA_INT length) { JAVA_OBJECT result = XMLVMArray_createSingleDimensionWithData(__CLASS_byte_1ARRAY, (int) length, bytes); return result; } JAVA_OBJECT newJavaNetInetAddressGenericBS (JAVA_ARRAY_BYTE* address, U_32 length, const char* hostName, U_32 scope_id) { org_xmlvm_runtime_XMLVMArray* byte_array; java_lang_String* aString; BOOLEAN isAnyAddress = 1; static JAVA_ARRAY_BYTE IPv4ANY[4] = { 0, 0, 0, 0 }; static JAVA_ARRAY_BYTE IPv6ANY[16] = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }; U_32 i = 0; JAVA_OBJECT result = JAVA_NULL; aString = xmlvm_create_java_string(hostName); /* check if the address being returned is the any address. If so we need to check the prefer flags to see how it should be returned (either as IPv4 Any or IPv6 ANY) */ if (harmony_supports_ipv6 ()) { /* Figure out if it is the any address */ for (i = 0; i < length; i++) { if (address[i] != 0) { isAnyAddress = 0; break; } } } else { /* just do what we used to do without checking */ isAnyAddress = 0; } /* If it is the any address then set up to return appropriately based on the flags */ if (isAnyAddress) { if ((!preferIPv4Stack ()) && (preferIPv6Addresses ())) { if ((byte_array = newJavaByteArray (IPv6ANY, sizeof (IPv6ANY))) == NULL) { return NULL; } } else { if ((byte_array = newJavaByteArray (IPv4ANY, sizeof (IPv4ANY))) == NULL) { return NULL; } } } else { /* not any so just set up to return the address normally */ if ((byte_array = newJavaByteArray (address, length)) == NULL) { return NULL; } } if (harmony_supports_ipv6 ()) { #ifdef SUPPORTS_SCOPED_GETBYADDR if (scope_id != 0) { result = java_net_InetAddress_getByAddress___java_lang_String_byte_1ARRAY_int(aString, byte_array, scope_id); } else { #endif result = java_net_InetAddress_getByAddress___java_lang_String_byte_1ARRAY(aString, byte_array); #ifdef SUPPORTS_SCOPED_GETBYADDR } #endif } else { result = __NEW_java_net_InetAddress(); java_net_InetAddress___INIT____byte_1ARRAY_java_lang_String(result, byte_array, aString); //return result; } return result; } I_32 hysock_socket (hysocket_t * handle, I_32 family, I_32 socktype, I_32 protocol) { I_32 rc = 0; int sock; /* initialize return hysocket_t to invalid socket */ *handle = INVALID_SOCKET; if (protocol != HYSOCK_DEFPROTOCOL) { rc = HYPORT_ERROR_SOCKET_BADPROTO; } else if ((socktype != HYSOCK_STREAM) && (socktype != HYSOCK_DGRAM)) { rc = HYPORT_ERROR_SOCKET_BADTYPE; } else if (family != HYADDR_FAMILY_AFINET6 && family != HYADDR_FAMILY_AFINET4 && family != HYADDR_FAMILY_UNSPEC) { rc = HYPORT_ERROR_SOCKET_BADAF; } if (rc == 0) { #if defined(IPv6_FUNCTION_SUPPORT) if (family != HYADDR_FAMILY_AFINET4) { family = HYADDR_FAMILY_AFINET6; sock = socket (AF_INET6, ((socktype == HYSOCK_STREAM) ? SOCK_STREAM : SOCK_DGRAM), 0); if (sock < 0) { rc = errno; HYSOCKDEBUG ("\n", rc); return portLibrary->error_set_last_error (portLibrary, rc, findError (rc)); } } else { #endif sock = socket (AF_INET, ((socktype == HYSOCK_STREAM) ? SOCK_STREAM : SOCK_DGRAM), 0); if (sock < 0) { rc = errno; return hyerror_set_last_error(rc, findError(rc)); } #if defined(IPv6_FUNCTION_SUPPORT) } #endif } if (rc == 0) { /*Tag this descriptor as being non-inheritable */ I_32 fdflags = fcntl (sock, F_GETFD, 0); fcntl (sock, F_SETFD, fdflags | FD_CLOEXEC); /* set up the socket structure */ *handle = XMLVM_ATOMIC_MALLOC(sizeof (struct hysocket_struct)); #if (defined(VALIDATE_ALLOCATIONS)) if (*handle == NULL) { close (sock); *handle = INVALID_SOCKET; return HYPORT_ERROR_SOCKET_NOBUFFERS; } #endif (*handle)->sock = sock; (*handle)->family = family; } return rc; } I_32 hysock_sockaddr_init6 (hysockaddr_t handle, U_8 * addr, I_32 addrlength, I_16 family, U_16 nPort, U_32 flowinfo, U_32 scope_id, hysocket_t sock) { OSSOCKADDR *sockaddr; #if defined(IPv6_FUNCTION_SUPPORT) OSSOCKADDR_IN6 *sockaddr_6; #endif XMLVM_BZERO(handle, sizeof (struct hysockaddr_struct)); if (family == HYADDR_FAMILY_AFINET4) { #if defined(IPv6_FUNCTION_SUPPORT) if (hysock_socketIsValid (portLibrary, sock) && (((struct hysocket_struct *) sock)->family == HYADDR_FAMILY_AFINET6)) { /* to talk IPv4 on an IPv6 socket we need to map the IPv4 address to an IPv6 format. If mapAddress is true then we do this */ sockaddr_6 = (OSSOCKADDR_IN6 *) & handle->addr; memset (sockaddr_6->sin6_addr.s6_addr, 0, 16); memcpy (&(sockaddr_6->sin6_addr.s6_addr[12]), addr, addrlength); /* do a check if it is the any address. we know the top 4 bytes of sockaddr_6->sin6_addr.s6_addr are 0's as we just cleared the, so we use them to do the check */ if (memcmp (sockaddr_6->sin6_addr.s6_addr, addr, addrlength) != 0) { /* if it is the any address then use the IPv6 any address */ sockaddr_6->sin6_addr.s6_addr[10] = 0xFF; sockaddr_6->sin6_addr.s6_addr[11] = 0xFF; } sockaddr_6->sin6_port = nPort; sockaddr_6->sin6_family = OS_AF_INET6; sockaddr_6->sin6_scope_id = scope_id; sockaddr_6->sin6_flowinfo = htonl (flowinfo); #if defined(FREEBSD) sockaddr_6->sin6_len = sizeof(OSSOCKADDR_IN6); #endif } else { #endif /* just initialize the IPv4 address as is as it will be used with an IPv4 Socket */ sockaddr = (OSSOCKADDR *) & handle->addr; XMLVM_MEMCPY (&sockaddr->sin_addr.s_addr, addr, addrlength); sockaddr->sin_port = nPort; sockaddr->sin_family = OS_AF_INET4; #if defined(FREEBSD) sockaddr->sin_len = sizeof(OSSOCKADDR); #endif #if defined(IPv6_FUNCTION_SUPPORT) } #endif } #if defined(IPv6_FUNCTION_SUPPORT) else if (family == HYADDR_FAMILY_AFINET6) { sockaddr_6 = (OSSOCKADDR_IN6 *) & handle->addr; memcpy (&sockaddr_6->sin6_addr.s6_addr, addr, addrlength); sockaddr_6->sin6_port = nPort; sockaddr_6->sin6_family = OS_AF_INET6; sockaddr_6->sin6_scope_id = scope_id; sockaddr_6->sin6_flowinfo = htonl (flowinfo); #if defined(SIN6_LEN) sockaddr_6->sin6_len = sizeof(OSSOCKADDR_IN6); #endif } #endif else { sockaddr = (OSSOCKADDR *) & handle->addr; memcpy (&sockaddr->sin_addr.s_addr, addr, HYSOCK_INADDR_LEN); sockaddr->sin_port = nPort; sockaddr->sin_family = map_addr_family_Hy_to_OS (family); #if defined(FREEBSD) sockaddr->sin_len = sizeof(OSSOCKADDR); #endif } return 0; } I_32 hysock_getnameinfo (hysockaddr_t in_addr, I_32 sockaddr_size, char *name, I_32 name_length, int flags) { /* If we have the IPv6 functions available we will call them, otherwise we'll call the IPv4 function */ #if defined(IPv6_FUNCTION_SUPPORT) int rc = 0; rc = getnameinfo ((OSADDR *) & in_addr->addr, sizeof (in_addr->addr), name, name_length, NULL, 0, flags); if (rc != 0) { rc = errno; HYSOCKDEBUG ("\n", rc); return portLibrary->error_set_last_error (portLibrary, rc, findError (rc)); } return 0; #else /* IPv6_FUNCTION_SUPPORT */ #if !HOSTENT_DATA_R OSHOSTENT *result; #endif #if GLIBC_R||OTHER_R BOOLEAN allocBuffer = FALSE; #endif int herr = 0; int i = 0; int rc = 0; int length; OSSOCKADDR *addr; #if HOSTENT_DATA_R||GLIBC_R||OTHER_R||NO_R PortlibPTBuffers_t ptBuffers; #endif /* HOSTENT_DATA_R || GLIBC_R || OTHER_R || NO_R */ addr = (OSSOCKADDR *) & in_addr->addr; if (addr->sin_family == OS_AF_INET4) { length = 4; } else { length = 16; } #if HOSTENT_DATA_R||GLIBC_R||OTHER_R||NO_R ptBuffers = hyport_tls_get (); if (NULL == ptBuffers) { return HYPORT_ERROR_SOCKET_SYSTEMFULL; } #define hostentBuffer (&ptBuffers->hostent) #endif /* HOSTENT_DATA_R || GLIBC_R || OTHER_R || NO_R */ /* one of several threadsafe gethostbyaddr calls must be made depending on the current platform */ /* if there is a transient error (HYPORT_ERROR_SOCKET_UNIX_TRYAGAIN), try making the call again */ for (i = 0; i < MAX_RETRIES; i++) { #if HOSTENT_DATA_R #define dataBuffer (ptBuffers->hostent_data) if (!dataBuffer) { dataBuffer = portLibrary->mem_allocate_memory (portLibrary, sizeof (OSHOSTENT_DATA)); if (!dataBuffer) { return HYPORT_ERROR_SOCKET_SYSTEMFULL; } } herr = gethostbyaddr_r (&addr->sin_addr, length, addr->sin_family, hostentBuffer, dataBuffer); #undef dataBuffer #elif ORIGINAL_R || NO_R #if NO_R MUTEX_ENTER (hostentLock); #endif result = gethostbyaddr (&addr->sin_addr, length, addr->sin_family); #if NO_R if (result) { herr = copy_hostent (result, &ptBuffers); } MUTEX_EXIT (hostentLock); if (herr != 0) { return herr; } #endif herr = h_errno; #elif GLIBC_R || OTHER_R #define buffer (ptBuffers->gethostBuffer) #define bufferSize (ptBuffers->gethostBufferSize) if (!buffer) { bufferSize = GET_HOST_BUFFER_SIZE; } while (TRUE) { if (allocBuffer == TRUE || !buffer) { /* The buffer is allocated bufferSize + EXTRA_SPACE, while gethostby*_r is only aware of bufferSize * because there seems to be a bug on Linux 386 where gethostbyname_r writes past the end of the * buffer. This bug has not been observed on other platforms, but EXTRA_SPACE is added anyway as a precaution.*/ buffer = XMLVM_ATOMIC_MALLOC(bufferSize + EXTRA_SPACE); if (!buffer) { return HYPORT_ERROR_SOCKET_SYSTEMFULL; } allocBuffer = FALSE; } #if GLIBC_R gethostbyaddr_r ((char *) &addr->sin_addr, length, addr->sin_family, hostentBuffer, buffer, bufferSize, &result, &herr); #elif OTHER_R result = gethostbyaddr_r ((char *) &addr->sin_addr, length, addr->sin_family, hostentBuffer, buffer, bufferSize, &herr); #endif /* GLIBC_R */ /* allocate more space if the buffer is too small */ if (errno == ERANGE || herr == ERANGE) { XMLVM_FREE(buffer); bufferSize *= 2; allocBuffer = TRUE; } else { break; } } #undef buffer #undef bufferSize #endif if (herr != HYPORT_ERROR_SOCKET_UNIX_TRYAGAIN) { break; } } #if HOSTENT_DATA_R if (herr != 0) { herr = h_errno; HYSOCKDEBUGH ("\n", herr); return portLibrary->error_set_last_error (portLibrary, herr, findHostError (herr)); } #else if (result == NULL) { HYSOCKDEBUGH ("\n", herr); return hyerror_set_last_error(herr, findError(herr)); } #endif else { memset (name, 0, sizeof (char) * name_length); #if HOSTENT_DATA_R||NO_R strcpy (name, hostentBuffer->h_name); #else strcpy (name, result->h_name); #endif } #if HOSTENT_DATA_R||GLIBC_R||OTHER_R #undef hostentBuffer #endif /*HOSTENT_DATA_R || GLIBC_R || OTHER_R */ return 0; #endif /* IPv6_FUNCTION_SUPPORT */ } I_32 hysock_bind (hysocket_t sock, hysockaddr_t addr) { I_32 rc = 0; I_32 length = getAddrLength(addr); if (bind (SOCKET_CAST (sock), (struct sockaddr *) &addr->addr, length) < 0) { rc = errno; rc = hyerror_set_last_error(rc, HYPORT_ERROR_SOCKET_ADDRNOTAVAIL); } return rc; } I_32 hysock_getsockname (hysocket_t handle, hysockaddr_t addrHandle) { socklen_t addrlen = sizeof (addrHandle->addr); #ifndef XMLVM_ASM_JS if (getsockname (SOCKET_CAST (handle), (struct sockaddr *) &addrHandle->addr, &addrlen) != 0) { I_32 err = errno; return hyerror_set_last_error(err, findError(err)); } #endif return 0; } I_32 hysock_connect (hysocket_t sock, hysockaddr_t addr) { I_32 rc = 0; I_32 length = getAddrLength(addr); if (connect (SOCKET_CAST (sock), (struct sockaddr *) &addr->addr, length) < 0) { rc = errno; rc = hyerror_set_last_error(rc, HYPORT_ERROR_SOCKET_OPFAILED); } return rc; } I_32 hysock_write (hysocket_t sock, U_8 * buf, I_32 nbyte, I_32 flags) { I_32 bytesSent = 0; bytesSent = send (SOCKET_CAST (sock), buf, nbyte, flags); if (-1 == bytesSent) { I_32 err = errno; return hyerror_set_last_error(err, findError(err)); } else { return bytesSent; } } I_32 hysock_read (hysocket_t sock, U_8 * buf, I_32 nbyte, I_32 flags) { I_32 bytesRec = 0; bytesRec = recv (SOCKET_CAST (sock), buf, nbyte, flags); if (-1 == bytesRec) { I_32 err = errno; return hyerror_set_last_error(err, findError(err)); } else { return bytesRec; } } I_32 hysock_close (hysocket_t * sock) { I_32 rc = 0; if (*sock == INVALID_SOCKET) { return hyerror_set_last_error(HYPORT_ERROR_SOCKET_UNIX_EBADF, HYPORT_ERROR_SOCKET_BADSOCKET); } if (close (SOCKET_CAST (*sock)) < 0) { rc = errno; rc = hyerror_set_last_error(rc, HYPORT_ERROR_SOCKET_BADSOCKET); } *sock = INVALID_SOCKET; return rc; } I_32 hysock_listen (hysocket_t sock, I_32 backlog) { I_32 rc = 0; if (listen (SOCKET_CAST (sock), backlog) < 0) { rc = errno; HYSOCKDEBUG ("\n", rc); rc = hyerror_set_last_error(rc, HYPORT_ERROR_SOCKET_OPFAILED); } return rc; } I_32 hysock_accept (hysocket_t serverSock, hysockaddr_t addrHandle, hysocket_t * sockHandle) { #if defined(LINUX) #define ACCEPTCAST (socklen_t *) #else #define ACCEPTCAST #endif I_32 rc = 0; int sc; socklen_t fromlen = sizeof (addrHandle->addr); *sockHandle = INVALID_SOCKET; sc = accept (SOCKET_CAST (serverSock), (struct sockaddr *) &addrHandle->addr, ACCEPTCAST & fromlen); if (sc < 0) { rc = hyerror_set_last_error(rc, HYPORT_ERROR_SOCKET_ADDRNOTAVAIL); } if (rc == 0) { //*sockHandle = portLibrary->mem_allocate_memory (portLibrary, sizeof (struct hysocket_struct)); *sockHandle = XMLVM_ATOMIC_MALLOC(sizeof (struct hysocket_struct)); #if (defined(VALIDATE_ALLOCATIONS)) if (*sockHandle == NULL) { close (sc); *sockHandle = INVALID_SOCKET; return HYPORT_ERROR_SOCKET_NOBUFFERS; } #endif SOCKET_CAST (*sockHandle) = sc; (*sockHandle)->family = serverSock->family; } return rc; } I_32 hysock_timeval_init (U_32 secTime, U_32 uSecTime, hytimeval_t timeP) { memset (timeP, 0, sizeof (struct hytimeval_struct)); timeP->time.tv_sec = secTime; timeP->time.tv_usec = uSecTime; return 0; } I_32 hysock_fdset_init (hysocket_t socketP) { PortlibPTBuffers_t ptBuffers; hyfdset_t fdset; ptBuffers = hyport_tls_get (); if (NULL == ptBuffers) { return HYPORT_ERROR_SOCKET_SYSTEMFULL; } if (NULL == ptBuffers->fdset) { ptBuffers->fdset = XMLVM_ATOMIC_MALLOC(sizeof (struct hyfdset_struct)); if (NULL == ptBuffers->fdset) { return HYPORT_ERROR_SOCKET_SYSTEMFULL; } } fdset = ptBuffers->fdset; memset (fdset, 0, sizeof (struct hyfdset_struct)); FD_ZERO (&fdset->handle); FD_SET (SOCKET_CAST (socketP), &fdset->handle); return 0; } I_32 hysock_fdset_size (hysocket_t handle) { return SOCKET_CAST (handle) + 1; } I_32 hysock_select (I_32 nfds, hyfdset_t readfds, hyfdset_t writefds, hyfdset_t exceptfds, hytimeval_t timeout) { I_32 rc = 0; I_32 result = 0; if (nfds >= FD_SETSIZE) { rc = hyerror_set_last_error(rc, HYPORT_ERROR_SOCKET_UNIX_EINVAL); return -1; } result = select (nfds, readfds == NULL ? NULL : &readfds->handle, writefds == NULL ? NULL : &writefds->handle, exceptfds == NULL ? NULL : &exceptfds->handle, timeout == NULL ? NULL : &timeout->time); if (result == -1) { HYSOCKDEBUG ("