Bug 324234 - Using g_io_add_watch_full() to wait for connect() to return

2008-08-20  Tor Lillqvist  <tml@novell.com>

	Bug 324234 - Using g_io_add_watch_full() to wait for connect() to
	return on a non-blocking socket returns prematurely

	Bug 548278 - Async GETs connections are always terminated
	unexpectedly on Windows

	* glib/giowin32.c: Add one more state variable to the
	GIOWin32Channel struct, ever_writable. Initialise it to FALSE, set
	to TRUE when the WSAEventSelect() indicates FD_WRITE, and never
	reset to FALSE.

	Don't do the WSASetEvent() in g_io_win32_prepare() unless
	ever_writable is TRUE. Don't automatically indicate G_IO_OUT in
	g_io_win32_check() unless ever_writable is TRUE.

	This fixes the behaviour of the test case program in bug #548278,
	and the "Testcase for the spurious OUT event bug" in bug
	#324234. It also doesn't seem to break anything. Not that there is
	any exhaustive test suite...

	Add a comment with a list of bugs that are related to the code in
	this file.


svn path=/trunk/; revision=7372
This commit is contained in:
Tor Lillqvist 2008-08-19 23:48:16 +00:00 committed by Tor Lillqvist
parent fcac96ed02
commit f7f48d5c08
2 changed files with 76 additions and 3 deletions

View File

@ -1,3 +1,28 @@
2008-08-20 Tor Lillqvist <tml@novell.com>
Bug 324234 - Using g_io_add_watch_full() to wait for connect() to
return on a non-blocking socket returns prematurely
Bug 548278 - Async GETs connections are always terminated
unexpectedly on Windows
* glib/giowin32.c: Add one more state variable to the
GIOWin32Channel struct, ever_writable. Initialise it to FALSE, set
to TRUE when the WSAEventSelect() indicates FD_WRITE, and never
reset to FALSE.
Don't do the WSASetEvent() in g_io_win32_prepare() unless
ever_writable is TRUE. Don't automatically indicate G_IO_OUT in
g_io_win32_check() unless ever_writable is TRUE.
This fixes the behaviour of the test case program in bug #548278,
and the "Testcase for the spurious OUT event bug" in bug
#324234. It also doesn't seem to break anything. Not that there is
any exhaustive test suite...
Add a comment with a list of bugs that are related to the code in
this file.
2008-08-18 Matthias Clasen <mclasen@redhat.com> 2008-08-18 Matthias Clasen <mclasen@redhat.com>
* configure.in: Bump version * configure.in: Bump version

View File

