Fix connection event error handling.

Commit 86a99c6b99 changed the way we
handle connection events to protect against spurious event loop
callbacks. Unfortunately, it turns out that calling connect() twice on
the same socket results in different behaviors depending on the platform
(even though it seems well defined in POSIX). On Windows this resulted
in the connection handling code being unable to react to connection
errors (such as connection refused), always hitting the timeout; on
Linux this resulted in spurious error messages about connect() returning
success.

In POSIX and on Linux, using connect() on a socket where the previous
attempt failed will attempt to connect again, resulting in unnecessary
network activity. Using getsockopt(SO_ERROR) before connect() solves
that, but introduces a race condition if a connection failure happens
between the two calls.

For this reason, this commit switches from connect() to a zero-sized
send() call, which is more consistent (though not completely, see the
truth table in the comments) and simpler to use for that purpose. Note
that Windows explictly support empty send() calls; POSIX says nothing
on the subject, but testing shows it works at least on Linux.

(Surprisingly enough, Windows seems more POSIX-compliant than Linux on
this one!)
This commit is contained in:
Etienne Dechamps 2014-06-28 11:13:29 +01:00
parent 86a99c6b99
commit cc284e7c5d
2 changed files with 28 additions and 22 deletions

View file

@ -401,30 +401,38 @@ static void handle_meta_io(void *data, int flags) {
connection_t *c = data; connection_t *c = data;
if(c->status.connecting) { if(c->status.connecting) {
/* The event loop does not protect against spurious events. Verify that we are actually connected. */ /*
if (connect(c->socket, &c->address.sa, sizeof(c->address)) == 0) The event loop does not protect against spurious events. Verify that we are actually connected
logger(DEBUG_CONNECTIONS, LOG_DEBUG, "Error while connecting to %s (%s): redundant connect() unexpectedly succeeded", c->name, c->hostname); by issuing an empty send() call.
else if (!sockisconn(sockerrno)) {
if (!sockalready(sockerrno)) { Note that the behavior of send() on potentially unconnected sockets differ between platforms:
logger(DEBUG_CONNECTIONS, LOG_DEBUG, "Error while checking connection status for %s (%s): %s", c->name, c->hostname, sockstrerror(sockerrno)); +------------+-----------+-------------+-----------+
| Event | POSIX | Linux | Windows |
+------------+-----------+-------------+-----------+
| Spurious | ENOTCONN | EWOULDBLOCK | ENOTCONN |
| Failed | ENOTCONN | (cause) | ENOTCONN |
| Successful | (success) | (success) | (success) |
+------------+-----------+-------------+-----------+
*/
if (send(c->socket, NULL, 0, 0) != 0) {
if (sockwouldblock(sockerrno))
return;
int socket_error;
if (!socknotconn(sockerrno))
socket_error = sockerrno;
else {
int len = sizeof socket_error;
getsockopt(c->socket, SOL_SOCKET, SO_ERROR, (void *)&socket_error, &len);
}
if (socket_error) {
logger(DEBUG_CONNECTIONS, LOG_DEBUG, "Error while connecting to %s (%s): %s", c->name, c->hostname, sockstrerror(socket_error));
terminate_connection(c, false); terminate_connection(c, false);
} }
return; return;
} }
c->status.connecting = false; c->status.connecting = false;
int result;
socklen_t len = sizeof result;
getsockopt(c->socket, SOL_SOCKET, SO_ERROR, (void *)&result, &len);
if(!result)
finish_connecting(c); finish_connecting(c);
else {
logger(DEBUG_CONNECTIONS, LOG_DEBUG, "Error while connecting to %s (%s): %s", c->name, c->hostname, sockstrerror(result));
terminate_connection(c, false);
return;
}
} }
if(flags & IO_WRITE) if(flags & IO_WRITE)

View file

@ -37,8 +37,7 @@ extern const char *winerror(int);
#define sockmsgsize(x) ((x) == WSAEMSGSIZE) #define sockmsgsize(x) ((x) == WSAEMSGSIZE)
#define sockinprogress(x) ((x) == WSAEINPROGRESS || (x) == WSAEWOULDBLOCK) #define sockinprogress(x) ((x) == WSAEINPROGRESS || (x) == WSAEWOULDBLOCK)
#define sockinuse(x) ((x) == WSAEADDRINUSE) #define sockinuse(x) ((x) == WSAEADDRINUSE)
#define sockalready(x) ((x) == WSAEALREADY || (x) == WSAEINVAL || (x) == WSAEWOULDBLOCK) /* See MSDN for connect() */ #define socknotconn(x) ((x) == WSAENOTCONN)
#define sockisconn(x) ((x) == WSAEISCONN)
#else #else
#define sockerrno errno #define sockerrno errno
#define sockstrerror(x) strerror(x) #define sockstrerror(x) strerror(x)
@ -46,8 +45,7 @@ extern const char *winerror(int);
#define sockmsgsize(x) ((x) == EMSGSIZE) #define sockmsgsize(x) ((x) == EMSGSIZE)
#define sockinprogress(x) ((x) == EINPROGRESS) #define sockinprogress(x) ((x) == EINPROGRESS)
#define sockinuse(x) ((x) == EADDRINUSE) #define sockinuse(x) ((x) == EADDRINUSE)
#define sockalready(x) ((x) == EALREADY) #define socknotconn(x) ((x) == ENOTCONN)
#define sockisconn(x) ((x) == EISCONN)
#endif #endif
extern unsigned int bitfield_to_int(const void *bitfield, size_t size); extern unsigned int bitfield_to_int(const void *bitfield, size_t size);