Fix iOS build for XCode 4.6.
[chromium-blink-merge.git] / base / synchronization / lock_unittest.cc
bloba048f8570c2eb455ee9b9f56c39e0a7e43eeeb16
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "base/synchronization/lock.h"
7 #include <stdlib.h>
9 #include "base/compiler_specific.h"
10 #include "base/threading/platform_thread.h"
11 #include "testing/gtest/include/gtest/gtest.h"
13 namespace base {
15 // Basic test to make sure that Acquire()/Release()/Try() don't crash ----------
17 class BasicLockTestThread : public PlatformThread::Delegate {
18 public:
19 BasicLockTestThread(Lock* lock) : lock_(lock), acquired_(0) {}
21 virtual void ThreadMain() OVERRIDE {
22 for (int i = 0; i < 10; i++) {
23 lock_->Acquire();
24 acquired_++;
25 lock_->Release();
27 for (int i = 0; i < 10; i++) {
28 lock_->Acquire();
29 acquired_++;
30 PlatformThread::Sleep(TimeDelta::FromMilliseconds(rand() % 20));
31 lock_->Release();
33 for (int i = 0; i < 10; i++) {
34 if (lock_->Try()) {
35 acquired_++;
36 PlatformThread::Sleep(TimeDelta::FromMilliseconds(rand() % 20));
37 lock_->Release();
42 int acquired() const { return acquired_; }
44 private:
45 Lock* lock_;
46 int acquired_;
48 DISALLOW_COPY_AND_ASSIGN(BasicLockTestThread);
51 TEST(LockTest, Basic) {
52 Lock lock;
53 BasicLockTestThread thread(&lock);
54 PlatformThreadHandle handle = kNullThreadHandle;
56 ASSERT_TRUE(PlatformThread::Create(0, &thread, &handle));
58 int acquired = 0;
59 for (int i = 0; i < 5; i++) {
60 lock.Acquire();
61 acquired++;
62 lock.Release();
64 for (int i = 0; i < 10; i++) {
65 lock.Acquire();
66 acquired++;
67 PlatformThread::Sleep(TimeDelta::FromMilliseconds(rand() % 20));
68 lock.Release();
70 for (int i = 0; i < 10; i++) {
71 if (lock.Try()) {
72 acquired++;
73 PlatformThread::Sleep(TimeDelta::FromMilliseconds(rand() % 20));
74 lock.Release();
77 for (int i = 0; i < 5; i++) {
78 lock.Acquire();
79 acquired++;
80 PlatformThread::Sleep(TimeDelta::FromMilliseconds(rand() % 20));
81 lock.Release();
84 PlatformThread::Join(handle);
86 EXPECT_GE(acquired, 20);
87 EXPECT_GE(thread.acquired(), 20);
90 // Test that Try() works as expected -------------------------------------------
92 class TryLockTestThread : public PlatformThread::Delegate {
93 public:
94 TryLockTestThread(Lock* lock) : lock_(lock), got_lock_(false) {}
96 virtual void ThreadMain() OVERRIDE {
97 got_lock_ = lock_->Try();
98 if (got_lock_)
99 lock_->Release();
102 bool got_lock() const { return got_lock_; }
104 private:
105 Lock* lock_;
106 bool got_lock_;
108 DISALLOW_COPY_AND_ASSIGN(TryLockTestThread);
111 TEST(LockTest, TryLock) {
112 Lock lock;
114 ASSERT_TRUE(lock.Try());
115 // We now have the lock....
117 // This thread will not be able to get the lock.
119 TryLockTestThread thread(&lock);
120 PlatformThreadHandle handle = kNullThreadHandle;
122 ASSERT_TRUE(PlatformThread::Create(0, &thread, &handle));
124 PlatformThread::Join(handle);
126 ASSERT_FALSE(thread.got_lock());
129 lock.Release();
131 // This thread will....
133 TryLockTestThread thread(&lock);
134 PlatformThreadHandle handle = kNullThreadHandle;
136 ASSERT_TRUE(PlatformThread::Create(0, &thread, &handle));
138 PlatformThread::Join(handle);
140 ASSERT_TRUE(thread.got_lock());
141 // But it released it....
142 ASSERT_TRUE(lock.Try());
145 lock.Release();
148 // Tests that locks actually exclude -------------------------------------------
150 class MutexLockTestThread : public PlatformThread::Delegate {
151 public:
152 MutexLockTestThread(Lock* lock, int* value) : lock_(lock), value_(value) {}
154 // Static helper which can also be called from the main thread.
155 static void DoStuff(Lock* lock, int* value) {
156 for (int i = 0; i < 40; i++) {
157 lock->Acquire();
158 int v = *value;
159 PlatformThread::Sleep(TimeDelta::FromMilliseconds(rand() % 10));
160 *value = v + 1;
161 lock->Release();
165 virtual void ThreadMain() OVERRIDE {
166 DoStuff(lock_, value_);
169 private:
170 Lock* lock_;
171 int* value_;
173 DISALLOW_COPY_AND_ASSIGN(MutexLockTestThread);
176 TEST(LockTest, MutexTwoThreads) {
177 Lock lock;
178 int value = 0;
180 MutexLockTestThread thread(&lock, &value);
181 PlatformThreadHandle handle = kNullThreadHandle;
183 ASSERT_TRUE(PlatformThread::Create(0, &thread, &handle));
185 MutexLockTestThread::DoStuff(&lock, &value);
187 PlatformThread::Join(handle);
189 EXPECT_EQ(2 * 40, value);
192 TEST(LockTest, MutexFourThreads) {
193 Lock lock;
194 int value = 0;
196 MutexLockTestThread thread1(&lock, &value);
197 MutexLockTestThread thread2(&lock, &value);
198 MutexLockTestThread thread3(&lock, &value);
199 PlatformThreadHandle handle1 = kNullThreadHandle;
200 PlatformThreadHandle handle2 = kNullThreadHandle;
201 PlatformThreadHandle handle3 = kNullThreadHandle;
203 ASSERT_TRUE(PlatformThread::Create(0, &thread1, &handle1));
204 ASSERT_TRUE(PlatformThread::Create(0, &thread2, &handle2));
205 ASSERT_TRUE(PlatformThread::Create(0, &thread3, &handle3));
207 MutexLockTestThread::DoStuff(&lock, &value);
209 PlatformThread::Join(handle1);
210 PlatformThread::Join(handle2);
211 PlatformThread::Join(handle3);
213 EXPECT_EQ(4 * 40, value);
216 } // namespace base