You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
132 lines
3.3 KiB
132 lines
3.3 KiB
/*
|
|
* 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();
|
|
}
|