| 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 |