1*d9f75844SAndroid Build Coastguard Worker /* 2*d9f75844SAndroid Build Coastguard Worker * Copyright (c) 2016 The WebRTC project authors. All Rights Reserved. 3*d9f75844SAndroid Build Coastguard Worker * 4*d9f75844SAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license 5*d9f75844SAndroid Build Coastguard Worker * that can be found in the LICENSE file in the root of the source 6*d9f75844SAndroid Build Coastguard Worker * tree. An additional intellectual property rights grant can be found 7*d9f75844SAndroid Build Coastguard Worker * in the file PATENTS. All contributing project authors may 8*d9f75844SAndroid Build Coastguard Worker * be found in the AUTHORS file in the root of the source tree. 9*d9f75844SAndroid Build Coastguard Worker */ 10*d9f75844SAndroid Build Coastguard Worker 11*d9f75844SAndroid Build Coastguard Worker #ifndef RTC_BASE_RACE_CHECKER_H_ 12*d9f75844SAndroid Build Coastguard Worker #define RTC_BASE_RACE_CHECKER_H_ 13*d9f75844SAndroid Build Coastguard Worker 14*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/checks.h" 15*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/platform_thread_types.h" 16*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/thread_annotations.h" 17*d9f75844SAndroid Build Coastguard Worker 18*d9f75844SAndroid Build Coastguard Worker namespace rtc { 19*d9f75844SAndroid Build Coastguard Worker 20*d9f75844SAndroid Build Coastguard Worker namespace internal { 21*d9f75844SAndroid Build Coastguard Worker class RaceCheckerScope; 22*d9f75844SAndroid Build Coastguard Worker } // namespace internal 23*d9f75844SAndroid Build Coastguard Worker 24*d9f75844SAndroid Build Coastguard Worker // Best-effort race-checking implementation. This primitive uses no 25*d9f75844SAndroid Build Coastguard Worker // synchronization at all to be as-fast-as-possible in the non-racy case. 26*d9f75844SAndroid Build Coastguard Worker class RTC_LOCKABLE RaceChecker { 27*d9f75844SAndroid Build Coastguard Worker public: 28*d9f75844SAndroid Build Coastguard Worker friend class internal::RaceCheckerScope; 29*d9f75844SAndroid Build Coastguard Worker RaceChecker(); 30*d9f75844SAndroid Build Coastguard Worker 31*d9f75844SAndroid Build Coastguard Worker private: 32*d9f75844SAndroid Build Coastguard Worker bool Acquire() const RTC_EXCLUSIVE_LOCK_FUNCTION(); 33*d9f75844SAndroid Build Coastguard Worker void Release() const RTC_UNLOCK_FUNCTION(); 34*d9f75844SAndroid Build Coastguard Worker 35*d9f75844SAndroid Build Coastguard Worker // Volatile to prevent code being optimized away in Acquire()/Release(). 36*d9f75844SAndroid Build Coastguard Worker mutable volatile int access_count_ = 0; 37*d9f75844SAndroid Build Coastguard Worker mutable volatile PlatformThreadRef accessing_thread_; 38*d9f75844SAndroid Build Coastguard Worker }; 39*d9f75844SAndroid Build Coastguard Worker 40*d9f75844SAndroid Build Coastguard Worker namespace internal { 41*d9f75844SAndroid Build Coastguard Worker class RTC_SCOPED_LOCKABLE RaceCheckerScope { 42*d9f75844SAndroid Build Coastguard Worker public: 43*d9f75844SAndroid Build Coastguard Worker explicit RaceCheckerScope(const RaceChecker* race_checker) 44*d9f75844SAndroid Build Coastguard Worker RTC_EXCLUSIVE_LOCK_FUNCTION(race_checker); 45*d9f75844SAndroid Build Coastguard Worker 46*d9f75844SAndroid Build Coastguard Worker bool RaceDetected() const; 47*d9f75844SAndroid Build Coastguard Worker ~RaceCheckerScope() RTC_UNLOCK_FUNCTION(); 48*d9f75844SAndroid Build Coastguard Worker 49*d9f75844SAndroid Build Coastguard Worker private: 50*d9f75844SAndroid Build Coastguard Worker const RaceChecker* const race_checker_; 51*d9f75844SAndroid Build Coastguard Worker const bool race_check_ok_; 52*d9f75844SAndroid Build Coastguard Worker }; 53*d9f75844SAndroid Build Coastguard Worker 54*d9f75844SAndroid Build Coastguard Worker class RTC_SCOPED_LOCKABLE RaceCheckerScopeDoNothing { 55*d9f75844SAndroid Build Coastguard Worker public: RaceCheckerScopeDoNothing(const RaceChecker * race_checker)56*d9f75844SAndroid Build Coastguard Worker explicit RaceCheckerScopeDoNothing(const RaceChecker* race_checker) 57*d9f75844SAndroid Build Coastguard Worker RTC_EXCLUSIVE_LOCK_FUNCTION(race_checker) {} 58*d9f75844SAndroid Build Coastguard Worker RTC_UNLOCK_FUNCTION()59*d9f75844SAndroid Build Coastguard Worker ~RaceCheckerScopeDoNothing() RTC_UNLOCK_FUNCTION() {} 60*d9f75844SAndroid Build Coastguard Worker }; 61*d9f75844SAndroid Build Coastguard Worker 62*d9f75844SAndroid Build Coastguard Worker } // namespace internal 63*d9f75844SAndroid Build Coastguard Worker } // namespace rtc 64*d9f75844SAndroid Build Coastguard Worker 65*d9f75844SAndroid Build Coastguard Worker #define RTC_CHECK_RUNS_SERIALIZED(x) \ 66*d9f75844SAndroid Build Coastguard Worker rtc::internal::RaceCheckerScope race_checker(x); \ 67*d9f75844SAndroid Build Coastguard Worker RTC_CHECK(!race_checker.RaceDetected()) 68*d9f75844SAndroid Build Coastguard Worker 69*d9f75844SAndroid Build Coastguard Worker #if RTC_DCHECK_IS_ON 70*d9f75844SAndroid Build Coastguard Worker #define RTC_DCHECK_RUNS_SERIALIZED(x) \ 71*d9f75844SAndroid Build Coastguard Worker rtc::internal::RaceCheckerScope race_checker(x); \ 72*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK(!race_checker.RaceDetected()) 73*d9f75844SAndroid Build Coastguard Worker #else 74*d9f75844SAndroid Build Coastguard Worker #define RTC_DCHECK_RUNS_SERIALIZED(x) \ 75*d9f75844SAndroid Build Coastguard Worker rtc::internal::RaceCheckerScopeDoNothing race_checker(x) 76*d9f75844SAndroid Build Coastguard Worker #endif 77*d9f75844SAndroid Build Coastguard Worker 78*d9f75844SAndroid Build Coastguard Worker #endif // RTC_BASE_RACE_CHECKER_H_ 79