// first of all, check whether we hadn't been cancelled already and don't
// start the user code at all then
- if ( thread->m_internal->GetState() == STATE_EXITED )
+ bool isExited = (thread->m_internal->GetState() == STATE_EXITED);
+
+ if ( isExited )
{
rc = (THREAD_RETVAL)-1;
}
}
rc = (THREAD_RETVAL)thread->Entry();
+ }
+
+ thread->OnExit();
+ // save IsDetached because thread object can be deleted by joinable
+ // threads after state is changed to STATE_EXITED.
+ bool isDetached = thread->IsDetached();
+
+ if (!isExited)
+ {
// enter m_critsect before changing the thread state
wxCriticalSectionLocker lock(thread->m_critsect);
-
thread->m_internal->SetState(STATE_EXITED);
}
- thread->OnExit();
-
// the thread may delete itself now if it wants, we don't need it any more
- thread->m_internal->LetDie();
+ if (isDetached) thread->m_internal->LetDie();
return rc;
}
// although the thread might be already in the EXITED state it might not
// have terminated yet and so we are not sure that it has actually
// terminated if the "if" above hadn't been taken
- do
+ for ( ;; )
{
if ( !::GetExitCodeThread(m_hThread, (LPDWORD)&rc) )
{
wxLogLastError(wxT("GetExitCodeThread"));
rc = (wxThread::ExitCode)-1;
+
+ break;
}
- } while ( (DWORD)rc == STILL_ACTIVE );
+
+ if ( (DWORD)rc != STILL_ACTIVE )
+ break;
+
+ // give the other thread some time to terminate, otherwise we may be
+ // starving it
+ ::Sleep(1);
+ }
if ( pRc )
*pRc = rc;