/* * 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. */ #include "rtc_base/thread_annotations.h" #include "test/gtest.h" namespace { class RTC_LOCKABLE Lock { public: void EnterWrite() const RTC_EXCLUSIVE_LOCK_FUNCTION() {} void EnterRead() const RTC_SHARED_LOCK_FUNCTION() {} bool TryEnterWrite() const RTC_EXCLUSIVE_TRYLOCK_FUNCTION(true) { return true; } bool TryEnterRead() const RTC_SHARED_TRYLOCK_FUNCTION(true) { return true; } void Leave() const RTC_UNLOCK_FUNCTION() {} }; class RTC_SCOPED_LOCKABLE ScopeLock { public: explicit ScopeLock(const Lock& lock) RTC_EXCLUSIVE_LOCK_FUNCTION(lock) {} ~ScopeLock() RTC_UNLOCK_FUNCTION() {} }; class ThreadSafe { public: ThreadSafe() { pt_protected_by_lock_ = new int; } ~ThreadSafe() { delete pt_protected_by_lock_; } void LockInOrder() { beforelock_.EnterWrite(); lock_.EnterWrite(); pt_lock_.EnterWrite(); pt_lock_.Leave(); lock_.Leave(); beforelock_.Leave(); } void UnprotectedFunction() RTC_LOCKS_EXCLUDED(lock_, pt_lock_) { // Can access unprotected Value. unprotected_ = 15; // Can access pointers themself, but not data they point to. int* tmp = pt_protected_by_lock_; pt_protected_by_lock_ = tmp; } void ReadProtected() { lock_.EnterRead(); unprotected_ = protected_by_lock_; lock_.Leave(); if (pt_lock_.TryEnterRead()) { unprotected_ = *pt_protected_by_lock_; pt_lock_.Leave(); } } void WriteProtected() { lock_.EnterWrite(); protected_by_lock_ = unprotected_; lock_.Leave(); if (pt_lock_.TryEnterWrite()) { *pt_protected_by_lock_ = unprotected_; pt_lock_.Leave(); } } void CallReadProtectedFunction() { lock_.EnterRead(); pt_lock_.EnterRead(); ReadProtectedFunction(); pt_lock_.Leave(); lock_.Leave(); } void CallWriteProtectedFunction() { ScopeLock scope_lock(GetLock()); ScopeLock pt_scope_lock(pt_lock_); WriteProtectedFunction(); } private: void ReadProtectedFunction() RTC_SHARED_LOCKS_REQUIRED(lock_, pt_lock_) { unprotected_ = protected_by_lock_; unprotected_ = *pt_protected_by_lock_; } void WriteProtectedFunction() RTC_EXCLUSIVE_LOCKS_REQUIRED(lock_, pt_lock_) { int x = protected_by_lock_; *pt_protected_by_lock_ = x; protected_by_lock_ = unprotected_; } const Lock& GetLock() RTC_LOCK_RETURNED(lock_) { return lock_; } Lock beforelock_ RTC_ACQUIRED_BEFORE(lock_); Lock lock_; Lock pt_lock_ RTC_ACQUIRED_AFTER(lock_); int unprotected_ = 0; int protected_by_lock_ RTC_GUARDED_BY(lock_) = 0; int* pt_protected_by_lock_ RTC_PT_GUARDED_BY(pt_lock_); }; } // namespace TEST(ThreadAnnotationsTest, Test) { // This test ensure thread annotations doesn't break compilation. // Thus no run-time expectations. ThreadSafe t; t.LockInOrder(); t.UnprotectedFunction(); t.ReadProtected(); t.WriteProtected(); t.CallReadProtectedFunction(); t.CallWriteProtectedFunction(); }