|  | /* | 
|  | *  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(); | 
|  | } |