X-Git-Url: https://git.saurik.com/wxWidgets.git/blobdiff_plain/22b6de6a70900550b5411f14fffc9d48adbbf222..3b84c8078e7be6fdb8afb369e559c331b5416da8:/src/common/selectdispatcher.cpp diff --git a/src/common/selectdispatcher.cpp b/src/common/selectdispatcher.cpp index 138fa3cc8c..1365319347 100644 --- a/src/common/selectdispatcher.cpp +++ b/src/common/selectdispatcher.cpp @@ -30,7 +30,8 @@ #include "wx/intl.h" #endif -#ifdef HAVE_SYS_SELECT_H +#if defined(HAVE_SYS_SELECT_H) || defined(__WATCOMC__) + #include #include #endif @@ -110,7 +111,7 @@ int wxSelectSets::Select(int nfds, struct timeval *tv) return select(nfds, &m_fds[Read], &m_fds[Write], &m_fds[Except], tv); } -void wxSelectSets::Handle(int fd, wxFDIOHandler& handler) const +bool wxSelectSets::Handle(int fd, wxFDIOHandler& handler) const { for ( int n = 0; n < Max; n++ ) { @@ -119,28 +120,19 @@ void wxSelectSets::Handle(int fd, wxFDIOHandler& handler) const wxLogTrace(wxSelectDispatcher_Trace, _T("Got %s event on fd %d"), ms_names[n], fd); (handler.*ms_handlers[n])(); - // callback can modify sets and destroy handler, returning from - // here guarantees that one event is processed at a time - return; + // callback can modify sets and destroy handler + // this forces that one event can be processed at one time + return true; } } + + return false; } // ---------------------------------------------------------------------------- // wxSelectDispatcher // ---------------------------------------------------------------------------- -/* static */ -wxSelectDispatcher *wxSelectDispatcher::Create() -{ - return new wxSelectDispatcher; -} - -wxSelectDispatcher::wxSelectDispatcher() -{ - m_maxFD = -1; -} - bool wxSelectDispatcher::RegisterFD(int fd, wxFDIOHandler *handler, int flags) { if ( !wxMappedFDIODispatcher::RegisterFD(fd, handler, flags) ) @@ -200,8 +192,9 @@ bool wxSelectDispatcher::UnregisterFD(int fd) return true; } -void wxSelectDispatcher::ProcessSets(const wxSelectSets& sets) +int wxSelectDispatcher::ProcessSets(const wxSelectSets& sets) { + int numEvents = 0; for ( int fd = 0; fd <= m_maxFD; fd++ ) { if ( !sets.HasFD(fd) ) @@ -214,11 +207,14 @@ void wxSelectDispatcher::ProcessSets(const wxSelectSets& sets) continue; } - sets.Handle(fd, *handler); + if ( sets.Handle(fd, *handler) ) + numEvents++; } + + return numEvents; } -void wxSelectDispatcher::Dispatch(int timeout) +int wxSelectDispatcher::DoSelect(wxSelectSets& sets, int timeout) const { struct timeval tv, *ptv; @@ -233,24 +229,37 @@ void wxSelectDispatcher::Dispatch(int timeout) ptv = NULL; } - wxSelectSets sets = m_sets; + int ret = sets.Select(m_maxFD + 1, ptv); - const int ret = sets.Select(m_maxFD + 1, ptv); - switch ( ret ) + // TODO: we need to restart select() in this case but for now just return + // as if timeout expired + if ( ret == -1 && errno == EINTR ) + ret = 0; + + return ret; +} + +bool wxSelectDispatcher::HasPending() const +{ + wxSelectSets sets(m_sets); + return DoSelect(sets, 0) > 0; +} + +int wxSelectDispatcher::Dispatch(int timeout) +{ + wxSelectSets sets(m_sets); + switch ( DoSelect(sets, timeout) ) { case -1: - if ( errno != EINTR ) - { - wxLogSysError(_("Failed to monitor I/O channels")); - } - break; + wxLogSysError(_("Failed to monitor I/O channels")); + return -1; case 0: // timeout expired without anything happening - break; + return 0; default: - ProcessSets(sets); + return ProcessSets(sets); } }