OLD | NEW |
| (Empty) |
1 // Copyright 2014 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. | |
4 | |
5 #include "mojo/edk/system/channel.h" | |
6 | |
7 #include "base/bind.h" | |
8 #include "base/location.h" | |
9 #include "base/message_loop/message_loop.h" | |
10 #include "base/test/test_io_thread.h" | |
11 #include "mojo/edk/embedder/platform_channel_pair.h" | |
12 #include "mojo/edk/embedder/simple_platform_support.h" | |
13 #include "mojo/edk/system/channel_endpoint.h" | |
14 #include "mojo/edk/system/message_in_transit.h" | |
15 #include "mojo/edk/system/message_pipe.h" | |
16 #include "mojo/edk/system/raw_channel.h" | |
17 #include "mojo/edk/system/test_utils.h" | |
18 #include "mojo/edk/system/waiter.h" | |
19 #include "testing/gtest/include/gtest/gtest.h" | |
20 | |
21 namespace mojo { | |
22 namespace system { | |
23 namespace { | |
24 | |
25 enum Tristate { TRISTATE_UNKNOWN = -1, TRISTATE_FALSE = 0, TRISTATE_TRUE = 1 }; | |
26 | |
27 Tristate BoolToTristate(bool b) { | |
28 return b ? TRISTATE_TRUE : TRISTATE_FALSE; | |
29 } | |
30 | |
31 class ChannelTest : public testing::Test { | |
32 public: | |
33 ChannelTest() | |
34 : io_thread_(base::TestIOThread::kAutoStart), | |
35 init_result_(TRISTATE_UNKNOWN) {} | |
36 virtual ~ChannelTest() {} | |
37 | |
38 virtual void SetUp() override { | |
39 io_thread_.PostTaskAndWait( | |
40 FROM_HERE, | |
41 base::Bind(&ChannelTest::SetUpOnIOThread, base::Unretained(this))); | |
42 } | |
43 | |
44 void CreateChannelOnIOThread() { | |
45 CHECK_EQ(base::MessageLoop::current(), io_thread()->message_loop()); | |
46 channel_ = new Channel(&platform_support_); | |
47 } | |
48 | |
49 void InitChannelOnIOThread() { | |
50 CHECK_EQ(base::MessageLoop::current(), io_thread()->message_loop()); | |
51 | |
52 CHECK(raw_channel_); | |
53 CHECK(channel_.get()); | |
54 CHECK_EQ(init_result_, TRISTATE_UNKNOWN); | |
55 | |
56 init_result_ = BoolToTristate(channel_->Init(raw_channel_.Pass())); | |
57 } | |
58 | |
59 void ShutdownChannelOnIOThread() { | |
60 CHECK_EQ(base::MessageLoop::current(), io_thread()->message_loop()); | |
61 | |
62 CHECK(channel_.get()); | |
63 channel_->Shutdown(); | |
64 } | |
65 | |
66 base::TestIOThread* io_thread() { return &io_thread_; } | |
67 RawChannel* raw_channel() { return raw_channel_.get(); } | |
68 scoped_ptr<RawChannel>* mutable_raw_channel() { return &raw_channel_; } | |
69 Channel* channel() { return channel_.get(); } | |
70 scoped_refptr<Channel>* mutable_channel() { return &channel_; } | |
71 Tristate init_result() const { return init_result_; } | |
72 | |
73 private: | |
74 void SetUpOnIOThread() { | |
75 CHECK_EQ(base::MessageLoop::current(), io_thread()->message_loop()); | |
76 | |
77 embedder::PlatformChannelPair channel_pair; | |
78 raw_channel_ = RawChannel::Create(channel_pair.PassServerHandle()).Pass(); | |
79 other_platform_handle_ = channel_pair.PassClientHandle(); | |
80 } | |
81 | |
82 embedder::SimplePlatformSupport platform_support_; | |
83 base::TestIOThread io_thread_; | |
84 scoped_ptr<RawChannel> raw_channel_; | |
85 embedder::ScopedPlatformHandle other_platform_handle_; | |
86 scoped_refptr<Channel> channel_; | |
87 | |
88 Tristate init_result_; | |
89 | |
90 DISALLOW_COPY_AND_ASSIGN(ChannelTest); | |
91 }; | |
92 | |
93 // ChannelTest.InitShutdown ---------------------------------------------------- | |
94 | |
95 TEST_F(ChannelTest, InitShutdown) { | |
96 io_thread()->PostTaskAndWait(FROM_HERE, | |
97 base::Bind(&ChannelTest::CreateChannelOnIOThread, | |
98 base::Unretained(this))); | |
99 ASSERT_TRUE(channel()); | |
100 | |
101 io_thread()->PostTaskAndWait( | |
102 FROM_HERE, | |
103 base::Bind(&ChannelTest::InitChannelOnIOThread, base::Unretained(this))); | |
104 EXPECT_EQ(TRISTATE_TRUE, init_result()); | |
105 | |
106 io_thread()->PostTaskAndWait( | |
107 FROM_HERE, | |
108 base::Bind(&ChannelTest::ShutdownChannelOnIOThread, | |
109 base::Unretained(this))); | |
110 | |
111 // Okay to destroy |Channel| on not-the-I/O-thread. | |
112 EXPECT_TRUE(channel()->HasOneRef()); | |
113 *mutable_channel() = nullptr; | |
114 } | |
115 | |
116 // ChannelTest.InitFails ------------------------------------------------------- | |
117 | |
118 class MockRawChannelOnInitFails : public RawChannel { | |
119 public: | |
120 MockRawChannelOnInitFails() : on_init_called_(false) {} | |
121 virtual ~MockRawChannelOnInitFails() {} | |
122 | |
123 // |RawChannel| public methods: | |
124 virtual size_t GetSerializedPlatformHandleSize() const override { return 0; } | |
125 | |
126 private: | |
127 // |RawChannel| protected methods: | |
128 virtual IOResult Read(size_t*) override { | |
129 CHECK(false); | |
130 return IO_FAILED_UNKNOWN; | |
131 } | |
132 virtual IOResult ScheduleRead() override { | |
133 CHECK(false); | |
134 return IO_FAILED_UNKNOWN; | |
135 } | |
136 virtual embedder::ScopedPlatformHandleVectorPtr GetReadPlatformHandles( | |
137 size_t, | |
138 const void*) override { | |
139 CHECK(false); | |
140 return embedder::ScopedPlatformHandleVectorPtr(); | |
141 } | |
142 virtual IOResult WriteNoLock(size_t*, size_t*) override { | |
143 CHECK(false); | |
144 return IO_FAILED_UNKNOWN; | |
145 } | |
146 virtual IOResult ScheduleWriteNoLock() override { | |
147 CHECK(false); | |
148 return IO_FAILED_UNKNOWN; | |
149 } | |
150 virtual bool OnInit() override { | |
151 EXPECT_FALSE(on_init_called_); | |
152 on_init_called_ = true; | |
153 return false; | |
154 } | |
155 virtual void OnShutdownNoLock(scoped_ptr<ReadBuffer>, | |
156 scoped_ptr<WriteBuffer>) override { | |
157 CHECK(false); | |
158 } | |
159 | |
160 bool on_init_called_; | |
161 | |
162 DISALLOW_COPY_AND_ASSIGN(MockRawChannelOnInitFails); | |
163 }; | |
164 | |
165 TEST_F(ChannelTest, InitFails) { | |
166 io_thread()->PostTaskAndWait(FROM_HERE, | |
167 base::Bind(&ChannelTest::CreateChannelOnIOThread, | |
168 base::Unretained(this))); | |
169 ASSERT_TRUE(channel()); | |
170 | |
171 ASSERT_TRUE(raw_channel()); | |
172 mutable_raw_channel()->reset(new MockRawChannelOnInitFails()); | |
173 | |
174 io_thread()->PostTaskAndWait( | |
175 FROM_HERE, | |
176 base::Bind(&ChannelTest::InitChannelOnIOThread, base::Unretained(this))); | |
177 EXPECT_EQ(TRISTATE_FALSE, init_result()); | |
178 | |
179 // Should destroy |Channel| with no |Shutdown()| (on not-the-I/O-thread). | |
180 EXPECT_TRUE(channel()->HasOneRef()); | |
181 *mutable_channel() = nullptr; | |
182 } | |
183 | |
184 // ChannelTest.CloseBeforeRun -------------------------------------------------- | |
185 | |
186 TEST_F(ChannelTest, CloseBeforeRun) { | |
187 io_thread()->PostTaskAndWait(FROM_HERE, | |
188 base::Bind(&ChannelTest::CreateChannelOnIOThread, | |
189 base::Unretained(this))); | |
190 ASSERT_TRUE(channel()); | |
191 | |
192 io_thread()->PostTaskAndWait( | |
193 FROM_HERE, | |
194 base::Bind(&ChannelTest::InitChannelOnIOThread, base::Unretained(this))); | |
195 EXPECT_EQ(TRISTATE_TRUE, init_result()); | |
196 | |
197 scoped_refptr<ChannelEndpoint> channel_endpoint; | |
198 scoped_refptr<MessagePipe> mp( | |
199 MessagePipe::CreateLocalProxy(&channel_endpoint)); | |
200 | |
201 MessageInTransit::EndpointId local_id = | |
202 channel()->AttachEndpoint(channel_endpoint); | |
203 EXPECT_EQ(Channel::kBootstrapEndpointId, local_id); | |
204 | |
205 mp->Close(0); | |
206 | |
207 // TODO(vtl): Currently, the |Close()| above won't detach (since it thinks | |
208 // we're still expecting a "run" message from the other side), so the | |
209 // |RunMessagePipeEndpoint()| below will return true. We need to refactor | |
210 // |AttachEndpoint()| to indicate whether |Run...()| will necessarily be | |
211 // called or not. (Then, in the case that it may not be called, this will | |
212 // return false.) | |
213 EXPECT_TRUE(channel()->RunMessagePipeEndpoint(local_id, | |
214 Channel::kBootstrapEndpointId)); | |
215 | |
216 io_thread()->PostTaskAndWait( | |
217 FROM_HERE, | |
218 base::Bind(&ChannelTest::ShutdownChannelOnIOThread, | |
219 base::Unretained(this))); | |
220 | |
221 EXPECT_TRUE(channel()->HasOneRef()); | |
222 } | |
223 | |
224 // ChannelTest.ShutdownAfterAttachAndRun --------------------------------------- | |
225 | |
226 TEST_F(ChannelTest, ShutdownAfterAttach) { | |
227 io_thread()->PostTaskAndWait(FROM_HERE, | |
228 base::Bind(&ChannelTest::CreateChannelOnIOThread, | |
229 base::Unretained(this))); | |
230 ASSERT_TRUE(channel()); | |
231 | |
232 io_thread()->PostTaskAndWait( | |
233 FROM_HERE, | |
234 base::Bind(&ChannelTest::InitChannelOnIOThread, base::Unretained(this))); | |
235 EXPECT_EQ(TRISTATE_TRUE, init_result()); | |
236 | |
237 scoped_refptr<ChannelEndpoint> channel_endpoint; | |
238 scoped_refptr<MessagePipe> mp( | |
239 MessagePipe::CreateLocalProxy(&channel_endpoint)); | |
240 | |
241 MessageInTransit::EndpointId local_id = | |
242 channel()->AttachEndpoint(channel_endpoint); | |
243 EXPECT_EQ(Channel::kBootstrapEndpointId, local_id); | |
244 | |
245 // TODO(vtl): Currently, we always "expect" a |RunMessagePipeEndpoint()| after | |
246 // an |AttachEndpoint()| (which is actually incorrect). We need to refactor | |
247 // |AttachEndpoint()| to indicate whether |Run...()| will necessarily be | |
248 // called or not. (Then, in the case that it may not be called, we should test | |
249 // a |Shutdown()| without the |Run...()|.) | |
250 EXPECT_TRUE(channel()->RunMessagePipeEndpoint(local_id, | |
251 Channel::kBootstrapEndpointId)); | |
252 | |
253 Waiter waiter; | |
254 waiter.Init(); | |
255 ASSERT_EQ( | |
256 MOJO_RESULT_OK, | |
257 mp->AddWaiter(0, &waiter, MOJO_HANDLE_SIGNAL_READABLE, 123, nullptr)); | |
258 | |
259 // Don't wait for the shutdown to run ... | |
260 io_thread()->PostTask(FROM_HERE, | |
261 base::Bind(&ChannelTest::ShutdownChannelOnIOThread, | |
262 base::Unretained(this))); | |
263 | |
264 // ... since this |Wait()| should fail once the channel is shut down. | |
265 EXPECT_EQ(MOJO_RESULT_FAILED_PRECONDITION, | |
266 waiter.Wait(MOJO_DEADLINE_INDEFINITE, nullptr)); | |
267 HandleSignalsState hss; | |
268 mp->RemoveWaiter(0, &waiter, &hss); | |
269 EXPECT_EQ(0u, hss.satisfied_signals); | |
270 EXPECT_EQ(0u, hss.satisfiable_signals); | |
271 | |
272 mp->Close(0); | |
273 | |
274 EXPECT_TRUE(channel()->HasOneRef()); | |
275 } | |
276 | |
277 // ChannelTest.WaitAfterAttachRunAndShutdown ----------------------------------- | |
278 | |
279 TEST_F(ChannelTest, WaitAfterAttachRunAndShutdown) { | |
280 io_thread()->PostTaskAndWait(FROM_HERE, | |
281 base::Bind(&ChannelTest::CreateChannelOnIOThread, | |
282 base::Unretained(this))); | |
283 ASSERT_TRUE(channel()); | |
284 | |
285 io_thread()->PostTaskAndWait( | |
286 FROM_HERE, | |
287 base::Bind(&ChannelTest::InitChannelOnIOThread, base::Unretained(this))); | |
288 EXPECT_EQ(TRISTATE_TRUE, init_result()); | |
289 | |
290 scoped_refptr<ChannelEndpoint> channel_endpoint; | |
291 scoped_refptr<MessagePipe> mp( | |
292 MessagePipe::CreateLocalProxy(&channel_endpoint)); | |
293 | |
294 MessageInTransit::EndpointId local_id = | |
295 channel()->AttachEndpoint(channel_endpoint); | |
296 EXPECT_EQ(Channel::kBootstrapEndpointId, local_id); | |
297 | |
298 EXPECT_TRUE(channel()->RunMessagePipeEndpoint(local_id, | |
299 Channel::kBootstrapEndpointId)); | |
300 | |
301 io_thread()->PostTaskAndWait( | |
302 FROM_HERE, | |
303 base::Bind(&ChannelTest::ShutdownChannelOnIOThread, | |
304 base::Unretained(this))); | |
305 | |
306 Waiter waiter; | |
307 waiter.Init(); | |
308 HandleSignalsState hss; | |
309 EXPECT_EQ(MOJO_RESULT_FAILED_PRECONDITION, | |
310 mp->AddWaiter(0, &waiter, MOJO_HANDLE_SIGNAL_READABLE, 123, &hss)); | |
311 EXPECT_EQ(0u, hss.satisfied_signals); | |
312 EXPECT_EQ(0u, hss.satisfiable_signals); | |
313 | |
314 mp->Close(0); | |
315 | |
316 EXPECT_TRUE(channel()->HasOneRef()); | |
317 } | |
318 | |
319 // TODO(vtl): More. ------------------------------------------------------------ | |
320 | |
321 } // namespace | |
322 } // namespace system | |
323 } // namespace mojo | |
OLD | NEW |