1 /////////////////////////////////////////////////////////////////////////////
3 // Purpose: wxThread Implementation
4 // Author: Original from Wolfram Gloger/Guilhem Lavaux
5 // Modified by: Vadim Zeitlin to make it work :-)
8 // Copyright: (c) Wolfram Gloger (1996, 1997); Guilhem Lavaux (1998),
9 // Vadim Zeitlin (1999)
10 // Licence: wxWindows licence
11 /////////////////////////////////////////////////////////////////////////////
14 #pragma implementation "thread.h"
17 // ----------------------------------------------------------------------------
19 // ----------------------------------------------------------------------------
21 // For compilers that support precompilation, includes "wx.h".
22 #include "wx/wxprec.h"
24 #if defined(__BORLANDC__)
38 #include "wx/module.h"
39 #include "wx/thread.h"
41 // the possible states of the thread ("=>" shows all possible transitions from
45 STATE_NEW
, // didn't start execution yet (=> RUNNING)
46 STATE_RUNNING
, // thread is running (=> PAUSED, CANCELED)
47 STATE_PAUSED
, // thread is temporarily suspended (=> RUNNING)
48 STATE_CANCELED
, // thread should terminate a.s.a.p. (=> EXITED)
49 STATE_EXITED
// thread is terminating
52 // ----------------------------------------------------------------------------
54 // ----------------------------------------------------------------------------
56 // TLS index of the slot where we store the pointer to the current thread
57 static DWORD s_tlsThisThread
= 0xFFFFFFFF;
59 // id of the main thread - the one which can call GUI functions without first
60 // calling wxMutexGuiEnter()
61 static DWORD s_idMainThread
= 0;
63 // if it's FALSE, some secondary thread is holding the GUI lock
64 static bool s_bGuiOwnedByMainThread
= TRUE
;
66 // critical section which controls access to all GUI functions: any secondary
67 // thread (i.e. except the main one) must enter this crit section before doing
69 static wxCriticalSection
*s_critsectGui
= NULL
;
71 // critical section which protects s_nWaitingForGui variable
72 static wxCriticalSection
*s_critsectWaitingForGui
= NULL
;
74 // number of threads waiting for GUI in wxMutexGuiEnter()
75 static size_t s_nWaitingForGui
= 0;
77 // are we waiting for a thread termination?
78 static bool s_waitingForThread
= FALSE
;
80 // ============================================================================
81 // Windows implementation of thread classes
82 // ============================================================================
84 // ----------------------------------------------------------------------------
85 // wxMutex implementation
86 // ----------------------------------------------------------------------------
95 p_internal
= new wxMutexInternal
;
96 p_internal
->p_mutex
= CreateMutex(NULL
, FALSE
, NULL
);
97 if ( !p_internal
->p_mutex
)
99 wxLogSysError(_("Can not create mutex."));
108 wxLogDebug("Warning: freeing a locked mutex (%d locks).", m_locked
);
109 CloseHandle(p_internal
->p_mutex
);
112 wxMutexError
wxMutex::Lock()
116 ret
= WaitForSingleObject(p_internal
->p_mutex
, INFINITE
);
127 wxLogSysError(_("Couldn't acquire a mutex lock"));
128 return wxMUTEX_MISC_ERROR
;
132 wxFAIL_MSG("impossible return value in wxMutex::Lock");
136 return wxMUTEX_NO_ERROR
;
139 wxMutexError
wxMutex::TryLock()
143 ret
= WaitForSingleObject(p_internal
->p_mutex
, 0);
144 if (ret
== WAIT_TIMEOUT
|| ret
== WAIT_ABANDONED
)
148 return wxMUTEX_NO_ERROR
;
151 wxMutexError
wxMutex::Unlock()
156 BOOL ret
= ReleaseMutex(p_internal
->p_mutex
);
159 wxLogSysError(_("Couldn't release a mutex"));
160 return wxMUTEX_MISC_ERROR
;
163 return wxMUTEX_NO_ERROR
;
166 // ----------------------------------------------------------------------------
167 // wxCondition implementation
168 // ----------------------------------------------------------------------------
170 class wxConditionInternal
177 wxCondition::wxCondition()
179 p_internal
= new wxConditionInternal
;
180 p_internal
->event
= CreateEvent(NULL
, FALSE
, FALSE
, NULL
);
181 if ( !p_internal
->event
)
183 wxLogSysError(_("Can not create event object."));
186 p_internal
->waiters
= 0;
189 wxCondition::~wxCondition()
191 CloseHandle(p_internal
->event
);
194 void wxCondition::Wait(wxMutex
& mutex
)
197 p_internal
->waiters
++;
198 WaitForSingleObject(p_internal
->event
, INFINITE
);
199 p_internal
->waiters
--;
203 bool wxCondition::Wait(wxMutex
& mutex
,
210 p_internal
->waiters
++;
211 ret
= WaitForSingleObject(p_internal
->event
, (sec
*1000)+(nsec
/1000000));
212 p_internal
->waiters
--;
215 return (ret
!= WAIT_TIMEOUT
);
218 void wxCondition::Signal()
220 SetEvent(p_internal
->event
);
223 void wxCondition::Broadcast()
227 for (i
=0;i
<p_internal
->waiters
;i
++)
229 if ( SetEvent(p_internal
->event
) == 0 )
231 wxLogSysError(_("Couldn't change the state of event object."));
236 // ----------------------------------------------------------------------------
237 // wxCriticalSection implementation
238 // ----------------------------------------------------------------------------
240 class wxCriticalSectionInternal
243 // init the critical section object
244 wxCriticalSectionInternal()
245 { ::InitializeCriticalSection(&m_data
); }
247 // implicit cast to the associated data
248 operator CRITICAL_SECTION
*() { return &m_data
; }
250 // free the associated ressources
251 ~wxCriticalSectionInternal()
252 { ::DeleteCriticalSection(&m_data
); }
255 CRITICAL_SECTION m_data
;
258 wxCriticalSection::wxCriticalSection()
260 m_critsect
= new wxCriticalSectionInternal
;
263 wxCriticalSection::~wxCriticalSection()
268 void wxCriticalSection::Enter()
270 ::EnterCriticalSection(*m_critsect
);
273 void wxCriticalSection::Leave()
275 ::LeaveCriticalSection(*m_critsect
);
278 // ----------------------------------------------------------------------------
279 // wxThread implementation
280 // ----------------------------------------------------------------------------
282 // wxThreadInternal class
283 // ----------------------
285 class wxThreadInternal
292 m_priority
= WXTHREAD_DEFAULT_PRIORITY
;
295 // create a new (suspended) thread (for the given thread object)
296 bool Create(wxThread
*thread
);
298 // suspend/resume/terminate
301 void Cancel() { m_state
= STATE_CANCELED
; }
304 void SetState(wxThreadState state
) { m_state
= state
; }
305 wxThreadState
GetState() const { return m_state
; }
308 void SetPriority(unsigned int priority
) { m_priority
= priority
; }
309 unsigned int GetPriority() const { return m_priority
; }
311 // thread handle and id
312 HANDLE
GetHandle() const { return m_hThread
; }
313 DWORD
GetId() const { return m_tid
; }
316 static DWORD
WinThreadStart(wxThread
*thread
);
319 HANDLE m_hThread
; // handle of the thread
320 wxThreadState m_state
; // state, see wxThreadState enum
321 unsigned int m_priority
; // thread priority in "wx" units
322 DWORD m_tid
; // thread id
325 DWORD
wxThreadInternal::WinThreadStart(wxThread
*thread
)
327 // store the thread object in the TLS
328 if ( !::TlsSetValue(s_tlsThisThread
, thread
) )
330 wxLogSysError(_("Can not start thread: error writing TLS."));
335 DWORD ret
= (DWORD
)thread
->Entry();
336 thread
->p_internal
->SetState(STATE_EXITED
);
344 bool wxThreadInternal::Create(wxThread
*thread
)
346 m_hThread
= ::CreateThread
348 NULL
, // default security
349 0, // default stack size
350 (LPTHREAD_START_ROUTINE
) // thread entry point
351 wxThreadInternal::WinThreadStart
, //
352 (LPVOID
)thread
, // parameter
353 CREATE_SUSPENDED
, // flags
354 &m_tid
// [out] thread id
357 if ( m_hThread
== NULL
)
359 wxLogSysError(_("Can't create thread"));
364 // translate wxWindows priority to the Windows one
366 if (m_priority
<= 20)
367 win_priority
= THREAD_PRIORITY_LOWEST
;
368 else if (m_priority
<= 40)
369 win_priority
= THREAD_PRIORITY_BELOW_NORMAL
;
370 else if (m_priority
<= 60)
371 win_priority
= THREAD_PRIORITY_NORMAL
;
372 else if (m_priority
<= 80)
373 win_priority
= THREAD_PRIORITY_ABOVE_NORMAL
;
374 else if (m_priority
<= 100)
375 win_priority
= THREAD_PRIORITY_HIGHEST
;
378 wxFAIL_MSG("invalid value of thread priority parameter");
379 win_priority
= THREAD_PRIORITY_NORMAL
;
382 if ( ::SetThreadPriority(m_hThread
, win_priority
) == 0 )
384 wxLogSysError(_("Can't set thread priority"));
390 bool wxThreadInternal::Suspend()
392 DWORD nSuspendCount
= ::SuspendThread(m_hThread
);
393 if ( nSuspendCount
== (DWORD
)-1 )
395 wxLogSysError(_("Can not suspend thread %x"), m_hThread
);
400 m_state
= STATE_PAUSED
;
405 bool wxThreadInternal::Resume()
407 DWORD nSuspendCount
= ::ResumeThread(m_hThread
);
408 if ( nSuspendCount
== (DWORD
)-1 )
410 wxLogSysError(_("Can not resume thread %x"), m_hThread
);
415 m_state
= STATE_RUNNING
;
423 wxThread
*wxThread::This()
425 wxThread
*thread
= (wxThread
*)::TlsGetValue(s_tlsThisThread
);
427 // be careful, 0 may be a valid return value as well
428 if ( !thread
&& (::GetLastError() != NO_ERROR
) )
430 wxLogSysError(_("Couldn't get the current thread pointer"));
438 bool wxThread::IsMain()
440 return ::GetCurrentThreadId() == s_idMainThread
;
443 void wxThread::Yield()
445 // 0 argument to Sleep() is special
449 void wxThread::Sleep(unsigned long milliseconds
)
451 ::Sleep(milliseconds
);
454 // create/start thread
455 // -------------------
457 wxThreadError
wxThread::Create()
459 if ( !p_internal
->Create(this) )
460 return wxTHREAD_NO_RESOURCE
;
462 return wxTHREAD_NO_ERROR
;
465 wxThreadError
wxThread::Run()
467 wxCriticalSectionLocker
lock(m_critsect
);
469 if ( p_internal
->GetState() != STATE_NEW
)
471 // actually, it may be almost any state at all, not only STATE_RUNNING
472 return wxTHREAD_RUNNING
;
478 // suspend/resume thread
479 // ---------------------
481 wxThreadError
wxThread::Pause()
483 wxCriticalSectionLocker
lock(m_critsect
);
485 return p_internal
->Suspend() ? wxTHREAD_NO_ERROR
: wxTHREAD_MISC_ERROR
;
488 wxThreadError
wxThread::Resume()
490 wxCriticalSectionLocker
lock(m_critsect
);
492 return p_internal
->Resume() ? wxTHREAD_NO_ERROR
: wxTHREAD_MISC_ERROR
;
498 wxThread::ExitCode
wxThread::Delete()
502 // Delete() is always safe to call, so consider all possible states
510 // set flag for wxIsWaitingForThread()
511 s_waitingForThread
= TRUE
;
518 wxCriticalSectionLocker
lock(m_critsect
);
520 p_internal
->Cancel();
521 hThread
= p_internal
->GetHandle();
524 // we can't just wait for the thread to terminate because it might be
525 // calling some GUI functions and so it will never terminate before we
526 // process the Windows messages that result from these functions
530 result
= ::MsgWaitForMultipleObjects
532 1, // number of objects to wait for
533 &hThread
, // the objects
534 FALSE
, // don't wait for all objects
535 INFINITE
, // no timeout
536 QS_ALLEVENTS
// return as soon as there are any events
543 wxLogSysError(_("Can not wait for thread termination"));
548 // thread we're waiting for terminated
551 case WAIT_OBJECT_0
+ 1:
552 // new message arrived, process it
553 if ( !wxTheApp
->DoMessage() )
555 // WM_QUIT received: kill the thread
563 // give the thread we're waiting for chance to exit
564 // from the GUI call it might have been in
565 if ( (s_nWaitingForGui
> 0) && wxGuiOwnedByMainThread() )
574 wxFAIL_MSG("unexpected result of MsgWaitForMultipleObject");
576 } while ( result
!= WAIT_OBJECT_0
);
580 s_waitingForThread
= FALSE
;
585 if ( !::GetExitCodeThread(hThread
, (LPDWORD
)&rc
) )
587 wxLogLastError("GetExitCodeThread");
592 wxASSERT_MSG( (LPVOID
)rc
!= (LPVOID
)STILL_ACTIVE
,
593 "thread must be already terminated." );
595 ::CloseHandle(hThread
);
601 wxThreadError
wxThread::Kill()
604 return wxTHREAD_NOT_RUNNING
;
606 if ( !::TerminateThread(p_internal
->GetHandle(), (DWORD
)-1) )
608 wxLogSysError(_("Couldn't terminate thread"));
610 return wxTHREAD_MISC_ERROR
;
615 return wxTHREAD_NO_ERROR
;
618 void wxThread::Exit(void *status
)
622 ::ExitThread((DWORD
)status
);
624 wxFAIL_MSG("Couldn't return from ExitThread()!");
627 void wxThread::SetPriority(unsigned int prio
)
629 wxCriticalSectionLocker
lock(m_critsect
);
631 p_internal
->SetPriority(prio
);
634 unsigned int wxThread::GetPriority() const
636 wxCriticalSectionLocker
lock((wxCriticalSection
&)m_critsect
);
638 return p_internal
->GetPriority();
641 unsigned long wxThread::GetID() const
643 wxCriticalSectionLocker
lock((wxCriticalSection
&)m_critsect
);
645 return (unsigned long)p_internal
->GetId();
648 bool wxThread::IsRunning() const
650 wxCriticalSectionLocker
lock((wxCriticalSection
&)m_critsect
);
652 return p_internal
->GetState() == STATE_RUNNING
;
655 bool wxThread::IsAlive() const
657 wxCriticalSectionLocker
lock((wxCriticalSection
&)m_critsect
);
659 return (p_internal
->GetState() == STATE_RUNNING
) ||
660 (p_internal
->GetState() == STATE_PAUSED
);
663 bool wxThread::TestDestroy() const
665 wxCriticalSectionLocker
lock((wxCriticalSection
&)m_critsect
);
667 return p_internal
->GetState() == STATE_CANCELED
;
672 p_internal
= new wxThreadInternal();
675 wxThread::~wxThread()
680 // ----------------------------------------------------------------------------
681 // Automatic initialization for thread module
682 // ----------------------------------------------------------------------------
684 class wxThreadModule
: public wxModule
687 virtual bool OnInit();
688 virtual void OnExit();
691 DECLARE_DYNAMIC_CLASS(wxThreadModule
)
694 IMPLEMENT_DYNAMIC_CLASS(wxThreadModule
, wxModule
)
696 bool wxThreadModule::OnInit()
698 // allocate TLS index for storing the pointer to the current thread
699 s_tlsThisThread
= ::TlsAlloc();
700 if ( s_tlsThisThread
== 0xFFFFFFFF )
702 // in normal circumstances it will only happen if all other
703 // TLS_MINIMUM_AVAILABLE (>= 64) indices are already taken - in other
704 // words, this should never happen
705 wxLogSysError(_("Thread module initialization failed: "
706 "impossible to allocate index in thread "
712 // main thread doesn't have associated wxThread object, so store 0 in the
714 if ( !::TlsSetValue(s_tlsThisThread
, (LPVOID
)0) )
716 ::TlsFree(s_tlsThisThread
);
717 s_tlsThisThread
= 0xFFFFFFFF;
719 wxLogSysError(_("Thread module initialization failed: "
720 "can not store value in thread local storage"));
725 s_critsectWaitingForGui
= new wxCriticalSection();
727 s_critsectGui
= new wxCriticalSection();
728 s_critsectGui
->Enter();
730 // no error return for GetCurrentThreadId()
731 s_idMainThread
= ::GetCurrentThreadId();
736 void wxThreadModule::OnExit()
738 if ( !::TlsFree(s_tlsThisThread
) )
740 wxLogLastError("TlsFree failed.");
745 s_critsectGui
->Leave();
746 delete s_critsectGui
;
747 s_critsectGui
= NULL
;
750 wxDELETE(s_critsectWaitingForGui
);
753 // ----------------------------------------------------------------------------
754 // under Windows, these functions are implemented usign a critical section and
755 // not a mutex, so the names are a bit confusing
756 // ----------------------------------------------------------------------------
758 void WXDLLEXPORT
wxMutexGuiEnter()
760 // this would dead lock everything...
761 wxASSERT_MSG( !wxThread::IsMain(),
762 "main thread doesn't want to block in wxMutexGuiEnter()!" );
764 // the order in which we enter the critical sections here is crucial!!
766 // set the flag telling to the main thread that we want to do some GUI
768 wxCriticalSectionLocker
enter(*s_critsectWaitingForGui
);
773 wxWakeUpMainThread();
775 // now we may block here because the main thread will soon let us in
776 // (during the next iteration of OnIdle())
777 s_critsectGui
->Enter();
780 void WXDLLEXPORT
wxMutexGuiLeave()
782 wxCriticalSectionLocker
enter(*s_critsectWaitingForGui
);
784 if ( wxThread::IsMain() )
786 s_bGuiOwnedByMainThread
= FALSE
;
790 // decrement the number of waiters now
791 wxASSERT_MSG( s_nWaitingForGui
> 0,
792 "calling wxMutexGuiLeave() without entering it first?" );
796 wxWakeUpMainThread();
799 s_critsectGui
->Leave();
802 void WXDLLEXPORT
wxMutexGuiLeaveOrEnter()
804 wxASSERT_MSG( wxThread::IsMain(),
805 "only main thread may call wxMutexGuiLeaveOrEnter()!" );
807 wxCriticalSectionLocker
enter(*s_critsectWaitingForGui
);
809 if ( s_nWaitingForGui
== 0 )
811 // no threads are waiting for GUI - so we may acquire the lock without
812 // any danger (but only if we don't already have it)
813 if ( !wxGuiOwnedByMainThread() )
815 s_critsectGui
->Enter();
817 s_bGuiOwnedByMainThread
= TRUE
;
819 //else: already have it, nothing to do
823 // some threads are waiting, release the GUI lock if we have it
824 if ( wxGuiOwnedByMainThread() )
828 //else: some other worker thread is doing GUI
832 bool WXDLLEXPORT
wxGuiOwnedByMainThread()
834 return s_bGuiOwnedByMainThread
;
837 // wake up the main thread if it's in ::GetMessage()
838 void WXDLLEXPORT
wxWakeUpMainThread()
840 // sending any message would do - hopefully WM_NULL is harmless enough
841 if ( !::PostThreadMessage(s_idMainThread
, WM_NULL
, 0, 0) )
843 // should never happen
844 wxLogLastError("PostThreadMessage(WM_NULL)");
848 bool WXDLLEXPORT
wxIsWaitingForThread()
850 return s_waitingForThread
;
853 #endif // wxUSE_THREADS