147 lines
5.0 KiB
C++
147 lines
5.0 KiB
C++
/*
|
|
* 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 <atomic>
|
|
|
|
#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<rtc::PlatformThreadRef> 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<int> 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_
|