79 lines
2.3 KiB
C
79 lines
2.3 KiB
C
|
/*
|
||
|
* Copyright (c) 2016 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_RACE_CHECKER_H_
|
||
|
#define RTC_BASE_RACE_CHECKER_H_
|
||
|
|
||
|
#include "rtc_base/checks.h"
|
||
|
#include "rtc_base/platform_thread_types.h"
|
||
|
#include "rtc_base/thread_annotations.h"
|
||
|
|
||
|
namespace rtc {
|
||
|
|
||
|
namespace internal {
|
||
|
class RaceCheckerScope;
|
||
|
} // namespace internal
|
||
|
|
||
|
// Best-effort race-checking implementation. This primitive uses no
|
||
|
// synchronization at all to be as-fast-as-possible in the non-racy case.
|
||
|
class RTC_LOCKABLE RaceChecker {
|
||
|
public:
|
||
|
friend class internal::RaceCheckerScope;
|
||
|
RaceChecker();
|
||
|
|
||
|
private:
|
||
|
bool Acquire() const RTC_EXCLUSIVE_LOCK_FUNCTION();
|
||
|
void Release() const RTC_UNLOCK_FUNCTION();
|
||
|
|
||
|
// Volatile to prevent code being optimized away in Acquire()/Release().
|
||
|
mutable volatile int access_count_ = 0;
|
||
|
mutable volatile PlatformThreadRef accessing_thread_;
|
||
|
};
|
||
|
|
||
|
namespace internal {
|
||
|
class RTC_SCOPED_LOCKABLE RaceCheckerScope {
|
||
|
public:
|
||
|
explicit RaceCheckerScope(const RaceChecker* race_checker)
|
||
|
RTC_EXCLUSIVE_LOCK_FUNCTION(race_checker);
|
||
|
|
||
|
bool RaceDetected() const;
|
||
|
~RaceCheckerScope() RTC_UNLOCK_FUNCTION();
|
||
|
|
||
|
private:
|
||
|
const RaceChecker* const race_checker_;
|
||
|
const bool race_check_ok_;
|
||
|
};
|
||
|
|
||
|
class RTC_SCOPED_LOCKABLE RaceCheckerScopeDoNothing {
|
||
|
public:
|
||
|
explicit RaceCheckerScopeDoNothing(const RaceChecker* race_checker)
|
||
|
RTC_EXCLUSIVE_LOCK_FUNCTION(race_checker) {}
|
||
|
|
||
|
~RaceCheckerScopeDoNothing() RTC_UNLOCK_FUNCTION() {}
|
||
|
};
|
||
|
|
||
|
} // namespace internal
|
||
|
} // namespace rtc
|
||
|
|
||
|
#define RTC_CHECK_RUNS_SERIALIZED(x) \
|
||
|
rtc::internal::RaceCheckerScope race_checker(x); \
|
||
|
RTC_CHECK(!race_checker.RaceDetected())
|
||
|
|
||
|
#if RTC_DCHECK_IS_ON
|
||
|
#define RTC_DCHECK_RUNS_SERIALIZED(x) \
|
||
|
rtc::internal::RaceCheckerScope race_checker(x); \
|
||
|
RTC_DCHECK(!race_checker.RaceDetected())
|
||
|
#else
|
||
|
#define RTC_DCHECK_RUNS_SERIALIZED(x) \
|
||
|
rtc::internal::RaceCheckerScopeDoNothing race_checker(x)
|
||
|
#endif
|
||
|
|
||
|
#endif // RTC_BASE_RACE_CHECKER_H_
|