📄 natplaindatagramsocketimpl.cc
字号:
/* Copyright (C) 1999, 2000, 2002 Free Software Foundation This file is part of libgcj.This software is copyrighted work licensed under the terms of theLibgcj License. Please consult the file "LIBGCJ_LICENSE" fordetails. */#include <config.h>#include <platform.h>#ifdef WIN32#include <errno.h>#include <string.h>#else /* WIN32 */#ifdef HAVE_NETINET_IN_H#include <netinet/in.h>#endif#ifdef HAVE_ARPA_INET_H#include <arpa/inet.h>#endif#include <errno.h>#include <string.h>#endif /* WIN32 */#if HAVE_BSTRING_H// Needed for bzero, implicitly used by FD_ZERO on IRIX 5.2 #include <bstring.h>#endif#include <gcj/cni.h>#include <java/io/IOException.h>#include <java/io/InterruptedIOException.h>#include <java/net/BindException.h>#include <java/net/SocketException.h>#include <java/net/PlainDatagramSocketImpl.h>#include <java/net/InetAddress.h>#include <java/net/NetworkInterface.h>#include <java/net/DatagramPacket.h>#include <java/net/PortUnreachableException.h>#include <java/lang/InternalError.h>#include <java/lang/Object.h>#include <java/lang/Boolean.h>#include <java/lang/Integer.h>#ifdef DISABLE_JAVA_NETvoidjava::net::PlainDatagramSocketImpl::create (){ throw new SocketException ( JvNewStringLatin1 ("DatagramSocketImpl.create: unimplemented"));}voidjava::net::PlainDatagramSocketImpl::bind (jint, java::net::InetAddress *){ throw new BindException ( JvNewStringLatin1 ("DatagramSocketImpl.bind: unimplemented"));}voidjava::net::PlainDatagramSocketImpl::connect (java::net::InetAddress *, jint){ throw new SocketException ( JvNewStringLatin1 ("DatagramSocketImpl.connect: unimplemented"));}voidjava::net::PlainDatagramSocketImpl::disconnect (){ throw new SocketException ( JvNewStringLatin1 ("DatagramSocketImpl.disconnect: unimplemented"));}jintjava::net::PlainDatagramSocketImpl::peek (java::net::InetAddress *){ throw new java::io::IOException ( JvNewStringLatin1 ("DatagramSocketImpl.peek: unimplemented"));}jintjava::net::PlainDatagramSocketImpl::peekData(java::net::DatagramPacket *){ throw new java::io::IOException ( JvNewStringLatin1 ("DatagramSocketImpl.peekData: unimplemented"));}voidjava::net::PlainDatagramSocketImpl::close (){ throw new java::io::IOException ( JvNewStringLatin1 ("DatagramSocketImpl.close: unimplemented"));}voidjava::net::PlainDatagramSocketImpl::send (java::net::DatagramPacket *){ throw new java::io::IOException ( JvNewStringLatin1 ("DatagramSocketImpl.send: unimplemented"));}voidjava::net::PlainDatagramSocketImpl::receive (java::net::DatagramPacket *){ throw new java::io::IOException ( JvNewStringLatin1 ("DatagramSocketImpl.receive: unimplemented"));}voidjava::net::PlainDatagramSocketImpl::setTimeToLive (jint){ throw new java::io::IOException ( JvNewStringLatin1 ("DatagramSocketImpl.setTimeToLive: unimplemented"));}jintjava::net::PlainDatagramSocketImpl::getTimeToLive (){ throw new java::io::IOException ( JvNewStringLatin1 ("DatagramSocketImpl.getTimeToLive: unimplemented"));}voidjava::net::PlainDatagramSocketImpl::mcastGrp (java::net::InetAddress *, java::net::NetworkInterface *, jboolean){ throw new java::io::IOException ( JvNewStringLatin1 ("DatagramSocketImpl.mcastGrp: unimplemented"));}voidjava::net::PlainDatagramSocketImpl::setOption (jint, java::lang::Object *){ throw new SocketException ( JvNewStringLatin1 ("DatagramSocketImpl.setOption: unimplemented"));}java::lang::Object *java::net::PlainDatagramSocketImpl::getOption (jint){ throw new SocketException ( JvNewStringLatin1 ("DatagramSocketImpl.getOption: unimplemented"));}#else /* DISABLE_JAVA_NET */union SockAddr{ struct sockaddr_in address;#ifdef HAVE_INET6 struct sockaddr_in6 address6;#endif};union McastReq{#if HAVE_STRUCT_IP_MREQ struct ip_mreq mreq;#endif#if HAVE_STRUCT_IPV6_MREQ struct ipv6_mreq mreq6;#endif};union InAddr{ struct in_addr addr;#ifdef HAVE_INET6 struct in6_addr addr6;#endif};// FIXME: routines here and/or in natPlainSocketImpl.cc could throw// NoRouteToHostException; also consider UnknownHostException, ConnectException.voidjava::net::PlainDatagramSocketImpl::create (){ int sock = _Jv_socket (AF_INET, SOCK_DGRAM, 0); if (sock < 0) { char* strerr = strerror (errno); throw new java::net::SocketException (JvNewStringUTF (strerr)); } _Jv_platform_close_on_exec (sock); // We use fnum in place of fd here. From leaving fd null we avoid // the double close problem in FileDescriptor.finalize. fnum = sock;}voidjava::net::PlainDatagramSocketImpl::bind (jint lport, java::net::InetAddress *host){ union SockAddr u; struct sockaddr *ptr = (struct sockaddr *) &u.address; // FIXME: Use getaddrinfo() to get actual protocol instead of assuming ipv4. jbyteArray haddress = host->addr; jbyte *bytes = elements (haddress); int len = haddress->length; if (len == 4) { u.address.sin_family = AF_INET; if (host != NULL) memcpy (&u.address.sin_addr, bytes, len); else u.address.sin_addr.s_addr = htonl (INADDR_ANY); len = sizeof (struct sockaddr_in); u.address.sin_port = htons (lport); }#ifdef HAVE_INET6 else if (len == 16) { u.address6.sin6_family = AF_INET6; memcpy (&u.address6.sin6_addr, bytes, len); len = sizeof (struct sockaddr_in6); u.address6.sin6_port = htons (lport); }#endif else throw new java::net::SocketException (JvNewStringUTF ("invalid length")); if (_Jv_bind (fnum, ptr, len) == 0) { socklen_t addrlen = sizeof(u); if (lport != 0) localPort = lport; else if (::getsockname (fnum, (sockaddr*) &u, &addrlen) == 0) localPort = ntohs (u.address.sin_port); else goto error; /* Allow broadcast by default. */ int broadcast = 1; if (::setsockopt (fnum, SOL_SOCKET, SO_BROADCAST, (char *) &broadcast, sizeof (broadcast)) != 0) goto error; return; } error: char* strerr = strerror (errno); throw new java::net::BindException (JvNewStringUTF (strerr));}voidjava::net::PlainDatagramSocketImpl::connect (java::net::InetAddress *, jint){ throw new ::java::lang::InternalError (JvNewStringLatin1 ( "PlainDatagramSocketImpl::connect: not implemented yet"));}voidjava::net::PlainDatagramSocketImpl::disconnect (){ throw new ::java::lang::InternalError (JvNewStringLatin1 ( "PlainDatagramSocketImpl::disconnect: not implemented yet"));}jintjava::net::PlainDatagramSocketImpl::peek (java::net::InetAddress *i){ // FIXME: Deal with Multicast and if the socket is connected. union SockAddr u; socklen_t addrlen = sizeof(u); ssize_t retlen = ::recvfrom (fnum, (char *) NULL, 0, MSG_PEEK, (sockaddr*) &u, &addrlen); if (retlen < 0) goto error; // FIXME: Deal with Multicast addressing and if the socket is connected. jbyteArray raddr; jint rport; if (u.address.sin_family == AF_INET) { raddr = JvNewByteArray (4); memcpy (elements (raddr), &u.address.sin_addr, 4); rport = ntohs (u.address.sin_port); }#ifdef HAVE_INET6 else if (u.address.sin_family == AF_INET6) { raddr = JvNewByteArray (16); memcpy (elements (raddr), &u.address6.sin6_addr, 16); rport = ntohs (u.address6.sin6_port); }#endif else throw new java::net::SocketException (JvNewStringUTF ("invalid family")); i->addr = raddr; return rport; error: char* strerr = strerror (errno); if (errno == ECONNREFUSED) throw new PortUnreachableException (JvNewStringUTF (strerr)); throw new java::io::IOException (JvNewStringUTF (strerr));}jintjava::net::PlainDatagramSocketImpl::peekData(java::net::DatagramPacket *p){ // FIXME: Deal with Multicast and if the socket is connected. union SockAddr u; socklen_t addrlen = sizeof(u); jbyte *dbytes = elements (p->getData()); ssize_t retlen = 0;// FIXME: implement timeout support for Win32#ifndef WIN32 // Do timeouts via select since SO_RCVTIMEO is not always available. if (timeout > 0 && fnum >= 0 && fnum < FD_SETSIZE) { fd_set rset; struct timeval tv; FD_ZERO(&rset); FD_SET(fnum, &rset); tv.tv_sec = timeout / 1000; tv.tv_usec = (timeout % 1000) * 1000; int retval; if ((retval = _Jv_select (fnum + 1, &rset, NULL, NULL, &tv)) < 0) goto error; else if (retval == 0) throw new java::io::InterruptedIOException (); }#endif /* WIN32 */ retlen = ::recvfrom (fnum, (char *) dbytes, p->getLength(), MSG_PEEK, (sockaddr*) &u, &addrlen); if (retlen < 0) goto error; // FIXME: Deal with Multicast addressing and if the socket is connected. jbyteArray raddr; jint rport; if (u.address.sin_family == AF_INET) { raddr = JvNewByteArray (4); memcpy (elements (raddr), &u.address.sin_addr, 4); rport = ntohs (u.address.sin_port); }#ifdef HAVE_INET6 else if (u.address.sin_family == AF_INET6) { raddr = JvNewByteArray (16); memcpy (elements (raddr), &u.address6.sin6_addr, 16); rport = ntohs (u.address6.sin6_port); }#endif else throw new java::net::SocketException (JvNewStringUTF ("invalid family")); p->setAddress (new InetAddress (raddr, NULL)); p->setPort (rport); p->setLength ((jint) retlen); return rport; error: char* strerr = strerror (errno); if (errno == ECONNREFUSED) throw new PortUnreachableException (JvNewStringUTF (strerr)); throw new java::io::IOException (JvNewStringUTF (strerr));}// Close(shutdown) the socket.voidjava::net::PlainDatagramSocketImpl::close (){ // Avoid races from asynchronous finalization. JvSynchronize sync (this); // The method isn't declared to throw anything, so we disregard // the return value. _Jv_close (fnum); fnum = -1; timeout = 0;}voidjava::net::PlainDatagramSocketImpl::send (java::net::DatagramPacket *p){ // FIXME: Deal with Multicast and if the socket is connected. jint rport = p->getPort(); union SockAddr u; jbyteArray haddress = p->getAddress()->addr; jbyte *bytes = elements (haddress); int len = haddress->length; struct sockaddr *ptr = (struct sockaddr *) &u.address; jbyte *dbytes = elements (p->getData()); if (len == 4) { u.address.sin_family = AF_INET; memcpy (&u.address.sin_addr, bytes, len); len = sizeof (struct sockaddr_in); u.address.sin_port = htons (rport); }#ifdef HAVE_INET6 else if (len == 16) { u.address6.sin6_family = AF_INET6; memcpy (&u.address6.sin6_addr, bytes, len); len = sizeof (struct sockaddr_in6); u.address6.sin6_port = htons (rport); }#endif else throw new java::net::SocketException (JvNewStringUTF ("invalid length")); if (::sendto (fnum, (char *) dbytes, p->getLength(), 0, ptr, len) >= 0) return; char* strerr = strerror (errno); if (errno == ECONNREFUSED) throw new PortUnreachableException (JvNewStringUTF (strerr));
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -