mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2025-01-27 22:46:15 +01:00
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:
parent
fcac96ed02
commit
f7f48d5c08
25
ChangeLog
25
ChangeLog
@ -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
|
||||||
|
@ -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;
|
||||||
|
Loading…
Reference in New Issue
Block a user