From c86517fcb673515f2fdb73bd3c133af599148416 Mon Sep 17 00:00:00 2001 From: Daniel Walker Date: Thu, 23 Aug 2001 23:25:33 +0000 Subject: [PATCH] - Fix for nonblocking sockets using WSAEventSelect() (patch from Ove Kaaven). - Changed WSAEnumNetworkEvents() so it only returns events that the application is looking for. - Changed sock_poll_event() to interpret a POLLIN event with zero bytes waiting to be read as a POLLHUP. --- dlls/winsock/socket.c | 2 +- server/sock.c | 22 +++++++++++++++------- 2 files changed, 16 insertions(+), 8 deletions(-) diff --git a/dlls/winsock/socket.c b/dlls/winsock/socket.c index 70d04bcc95a..a11541e8d5d 100644 --- a/dlls/winsock/socket.c +++ b/dlls/winsock/socket.c @@ -2603,7 +2603,7 @@ int WINAPI WSAEnumNetworkEvents(SOCKET s, WSAEVENT hEvent, LPWSANETWORKEVENTS lp req->c_event = hEvent; if (!(ret = SERVER_CALL())) { - lpEvent->lNetworkEvents = req->pmask; + lpEvent->lNetworkEvents = req->pmask & req->mask; memcpy(lpEvent->iErrorCode, server_data_ptr(req), server_data_size(req) ); } } diff --git a/server/sock.c b/server/sock.c index b5c77a8e3f0..070146d4ca1 100644 --- a/server/sock.c +++ b/server/sock.c @@ -84,7 +84,7 @@ static void sock_reselect( struct sock *sock ) if (sock->obj.select == -1) { /* previously unconnected socket, is this reselect supposed to connect it? */ - if (!sock->state) return; + if (!(sock->state & ~WS_FD_NONBLOCKING)) return; /* ok, it is, attach it to the wineserver's main poll loop */ add_select_user( &sock->obj ); } @@ -161,12 +161,20 @@ static void sock_poll_event( struct object *obj, int event ) /* normal data flow */ if (event & POLLIN) { - /* incoming data */ - sock->pmask |= FD_READ; - sock->hmask |= FD_READ; - sock->errors[FD_READ_BIT] = 0; - if (debug_level) - fprintf(stderr, "socket %d is readable\n", sock->obj.fd ); + char dummy; + + /* Linux 2.4 doesn't report POLLHUP if only one side of the socket + * has been closed, so we need to check for it explicitly here */ + if (!recv( sock->obj.fd, &dummy, 1, MSG_PEEK )) event = POLLHUP; + else + { + /* incoming data */ + sock->pmask |= FD_READ; + sock->hmask |= FD_READ; + sock->errors[FD_READ_BIT] = 0; + if (debug_level) + fprintf(stderr, "socket %d is readable\n", sock->obj.fd ); + } } if (event & POLLOUT) {