/* * Copyright 2020 The WebRTC Project Authors. All rights reserved. * * Use of this source code is governed by a BSD-style license * that can be found in the LICENSE file in the root of the source * tree. An additional intellectual property rights grant can be found * in the file PATENTS. All contributing project authors may * be found in the AUTHORS file in the root of the source tree. */ #ifndef RTC_BASE_SYNCHRONIZATION_MUTEX_H_ #define RTC_BASE_SYNCHRONIZATION_MUTEX_H_ #include #include "absl/base/const_init.h" #include "rtc_base/checks.h" #include "rtc_base/platform_thread_types.h" #include "rtc_base/system/unused.h" #include "rtc_base/thread_annotations.h" #if defined(WEBRTC_ABSL_MUTEX) #include "rtc_base/synchronization/mutex_abseil.h" // nogncheck #elif defined(WEBRTC_WIN) #include "rtc_base/synchronization/mutex_critical_section.h" #elif defined(WEBRTC_POSIX) #include "rtc_base/synchronization/mutex_pthread.h" #else #error Unsupported platform. #endif namespace webrtc { // The Mutex guarantees exclusive access and aims to follow Abseil semantics // (i.e. non-reentrant etc). class RTC_LOCKABLE Mutex final { public: Mutex() = default; Mutex(const Mutex&) = delete; Mutex& operator=(const Mutex&) = delete; void Lock() RTC_EXCLUSIVE_LOCK_FUNCTION() { rtc::PlatformThreadRef current = CurrentThreadRefAssertingNotBeingHolder(); impl_.Lock(); // |holder_| changes from 0 to CurrentThreadRef(). holder_.store(current, std::memory_order_relaxed); } RTC_WARN_UNUSED_RESULT bool TryLock() RTC_EXCLUSIVE_TRYLOCK_FUNCTION(true) { rtc::PlatformThreadRef current = CurrentThreadRefAssertingNotBeingHolder(); if (impl_.TryLock()) { // |holder_| changes from 0 to CurrentThreadRef(). holder_.store(current, std::memory_order_relaxed); return true; } return false; } void Unlock() RTC_UNLOCK_FUNCTION() { // |holder_| changes from CurrentThreadRef() to 0. If something else than // CurrentThreadRef() is stored in |holder_|, the Unlock results in // undefined behavior as mutexes can't be unlocked from another thread than // the one that locked it, or called while not being locked. holder_.store(0, std::memory_order_relaxed); impl_.Unlock(); } private: rtc::PlatformThreadRef CurrentThreadRefAssertingNotBeingHolder() { rtc::PlatformThreadRef holder = holder_.load(std::memory_order_relaxed); rtc::PlatformThreadRef current = rtc::CurrentThreadRef(); // TODO(bugs.webrtc.org/11567): remove this temporary check after migrating // fully to Mutex. RTC_CHECK_NE(holder, current); return current; } MutexImpl impl_; // TODO(bugs.webrtc.org/11567): remove |holder_| after migrating fully to // Mutex. // |holder_| contains the PlatformThreadRef of the thread currently holding // the lock, or 0. // Remarks on the used memory orders: the atomic load in // CurrentThreadRefAssertingNotBeingHolder() observes either of two things: // 1. our own previous write to holder_ with our thread ID. // 2. another thread (with ID y) writing y and then 0 from an initial value of // 0. If we're observing case 1, our own stores are obviously ordered before // the load, and hit the CHECK. If we're observing case 2, the value observed // w.r.t |impl_| being locked depends on the memory order. Since we only care // that it's different from CurrentThreadRef()), we use the more performant // option, memory_order_relaxed. std::atomic holder_ = {0}; }; // MutexLock, for serializing execution through a scope. class RTC_SCOPED_LOCKABLE MutexLock final { public: MutexLock(const MutexLock&) = delete; MutexLock& operator=(const MutexLock&) = delete; explicit MutexLock(Mutex* mutex) RTC_EXCLUSIVE_LOCK_FUNCTION(mutex) : mutex_(mutex) { mutex->Lock(); } ~MutexLock() RTC_UNLOCK_FUNCTION() { mutex_->Unlock(); } private: Mutex* mutex_; }; // A mutex used to protect global variables. Do NOT use for other purposes. #if defined(WEBRTC_ABSL_MUTEX) using GlobalMutex = absl::Mutex; using GlobalMutexLock = absl::MutexLock; #else class RTC_LOCKABLE GlobalMutex final { public: GlobalMutex(const GlobalMutex&) = delete; GlobalMutex& operator=(const GlobalMutex&) = delete; constexpr explicit GlobalMutex(absl::ConstInitType /*unused*/) : mutex_locked_(0) {} void Lock() RTC_EXCLUSIVE_LOCK_FUNCTION(); void Unlock() RTC_UNLOCK_FUNCTION(); private: std::atomic mutex_locked_; // 0 means lock not taken, 1 means taken. }; // GlobalMutexLock, for serializing execution through a scope. class RTC_SCOPED_LOCKABLE GlobalMutexLock final { public: GlobalMutexLock(const GlobalMutexLock&) = delete; GlobalMutexLock& operator=(const GlobalMutexLock&) = delete; explicit GlobalMutexLock(GlobalMutex* mutex) RTC_EXCLUSIVE_LOCK_FUNCTION(mutex_); ~GlobalMutexLock() RTC_UNLOCK_FUNCTION(); private: GlobalMutex* mutex_; }; #endif // if defined(WEBRTC_ABSL_MUTEX) } // namespace webrtc #endif // RTC_BASE_SYNCHRONIZATION_MUTEX_H_