| // Copyright (C) 2014 The Android Open Source Project |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| #pragma once |
| |
| #include "aemu/base/Compiler.h" |
| |
| #ifdef _WIN32 |
| #define WIN32_LEAN_AND_MEAN 1 |
| #include <windows.h> |
| #else |
| #include <pthread.h> |
| #endif |
| |
| #include <assert.h> |
| |
| #define AEMU_DEBUG 0 |
| |
| #if AEMU_DEBUG |
| #define AEMU_IF_DEBUG(x) x |
| #else |
| #define AEMU_IF_DEBUG(x) |
| #endif |
| |
| namespace gfxstream { |
| namespace guest { |
| |
| template <class Lockable> |
| class AutoLock; |
| |
| class AutoWriteLock; |
| class AutoReadLock; |
| |
| // A wrapper class for mutexes only suitable for using in static context, |
| // where it's OK to leak the underlying system object. |
| // Use Lock / RecursiveLock for scoped or member locks. |
| template <bool IsRecursive> |
| class StaticLock; |
| |
| template <> |
| class StaticLock<false> { |
| public: |
| using AutoLock = gfxstream::guest::AutoLock<StaticLock>; |
| |
| constexpr StaticLock() = default; |
| |
| // Acquire the lock. |
| void lock() { |
| #ifdef _WIN32 |
| ::AcquireSRWLockExclusive(&mLock); |
| #else |
| ::pthread_mutex_lock(&mLock); |
| #endif |
| AEMU_IF_DEBUG(mIsLocked = true;) |
| } |
| |
| bool tryLock() { |
| bool ret = false; |
| #ifdef _WIN32 |
| ret = ::TryAcquireSRWLockExclusive(&mLock); |
| #else |
| ret = ::pthread_mutex_trylock(&mLock) == 0; |
| #endif |
| AEMU_IF_DEBUG(mIsLocked = ret;) |
| return ret; |
| } |
| |
| AEMU_IF_DEBUG(bool isLocked() const { return mIsLocked; }) |
| |
| // Release the lock. |
| void unlock() { |
| AEMU_IF_DEBUG(mIsLocked = false;) |
| #ifdef _WIN32 |
| ::ReleaseSRWLockExclusive(&mLock); |
| #else |
| ::pthread_mutex_unlock(&mLock); |
| #endif |
| } |
| |
| protected: |
| friend class ConditionVariable; |
| |
| #ifdef _WIN32 |
| // Benchmarks show that on Windows SRWLOCK performs a little bit better than |
| // CRITICAL_SECTION for uncontended mode and much better in case of |
| // contention. |
| SRWLOCK mLock = SRWLOCK_INIT; |
| #else |
| pthread_mutex_t mLock = PTHREAD_MUTEX_INITIALIZER; |
| #endif |
| // Both POSIX threads and WinAPI don't allow move (undefined behavior). |
| DISALLOW_COPY_ASSIGN_AND_MOVE(StaticLock); |
| |
| AEMU_IF_DEBUG(bool mIsLocked = false;) |
| }; |
| |
| template <> |
| class StaticLock<true> { |
| public: |
| using AutoLock = gfxstream::guest::AutoLock<StaticLock>; |
| |
| StaticLock() { |
| #ifdef _WIN32 |
| ::InitializeCriticalSectionAndSpinCount(&mLock, 0x400); |
| #else |
| pthread_mutexattr_t attr; |
| pthread_mutexattr_init(&attr); |
| pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_RECURSIVE); |
| pthread_mutex_init(&mLock, &attr); |
| #endif |
| } |
| |
| // Acquire the lock. |
| void lock() { |
| #ifdef _WIN32 |
| ::EnterCriticalSection(&mLock); |
| #else |
| ::pthread_mutex_lock(&mLock); |
| #endif |
| AEMU_IF_DEBUG(mIsLocked = true;) |
| } |
| |
| bool tryLock() { |
| bool ret = false; |
| #ifdef _WIN32 |
| ret = ::TryEnterCriticalSection(&mLock); |
| #else |
| ret = ::pthread_mutex_trylock(&mLock) == 0; |
| #endif |
| AEMU_IF_DEBUG(mIsLocked = ret;) |
| return ret; |
| } |
| |
| AEMU_IF_DEBUG(bool isLocked() const { return mIsLocked; }) |
| |
| // Release the lock. |
| void unlock() { |
| AEMU_IF_DEBUG(mIsLocked = false;) |
| #ifdef _WIN32 |
| ::LeaveCriticalSection(&mLock); |
| #else |
| ::pthread_mutex_unlock(&mLock); |
| #endif |
| } |
| |
| protected: |
| friend class ConditionVariable; |
| |
| #ifdef _WIN32 |
| // We use CRITICAL_SECTION since it always allow recursive access. |
| CRITICAL_SECTION mLock; |
| #else |
| pthread_mutex_t mLock = PTHREAD_MUTEX_INITIALIZER; |
| #endif |
| // Both POSIX threads and WinAPI don't allow move (undefined behavior). |
| DISALLOW_COPY_ASSIGN_AND_MOVE(StaticLock); |
| |
| AEMU_IF_DEBUG(bool mIsLocked = false;) |
| }; |
| |
| // Simple wrapper class for mutexes used in non-static context. |
| class Lock : public StaticLock<false> { |
| public: |
| using StaticLock::AutoLock; |
| |
| constexpr Lock() = default; |
| #ifndef _WIN32 |
| // The only difference is that POSIX requires a deallocation function call |
| // for its mutexes. |
| ~Lock() { ::pthread_mutex_destroy(&mLock); } |
| #endif |
| }; |
| |
| // Simple wrapper class for mutexes used in non-static context. |
| class RecursiveLock : public StaticLock<true> { |
| public: |
| using StaticLock::AutoLock; |
| |
| RecursiveLock() = default; |
| |
| ~RecursiveLock() { |
| #ifdef _WIN32 |
| ::DeleteCriticalSection(&mLock); |
| #else |
| ::pthread_mutex_destroy(&mLock); |
| #endif |
| } |
| }; |
| |
| class ReadWriteLock { |
| public: |
| using AutoWriteLock = gfxstream::guest::AutoWriteLock; |
| using AutoReadLock = gfxstream::guest::AutoReadLock; |
| |
| #ifdef _WIN32 |
| constexpr ReadWriteLock() = default; |
| ~ReadWriteLock() = default; |
| void lockRead() { ::AcquireSRWLockShared(&mLock); } |
| void unlockRead() { ::ReleaseSRWLockShared(&mLock); } |
| void lockWrite() { ::AcquireSRWLockExclusive(&mLock); } |
| void unlockWrite() { ::ReleaseSRWLockExclusive(&mLock); } |
| |
| private: |
| SRWLOCK mLock = SRWLOCK_INIT; |
| #else // !_WIN32 |
| ReadWriteLock() { ::pthread_rwlock_init(&mLock, NULL); } |
| ~ReadWriteLock() { ::pthread_rwlock_destroy(&mLock); } |
| void lockRead() { ::pthread_rwlock_rdlock(&mLock); } |
| void unlockRead() { ::pthread_rwlock_unlock(&mLock); } |
| void lockWrite() { ::pthread_rwlock_wrlock(&mLock); } |
| void unlockWrite() { ::pthread_rwlock_unlock(&mLock); } |
| |
| private: |
| pthread_rwlock_t mLock; |
| #endif // !_WIN32 |
| |
| friend class ConditionVariable; |
| DISALLOW_COPY_ASSIGN_AND_MOVE(ReadWriteLock); |
| }; |
| |
| // Helper class to lock / unlock a mutex automatically on scope |
| // entry and exit. |
| // NB: not thread-safe (as opposed to the Lock class) |
| template <class Lockable> |
| class AutoLock { |
| public: |
| AutoLock(Lockable& lock) : mLock(lock) { mLock.lock(); } |
| |
| AutoLock(AutoLock<Lockable>&& other) : mLock(other.mLock), mLocked(other.mLocked) { |
| other.mLocked = false; |
| } |
| |
| void lock() { |
| assert(!mLocked); |
| mLock.lock(); |
| mLocked = true; |
| } |
| |
| void unlock() { |
| assert(mLocked); |
| mLock.unlock(); |
| mLocked = false; |
| } |
| |
| bool isLocked() const { return mLocked; } |
| |
| ~AutoLock() { |
| if (mLocked) { |
| mLock.unlock(); |
| } |
| } |
| |
| private: |
| Lockable& mLock; |
| bool mLocked = true; |
| |
| friend class ConditionVariable; |
| // Don't allow move because this class has a non-movable object. |
| DISALLOW_COPY_AND_ASSIGN(AutoLock); |
| }; |
| |
| class AutoWriteLock { |
| public: |
| AutoWriteLock(ReadWriteLock& lock) : mLock(lock) { mLock.lockWrite(); } |
| |
| void lockWrite() { |
| assert(!mWriteLocked); |
| mLock.lockWrite(); |
| mWriteLocked = true; |
| } |
| |
| void unlockWrite() { |
| assert(mWriteLocked); |
| mLock.unlockWrite(); |
| mWriteLocked = false; |
| } |
| |
| ~AutoWriteLock() { |
| if (mWriteLocked) { |
| mLock.unlockWrite(); |
| } |
| } |
| |
| private: |
| ReadWriteLock& mLock; |
| bool mWriteLocked = true; |
| // This class has a non-movable object. |
| DISALLOW_COPY_ASSIGN_AND_MOVE(AutoWriteLock); |
| }; |
| |
| class AutoReadLock { |
| public: |
| AutoReadLock(ReadWriteLock& lock) : mLock(lock) { mLock.lockRead(); } |
| |
| void lockRead() { |
| assert(!mReadLocked); |
| mLock.lockRead(); |
| mReadLocked = true; |
| } |
| |
| void unlockRead() { |
| assert(mReadLocked); |
| mLock.unlockRead(); |
| mReadLocked = false; |
| } |
| |
| ~AutoReadLock() { |
| if (mReadLocked) { |
| mLock.unlockRead(); |
| } |
| } |
| |
| private: |
| ReadWriteLock& mLock; |
| bool mReadLocked = true; |
| // This class has a non-movable object. |
| DISALLOW_COPY_ASSIGN_AND_MOVE(AutoReadLock); |
| }; |
| |
| } // namespace guest |
| } // namespace gfxstream |