1 ///////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     wxThread (Posix) Implementation 
   4 // Author:      Original from Wolfram Gloger/Guilhem Lavaux 
   8 // Copyright:   (c) Wolfram Gloger (1996, 1997) 
   9 //                  Guilhem Lavaux (1998) 
  10 //                  Vadim Zeitlin (1999) 
  11 //                  Robert Roebling (1999) 
  12 // Licence:     wxWindows licence 
  13 ///////////////////////////////////////////////////////////////////////////// 
  15 // ============================================================================ 
  17 // ============================================================================ 
  19 // ---------------------------------------------------------------------------- 
  21 // ---------------------------------------------------------------------------- 
  24     #pragma implementation "thread.h" 
  27 // With simple makefiles, we must ignore the file body if not using 
  33 #include "wx/thread.h" 
  34 #include "wx/module.h" 
  38 #include "wx/dynarray.h" 
  50 // ---------------------------------------------------------------------------- 
  52 // ---------------------------------------------------------------------------- 
  54 // the possible states of the thread and transitions from them 
  57     STATE_NEW
,          // didn't start execution yet (=> RUNNING) 
  58     STATE_RUNNING
,      // running (=> PAUSED or EXITED) 
  59     STATE_PAUSED
,       // suspended (=> RUNNING or EXITED) 
  60     STATE_EXITED        
// thread doesn't exist any more 
  63 // ---------------------------------------------------------------------------- 
  65 // ---------------------------------------------------------------------------- 
  67 WX_DEFINE_ARRAY(wxThread 
*, wxArrayThread
); 
  69 // ----------------------------------------------------------------------------- 
  71 // ----------------------------------------------------------------------------- 
  73 // we keep the list of all threads created by the application to be able to 
  74 // terminate them on exit if there are some left - otherwise the process would 
  76 static wxArrayThread gs_allThreads
; 
  78 // the id of the main thread 
  79 static pthread_t gs_tidMain
; 
  81 // the key for the pointer to the associated wxThread object 
  82 static pthread_key_t gs_keySelf
; 
  84 // this mutex must be acquired before any call to a GUI function 
  85 static wxMutex 
*gs_mutexGui
; 
  87 // ============================================================================ 
  89 // ============================================================================ 
  91 //-------------------------------------------------------------------- 
  92 // wxMutex (Posix implementation) 
  93 //-------------------------------------------------------------------- 
  98     pthread_mutex_t p_mutex
; 
 103     p_internal 
= new wxMutexInternal
; 
 104     pthread_mutex_init( &(p_internal
->p_mutex
), (const pthread_mutexattr_t
*) NULL 
); 
 111         wxLogDebug(_T("Freeing a locked mutex (%d locks)"), m_locked
); 
 113     pthread_mutex_destroy( &(p_internal
->p_mutex
) ); 
 117 wxMutexError 
wxMutex::Lock() 
 119     int err 
= pthread_mutex_lock( &(p_internal
->p_mutex
) ); 
 122         wxLogDebug(_T("Locking this mutex would lead to deadlock!")); 
 124         return wxMUTEX_DEAD_LOCK
; 
 129     return wxMUTEX_NO_ERROR
; 
 132 wxMutexError 
wxMutex::TryLock() 
 139     int err 
= pthread_mutex_trylock( &(p_internal
->p_mutex
) ); 
 142         case EBUSY
: return wxMUTEX_BUSY
; 
 147     return wxMUTEX_NO_ERROR
; 
 150 wxMutexError 
wxMutex::Unlock() 
 158         wxLogDebug(_T("Unlocking not locked mutex.")); 
 160         return wxMUTEX_UNLOCKED
; 
 163     pthread_mutex_unlock( &(p_internal
->p_mutex
) ); 
 165     return wxMUTEX_NO_ERROR
; 
 168 //-------------------------------------------------------------------- 
 169 // wxCondition (Posix implementation) 
 170 //-------------------------------------------------------------------- 
 172 class wxConditionInternal
 
 175   pthread_cond_t p_condition
; 
 178 wxCondition::wxCondition() 
 180     p_internal 
