// 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" #include "aemu/base/ThreadAnnotations.h" #include #ifdef _WIN32 #define WIN32_LEAN_AND_MEAN 1 #include #else #include #endif #include namespace android { namespace base { 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 for scoped or // member locks. class CAPABILITY("mutex") StaticLock { public: using AutoLock = android::base::AutoLock; constexpr StaticLock() = default; // Acquire the lock. void lock() ACQUIRE() { #ifdef _WIN32 ::AcquireSRWLockExclusive(&mLock); #else ::pthread_mutex_lock(&mLock); #endif } bool tryLock() TRY_ACQUIRE(true) { bool ret = false; #ifdef _WIN32 ret = ::TryAcquireSRWLockExclusive(&mLock); #else ret = ::pthread_mutex_trylock(&mLock) == 0; #endif return ret; } // Release the lock. void unlock() RELEASE() { #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); }; // Simple wrapper class for mutexes used in non-static context. class Lock : public StaticLock { 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 }; class ReadWriteLock { public: using AutoWriteLock = android::base::AutoWriteLock; using AutoReadLock = android::base::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) class SCOPED_CAPABILITY AutoLock { public: AutoLock(StaticLock& lock) ACQUIRE(mLock) : mLock(lock) { mLock.lock(); } AutoLock(AutoLock&& other) : mLock(other.mLock), mLocked(other.mLocked) { other.mLocked = false; } void lock() ACQUIRE(mLock) { assert(!mLocked); mLock.lock(); mLocked = true; } void unlock() RELEASE(mLock) { assert(mLocked); mLock.unlock(); mLocked = false; } bool isLocked() const { return mLocked; } ~AutoLock() RELEASE() { if (mLocked) { mLock.unlock(); } } private: StaticLock& 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); }; // Sequence lock implementation // See https://en.wikipedia.org/wiki/Seqlock for more info static inline __attribute__((always_inline)) void SmpWmb() { #if defined(__aarch64__) asm volatile("dmb ishst" ::: "memory"); #elif defined(__x86_64__) std::atomic_thread_fence(std::memory_order_release); #elif defined(__riscv) && (__riscv_xlen == 64) std::atomic_thread_fence(std::memory_order_release); #else #error "Unimplemented SmpWmb for current CPU architecture" #endif } static inline __attribute__((always_inline)) void SmpRmb() { #if defined(__aarch64__) asm volatile("dmb ishld" ::: "memory"); #elif defined(__x86_64__) std::atomic_thread_fence(std::memory_order_acquire); #elif defined(__riscv) && (__riscv_xlen == 64) std::atomic_thread_fence(std::memory_order_acquire); #else #error "Unimplemented SmpRmb for current CPU architecture" #endif } class SeqLock { public: void beginWrite() ACQUIRE(mWriteLock) { mWriteLock.lock(); mSeq.fetch_add(1, std::memory_order_release); SmpWmb(); } void endWrite() RELEASE(mWriteLock) { SmpWmb(); mSeq.fetch_add(1, std::memory_order_release); mWriteLock.unlock(); } #ifdef __cplusplus # define SEQLOCK_LIKELY( exp ) (__builtin_expect( !!(exp), true )) # define SEQLOCK_UNLIKELY( exp ) (__builtin_expect( !!(exp), false )) #else # define SEQLOCK_LIKELY( exp ) (__builtin_expect( !!(exp), 1 )) # define SEQLOCK_UNLIKELY( exp ) (__builtin_expect( !!(exp), 0 )) #endif uint32_t beginRead() { uint32_t res; repeat: res = mSeq.load(std::memory_order_acquire); if (SEQLOCK_UNLIKELY(res & 1)) { goto repeat; } SmpRmb(); return res; } bool shouldRetryRead(uint32_t prevSeq) { SmpRmb(); uint32_t res = mSeq.load(std::memory_order_acquire); return (res != prevSeq); } // Convenience class for write class ScopedWrite { public: ScopedWrite(SeqLock* lock) : mLock(lock) { mLock->beginWrite(); } ~ScopedWrite() { mLock->endWrite(); } private: SeqLock* mLock; }; // Convenience macro for read (no std::function due to its considerable overhead) #define AEMU_SEQLOCK_READ_WITH_RETRY(lock, readStuff) { uint32_t aemu_seqlock_curr_seq; do { \ aemu_seqlock_curr_seq = (lock)->beginRead(); \ readStuff; \ } while ((lock)->shouldRetryRead(aemu_seqlock_curr_seq)); } private: std::atomic mSeq { 0 }; // The sequence number Lock mWriteLock; // Just use a normal mutex to protect writes }; } // namespace base } // namespace android