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() {}
25 friend class base::RefCountedThreadSafe<TrivialDispatcher>;
26 virtual ~TrivialDispatcher() {}
28 virtual scoped_refptr<Dispatcher>
29 CreateEquivalentDispatcherAndCloseImplNoLock() OVERRIDE {
30 return scoped_refptr<Dispatcher>(new TrivialDispatcher());
33 DISALLOW_COPY_AND_ASSIGN(TrivialDispatcher);
36 TEST(DispatcherTest, Basic) {
37 scoped_refptr<Dispatcher> d(new TrivialDispatcher());
39 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
40 d->WriteMessage(NULL, 0, NULL, MOJO_WRITE_MESSAGE_FLAG_NONE));
41 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
42 d->ReadMessage(NULL, NULL, NULL, NULL,
43 MOJO_WRITE_MESSAGE_FLAG_NONE));
44 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
45 d->WriteData(NULL, NULL, MOJO_WRITE_DATA_FLAG_NONE));
46 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
47 d->BeginWriteData(NULL, NULL, MOJO_WRITE_DATA_FLAG_NONE));
48 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
50 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
51 d->ReadData(NULL, NULL, MOJO_READ_DATA_FLAG_NONE));
52 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
53 d->BeginReadData(NULL, NULL, MOJO_READ_DATA_FLAG_NONE));
54 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
58 EXPECT_EQ(MOJO_RESULT_FAILED_PRECONDITION,
59 d->AddWaiter(&w, MOJO_WAIT_FLAG_EVERYTHING, 0));
60 // Okay to remove even if it wasn't added (or was already removed).
64 EXPECT_EQ(MOJO_RESULT_OK, d->Close());
66 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
67 d->WriteMessage(NULL, 0, NULL, MOJO_WRITE_MESSAGE_FLAG_NONE));
68 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
69 d->ReadMessage(NULL, NULL, NULL, NULL,
70 MOJO_WRITE_MESSAGE_FLAG_NONE));
71 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
72 d->WriteData(NULL, NULL, MOJO_WRITE_DATA_FLAG_NONE));
73 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
74 d->BeginWriteData(NULL, NULL, MOJO_WRITE_DATA_FLAG_NONE));
75 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
77 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
78 d->ReadData(NULL, NULL, MOJO_READ_DATA_FLAG_NONE));
79 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
80 d->BeginReadData(NULL, NULL, MOJO_READ_DATA_FLAG_NONE));
81 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
83 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
84 d->AddWaiter(&w, MOJO_WAIT_FLAG_EVERYTHING, 0));
88 class ThreadSafetyStressThread : public base::SimpleThread {
106 ThreadSafetyStressThread(base::WaitableEvent* event,
107 scoped_refptr<Dispatcher> dispatcher,
109 : base::SimpleThread("thread_safety_stress_thread"),
111 dispatcher_(dispatcher),
114 CHECK_LT(op_, DISPATCHER_OP_COUNT);
117 virtual ~ThreadSafetyStressThread() {
122 virtual void Run() OVERRIDE {
128 MojoResult r = dispatcher_->Close();
129 EXPECT_TRUE(r == MOJO_RESULT_OK || r == MOJO_RESULT_INVALID_ARGUMENT)
134 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
135 dispatcher_->WriteMessage(NULL, 0, NULL,
136 MOJO_WRITE_MESSAGE_FLAG_NONE));
139 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
140 dispatcher_->ReadMessage(NULL, NULL, NULL, NULL,
141 MOJO_WRITE_MESSAGE_FLAG_NONE));
144 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
145 dispatcher_->WriteData(NULL, NULL,
146 MOJO_WRITE_DATA_FLAG_NONE));
148 case BEGIN_WRITE_DATA:
149 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
150 dispatcher_->BeginWriteData(NULL, NULL,
151 MOJO_WRITE_DATA_FLAG_NONE));
154 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
155 dispatcher_->EndWriteData(0));
158 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
159 dispatcher_->ReadData(NULL, NULL, MOJO_READ_DATA_FLAG_NONE));
161 case BEGIN_READ_DATA:
162 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
163 dispatcher_->BeginReadData(NULL, NULL,
164 MOJO_READ_DATA_FLAG_NONE));
167 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT,
168 dispatcher_->EndReadData(0));
171 MojoResult r = dispatcher_->AddWaiter(&waiter_,
172 MOJO_WAIT_FLAG_EVERYTHING, 0);
173 EXPECT_TRUE(r == MOJO_RESULT_FAILED_PRECONDITION ||
174 r == MOJO_RESULT_INVALID_ARGUMENT);
178 dispatcher_->RemoveWaiter(&waiter_);
185 // Always try to remove the waiter, in case we added it.
186 dispatcher_->RemoveWaiter(&waiter_);
189 base::WaitableEvent* const event_;
190 const scoped_refptr<Dispatcher> dispatcher_;
191 const DispatcherOp op_;
195 DISALLOW_COPY_AND_ASSIGN(ThreadSafetyStressThread);
198 TEST(DispatcherTest, ThreadSafetyStress) {
199 static const size_t kRepeatCount = 20;
200 static const size_t kNumThreads = 100;
202 for (size_t i = 0; i < kRepeatCount; i++) {
203 // Manual reset, not initially signalled.
204 base::WaitableEvent event(true, false);
205 scoped_refptr<Dispatcher> d(new TrivialDispatcher());
208 ScopedVector<ThreadSafetyStressThread> threads;
209 for (size_t j = 0; j < kNumThreads; j++) {
210 ThreadSafetyStressThread::DispatcherOp op =
211 static_cast<ThreadSafetyStressThread::DispatcherOp>(
212 (i+j) % ThreadSafetyStressThread::DISPATCHER_OP_COUNT);
213 threads.push_back(new ThreadSafetyStressThread(&event, d, op));
214 threads.back()->Start();
216 event.Signal(); // Kicks off real work on the threads.
217 } // Joins all the threads.
219 // One of the threads should already have closed the dispatcher.
220 EXPECT_EQ(MOJO_RESULT_INVALID_ARGUMENT, d->Close());
224 TEST(DispatcherTest, ThreadSafetyStressNoClose) {
225 static const size_t kRepeatCount = 20;
226 static const size_t kNumThreads = 100;
228 for (size_t i = 0; i < kRepeatCount; i++) {
229 // Manual reset, not initially signalled.
230 base::WaitableEvent event(true, false);
231 scoped_refptr<Dispatcher> d(new TrivialDispatcher());
234 ScopedVector<ThreadSafetyStressThread> threads;
235 for (size_t j = 0; j < kNumThreads; j++) {
236 ThreadSafetyStressThread::DispatcherOp op =
237 static_cast<ThreadSafetyStressThread::DispatcherOp>(
238 (i+j) % (ThreadSafetyStressThread::DISPATCHER_OP_COUNT-1) + 1);
239 threads.push_back(new ThreadSafetyStressThread(&event, d, op));
240 threads.back()->Start();
242 event.Signal(); // Kicks off real work on the threads.
243 } // Joins all the threads.
245 EXPECT_EQ(MOJO_RESULT_OK, d->Close());
250 } // namespace system