= new wxConditionInternal
; 
 181     pthread_cond_init( &(p_internal
->p_condition
), (const pthread_condattr_t 
*) NULL 
); 
 184 wxCondition::~wxCondition() 
 186     pthread_cond_destroy( &(p_internal
->p_condition
) ); 
 191 void wxCondition::Wait(wxMutex
& mutex
) 
 193     pthread_cond_wait( &(p_internal
->p_condition
), &(mutex
.p_internal
->p_mutex
) ); 
 196 bool wxCondition::Wait(wxMutex
& mutex
, unsigned long sec
, unsigned long nsec
) 
 198     struct timespec tspec
; 
 200     tspec
.tv_sec 
= time(0L)+sec
; 
 201     tspec
.tv_nsec 
= nsec
; 
 202     return (pthread_cond_timedwait(&(p_internal
->p_condition
), &(mutex
.p_internal
->p_mutex
), &tspec
) != ETIMEDOUT
); 
 205 void wxCondition::Signal() 
 207     pthread_cond_signal( &(p_internal
->p_condition
) ); 
 210 void wxCondition::Broadcast() 
 212     pthread_cond_broadcast( &(p_internal
->p_condition
) ); 
 215 //-------------------------------------------------------------------- 
 216 // wxThread (Posix implementation) 
 217 //-------------------------------------------------------------------- 
 219 class wxThreadInternal
 
 225     // thread entry function 
 226     static void *PthreadStart(void *ptr
); 
 231         // ask the thread to terminate 
 233         // wake up threads waiting for our termination 
 235         // go to sleep until Resume() is called 
 242     int GetPriority() const { return m_prio
; } 
 243     void SetPriority(int prio
) { m_prio 
= prio
; } 
 245     wxThreadState 
GetState() const { return m_state
; } 
 246     void SetState(wxThreadState state
) { m_state 
= state
; } 
 248     pthread_t 
GetId() const { return m_threadId
; } 
 249     pthread_t 
*GetIdPtr() { return &m_threadId
; } 
 251     void SetCancelFlag() { m_cancelled 
= TRUE
; } 
 252     bool WasCancelled() const { return m_cancelled
; } 
 255     pthread_t     m_threadId
;   // id of the thread 
 256     wxThreadState m_state
;      // see wxThreadState enum 
 257     int           m_prio
;       // in wxWindows units: from 0 to 100 
 259     // set when the thread should terminate 
 262     // this (mutex, cond) pair is used to synchronize the main thread and this 
 263     // thread in several situations: 
 264     //  1. The thread function blocks until condition is signaled by Run() when 
 265     //     it's initially created - this allows thread creation in "suspended" 
 267     //  2. The Delete() function blocks until the condition is signaled when the 
 272     // another (mutex, cond) pair for Pause()/Resume() usage 
 274     // VZ: it's possible that we might reuse the mutex and condition from above 
 275     //     for this too, but as I'm not at all sure that it won't create subtle 
 276     //     problems with race conditions between, say, Pause() and Delete() I 
 277     //     prefer this may be a bit less efficient but much safer solution 
 278     wxMutex     m_mutexSuspend
; 
 279     wxCondition m_condSuspend
; 
 282 void *wxThreadInternal::PthreadStart(void *ptr
) 
 284     wxThread 
*thread 
= (wxThread 
*)ptr
; 
 285     wxThreadInternal 
*pthread 
= thread
->p_internal
; 
 287     int rc 
= pthread_setspecific(gs_keySelf
, thread
); 
 290         wxLogSysError(rc
, _("Cannot start thread: error writing TLS")); 
 295     // wait for the condition to be signaled from Run() 
 296     // mutex state: currently locked by the thread which created us 
 297     pthread
->m_cond
.Wait(pthread
->m_mutex
); 
 299     // mutex state: locked again on exit of Wait() 
 301     // call the main entry 
 302     void* status 
= thread
->Entry(); 
 304     // terminate the thread 
 305     thread
->Exit(status
); 
 307     wxFAIL_MSG(_T("wxThread::Exit() can't return.")); 
 312 wxThreadInternal::wxThreadInternal() 
 317     // this mutex is locked during almost all thread lifetime - it will only be 
 318     // unlocked in the very end 
 321     // this mutex is used in Pause()/Resume() and is also locked all the time 
 322     // unless the thread is paused 
 323     m_mutexSuspend
.Lock(); 
 326 wxThreadInternal::~wxThreadInternal() 
 328     // GL: moved to SignalExit 
 329     // m_mutexSuspend.Unlock(); 
 331     // note that m_mutex will be unlocked by the thread which waits for our 
 335 wxThreadError 
