|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363 |
- /*
- * DISTRHO Plugin Framework (DPF)
- * Copyright (C) 2012-2016 Filipe Coelho <falktx@falktx.com>
- *
- * Permission to use, copy, modify, and/or distribute this software for any purpose with
- * or without fee is hereby granted, provided that the above copyright notice and this
- * permission notice appear in all copies.
- *
- * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD
- * TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN
- * NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
- * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER
- * IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
- * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
- */
-
- #ifndef DISTRHO_MUTEX_HPP_INCLUDED
- #define DISTRHO_MUTEX_HPP_INCLUDED
-
- #include "../DistrhoUtils.hpp"
-
- #ifdef DISTRHO_OS_WINDOWS
- # include <winsock2.h>
- # include <windows.h>
- #endif
-
- #include <pthread.h>
-
- START_NAMESPACE_DISTRHO
-
- class Signal;
-
- // -----------------------------------------------------------------------
- // Mutex class
-
- class Mutex
- {
- public:
- /*
- * Constructor.
- */
- Mutex(bool inheritPriority = true) noexcept
- : fMutex()
- {
- pthread_mutexattr_t attr;
- pthread_mutexattr_init(&attr);
- pthread_mutexattr_setprotocol(&attr, inheritPriority ? PTHREAD_PRIO_INHERIT : PTHREAD_PRIO_NONE);
- pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_NORMAL);
- pthread_mutex_init(&fMutex, &attr);
- pthread_mutexattr_destroy(&attr);
- }
-
- /*
- * Destructor.
- */
- ~Mutex() noexcept
- {
- pthread_mutex_destroy(&fMutex);
- }
-
- /*
- * Lock the mutex.
- */
- void lock() const noexcept
- {
- pthread_mutex_lock(&fMutex);
- }
-
- /*
- * Try to lock the mutex.
- * Returns true if successful.
- */
- bool tryLock() const noexcept
- {
- return (pthread_mutex_trylock(&fMutex) == 0);
- }
-
- /*
- * Unlock the mutex.
- */
- void unlock() const noexcept
- {
- pthread_mutex_unlock(&fMutex);
- }
-
- private:
- mutable pthread_mutex_t fMutex;
-
- DISTRHO_PREVENT_HEAP_ALLOCATION
- DISTRHO_DECLARE_NON_COPY_CLASS(Mutex)
- };
-
- // -----------------------------------------------------------------------
- // RecursiveMutex class
-
- class RecursiveMutex
- {
- public:
- /*
- * Constructor.
- */
- RecursiveMutex() noexcept
- #ifdef DISTRHO_OS_WINDOWS
- : fSection()
- #else
- : fMutex()
- #endif
- {
- #ifdef DISTRHO_OS_WINDOWS
- InitializeCriticalSection(&fSection);
- #else
- pthread_mutexattr_t attr;
- pthread_mutexattr_init(&attr);
- pthread_mutexattr_setprotocol(&attr, PTHREAD_PRIO_INHERIT);
- pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_RECURSIVE);
- pthread_mutex_init(&fMutex, &attr);
- pthread_mutexattr_destroy(&attr);
- #endif
- }
-
- /*
- * Destructor.
- */
- ~RecursiveMutex() noexcept
- {
- #ifdef DISTRHO_OS_WINDOWS
- DeleteCriticalSection(&fSection);
- #else
- pthread_mutex_destroy(&fMutex);
- #endif
- }
-
- /*
- * Lock the mutex.
- */
- void lock() const noexcept
- {
- #ifdef DISTRHO_OS_WINDOWS
- EnterCriticalSection(&fSection);
- #else
- pthread_mutex_lock(&fMutex);
- #endif
- }
-
- /*
- * Try to lock the mutex.
- * Returns true if successful.
- */
- bool tryLock() const noexcept
- {
- #ifdef DISTRHO_OS_WINDOWS
- return (TryEnterCriticalSection(&fSection) != FALSE);
- #else
- return (pthread_mutex_trylock(&fMutex) == 0);
- #endif
- }
-
- /*
- * Unlock the mutex.
- */
- void unlock() const noexcept
- {
- #ifdef DISTRHO_OS_WINDOWS
- LeaveCriticalSection(&fSection);
- #else
- pthread_mutex_unlock(&fMutex);
- #endif
- }
-
- private:
- #ifdef DISTRHO_OS_WINDOWS
- mutable CRITICAL_SECTION fSection;
- #else
- mutable pthread_mutex_t fMutex;
- #endif
-
- DISTRHO_PREVENT_HEAP_ALLOCATION
- DISTRHO_DECLARE_NON_COPY_CLASS(RecursiveMutex)
- };
-
- // -----------------------------------------------------------------------
- // Signal class
-
- class Signal
- {
- public:
- /*
- * Constructor.
- */
- Signal() noexcept
- : fCondition(),
- fMutex(),
- fTriggered(false)
- {
- pthread_condattr_t cattr;
- pthread_condattr_init(&cattr);
- pthread_condattr_setpshared(&cattr, PTHREAD_PROCESS_PRIVATE);
- pthread_cond_init(&fCondition, &cattr);
- pthread_condattr_destroy(&cattr);
-
- pthread_mutexattr_t mattr;
- pthread_mutexattr_init(&mattr);
- pthread_mutexattr_setprotocol(&mattr, PTHREAD_PRIO_INHERIT);
- pthread_mutexattr_settype(&mattr, PTHREAD_MUTEX_NORMAL);
- pthread_mutex_init(&fMutex, &mattr);
- pthread_mutexattr_destroy(&mattr);
- }
-
- /*
- * Destructor.
- */
- ~Signal() noexcept
- {
- pthread_cond_destroy(&fCondition);
- pthread_mutex_destroy(&fMutex);
- }
-
- /*
- * Wait for a signal.
- */
- void wait() noexcept
- {
- pthread_mutex_lock(&fMutex);
-
- while (! fTriggered)
- {
- try {
- pthread_cond_wait(&fCondition, &fMutex);
- } DISTRHO_SAFE_EXCEPTION("pthread_cond_wait");
- }
-
- fTriggered = false;
-
- pthread_mutex_unlock(&fMutex);
- }
-
- /*
- * Wake up all waiting threads.
- */
- void signal() noexcept
- {
- pthread_mutex_lock(&fMutex);
-
- if (! fTriggered)
- {
- fTriggered = true;
- pthread_cond_broadcast(&fCondition);
- }
-
- pthread_mutex_unlock(&fMutex);
- }
-
- private:
- pthread_cond_t fCondition;
- pthread_mutex_t fMutex;
- volatile bool fTriggered;
-
- DISTRHO_PREVENT_HEAP_ALLOCATION
- DISTRHO_DECLARE_NON_COPY_CLASS(Signal)
- };
-
- // -----------------------------------------------------------------------
- // Helper class to lock&unlock a mutex during a function scope.
-
- template <class Mutex>
- class ScopeLocker
- {
- public:
- ScopeLocker(const Mutex& mutex) noexcept
- : fMutex(mutex)
- {
- fMutex.lock();
- }
-
- ~ScopeLocker() noexcept
- {
- fMutex.unlock();
- }
-
- private:
- const Mutex& fMutex;
-
- DISTRHO_PREVENT_HEAP_ALLOCATION
- DISTRHO_DECLARE_NON_COPY_CLASS(ScopeLocker)
- };
-
- // -----------------------------------------------------------------------
- // Helper class to try-lock&unlock a mutex during a function scope.
-
- template <class Mutex>
- class ScopeTryLocker
- {
- public:
- ScopeTryLocker(const Mutex& mutex) noexcept
- : fMutex(mutex),
- fLocked(mutex.tryLock()) {}
-
- ~ScopeTryLocker() noexcept
- {
- if (fLocked)
- fMutex.unlock();
- }
-
- bool wasLocked() const noexcept
- {
- return fLocked;
- }
-
- bool wasNotLocked() const noexcept
- {
- return !fLocked;
- }
-
- private:
- const Mutex& fMutex;
- const bool fLocked;
-
- DISTRHO_PREVENT_HEAP_ALLOCATION
- DISTRHO_DECLARE_NON_COPY_CLASS(ScopeTryLocker)
- };
-
- // -----------------------------------------------------------------------
- // Helper class to unlock&lock a mutex during a function scope.
-
- template <class Mutex>
- class ScopeUnlocker
- {
- public:
- ScopeUnlocker(const Mutex& mutex) noexcept
- : fMutex(mutex)
- {
- fMutex.unlock();
- }
-
- ~ScopeUnlocker() noexcept
- {
- fMutex.lock();
- }
-
- private:
- const Mutex& fMutex;
-
- DISTRHO_PREVENT_HEAP_ALLOCATION
- DISTRHO_DECLARE_NON_COPY_CLASS(ScopeUnlocker)
- };
-
- // -----------------------------------------------------------------------
- // Define types
-
- typedef ScopeLocker<Mutex> MutexLocker;
- typedef ScopeLocker<RecursiveMutex> RecursiveMutexLocker;
-
- typedef ScopeTryLocker<Mutex> MutexTryLocker;
- typedef ScopeTryLocker<RecursiveMutex> RecursiveMutexTryLocker;
-
- typedef ScopeUnlocker<Mutex> MutexUnlocker;
- typedef ScopeUnlocker<RecursiveMutex> RecursiveMutexUnlocker;
-
- // -----------------------------------------------------------------------
-
- END_NAMESPACE_DISTRHO
-
- #endif // DISTRHO_MUTEX_HPP_INCLUDED
|