@ -29,6 +29,44 @@
* GLib at ftp://ftp.gtk.org/pub/gtk/. * GLib at ftp://ftp.gtk.org/pub/gtk/.
*/ */
/*
* Bugs that are related to the code in this file:
*
* Bug 137968 - Sometimes a GIOFunc on Win32 is called with zero condition
* http://bugzilla.gnome.org/show_bug.cgi?id=137968
*
* Bug 324234 - Using g_io_add_watch_full() to wait for connect() to return on a non-blocking socket returns prematurely
* http://bugzilla.gnome.org/show_bug.cgi?id=324234
*
* Bug 331214 - g_io_channel async socket io stalls
* http://bugzilla.gnome.org/show_bug.cgi?id=331214
*
* Bug 338943 - Multiple watches on the same socket
* http://bugzilla.gnome.org/show_bug.cgi?id=338943
*
* Bug 357674 - 2 serious bugs in giowin32.c making glib iochannels useless
* http://bugzilla.gnome.org/show_bug.cgi?id=357674
*
* Bug 425156 - GIOChannel deadlocks on a win32 socket
* http://bugzilla.gnome.org/show_bug.cgi?id=425156
*
* Bug 468910 - giofunc condition=0
* http://bugzilla.gnome.org/show_bug.cgi?id=468910
*
* Bug 500246 - Bug fixes for giowin32
* http://bugzilla.gnome.org/show_bug.cgi?id=500246
*
* Bug 548278 - Async GETs connections are always terminated unexpectedly on windows
* http://bugzilla.gnome.org/show_bug.cgi?id=548278
*
* Bug 548536 - giowin32 problem when adding and removing watches
* http://bugzilla.gnome.org/show_bug.cgi?id=548536
*
* When fixing bugs related to the code in this file, either the above
* bugs or others, make sure that the test programs attached to the
* above bugs continue to work.
*/
#include "config.h" #include "config.h"
#include "glib.h" #include "glib.h"
@ -116,6 +154,7 @@ struct _GIOWin32Channel {
int last_events; int last_events;
HANDLE event; HANDLE event;
gboolean write_would_have_blocked; gboolean write_would_have_blocked;
gboolean ever_writable;
}; };
#define LOCK(mutex) EnterCriticalSection (&mutex) #define LOCK(mutex) EnterCriticalSection (&mutex)
@ -242,6 +281,7 @@ g_io_channel_win32_init (GIOWin32Channel *channel)
channel->last_events = 0; channel->last_events = 0;
channel->event = NULL; channel->event = NULL;
channel->write_would_have_blocked = FALSE; channel->write_would_have_blocked = FALSE;
channel->ever_writable = FALSE;
InitializeCriticalSection (&channel->mutex); InitializeCriticalSection (&channel->mutex);
} }
@ -734,7 +774,9 @@ g_io_win32_prepare (GSource *source,
g_print ("\n setting last_events=0"); g_print ("\n setting last_events=0");
channel->last_events = 0; channel->last_events = 0;
if ((event_mask & FD_WRITE) && !channel->write_would_have_blocked) if ((event_mask & FD_WRITE) &&
channel->ever_writable &&
!channel->write_would_have_blocked)
{ {
if (channel->debug) if (channel->debug)
g_print (" WSASetEvent(%p)", (WSAEVENT) watch->pollfd.fd); g_print (" WSASetEvent(%p)", (WSAEVENT) watch->pollfd.fd);
@ -845,11 +887,15 @@ g_io_win32_check (GSource *source)
(HANDLE) watch->pollfd.fd); (HANDLE) watch->pollfd.fd);
ResetEvent ((HANDLE) watch->pollfd.fd); ResetEvent ((HANDLE) watch->pollfd.fd);
} }
else if (events.lNetworkEvents & FD_WRITE)
channel->ever_writable = TRUE;
channel->last_events = events.lNetworkEvents; channel->last_events = events.lNetworkEvents;
} }
watch->pollfd.revents = 0; watch->pollfd.revents = 0;
if (channel->last_events & (FD_READ | FD_ACCEPT)) if (channel->last_events & (FD_READ | FD_ACCEPT))
watch->pollfd.revents |= G_IO_IN; watch->pollfd.revents |= G_IO_IN;
if (channel->last_events & FD_WRITE) if (channel->last_events & FD_WRITE)
watch->pollfd.revents |= G_IO_OUT; watch->pollfd.revents |= G_IO_OUT;
else else
@ -869,10 +915,12 @@ g_io_win32_check (GSource *source)
} }
/* Regardless of WSAEnumNetworkEvents() result, if watching for /* Regardless of WSAEnumNetworkEvents() result, if watching for
* writability, unless last write would have blocked set * writability, and if we have ever got a FD_WRITE event, and
* G_IO_OUT. But never set both G_IO_OUT and G_IO_HUP. * unless last write would have blocked, set G_IO_OUT. But never
* set both G_IO_OUT and G_IO_HUP.
*/ */
if (!(watch->pollfd.revents & G_IO_HUP) && if (!(watch->pollfd.revents & G_IO_HUP) &&
channel->ever_writable &&
!channel->write_would_have_blocked && !channel->write_would_have_blocked &&
(channel->event_mask & FD_WRITE)) (channel->event_mask & FD_WRITE))
watch->pollfd.revents |= G_IO_OUT; watch->pollfd.revents |= G_IO_OUT;