wxThreadInternal::Run() 
 337     wxCHECK_MSG( GetState() == STATE_NEW
, wxTHREAD_RUNNING
, 
 338                  _T("thread may only be started once after successful Create()") ); 
 340     // the mutex was locked on Create(), so we will be able to lock it again 
 341     // only when the thread really starts executing and enters the wait - 
 342     // otherwise we might signal the condition before anybody is waiting for it 
 343     wxMutexLocker 
lock(m_mutex
); 
 346     m_state 
= STATE_RUNNING
; 
 348     return wxTHREAD_NO_ERROR
; 
 350     // now the mutex is unlocked back - but just to allow Wait() function to 
 351     // terminate by relocking it, so the net result is that the worker thread 
 352     // starts executing and the mutex is still locked 
 355 void wxThreadInternal::Wait() 
 357     wxCHECK_RET( WasCancelled(), _T("thread should have been cancelled first") ); 
 359     // if the thread we're waiting for is waiting for the GUI mutex, we will 
 360     // deadlock so make sure we release it temporarily 
 361     if ( wxThread::IsMain() ) 
 364     // entering Wait() releases the mutex thus allowing SignalExit() to acquire 
 365     // it and to signal us its termination 
 366     m_cond
.Wait(m_mutex
); 
 368     // mutex is still in the locked state - relocked on exit from Wait(), so 
 369     // unlock it - we don't need it any more, the thread has already terminated 
 372     // reacquire GUI mutex 
 373     if ( wxThread::IsMain() ) 
 377 void wxThreadInternal::SignalExit() 
 379     // GL: Unlock mutexSuspend here. 
 380     m_mutexSuspend
.Unlock(); 
 382     // as mutex is currently locked, this will block until some other thread 
 383     // (normally the same which created this one) unlocks it by entering Wait() 
 386     // wake up all the threads waiting for our termination 
 389     // after this call mutex will be finally unlocked 
 393 void wxThreadInternal::Pause() 
 395     // the state is set from the thread which pauses us first, this function 
 396     // is called later so the state should have been already set 
 397     wxCHECK_RET( m_state 
== STATE_PAUSED
, 
 398                  _T("thread must first be paused with wxThread::Pause().") ); 
 400     // don't pause the thread which is being terminated - this would lead to 
 401     // deadlock if the thread is paused after Delete() had called Resume() but 
 402     // before it had time to call Wait() 
 403     if ( WasCancelled() ) 
 406     // wait until the condition is signaled from Resume() 
 407     m_condSuspend
.Wait(m_mutexSuspend
); 
 410 void wxThreadInternal::Resume() 
 412     wxCHECK_RET( m_state 
== STATE_PAUSED
, 
 413                  _T("can't resume thread which is not suspended.") ); 
 415     // we will be able to lock this mutex only when Pause() starts waiting 
 416     wxMutexLocker 
lock(m_mutexSuspend
); 
 417     m_condSuspend
.Signal(); 
 419     SetState(STATE_RUNNING
); 
 422 // ----------------------------------------------------------------------------- 
 424 // ----------------------------------------------------------------------------- 
 426 wxThread 
*wxThread::This() 
 428     return (wxThread 
*)pthread_getspecific(gs_keySelf
); 
 431 bool wxThread::IsMain() 
 433     return (bool)pthread_equal(pthread_self(), gs_tidMain
); 
 436 void wxThread::Yield() 
 441 void wxThread::Sleep(unsigned long milliseconds
) 
 443     wxUsleep(milliseconds
); 
 446 // ----------------------------------------------------------------------------- 
 448 // ----------------------------------------------------------------------------- 
 452     // add this thread to the global list of all threads 
 453     gs_allThreads
.Add(this); 
 455     p_internal 
= new wxThreadInternal(); 
 458 wxThreadError 
wxThread::Create() 
 460     if (p_internal
->GetState() != STATE_NEW
) 
 461         return wxTHREAD_RUNNING
; 
 463     // set up the thread attribute: right now, we only set thread priority 
 465     pthread_attr_init(&attr
); 
 467 #ifdef HAVE_THREAD_PRIORITY_FUNCTIONS 
 469     if ( pthread_attr_getschedpolicy(&attr
, &prio
) != 0 ) 
 471         wxLogError(_("Cannot retrieve thread scheduling policy.")); 
 474     int min_prio 
