You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
395 lines
6.9 KiB
395 lines
6.9 KiB
/*****************************************************************************/
|
|
// Copyright 2006-2008 Adobe Systems Incorporated
|
|
// All Rights Reserved.
|
|
//
|
|
// NOTICE: Adobe permits you to use, modify, and distribute this file in
|
|
// accordance with the terms of the Adobe license agreement accompanying it.
|
|
/*****************************************************************************/
|
|
|
|
/* $Id: //mondo/dng_sdk_1_4/dng_sdk/source/dng_mutex.cpp#3 $ */
|
|
/* $DateTime: 2012/09/05 12:31:51 $ */
|
|
/* $Change: 847652 $ */
|
|
/* $Author: tknoll $ */
|
|
|
|
#include "dng_mutex.h"
|
|
|
|
#include "dng_assertions.h"
|
|
#include "dng_exceptions.h"
|
|
|
|
#include <stdlib.h>
|
|
|
|
/*****************************************************************************/
|
|
|
|
#if qDNGThreadSafe
|
|
|
|
namespace
|
|
{
|
|
|
|
class InnermostMutexHolder
|
|
{
|
|
|
|
private:
|
|
|
|
pthread_key_t fInnermostMutexKey;
|
|
|
|
public:
|
|
|
|
InnermostMutexHolder ()
|
|
|
|
: fInnermostMutexKey ()
|
|
|
|
{
|
|
|
|
int result = pthread_key_create (&fInnermostMutexKey, NULL);
|
|
|
|
DNG_ASSERT (result == 0, "pthread_key_create failed.");
|
|
|
|
if (result != 0)
|
|
ThrowProgramError ();
|
|
|
|
}
|
|
|
|
~InnermostMutexHolder ()
|
|
{
|
|
|
|
pthread_key_delete (fInnermostMutexKey);
|
|
|
|
}
|
|
|
|
void SetInnermostMutex (dng_mutex *mutex)
|
|
{
|
|
|
|
int result;
|
|
|
|
result = pthread_setspecific (fInnermostMutexKey, (void *)mutex);
|
|
|
|
DNG_ASSERT (result == 0, "pthread_setspecific failed.");
|
|
|
|
#if 0 // Hard failure here was causing crash on quit.
|
|
|
|
if (result != 0)
|
|
ThrowProgramError ();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
dng_mutex *GetInnermostMutex ()
|
|
{
|
|
|
|
void *result = pthread_getspecific (fInnermostMutexKey);
|
|
|
|
return reinterpret_cast<dng_mutex *> (result);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
InnermostMutexHolder gInnermostMutexHolder;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
/*****************************************************************************/
|
|
|
|
dng_mutex::dng_mutex (const char *mutexName, uint32 mutexLevel)
|
|
|
|
#if qDNGThreadSafe
|
|
|
|
: fPthreadMutex ()
|
|
, fMutexLevel (mutexLevel)
|
|
, fRecursiveLockCount (0)
|
|
, fPrevHeldMutex (NULL)
|
|
, fMutexName (mutexName)
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
#if qDNGThreadSafe
|
|
|
|
if (pthread_mutex_init (&fPthreadMutex, NULL) != 0)
|
|
{
|
|
ThrowMemoryFull ();
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
|
|
dng_mutex::~dng_mutex ()
|
|
{
|
|
|
|
#if qDNGThreadSafe
|
|
|
|
pthread_mutex_destroy (&fPthreadMutex);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
|
|
void dng_mutex::Lock ()
|
|
{
|
|
|
|
#if qDNGThreadSafe
|
|
|
|
dng_mutex *innermostMutex = gInnermostMutexHolder.GetInnermostMutex ();
|
|
|
|
if (innermostMutex != NULL)
|
|
{
|
|
|
|
if (innermostMutex == this)
|
|
{
|
|
|
|
fRecursiveLockCount++;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
bool lockOrderPreserved = fMutexLevel > innermostMutex->fMutexLevel /* ||
|
|
(fMutexLevel == innermostMutex->fMutexLevel && innermostMutex < this) */;
|
|
|
|
if (!lockOrderPreserved)
|
|
{
|
|
|
|
DNG_REPORT ("Lock ordering violation.");
|
|
|
|
#if qDNGDebug
|
|
|
|
dng_show_message_f ("This mutex: %s v Innermost mutex: %s",
|
|
this->MutexName (),
|
|
innermostMutex->MutexName ());
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pthread_mutex_lock (&fPthreadMutex);
|
|
|
|
fPrevHeldMutex = innermostMutex;
|
|
|
|
gInnermostMutexHolder.SetInnermostMutex (this);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
|
|
void dng_mutex::Unlock ()
|
|
{
|
|
|
|
#if qDNGThreadSafe
|
|
|
|
DNG_ASSERT (gInnermostMutexHolder.GetInnermostMutex () == this, "Mutexes unlocked out of order!!!");
|
|
|
|
if (fRecursiveLockCount > 0)
|
|
{
|
|
|
|
fRecursiveLockCount--;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
gInnermostMutexHolder.SetInnermostMutex (fPrevHeldMutex);
|
|
|
|
fPrevHeldMutex = NULL;
|
|
|
|
pthread_mutex_unlock (&fPthreadMutex);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
|
|
const char *dng_mutex::MutexName () const
|
|
{
|
|
|
|
#if qDNGThreadSafe
|
|
|
|
if (fMutexName)
|
|
return fMutexName;
|
|
|
|
#endif
|
|
|
|
return "< unknown >";
|
|
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
|
|
dng_lock_mutex::dng_lock_mutex (dng_mutex *mutex)
|
|
|
|
: fMutex (mutex)
|
|
|
|
{
|
|
|
|
if (fMutex)
|
|
fMutex->Lock ();
|
|
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
|
|
dng_lock_mutex::~dng_lock_mutex ()
|
|
{
|
|
|
|
if (fMutex)
|
|
fMutex->Unlock ();
|
|
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
|
|
dng_unlock_mutex::dng_unlock_mutex (dng_mutex *mutex)
|
|
|
|
: fMutex (mutex)
|
|
|
|
{
|
|
|
|
if (fMutex)
|
|
fMutex->Unlock ();
|
|
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
|
|
dng_unlock_mutex::~dng_unlock_mutex ()
|
|
{
|
|
|
|
if (fMutex)
|
|
fMutex->Lock ();
|
|
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
|
|
#if qDNGThreadSafe
|
|
|
|
/*****************************************************************************/
|
|
|
|
dng_condition::dng_condition ()
|
|
|
|
: fPthreadCondition ()
|
|
|
|
{
|
|
|
|
int result;
|
|
|
|
result = pthread_cond_init (&fPthreadCondition, NULL);
|
|
|
|
DNG_ASSERT (result == 0, "pthread_cond_init failed.");
|
|
|
|
if (result != 0)
|
|
{
|
|
ThrowProgramError ();
|
|
}
|
|
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
|
|
dng_condition::~dng_condition ()
|
|
{
|
|
|
|
pthread_cond_destroy (&fPthreadCondition);
|
|
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
|
|
bool dng_condition::Wait (dng_mutex &mutex, double timeoutSecs)
|
|
{
|
|
|
|
bool timedOut = false;
|
|
|
|
dng_mutex *innermostMutex = gInnermostMutexHolder.GetInnermostMutex ();
|
|
|
|
DNG_ASSERT (innermostMutex == &mutex, "Attempt to wait on non-innermost mutex.");
|
|
|
|
innermostMutex = mutex.fPrevHeldMutex;
|
|
|
|
gInnermostMutexHolder.SetInnermostMutex (innermostMutex);
|
|
|
|
mutex.fPrevHeldMutex = NULL;
|
|
|
|
if (timeoutSecs < 0)
|
|
{
|
|
|
|
pthread_cond_wait (&fPthreadCondition, &mutex.fPthreadMutex);
|
|
|
|
}
|
|
|
|
else
|
|
{
|
|
|
|
struct timespec now;
|
|
|
|
dng_pthread_now (&now);
|
|
|
|
timeoutSecs += now.tv_sec;
|
|
timeoutSecs += now.tv_nsec / 1000000000.0;
|
|
|
|
now.tv_sec = (long) timeoutSecs;
|
|
now.tv_nsec = (long) ((timeoutSecs - now.tv_sec) * 1000000000);
|
|
|
|
timedOut = (pthread_cond_timedwait (&fPthreadCondition, &mutex.fPthreadMutex, &now) == ETIMEDOUT);
|
|
|
|
}
|
|
|
|
mutex.fPrevHeldMutex = innermostMutex;
|
|
|
|
gInnermostMutexHolder.SetInnermostMutex (&mutex);
|
|
|
|
return !timedOut;
|
|
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
|
|
void dng_condition::Signal ()
|
|
{
|
|
|
|
int result;
|
|
|
|
result = pthread_cond_signal (&fPthreadCondition);
|
|
|
|
DNG_ASSERT (result == 0, "pthread_cond_signal failed.");
|
|
|
|
if (result != 0)
|
|
ThrowProgramError ();
|
|
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
|
|
void dng_condition::Broadcast ()
|
|
{
|
|
|
|
int result;
|
|
|
|
result = pthread_cond_broadcast (&fPthreadCondition);
|
|
|
|
DNG_ASSERT (result == 0, "pthread_cond_broadcast failed.");
|
|
|
|
if (result != 0)
|
|
ThrowProgramError ();
|
|
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
|
|
#endif // qDNGThreadSafe
|
|
|
|
/*****************************************************************************/
|