Test for reentrance into non-recursive wxMutex. If it happens, yield dead lock error, don't hang

git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@55521 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
This commit is contained in:
Robert Roebling
2008-09-08 08:00:03 +00:00
parent 44b25eac69
commit 375f364ef2
2 changed files with 23 additions and 7 deletions

View File

@@ -191,7 +191,6 @@ private:
HANDLE m_mutex; HANDLE m_mutex;
unsigned long m_owningThread; unsigned long m_owningThread;
bool m_isLocked;
wxMutexType m_type; wxMutexType m_type;
DECLARE_NO_COPY_CLASS(wxMutexInternal) DECLARE_NO_COPY_CLASS(wxMutexInternal)
@@ -210,7 +209,6 @@ wxMutexInternal::wxMutexInternal(wxMutexType mutexType)
m_type = mutexType; m_type = mutexType;
m_owningThread = 0; m_owningThread = 0;
m_isLocked = false;
if ( !m_mutex ) if ( !m_mutex )
{ {
@@ -243,7 +241,7 @@ wxMutexError wxMutexInternal::LockTimeout(DWORD milliseconds)
if (m_type == wxMUTEX_DEFAULT) if (m_type == wxMUTEX_DEFAULT)
{ {
// Don't allow recursive // Don't allow recursive
if (m_isLocked) if (m_owningThread != 0)
{ {
if (m_owningThread == wxThread::GetCurrentId()) if (m_owningThread == wxThread::GetCurrentId())
return wxMUTEX_DEAD_LOCK; return wxMUTEX_DEAD_LOCK;
@@ -283,7 +281,6 @@ wxMutexError wxMutexInternal::LockTimeout(DWORD milliseconds)
if (m_type == wxMUTEX_DEFAULT) if (m_type == wxMUTEX_DEFAULT)
{ {
// required for checking recursiveness // required for checking recursiveness
m_isLocked = true;
m_owningThread = wxThread::GetCurrentId(); m_owningThread = wxThread::GetCurrentId();
} }
@@ -292,15 +289,15 @@ wxMutexError wxMutexInternal::LockTimeout(DWORD milliseconds)
wxMutexError wxMutexInternal::Unlock() wxMutexError wxMutexInternal::Unlock()
{ {
// required for checking recursiveness
m_owningThread = 0;
if ( !::ReleaseMutex(m_mutex) ) if ( !::ReleaseMutex(m_mutex) )
{ {
wxLogLastError(_T("ReleaseMutex()")); wxLogLastError(_T("ReleaseMutex()"));
return wxMUTEX_MISC_ERROR; return wxMUTEX_MISC_ERROR;
} }
// required for checking recursiveness
m_isLocked = false;
return wxMUTEX_NO_ERROR; return wxMUTEX_NO_ERROR;
} }

View File

@@ -189,6 +189,8 @@ private:
private: private:
pthread_mutex_t m_mutex; pthread_mutex_t m_mutex;
bool m_isOk; bool m_isOk;
wxMutexType m_type;
unsigned long m_owningThread;
// wxConditionInternal uses our m_mutex // wxConditionInternal uses our m_mutex
friend class wxConditionInternal; friend class wxConditionInternal;
@@ -203,6 +205,9 @@ extern "C" int pthread_mutexattr_settype(pthread_mutexattr_t *, int);
wxMutexInternal::wxMutexInternal(wxMutexType mutexType) wxMutexInternal::wxMutexInternal(wxMutexType mutexType)
{ {
m_type = mutexType;
m_owningThread = 0;
int err; int err;
switch ( mutexType ) switch ( mutexType )
{ {
@@ -263,6 +268,12 @@ wxMutexInternal::~wxMutexInternal()
wxMutexError wxMutexInternal::Lock() wxMutexError wxMutexInternal::Lock()
{ {
if ((m_type == wxMUTEX_DEFAULT) && (m_owningThread != 0))
{
if (m_owningThread == wxThread::GetCurrentId())
return wxMUTEX_DEAD_LOCK;
}
return HandleLockResult(pthread_mutex_lock(&m_mutex)); return HandleLockResult(pthread_mutex_lock(&m_mutex));
} }
@@ -318,6 +329,8 @@ wxMutexError wxMutexInternal::Lock(unsigned long ms)
wxMutexError wxMutexInternal::HandleLockResult(int err) wxMutexError wxMutexInternal::HandleLockResult(int err)
{ {
// wxPrintf( "err %d\n", err );
switch ( err ) switch ( err )
{ {
case EDEADLK: case EDEADLK:
@@ -334,6 +347,8 @@ wxMutexError wxMutexInternal::HandleLockResult(int err)
return wxMUTEX_TIMEOUT; return wxMUTEX_TIMEOUT;
case 0: case 0:
if (m_type == wxMUTEX_DEFAULT)
m_owningThread = wxThread::GetCurrentId();
return wxMUTEX_NO_ERROR; return wxMUTEX_NO_ERROR;
default: default:
@@ -359,6 +374,8 @@ wxMutexError wxMutexInternal::TryLock()
break; break;
case 0: case 0:
if (m_type == wxMUTEX_DEFAULT)
m_owningThread = wxThread::GetCurrentId();
return wxMUTEX_NO_ERROR; return wxMUTEX_NO_ERROR;
default: default:
@@ -370,6 +387,8 @@ wxMutexError wxMutexInternal::TryLock()
wxMutexError wxMutexInternal::Unlock() wxMutexError wxMutexInternal::Unlock()
{ {
m_owningThread = 0;
int err = pthread_mutex_unlock(&m_mutex); int err = pthread_mutex_unlock(&m_mutex);
switch ( err ) switch ( err )
{ {