= sched_get_priority_min(prio
), 
 475         max_prio 
= sched_get_priority_max(prio
); 
 477     if ( min_prio 
== -1 || max_prio 
== -1 ) 
 479         wxLogError(_("Cannot get priority range for scheduling policy %d."), 
 484         struct sched_param sp
; 
 485         pthread_attr_getschedparam(&attr
, &sp
); 
 486         sp
.sched_priority 
= min_prio 
+ 
 487                            (p_internal
->GetPriority()*(max_prio
-min_prio
))/100; 
 488         pthread_attr_setschedparam(&attr
, &sp
); 
 490 #endif // HAVE_THREAD_PRIORITY_FUNCTIONS 
 492 #ifdef HAVE_PTHREAD_ATTR_SETSCOPE 
 493     // this will make the threads created by this process really concurrent 
 494     pthread_attr_setscope(&attr
, PTHREAD_SCOPE_SYSTEM
); 
 495 #endif // HAVE_PTHREAD_ATTR_SETSCOPE 
 497     // create the new OS thread object 
 498     int rc 
= pthread_create(p_internal
->GetIdPtr(), &attr
, 
 499                             wxThreadInternal::PthreadStart
, (void *)this); 
 500     pthread_attr_destroy(&attr
); 
 504         p_internal
->SetState(STATE_EXITED
); 
 505         return wxTHREAD_NO_RESOURCE
; 
 508     return wxTHREAD_NO_ERROR
; 
 511 wxThreadError 
wxThread::Run() 
 513     return p_internal
->Run(); 
 516 // ----------------------------------------------------------------------------- 
 518 // ----------------------------------------------------------------------------- 
 520 void wxThread::SetPriority(unsigned int prio
) 
 522     wxCHECK_RET( ((int)WXTHREAD_MIN_PRIORITY 
<= (int)prio
) && 
 523                  ((int)prio 
<= (int)WXTHREAD_MAX_PRIORITY
), 
 524                  _T("invalid thread priority") ); 
 526     wxCriticalSectionLocker 
lock(m_critsect
); 
 528     switch ( p_internal
->GetState() ) 
 531             // thread not yet started, priority will be set when it is 
 532             p_internal
->SetPriority(prio
); 
 537 #ifdef HAVE_THREAD_PRIORITY_FUNCTIONS 
 539                 struct sched_param sparam
; 
 540                 sparam
.sched_priority 
= prio
; 
 542                 if ( pthread_setschedparam(p_internal
->GetId(), 
 543                                            SCHED_OTHER
, &sparam
) != 0 ) 
 545                     wxLogError(_("Failed to set thread priority %d."), prio
); 
 548 #endif // HAVE_THREAD_PRIORITY_FUNCTIONS 
 553             wxFAIL_MSG(_T("impossible to set thread priority in this state")); 
 557 unsigned int wxThread::GetPriority() const 
 559     wxCriticalSectionLocker 
lock((wxCriticalSection 
&)m_critsect
); 
 561     return p_internal
->GetPriority(); 
 564 unsigned long wxThread::GetID() const 
 566     return (unsigned long)p_internal
->GetId(); 
 569 // ----------------------------------------------------------------------------- 
 571 // ----------------------------------------------------------------------------- 
 573 wxThreadError 
wxThread::Pause() 
 575     wxCriticalSectionLocker 
lock(m_critsect
); 
 577     if ( p_internal
->GetState() != STATE_RUNNING 
) 
 579         wxLogDebug(_T("Can't pause thread which is not running.")); 
 581         return wxTHREAD_NOT_RUNNING
; 
 584     p_internal
->SetState(STATE_PAUSED
); 
 586     return wxTHREAD_NO_ERROR
; 
 589 wxThreadError 
wxThread::Resume() 
 591     wxCriticalSectionLocker 
lock(m_critsect
); 
 593     if ( p_internal
->GetState() == STATE_PAUSED 
) 
 596         p_internal
->Resume(); 
 599         return wxTHREAD_NO_ERROR
; 
 603         wxLogDebug(_T("Attempt to resume a thread which is not paused.")); 
 605         return wxTHREAD_MISC_ERROR
; 
 609 // ----------------------------------------------------------------------------- 
 611 // ----------------------------------------------------------------------------- 
 613 wxThread::ExitCode 
wxThread::Delete() 
 616     wxThreadState state 
= p_internal
->GetState(); 
 619     // ask the thread to stop 
 620     p_internal
->SetCancelFlag(); 
 630             // resume the thread first 
 636             // wait until the thread stops 
 643 wxThreadError 
wxThread::Kill() 
 645     switch ( p_internal
->GetState() ) 
 649             return wxTHREAD_NOT_RUNNING
; 
 652 #ifdef HAVE_PTHREAD_CANCEL  
 653             if ( pthread_cancel(p_internal
->GetId()) != 0 ) 
 656                 wxLogError(_("Failed to terminate a thread.")); 
 658                 return wxTHREAD_MISC_ERROR
; 
 661             return wxTHREAD_NO_ERROR
; 
 665 void wxThread::Exit(void *status
) 
 667     // first call user-level clean up code 
 670     // next wake up the threads waiting for us (OTOH, this function won't return 
 671     // until someone waited for us!) 
 672     p_internal
->SignalExit(); 
 674     p_internal
->SetState(STATE_EXITED
); 
 676     // delete both C++ thread object and terminate the OS thread object 
 677     // GL: This is very ugly and buggy ... 
 679     pthread_exit(status
); 
 682 // also test whether we were paused 
 683 bool wxThread::TestDestroy() 
 685     wxCriticalSectionLocker 
lock(m_critsect
); 
 687     if ( p_internal
->GetState() == STATE_PAUSED 
) 
 689         // leave the crit section or the other threads will stop too if they try 
 690         // to call any of (seemingly harmless) IsXXX() functions while we sleep 
 695         // enter it back before it's finally left in lock object dtor 
 699     return p_internal
->WasCancelled(); 
 702 wxThread::~wxThread() 
 704     // remove this thread from the global array 
 705     gs_allThreads
.Remove(this); 
 708 // ----------------------------------------------------------------------------- 
 710 // ----------------------------------------------------------------------------- 
 712 bool wxThread::IsRunning() const 
 714     wxCriticalSectionLocker 
lock((wxCriticalSection 
&)m_critsect
); 
 716     return p_internal
->GetState() == STATE_RUNNING
; 
 719 bool wxThread::IsAlive() const 
 721     wxCriticalSectionLocker 
lock((wxCriticalSection
&)m_critsect
); 
 723     switch ( p_internal
->GetState() ) 
 734 bool wxThread::IsPaused() const 
 736     wxCriticalSectionLocker 
lock((wxCriticalSection
&)m_critsect
); 
 738     return (p_internal
->GetState() == STATE_PAUSED
); 
 741 //-------------------------------------------------------------------- 
 743 //-------------------------------------------------------------------- 
 745 class wxThreadModule 
: public wxModule
 
 748     virtual bool OnInit(); 
 749     virtual void OnExit(); 
 752     DECLARE_DYNAMIC_CLASS(wxThreadModule
) 
 755 IMPLEMENT_DYNAMIC_CLASS(wxThreadModule
, wxModule
) 
 757 bool wxThreadModule::OnInit() 
 759     int rc 
= pthread_key_create(&gs_keySelf
, NULL 
/* dtor function */); 
 762         wxLogSysError(rc
, _("Thread module initialization failed: " 
 763                             "failed to create thread key")); 
 768     gs_mutexGui 
= new wxMutex(); 
 772     gs_tidMain 
= pthread_self(); 
 778 void wxThreadModule::OnExit() 
 780     wxASSERT_MSG( wxThread::IsMain(), _T("only main thread can be here") ); 
 782     // terminate any threads left 
 783     size_t count 
= gs_allThreads
.GetCount(); 
 785         wxLogDebug(_T("Some threads were not terminated by the application.")); 
 787     for ( size_t n 
= 0u; n 
< count
; n
++ ) 
 789         gs_allThreads
[n
]->Delete(); 
 793     gs_mutexGui
->Unlock(); 
 800     (void)pthread_key_delete(gs_keySelf
); 
 803 // ---------------------------------------------------------------------------- 
 805 // ---------------------------------------------------------------------------- 
 807 void wxMutexGuiEnter() 
 812 void wxMutexGuiLeave() 
 814   gs_mutexGui
->Unlock();