/* * Carla Mutex * Copyright (C) 2013-2014 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 // ----------------------------------------------------------------------- // CarlaMutex class class CarlaMutex { public: /* * Constructor. */ CarlaMutex() noexcept : fTryLockWasCalled(false) { pthread_mutex_init(&fMutex, nullptr); } /* * 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. */ void lock() const noexcept { pthread_mutex_lock(&fMutex); } /* * 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; // The mutex mutable volatile bool fTryLockWasCalled; // true if "tryLock()" was called at least once CARLA_PREVENT_HEAP_ALLOCATION CARLA_DECLARE_NON_COPY_CLASS(CarlaMutex) }; // ----------------------------------------------------------------------- // CarlaCriticalSection class class CarlaCriticalSection { public: /* * Constructor. */ CarlaCriticalSection() noexcept { #ifdef CARLA_OS_WIN InitializeCriticalSection(&fSection); #else pthread_mutexattr_t atts; pthread_mutexattr_init(&atts); pthread_mutexattr_settype(&atts, PTHREAD_MUTEX_RECURSIVE); # ifndef CARLA_OS_ANDROID pthread_mutexattr_setprotocol(&atts, PTHREAD_PRIO_INHERIT); # endif pthread_mutex_init(&fMutex, &atts); pthread_mutexattr_destroy(&atts); #endif } /* * Destructor. */ ~CarlaCriticalSection() noexcept { #ifdef CARLA_OS_WIN DeleteCriticalSection(&fSection); #else pthread_mutex_destroy(&fMutex); #endif } /* * Enter section. */ void enter() const noexcept { #ifdef CARLA_OS_WIN EnterCriticalSection(&fSection); #else pthread_mutex_lock(&fMutex); #endif } /* * Try-Enter section. */ bool tryEnter() const noexcept { #ifdef CARLA_OS_WIN return (TryEnterCriticalSection(&fSection) != FALSE); #else return (pthread_mutex_trylock(&fMutex) == 0); #endif } /* * Leave section. */ void leave() 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_PREVENT_HEAP_ALLOCATION CARLA_DECLARE_NON_COPY_CLASS(CarlaCriticalSection) }; // ----------------------------------------------------------------------- // Helper class to lock&unlock a mutex during a function scope. class CarlaMutexLocker { public: CarlaMutexLocker(const CarlaMutex& mutex) noexcept : fMutex(mutex) { fMutex.lock(); } ~CarlaMutexLocker() noexcept { fMutex.unlock(); } private: const CarlaMutex& fMutex; CARLA_PREVENT_HEAP_ALLOCATION CARLA_DECLARE_NON_COPY_CLASS(CarlaMutexLocker) }; // ----------------------------------------------------------------------- // Helper class to unlock&lock a mutex during a function scope. class CarlaMutexUnlocker { public: CarlaMutexUnlocker(const CarlaMutex& mutex) noexcept : fMutex(mutex) { fMutex.unlock(); } ~CarlaMutexUnlocker() noexcept { fMutex.lock(); } private: const CarlaMutex& fMutex; CARLA_PREVENT_HEAP_ALLOCATION CARLA_DECLARE_NON_COPY_CLASS(CarlaMutexUnlocker) }; // ----------------------------------------------------------------------- // Helper class to enter&leave during a function scope. class CarlaCriticalSectionScope { public: CarlaCriticalSectionScope(const CarlaCriticalSection& cs) noexcept : fSection(cs) { fSection.enter(); } ~CarlaCriticalSectionScope() noexcept { fSection.leave(); } private: const CarlaCriticalSection& fSection; CARLA_PREVENT_HEAP_ALLOCATION CARLA_DECLARE_NON_COPY_CLASS(CarlaCriticalSectionScope) }; // ----------------------------------------------------------------------- #endif // CARLA_MUTEX_HPP_INCLUDED