diff options
author | Corinna Vinschen <corinna@vinschen.de> | 2010-08-30 14:39:43 +0400 |
---|---|---|
committer | Corinna Vinschen <corinna@vinschen.de> | 2010-08-30 14:39:43 +0400 |
commit | 92596190c4f48613d55ac7878c481d872f717d21 (patch) | |
tree | 2ae34a207f30df5c185020913a4aeea1789e99f2 /winsup/cygwin/select.cc | |
parent | 98edb049e44c1630efaf8635281af211ddad3684 (diff) |
* autoload.cc (MsgWaitForMultipleObjectsEx): Define.
(MsgWaitForMultipleObjects): Remove.
* select.cc (select_stuff::wait): Use MsgWaitForMultipleObjectsEx with
QS_ALLPOSTMESSAGE and, if possible, MWMO_INPUTAVAILABLE flags. Explain
why. Fix a potential crash due to a NULL pointer in WAIT_FAILED case.
(peek_windows): Use filter pattern on NT4. Explain why.
* wincap.h (wincaps::has_mwmo_inputavailable): New element.
* wincap.cc: Implement above element throughout.
Diffstat (limited to 'winsup/cygwin/select.cc')
-rw-r--r-- | winsup/cygwin/select.cc | 24 |
1 files changed, 21 insertions, 3 deletions
diff --git a/winsup/cygwin/select.cc b/winsup/cygwin/select.cc index 41636d179..611367b31 100644 --- a/winsup/cygwin/select.cc +++ b/winsup/cygwin/select.cc @@ -287,7 +287,17 @@ select_stuff::wait (fd_set *readfds, fd_set *writefds, fd_set *exceptfds, if (!windows_used) wait_ret = WaitForMultipleObjects (m, w4, FALSE, ms); else - wait_ret = MsgWaitForMultipleObjects (m, w4, FALSE, ms, QS_ALLINPUT); + /* Using MWMO_INPUTAVAILABLE is the officially supported solution for + the problem that the call to PeekMessage disarms the queue state + so that a subsequent MWFMO hangs, even if there are still messages + in the queue. Unfortunately this flag didn't exist prior to Win2K, + so for NT4 we fall back to a different usage of PeekMessage in + peek_windows. See there for more details. */ + wait_ret = + MsgWaitForMultipleObjectsEx (m, w4, ms, + QS_ALLINPUT | QS_ALLPOSTMESSAGE, + wincap.has_mwmo_inputavailable () + ? MWMO_INPUTAVAILABLE : 0); switch (wait_ret) { @@ -296,7 +306,8 @@ select_stuff::wait (fd_set *readfds, fd_set *writefds, fd_set *exceptfds, set_sig_errno (EINTR); return -1; case WAIT_FAILED: - select_printf ("WaitForMultipleObjects failed"); + system_printf ("WaitForMultipleObjects failed"); + s = &start; s->set_select_errno (); return -1; case WAIT_TIMEOUT: @@ -1531,7 +1542,14 @@ peek_windows (select_record *me, bool) if (me->read_selected && me->read_ready) return 1; - if (PeekMessage (&m, (HWND) h, 0, 0, PM_NOREMOVE)) + /* On NT4 we use a filter pattern which allows to use QS_ALLPOSTMESSAGE + to keep the queue state as unread. Note that this only works if the + application itself does not call PeekMessage or GetQueueState the wrong + way. But there's no way around it. On Win2K and later we rather use + MsgWaitForMultipleObjectsEx(MWMO_INPUTAVAILABLE). */ + if (PeekMessage (&m, (HWND) h, 0, + wincap.has_mwmo_inputavailable () ? 0 : UINT_MAX - 1, + PM_NOREMOVE)) { me->read_ready = true; select_printf ("window %d(%p) ready", me->fd, me->fh->get_handle ()); |