/* * Carla Mutex * Copyright (C) 2013-2023 Filipe Coelho * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as * published by the Free Software Foundation; either version 2 of * the License, or any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * For a full copy of the GNU General Public License see the doc/GPL.txt file. */ #ifndef CARLA_MUTEX_HPP_INCLUDED #define CARLA_MUTEX_HPP_INCLUDED #include "CarlaUtils.hpp" #include class CarlaSignal; // ----------------------------------------------------------------------- // CarlaMutex class class CarlaMutex { public: /* * Constructor. */ CarlaMutex(const bool inheritPriority = true) noexcept : fMutex(), fTryLockWasCalled(false) { pthread_mutexattr_t attr; pthread_mutexattr_init(&attr); #ifdef __GNUC__ pthread_mutexattr_setprotocol(&attr, inheritPriority ? PTHREAD_PRIO_INHERIT : PTHREAD_PRIO_NONE); #else // unsupported? (void)inheritPriority; #endif pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_NORMAL); pthread_mutex_init(&fMutex, &attr); pthread_mutexattr_destroy(&attr); } /* * Destructor. */ ~CarlaMutex() noexcept { pthread_mutex_destroy(&fMutex); } /* * Check if "tryLock()" was called before. */ bool wasTryLockCalled() const noexcept { const bool ret(fTryLockWasCalled); fTryLockWasCalled = false; return ret; } /* * Lock the mutex. */ bool lock() const noexcept { return (pthread_mutex_lock(&fMutex) == 0); } /* * Try to lock the mutex. * Returns true if successful. */ bool tryLock() const noexcept { fTryLockWasCalled = true; return (pthread_mutex_trylock(&fMutex) == 0); } /* * Unlock the mutex, optionally resetting the tryLock check. */ void unlock(const bool resetTryLock = false) const noexcept { if (resetTryLock) fTryLockWasCalled = false; pthread_mutex_unlock(&fMutex); } private: mutable pthread_mutex_t fMutex; mutable volatile bool fTryLockWasCalled; // true if "tryLock()" was called at least once CARLA_DECLARE_NON_COPYABLE(CarlaMutex) }; // ----------------------------------------------------------------------- // CarlaRecursiveMutex class class CarlaRecursiveMutex { public: /* * Constructor. */ CarlaRecursiveMutex() noexcept #ifdef CARLA_OS_WIN : fSection() #else : fMutex() #endif { #ifdef CARLA_OS_WIN 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. */ ~CarlaRecursiveMutex() noexcept { #ifdef CARLA_OS_WIN DeleteCriticalSection(&fSection); #else pthread_mutex_destroy(&fMutex); #endif } /* * Lock the mutex. */ bool lock() const noexcept { #ifdef CARLA_OS_WIN EnterCriticalSection(&fSection); return true; #else return (pthread_mutex_lock(&fMutex) == 0); #endif } /* * Try to lock the mutex. * Returns true if successful. */ bool tryLock() const noexcept { #ifdef CARLA_OS_WIN return (TryEnterCriticalSection(&fSection) != FALSE); #else return (pthread_mutex_trylock(&fMutex) == 0); #endif } /* * Unlock the mutex. */ void unlock() const noexcept { #ifdef CARLA_OS_WIN LeaveCriticalSection(&fSection); #else pthread_mutex_unlock(&fMutex); #endif } private: #ifdef CARLA_OS_WIN mutable CRITICAL_SECTION fSection; #else mutable pthread_mutex_t fMutex; #endif CARLA_DECLARE_NON_COPYABLE(CarlaRecursiveMutex) }; // ----------------------------------------------------------------------- // CarlaSignal class class CarlaSignal { public: /* * Constructor. */ CarlaSignal() 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); #ifdef __GNUC__ pthread_mutexattr_setprotocol(&mattr, PTHREAD_PRIO_INHERIT); #endif pthread_mutexattr_settype(&mattr, PTHREAD_MUTEX_NORMAL); pthread_mutex_init(&fMutex, &mattr); pthread_mutexattr_destroy(&mattr); } /* * Destructor. */ ~CarlaSignal() 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); } CARLA_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; CARLA_PREVENT_HEAP_ALLOCATION CARLA_DECLARE_NON_COPYABLE(CarlaSignal) }; // ----------------------------------------------------------------------- // Helper class to lock&unlock a mutex during a function scope. template class CarlaScopeLocker { public: CarlaScopeLocker(const Mutex& mutex) noexcept : fMutex(mutex) { fMutex.lock(); } ~CarlaScopeLocker() noexcept { fMutex.unlock(); } private: const Mutex& fMutex; CARLA_PREVENT_HEAP_ALLOCATION CARLA_DECLARE_NON_COPYABLE(CarlaScopeLocker) }; // ----------------------------------------------------------------------- // Helper class to try-lock&unlock a mutex during a function scope. template class CarlaScopeTryLocker { public: CarlaScopeTryLocker(const Mutex& mutex) noexcept : fMutex(mutex), fLocked(mutex.tryLock()) {} CarlaScopeTryLocker(const Mutex& mutex, const bool forceLock) noexcept : fMutex(mutex), fLocked(forceLock ? mutex.lock() : mutex.tryLock()) {} ~CarlaScopeTryLocker() noexcept { if (fLocked) fMutex.unlock(); } bool wasLocked() const noexcept { return fLocked; } bool wasNotLocked() const noexcept { return !fLocked; } bool tryAgain() const noexcept { return fMutex.tryLock(); } private: const Mutex& fMutex; const bool fLocked; CARLA_PREVENT_HEAP_ALLOCATION CARLA_DECLARE_NON_COPYABLE(CarlaScopeTryLocker) }; // ----------------------------------------------------------------------- // Helper class to unlock&lock a mutex during a function scope. template class CarlaScopeUnlocker { public: CarlaScopeUnlocker(const Mutex& mutex) noexcept : fMutex(mutex) { fMutex.unlock(); } ~CarlaScopeUnlocker() noexcept { fMutex.lock(); } private: const Mutex& fMutex; CARLA_PREVENT_HEAP_ALLOCATION CARLA_DECLARE_NON_COPYABLE(CarlaScopeUnlocker) }; // ----------------------------------------------------------------------- // Define types typedef CarlaScopeLocker CarlaMutexLocker; typedef CarlaScopeLocker CarlaRecursiveMutexLocker; typedef CarlaScopeTryLocker CarlaMutexTryLocker; typedef CarlaScopeTryLocker CarlaRecursiveMutexTryLocker; typedef CarlaScopeUnlocker CarlaMutexUnlocker; typedef CarlaScopeUnlocker CarlaRecursiveMutexUnlocker; // ----------------------------------------------------------------------- #endif // CARLA_MUTEX_HPP_INCLUDED