1 // Copyright 2013 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 "mojo/system/dispatcher.h"
7 #include "base/basictypes.h"
8 #include "base/memory/ref_counted.h"
9 #include "base/memory/scoped_vector.h"
10 #include "base/synchronization/waitable_event.h"
11 #include "base/threading/simple_thread.h"
12 #include "mojo/system/waiter.h"
13 #include "testing/gtest/include/gtest/gtest.h"
19 // Trivial subclass that makes the constructor public.
20 class TrivialDispatcher : public Dispatcher {
22 TrivialDispatcher() {}
24 virtual Type GetType() const OVERRIDE {
29 friend class base::RefCountedThreadSafe<TrivialDispatcher>;
30 virtual ~TrivialDispatcher() {}
32 virtual scoped_refptr<Dispatcher>
33 CreateEquivalentDispatcherAndCloseImplNoLock() OVERRIDE {
34 lock().AssertAcquired();
35 return scoped_refptr<Dispatcher>(new TrivialDispatcher());
38 DISALLOW_COPY_AND_ASSIGN(TrivialDispatcher);
41 TEST(DispatcherTest, Basic) {
42 scoped_refptr<Dispatcher> d(new TrivialDispatcher());
44 EXPECT_EQ(Dispatcher::kTypeUnknown, d->GetType());
46 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
47 d->WriteMessage(NULL, 0, NULL, MOJO_WRITE_MESSAGE_FLAG_NONE));
48 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
49 d->ReadMessage(NULL, NULL, NULL, NULL,
50 MOJO_WRITE_MESSAGE_FLAG_NONE));
51 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
52 d->WriteData(NULL, NULL, MOJO_WRITE_DATA_FLAG_NONE));
53 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
54 d->BeginWriteData(NULL, NULL, MOJO_WRITE_DATA_FLAG_NONE));
55 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
57 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
58 d->ReadData(NULL, NULL, MOJO_READ_DATA_FLAG_NONE));
59 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
60 d->BeginReadData(NULL, NULL, MOJO_READ_DATA_FLAG_NONE));
61 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
65 EXPECT_EQ(MOJO_RESULT_FAILED_PRECONDITION,
66 d->AddWaiter(&w, MOJO_WAIT_FLAG_EVERYTHING, 0));
67 // Okay to remove even if it wasn't added (or was already removed).
71 EXPECT_EQ(MOJO_RESULT_OK, d->Close());
73 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
74 d->WriteMessage(NULL, 0, NULL, MOJO_WRITE_MESSAGE_FLAG_NONE));
75 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
76 d->ReadMessage(NULL, NULL, NULL, NULL,
77 MOJO_WRITE_MESSAGE_FLAG_NONE));
78 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
79 d->WriteData(NULL, NULL, MOJO_WRITE_DATA_FLAG_NONE));
80 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
81 d->BeginWriteData(NULL, NULL, MOJO_WRITE_DATA_FLAG_NONE));
82 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
84 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
85 d->ReadData(NULL, NULL, MOJO_READ_DATA_FLAG_NONE));
86 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
87 d->BeginReadData(NULL, NULL, MOJO_READ_DATA_FLAG_NONE));
88 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
90 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
91 d->AddWaiter(&w, MOJO_WAIT_FLAG_EVERYTHING, 0));
95 class ThreadSafetyStressThread : public base::SimpleThread {
113 ThreadSafetyStressThread(base::WaitableEvent* event,
114 scoped_refptr<Dispatcher> dispatcher,
116 : base::SimpleThread("thread_safety_stress_thread"),
118 dispatcher_(dispatcher),
121 CHECK_LT(op_, DISPATCHER_OP_COUNT);
124 virtual ~ThreadSafetyStressThread() {
129 virtual void Run() OVERRIDE {
135 MojoResult r = dispatcher_->Close();
136 EXPECT_TRUE(r == MOJO_RESULT_OK || r == MOJO_RESULT_INVALID_ARGUMENT)
141 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
142 dispatcher_->WriteMessage(NULL, 0, NULL,
143 MOJO_WRITE_MESSAGE_FLAG_NONE));
146 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
147 dispatcher_->ReadMessage(NULL, NULL, NULL, NULL,
148 MOJO_WRITE_MESSAGE_FLAG_NONE));
151 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
152 dispatcher_->WriteData(NULL, NULL,
153 MOJO_WRITE_DATA_FLAG_NONE));
155 case BEGIN_WRITE_DATA:
156 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
157 dispatcher_->BeginWriteData(NULL, NULL,
158 MOJO_WRITE_DATA_FLAG_NONE));
161 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
162 dispatcher_->EndWriteData(0));
165 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
166 dispatcher_->ReadData(NULL, NULL, MOJO_READ_DATA_FLAG_NONE));
168 case BEGIN_READ_DATA:
169 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
170 dispatcher_->BeginReadData(NULL, NULL,
171 MOJO_READ_DATA_FLAG_NONE));
174 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
175 dispatcher_->EndReadData(0));
178 MojoResult r = dispatcher_->AddWaiter(&waiter_,
179 MOJO_WAIT_FLAG_EVERYTHING, 0);
180 EXPECT_TRUE(r == MOJO_RESULT_FAILED_PRECONDITION ||
181 r == MOJO_RESULT_INVALID_ARGUMENT);
185 dispatcher_->RemoveWaiter(&waiter_);
192 // Always try to remove the waiter, in case we added it.
193 dispatcher_->RemoveWaiter(&waiter_);
196 base::WaitableEvent* const event_;
197 const scoped_refptr<Dispatcher> dispatcher_;
198 const DispatcherOp op_;
202 DISALLOW_COPY_AND_ASSIGN(ThreadSafetyStressThread);
205 TEST(DispatcherTest, ThreadSafetyStress) {
206 static const size_t kRepeatCount = 20;
207 static const size_t kNumThreads = 100;
209 for (size_t i = 0; i < kRepeatCount; i++) {
210 // Manual reset, not initially signalled.
211 base::WaitableEvent event(true, false);
212 scoped_refptr<Dispatcher> d(new TrivialDispatcher());
215 ScopedVector<ThreadSafetyStressThread> threads;
216 for (size_t j = 0; j < kNumThreads; j++) {
217 ThreadSafetyStressThread::DispatcherOp op =
218 static_cast<ThreadSafetyStressThread::DispatcherOp>(
219 (i+j) % ThreadSafetyStressThread::DISPATCHER_OP_COUNT);
220 threads.push_back(new ThreadSafetyStressThread(&event, d, op));
221 threads.back()->Start();
223 event.Signal(); // Kicks off real work on the threads.
224 } // Joins all the threads.
226 // One of the threads should already have closed the dispatcher.
227 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT, d->Close());
231 TEST(DispatcherTest, ThreadSafetyStressNoClose) {
232 static const size_t kRepeatCount = 20;
233 static const size_t kNumThreads = 100;
235 for (size_t i = 0; i < kRepeatCount; i++) {
236 // Manual reset, not initially signalled.
237 base::WaitableEvent event(true, false);
238 scoped_refptr<Dispatcher> d(new TrivialDispatcher());
241 ScopedVector<ThreadSafetyStressThread> threads;
242 for (size_t j = 0; j < kNumThreads; j++) {
243 ThreadSafetyStressThread::DispatcherOp op =
244 static_cast<ThreadSafetyStressThread::DispatcherOp>(
245 (i+j) % (ThreadSafetyStressThread::DISPATCHER_OP_COUNT-1) + 1);
246 threads.push_back(new ThreadSafetyStressThread(&event, d, op));
247 threads.back()->Start();
249 event.Signal(); // Kicks off real work on the threads.
250 } // Joins all the threads.
252 EXPECT_EQ(MOJO_RESULT_OK, d->Close());
257 } // namespace system