summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorAdam Tkac <atkac@redhat.com>2008-11-14 15:56:15 +0000
committerAdam Tkac <atkac@redhat.com>2008-11-14 15:56:15 +0000
commit9cb6a4259975fbd21112a4dffdaef47d23ff9f26 (patch)
tree790e54c5fa61a677483c52785b357958dbc46356
parent20e0d711fdc2cdb2df7f3d4447c27832e982a24b (diff)
downloadtigervnc-9cb6a4259975fbd21112a4dffdaef47d23ff9f26.tar.gz
tigervnc-9cb6a4259975fbd21112a4dffdaef47d23ff9f26.zip
[Development] Basic IPv6 support to viewer.
git-svn-id: svn://svn.code.sf.net/p/tigervnc/code/trunk@3169 3789f03b-4d11-0410-bbf8-ca57d06f2519
-rw-r--r--common/configure.ac2
-rw-r--r--common/network/TcpSocket.cxx126
2 files changed, 94 insertions, 34 deletions
diff --git a/common/configure.ac b/common/configure.ac
index fabde362..a9fe241a 100644
--- a/common/configure.ac
+++ b/common/configure.ac
@@ -78,7 +78,7 @@ AC_ARG_WITH([included-jpeg],
AM_CONDITIONAL([INCLUDED_JPEG], [ test "x$INCLUDED_JPEG" = xyes ])
AC_CONFIG_SUBDIRS([jpeg])
-AC_CHECK_FUNCS_ONCE([vsnprintf strcasecmp strncasecmp])
+AC_CHECK_FUNCS_ONCE([vsnprintf strcasecmp strncasecmp getaddrinfo])
AC_CHECK_TYPES([socklen_t],
[AC_DEFINE([VNC_SOCKLEN_T], [socklen_t], [Use correct size])],
diff --git a/common/network/TcpSocket.cxx b/common/network/TcpSocket.cxx
index 777627b4..12c7266b 100644
--- a/common/network/TcpSocket.cxx
+++ b/common/network/TcpSocket.cxx
@@ -56,6 +56,16 @@
using namespace network;
using namespace rdr;
+typedef struct vnc_sockaddr {
+ union {
+ sockaddr sa;
+ sockaddr_in sin;
+#ifdef HAVE_GETADDRINFO
+ sockaddr_in6 sin6;
+#endif
+ } u;
+} vnc_sockaddr_t;
+
static rfb::LogWriter vlog("TcpSocket");
/* Tunnelling support. */
@@ -110,52 +120,102 @@ TcpSocket::TcpSocket(int sock, bool close)
TcpSocket::TcpSocket(const char *host, int port)
: closeFd(true)
{
- int sock;
+ int sock, err, result, family;
+ vnc_sockaddr_t sa;
+ VNC_SOCKLEN_T salen;
+#ifdef HAVE_GETADDRINFO
+ struct addrinfo *ai, *current, hints;
+#endif
// - Create a socket
initSockets();
- if ((sock = socket(AF_INET, SOCK_STREAM, 0)) < 0)
- throw SocketException("unable to create socket", errorNumber);
-
-#ifndef WIN32
- // - By default, close the socket on exec()
- fcntl(sock, F_SETFD, FD_CLOEXEC);
-#endif
- // - Connect it to something
+#ifdef HAVE_GETADDRINFO
+ memset(&hints, 0, sizeof(struct addrinfo));
+ hints.ai_family = AF_UNSPEC;
+ hints.ai_socktype = SOCK_STREAM;
+ hints.ai_canonname = NULL;
+ hints.ai_addr = NULL;
+ hints.ai_next = NULL;
+
+ if ((result = getaddrinfo(host, NULL, &hints, &ai)) != 0) {
+ throw Exception("unable to resolve host by name: %s",
+ gai_strerror(result));
+ }
- // Try processing the host as an IP address
- struct sockaddr_in addr;
- memset(&addr, 0, sizeof(addr));
- addr.sin_family = AF_INET;
- addr.sin_addr.s_addr = inet_addr((char *)host);
- addr.sin_port = htons(port);
- if ((int)addr.sin_addr.s_addr == -1) {
- // Host was not an IP address - try resolving as DNS name
- struct hostent *hostinfo;
- hostinfo = gethostbyname((char *)host);
- if (hostinfo && hostinfo->h_addr) {
- addr.sin_addr.s_addr = ((struct in_addr *)hostinfo->h_addr)->s_addr;
- } else {
- int e = errorNumber;
- closesocket(sock);
- throw SocketException("unable to resolve host by name", e);
+ for (current = ai; current != NULL; current = current->ai_next) {
+ family = current->ai_family;
+ if (family != AF_INET && family != AF_INET6)
+ continue;
+
+ salen = current->ai_addrlen;
+ memcpy(&sa, current->ai_addr, salen);
+
+ if (family == AF_INET)
+ sa.u.sin.sin_port = htons(port);
+ else
+ sa.u.sin6.sin6_port = htons(port);
+
+#else /* HAVE_GETADDRINFO */
+ family = AF_INET;
+ salen = sizeof(struct sockaddr_in);
+
+ /* Try processing the host as an IP address */
+ memset(&sa, 0, sizeof(sa));
+ sa.u.sin.sin_family = AF_INET;
+ sa.u.sin.sin_addr.s_addr = inet_addr((char *)host);
+ sa.u.sin.sin_port = htons(port);
+ if ((int)sa.u.sin.sin_addr.s_addr == -1) {
+ /* Host was not an IP address - try resolving as DNS name */
+ struct hostent *hostinfo;
+ hostinfo = gethostbyname((char *)host);
+ if (hostinfo && hostinfo->h_addr) {
+ sa.u.sin.sin_addr.s_addr = ((struct in_addr *)hostinfo->h_addr)->s_addr;
+ } else {
+ err = errorNumber;
+ throw SocketException("unable to resolve host by name", err);
+ }
+ }
+#endif /* HAVE_GETADDRINFO */
+
+ sock = socket (family, SOCK_STREAM, 0);
+ if (sock == -1) {
+ err = errorNumber;
+#ifdef HAVE_GETADDRINFO
+ freeaddrinfo(ai);
+#endif /* HAVE_GETADDRINFO */
+ throw SocketException("unable to create socket", err);
}
- }
- // Attempt to connect to the remote host
- for (;;) {
- if (connect(sock, (struct sockaddr *)&addr, sizeof(addr)) != 0) {
- int e = errorNumber;
+ /* Attempt to connect to the remote host */
+ while ((result = connect(sock, &sa.u.sa, sizeof(sa))) == -1) {
+ err = errorNumber;
#ifndef WIN32
- if (e == EINTR)
+ if (err == EINTR)
continue;
#endif
closesocket(sock);
- throw SocketException("unable to connect to host", e);
- } else break;
+ break;
+ }
+
+#ifdef HAVE_GETADDRINFO
+ if (result == 0)
+ break;
+ else
+ continue;
}
+ freeaddrinfo(ai);
+#endif /* HAVE_GETADDRINFO */
+
+ if (result == -1)
+ throw SocketException("unable connect to socket", err);
+
+#ifndef WIN32
+ // - By default, close the socket on exec()
+ fcntl(sock, F_SETFD, FD_CLOEXEC);
+#endif
+
// Disable Nagle's algorithm, to reduce latency
enableNagles(sock, false);