OLD | NEW |
1 // Copyright 2014 The Chromium Authors. All rights reserved. | 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 | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "third_party/mojo/src/mojo/edk/embedder/embedder.h" | 5 #include "mojo/edk/embedder/embedder.h" |
6 | |
7 #include <string.h> | |
8 | 6 |
9 #include "base/bind.h" | 7 #include "base/bind.h" |
10 #include "base/command_line.h" | 8 #include "base/command_line.h" |
11 #include "base/location.h" | |
12 #include "base/logging.h" | 9 #include "base/logging.h" |
13 #include "base/message_loop/message_loop.h" | 10 #include "base/message_loop/message_loop.h" |
14 #include "base/synchronization/waitable_event.h" | 11 #include "base/synchronization/waitable_event.h" |
15 #include "base/test/test_io_thread.h" | |
16 #include "base/test/test_timeouts.h" | 12 #include "base/test/test_timeouts.h" |
| 13 #include "mojo/edk/embedder/embedder.h" |
| 14 #include "mojo/edk/embedder/platform_channel_pair.h" |
| 15 #include "mojo/edk/embedder/simple_platform_support.h" |
| 16 #include "mojo/edk/embedder/test_embedder.h" |
| 17 #include "mojo/edk/system/test_utils.h" |
| 18 #include "mojo/edk/test/multiprocess_test_helper.h" |
| 19 #include "mojo/message_pump/message_pump_mojo.h" |
17 #include "mojo/public/c/system/core.h" | 20 #include "mojo/public/c/system/core.h" |
18 #include "mojo/public/cpp/system/handle.h" | 21 #include "mojo/public/cpp/system/handle.h" |
19 #include "mojo/public/cpp/system/macros.h" | 22 #include "mojo/public/cpp/system/macros.h" |
20 #include "mojo/public/cpp/system/message_pipe.h" | 23 #include "mojo/public/cpp/system/message_pipe.h" |
21 #include "testing/gtest/include/gtest/gtest.h" | 24 #include "testing/gtest/include/gtest/gtest.h" |
22 #include "third_party/mojo/src/mojo/edk/embedder/platform_channel_pair.h" | |
23 #include "third_party/mojo/src/mojo/edk/embedder/test_embedder.h" | |
24 #include "third_party/mojo/src/mojo/edk/system/mutex.h" | |
25 #include "third_party/mojo/src/mojo/edk/system/test_utils.h" | |
26 #include "third_party/mojo/src/mojo/edk/test/multiprocess_test_helper.h" | |
27 #include "third_party/mojo/src/mojo/edk/test/scoped_ipc_support.h" | |
28 | 25 |
29 namespace mojo { | 26 namespace mojo { |
30 namespace embedder { | 27 namespace edk { |
31 namespace { | 28 namespace { |
32 | 29 |
33 const MojoHandleSignals kSignalReadadableWritable = | 30 const MojoHandleSignals kSignalReadadableWritable = |
34 MOJO_HANDLE_SIGNAL_READABLE | MOJO_HANDLE_SIGNAL_WRITABLE; | 31 MOJO_HANDLE_SIGNAL_READABLE | MOJO_HANDLE_SIGNAL_WRITABLE; |
35 | 32 |
36 const MojoHandleSignals kSignalAll = MOJO_HANDLE_SIGNAL_READABLE | | 33 const MojoHandleSignals kSignalAll = MOJO_HANDLE_SIGNAL_READABLE | |
37 MOJO_HANDLE_SIGNAL_WRITABLE | | 34 MOJO_HANDLE_SIGNAL_WRITABLE | |
38 MOJO_HANDLE_SIGNAL_PEER_CLOSED; | 35 MOJO_HANDLE_SIGNAL_PEER_CLOSED; |
39 | 36 |
40 const char kConnectionIdFlag[] = "test-connection-id"; | 37 class EmbedderTest : public test::MojoSystemTest { |
41 | |
42 void DoNothing() {} | |
43 | |
44 class ScopedTestChannel { | |
45 public: | 38 public: |
46 // Creates a channel, which lives on the I/O thread given to | 39 EmbedderTest() {} |
47 // |InitIPCSupport()|. After construction, |bootstrap_message_pipe()| gives | 40 ~EmbedderTest() override {} |
48 // the Mojo handle for the bootstrap message pipe on this channel; it is up to | |
49 // the caller to close this handle. Note: The I/O thread must outlive this | |
50 // object (and its message loop must continue pumping messages while this | |
51 // object is alive). | |
52 explicit ScopedTestChannel(ScopedPlatformHandle platform_handle) | |
53 : bootstrap_message_pipe_(MOJO_HANDLE_INVALID), | |
54 event_(true, false), // Manual reset. | |
55 channel_info_(nullptr), | |
56 wait_on_shutdown_(true) { | |
57 bootstrap_message_pipe_ = | |
58 CreateChannel(platform_handle.Pass(), | |
59 base::Bind(&ScopedTestChannel::DidCreateChannel, | |
60 base::Unretained(this)), | |
61 nullptr) | |
62 .release() | |
63 .value(); | |
64 CHECK_NE(bootstrap_message_pipe_, MOJO_HANDLE_INVALID); | |
65 } | |
66 | |
67 // Destructor: Shuts down the channel. (As noted above, for this to happen, | |
68 // the I/O thread must be alive and pumping messages.) | |
69 ~ScopedTestChannel() { | |
70 // |WaitForChannelCreationCompletion()| must be called before destruction. | |
71 CHECK(event_.IsSignaled()); | |
72 event_.Reset(); | |
73 if (wait_on_shutdown_) { | |
74 DestroyChannel(channel_info_, | |
75 base::Bind(&ScopedTestChannel::DidDestroyChannel, | |
76 base::Unretained(this)), | |
77 nullptr); | |
78 event_.Wait(); | |
79 } else { | |
80 DestroyChannel(channel_info_, base::Bind(&DoNothing), nullptr); | |
81 } | |
82 } | |
83 | |
84 // Waits for channel creation to be completed. | |
85 void WaitForChannelCreationCompletion() { event_.Wait(); } | |
86 | |
87 MojoHandle bootstrap_message_pipe() const { return bootstrap_message_pipe_; } | |
88 | |
89 // Call only after |WaitForChannelCreationCompletion()|. Use only to check | |
90 // that it's not null. | |
91 const ChannelInfo* channel_info() const { return channel_info_; } | |
92 | |
93 // Don't wait for the channel shutdown to finish on destruction. Used to | |
94 // exercise races. | |
95 void NoWaitOnShutdown() { wait_on_shutdown_ = false; } | |
96 | 41 |
97 private: | 42 private: |
98 void DidCreateChannel(ChannelInfo* channel_info) { | |
99 CHECK(channel_info); | |
100 CHECK(!channel_info_); | |
101 channel_info_ = channel_info; | |
102 event_.Signal(); | |
103 } | |
104 | |
105 void DidDestroyChannel() { event_.Signal(); } | |
106 | |
107 // Valid from creation until whenever it gets closed (by the "owner" of this | |
108 // object). | |
109 // Note: We don't want use the C++ wrappers here, since we want to test the | |
110 // API at the lowest level. | |
111 MojoHandle bootstrap_message_pipe_; | |
112 | |
113 // Set after channel creation has been completed (i.e., the callback to | |
114 // |CreateChannel()| has been called). Also used in the destructor to wait for | |
115 // |DestroyChannel()| completion. | |
116 base::WaitableEvent event_; | |
117 | |
118 // Valid after channel creation completion until destruction. | |
119 ChannelInfo* channel_info_; | |
120 | |
121 // Whether the destructor should wait until the channel is destroyed. | |
122 bool wait_on_shutdown_; | |
123 | |
124 MOJO_DISALLOW_COPY_AND_ASSIGN(ScopedTestChannel); | |
125 }; | |
126 | |
127 class EmbedderTest : public testing::Test { | |
128 public: | |
129 EmbedderTest() : test_io_thread_(base::TestIOThread::kAutoStart) {} | |
130 ~EmbedderTest() override {} | |
131 | |
132 protected: | |
133 base::TestIOThread& test_io_thread() { return test_io_thread_; } | |
134 scoped_refptr<base::TaskRunner> test_io_task_runner() { | |
135 return test_io_thread_.task_runner(); | |
136 } | |
137 | |
138 private: | |
139 void SetUp() override { test::InitWithSimplePlatformSupport(); } | |
140 | |
141 void TearDown() override { EXPECT_TRUE(test::Shutdown()); } | |
142 | |
143 base::TestIOThread test_io_thread_; | |
144 | |
145 MOJO_DISALLOW_COPY_AND_ASSIGN(EmbedderTest); | 43 MOJO_DISALLOW_COPY_AND_ASSIGN(EmbedderTest); |
146 }; | 44 }; |
147 | 45 |
148 TEST_F(EmbedderTest, ChannelsBasic) { | 46 TEST_F(EmbedderTest, ChannelBasic) { |
149 mojo::test::ScopedIPCSupport ipc_support(test_io_task_runner()); | 47 MojoHandle server_mp, client_mp; |
150 | 48 ASSERT_EQ(MOJO_RESULT_OK, |
151 PlatformChannelPair channel_pair; | 49 MojoCreateMessagePipe(nullptr, &server_mp, &client_mp)); |
152 ScopedTestChannel server_channel(channel_pair.PassServerHandle()); | |
153 MojoHandle server_mp = server_channel.bootstrap_message_pipe(); | |
154 EXPECT_NE(server_mp, MOJO_HANDLE_INVALID); | |
155 ScopedTestChannel client_channel(channel_pair.PassClientHandle()); | |
156 MojoHandle client_mp = client_channel.bootstrap_message_pipe(); | |
157 EXPECT_NE(client_mp, MOJO_HANDLE_INVALID); | |
158 | 50 |
159 // We can write to a message pipe handle immediately. | 51 // We can write to a message pipe handle immediately. |
160 const char kHello[] = "hello"; | 52 const char kHello[] = "hello"; |
161 EXPECT_EQ( | 53 |
162 MOJO_RESULT_OK, | 54 size_t write_size = sizeof(kHello); |
163 MojoWriteMessage(server_mp, kHello, static_cast<uint32_t>(sizeof(kHello)), | 55 const char* write_buffer = kHello; |
164 nullptr, 0, MOJO_WRITE_MESSAGE_FLAG_NONE)); | 56 EXPECT_EQ(MOJO_RESULT_OK, |
| 57 MojoWriteMessage(server_mp, write_buffer, |
| 58 static_cast<uint32_t>(write_size), nullptr, 0, |
| 59 MOJO_WRITE_MESSAGE_FLAG_NONE)); |
165 | 60 |
166 // Now wait for the other side to become readable. | 61 // Now wait for the other side to become readable. |
167 MojoHandleSignalsState state; | 62 MojoHandleSignalsState state; |
168 EXPECT_EQ(MOJO_RESULT_OK, MojoWait(client_mp, MOJO_HANDLE_SIGNAL_READABLE, | 63 EXPECT_EQ(MOJO_RESULT_OK, MojoWait(client_mp, MOJO_HANDLE_SIGNAL_READABLE, |
169 MOJO_DEADLINE_INDEFINITE, &state)); | 64 MOJO_DEADLINE_INDEFINITE, &state)); |
170 EXPECT_EQ(kSignalReadadableWritable, state.satisfied_signals); | 65 EXPECT_EQ(kSignalReadadableWritable, state.satisfied_signals); |
171 EXPECT_EQ(kSignalAll, state.satisfiable_signals); | 66 EXPECT_EQ(kSignalAll, state.satisfiable_signals); |
172 | 67 |
173 char buffer[1000] = {}; | 68 char read_buffer[1000] = {}; |
174 uint32_t num_bytes = static_cast<uint32_t>(sizeof(buffer)); | 69 uint32_t num_bytes = static_cast<uint32_t>(sizeof(read_buffer)); |
175 EXPECT_EQ(MOJO_RESULT_OK, | 70 EXPECT_EQ(MOJO_RESULT_OK, |
176 MojoReadMessage(client_mp, buffer, &num_bytes, nullptr, nullptr, | 71 MojoReadMessage(client_mp, read_buffer, &num_bytes, nullptr, |
177 MOJO_READ_MESSAGE_FLAG_NONE)); | 72 nullptr, MOJO_READ_MESSAGE_FLAG_NONE)); |
178 EXPECT_EQ(sizeof(kHello), num_bytes); | 73 EXPECT_EQ(write_size, num_bytes); |
179 EXPECT_STREQ(kHello, buffer); | 74 EXPECT_STREQ(kHello, read_buffer); |
180 | 75 |
181 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(server_mp)); | 76 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(server_mp)); |
182 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(client_mp)); | 77 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(client_mp)); |
183 | |
184 // By this point, these waits should basically be no-ops (since we've waited | |
185 // for the client message pipe to become readable, which implies that both | |
186 // the server and client channels were completely created). | |
187 server_channel.WaitForChannelCreationCompletion(); | |
188 client_channel.WaitForChannelCreationCompletion(); | |
189 EXPECT_TRUE(server_channel.channel_info()); | |
190 EXPECT_TRUE(client_channel.channel_info()); | |
191 } | 78 } |
192 | 79 |
193 class TestAsyncWaiter { | 80 // Test sending a MP which has read messages out of the OS pipe but which have |
194 public: | 81 // not been consumed using MojoReadMessage yet. |
195 TestAsyncWaiter() : event_(true, false), wait_result_(MOJO_RESULT_UNKNOWN) {} | 82 TEST_F(EmbedderTest, SendReadableMessagePipe) { |
| 83 MojoHandle server_mp, client_mp; |
| 84 ASSERT_EQ(MOJO_RESULT_OK, |
| 85 MojoCreateMessagePipe(nullptr, &server_mp, &client_mp)); |
196 | 86 |
197 void Awake(MojoResult result) { | 87 MojoHandle server_mp2, client_mp2; |
198 system::MutexLocker l(&wait_result_mutex_); | 88 ASSERT_EQ(MOJO_RESULT_OK, |
199 wait_result_ = result; | 89 MojoCreateMessagePipe(nullptr, &server_mp2, &client_mp2)); |
200 event_.Signal(); | 90 |
| 91 // Write to server2 and wait for client2 to be readable before sending it. |
| 92 // client2's MessagePipeDispatcher will have the message below in its |
| 93 // message_queue_. For extra measures, also verify that this pending message |
| 94 // can contain a message pipe. |
| 95 MojoHandle server_mp3, client_mp3; |
| 96 ASSERT_EQ(MOJO_RESULT_OK, |
| 97 MojoCreateMessagePipe(nullptr, &server_mp3, &client_mp3)); |
| 98 const char kHello[] = "hello"; |
| 99 size_t write_size; |
| 100 const char* write_buffer; |
| 101 write_buffer = kHello; |
| 102 write_size = sizeof(kHello); |
| 103 EXPECT_EQ(MOJO_RESULT_OK, |
| 104 MojoWriteMessage(server_mp2, write_buffer, |
| 105 static_cast<uint32_t>(write_size), &client_mp3, 1, |
| 106 MOJO_WRITE_MESSAGE_FLAG_NONE)); |
| 107 |
| 108 MojoHandleSignalsState state; |
| 109 EXPECT_EQ(MOJO_RESULT_OK, MojoWait(client_mp2, MOJO_HANDLE_SIGNAL_READABLE, |
| 110 MOJO_DEADLINE_INDEFINITE, &state)); |
| 111 EXPECT_EQ(kSignalReadadableWritable, state.satisfied_signals); |
| 112 EXPECT_EQ(kSignalAll, state.satisfiable_signals); |
| 113 |
| 114 // Now send client2 |
| 115 EXPECT_EQ(MOJO_RESULT_OK, |
| 116 MojoWriteMessage(server_mp, write_buffer, |
| 117 static_cast<uint32_t>(write_size), &client_mp2, 1, |
| 118 MOJO_WRITE_MESSAGE_FLAG_NONE)); |
| 119 |
| 120 |
| 121 EXPECT_EQ(MOJO_RESULT_OK, MojoWait(client_mp, MOJO_HANDLE_SIGNAL_READABLE, |
| 122 MOJO_DEADLINE_INDEFINITE, &state)); |
| 123 EXPECT_EQ(kSignalReadadableWritable, state.satisfied_signals); |
| 124 EXPECT_EQ(kSignalAll, state.satisfiable_signals); |
| 125 |
| 126 char read_buffer[20000] = {}; |
| 127 uint32_t num_bytes = static_cast<uint32_t>(sizeof(read_buffer)); |
| 128 MojoHandle ports[10]; |
| 129 uint32_t num_ports; |
| 130 EXPECT_EQ(MOJO_RESULT_OK, |
| 131 MojoReadMessage(client_mp, read_buffer, &num_bytes, &ports[0], |
| 132 &num_ports, MOJO_READ_MESSAGE_FLAG_NONE)); |
| 133 EXPECT_EQ(write_size, num_bytes); |
| 134 EXPECT_STREQ(kHello, read_buffer); |
| 135 EXPECT_EQ(1, num_ports); |
| 136 |
| 137 |
| 138 client_mp2 = ports[0]; |
| 139 EXPECT_EQ(MOJO_RESULT_OK, MojoWait(client_mp2, MOJO_HANDLE_SIGNAL_READABLE, |
| 140 MOJO_DEADLINE_INDEFINITE, &state)); |
| 141 EXPECT_EQ(kSignalReadadableWritable, state.satisfied_signals); |
| 142 EXPECT_EQ(kSignalAll, state.satisfiable_signals); |
| 143 |
| 144 |
| 145 EXPECT_EQ(MOJO_RESULT_OK, |
| 146 MojoReadMessage(client_mp2, read_buffer, &num_bytes, &client_mp3, |
| 147 &num_ports, MOJO_READ_MESSAGE_FLAG_NONE)); |
| 148 EXPECT_EQ(write_size, num_bytes); |
| 149 EXPECT_STREQ(kHello, read_buffer); |
| 150 EXPECT_EQ(1, num_ports); |
| 151 |
| 152 |
| 153 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(server_mp3)); |
| 154 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(client_mp3)); |
| 155 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(server_mp2)); |
| 156 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(client_mp2)); |
| 157 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(server_mp)); |
| 158 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(client_mp)); |
| 159 } |
| 160 |
| 161 // TODO(jam): fix and renable |
| 162 TEST_F(EmbedderTest, DISABLED_SendMessagePipeWithWriteQueue) { |
| 163 MojoHandle server_mp, client_mp; |
| 164 ASSERT_EQ(MOJO_RESULT_OK, |
| 165 MojoCreateMessagePipe(nullptr, &server_mp, &client_mp)); |
| 166 |
| 167 MojoHandle server_mp2, client_mp2; |
| 168 ASSERT_EQ(MOJO_RESULT_OK, |
| 169 MojoCreateMessagePipe(nullptr, &server_mp2, &client_mp2)); |
| 170 |
| 171 static const size_t kNumMessages = 1001; |
| 172 for (size_t i = 0; i < kNumMessages; i++) { |
| 173 std::string write_buffer(i, 'A' + (i % 26)); |
| 174 ASSERT_EQ(MOJO_RESULT_OK, |
| 175 MojoWriteMessage(client_mp2, write_buffer.data(), |
| 176 write_buffer.size(), nullptr, 0, |
| 177 MOJO_WRITE_MESSAGE_FLAG_NONE)); |
201 } | 178 } |
202 | 179 |
203 bool TryWait() { return event_.TimedWait(TestTimeouts::action_timeout()); } | 180 // Now send client2 |
| 181 EXPECT_EQ(MOJO_RESULT_OK, |
| 182 MojoWriteMessage(server_mp, nullptr, 0, &client_mp2, 1, |
| 183 MOJO_WRITE_MESSAGE_FLAG_NONE)); |
204 | 184 |
205 MojoResult wait_result() const { | 185 MojoHandleSignalsState state; |
206 system::MutexLocker l(&wait_result_mutex_); | 186 EXPECT_EQ(MOJO_RESULT_OK, MojoWait(client_mp, MOJO_HANDLE_SIGNAL_READABLE, |
207 return wait_result_; | 187 MOJO_DEADLINE_INDEFINITE, &state)); |
208 } | 188 EXPECT_EQ(kSignalReadadableWritable, state.satisfied_signals); |
| 189 EXPECT_EQ(kSignalAll, state.satisfiable_signals); |
209 | 190 |
210 private: | 191 /// todo: read all msgs written.. |
211 base::WaitableEvent event_; | |
212 | 192 |
213 mutable system::Mutex wait_result_mutex_; | 193 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(server_mp2)); |
214 MojoResult wait_result_ MOJO_GUARDED_BY(wait_result_mutex_); | 194 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(client_mp2)); |
215 | 195 |
216 MOJO_DISALLOW_COPY_AND_ASSIGN(TestAsyncWaiter); | |
217 }; | |
218 | 196 |
219 void WriteHello(MessagePipeHandle pipe) { | 197 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(server_mp)); |
220 static const char kHello[] = "hello"; | 198 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(client_mp)); |
221 CHECK_EQ(MOJO_RESULT_OK, | |
222 WriteMessageRaw(pipe, kHello, static_cast<uint32_t>(sizeof(kHello)), | |
223 nullptr, 0, MOJO_WRITE_MESSAGE_FLAG_NONE)); | |
224 } | |
225 | |
226 void CloseScopedHandle(ScopedMessagePipeHandle handle) { | |
227 // Do nothing and the destructor will close it. | |
228 } | |
229 | |
230 TEST_F(EmbedderTest, AsyncWait) { | |
231 ScopedMessagePipeHandle client_mp; | |
232 ScopedMessagePipeHandle server_mp; | |
233 EXPECT_EQ(MOJO_RESULT_OK, CreateMessagePipe(nullptr, &client_mp, &server_mp)); | |
234 | |
235 TestAsyncWaiter waiter; | |
236 EXPECT_EQ(MOJO_RESULT_OK, | |
237 AsyncWait(client_mp.get().value(), MOJO_HANDLE_SIGNAL_READABLE, | |
238 base::Bind(&TestAsyncWaiter::Awake, | |
239 base::Unretained(&waiter)))); | |
240 | |
241 test_io_task_runner()->PostTask(FROM_HERE, | |
242 base::Bind(&WriteHello, server_mp.get())); | |
243 EXPECT_TRUE(waiter.TryWait()); | |
244 EXPECT_EQ(MOJO_RESULT_OK, waiter.wait_result()); | |
245 | |
246 // If message is in the queue, it does't allow us to wait. | |
247 TestAsyncWaiter waiter_that_doesnt_wait; | |
248 EXPECT_EQ(MOJO_RESULT_ALREADY_EXISTS, | |
249 AsyncWait(client_mp.get().value(), MOJO_HANDLE_SIGNAL_READABLE, | |
250 base::Bind(&TestAsyncWaiter::Awake, | |
251 base::Unretained(&waiter_that_doesnt_wait)))); | |
252 | |
253 char buffer[1000]; | |
254 uint32_t num_bytes = static_cast<uint32_t>(sizeof(buffer)); | |
255 CHECK_EQ(MOJO_RESULT_OK, | |
256 ReadMessageRaw(client_mp.get(), buffer, &num_bytes, nullptr, nullptr, | |
257 MOJO_READ_MESSAGE_FLAG_NONE)); | |
258 | |
259 TestAsyncWaiter unsatisfiable_waiter; | |
260 EXPECT_EQ(MOJO_RESULT_OK, | |
261 AsyncWait(client_mp.get().value(), MOJO_HANDLE_SIGNAL_READABLE, | |
262 base::Bind(&TestAsyncWaiter::Awake, | |
263 base::Unretained(&unsatisfiable_waiter)))); | |
264 | |
265 test_io_task_runner()->PostTask( | |
266 FROM_HERE, | |
267 base::Bind(&CloseScopedHandle, base::Passed(server_mp.Pass()))); | |
268 | |
269 EXPECT_TRUE(unsatisfiable_waiter.TryWait()); | |
270 EXPECT_EQ(MOJO_RESULT_FAILED_PRECONDITION, | |
271 unsatisfiable_waiter.wait_result()); | |
272 } | 199 } |
273 | 200 |
274 TEST_F(EmbedderTest, ChannelsHandlePassing) { | 201 TEST_F(EmbedderTest, ChannelsHandlePassing) { |
275 mojo::test::ScopedIPCSupport ipc_support(test_io_task_runner()); | 202 MojoHandle server_mp, client_mp; |
276 | 203 ASSERT_EQ(MOJO_RESULT_OK, |
277 PlatformChannelPair channel_pair; | 204 MojoCreateMessagePipe(nullptr, &server_mp, &client_mp)); |
278 ScopedTestChannel server_channel(channel_pair.PassServerHandle()); | |
279 MojoHandle server_mp = server_channel.bootstrap_message_pipe(); | |
280 EXPECT_NE(server_mp, MOJO_HANDLE_INVALID); | 205 EXPECT_NE(server_mp, MOJO_HANDLE_INVALID); |
281 ScopedTestChannel client_channel(channel_pair.PassClientHandle()); | |
282 MojoHandle client_mp = client_channel.bootstrap_message_pipe(); | |
283 EXPECT_NE(client_mp, MOJO_HANDLE_INVALID); | 206 EXPECT_NE(client_mp, MOJO_HANDLE_INVALID); |
284 | 207 |
285 MojoHandle h0, h1; | 208 MojoHandle h0, h1; |
286 EXPECT_EQ(MOJO_RESULT_OK, MojoCreateMessagePipe(nullptr, &h0, &h1)); | 209 EXPECT_EQ(MOJO_RESULT_OK, MojoCreateMessagePipe(nullptr, &h0, &h1)); |
287 | 210 |
288 // Write a message to |h0| (attaching nothing). | 211 // Write a message to |h0| (attaching nothing). |
289 const char kHello[] = "hello"; | 212 const char kHello[] = "hello"; |
290 EXPECT_EQ(MOJO_RESULT_OK, | 213 EXPECT_EQ(MOJO_RESULT_OK, |
291 MojoWriteMessage(h0, kHello, static_cast<uint32_t>(sizeof(kHello)), | 214 MojoWriteMessage(h0, kHello, static_cast<uint32_t>(sizeof(kHello)), |
292 nullptr, 0, MOJO_WRITE_MESSAGE_FLAG_NONE)); | 215 nullptr, 0, MOJO_WRITE_MESSAGE_FLAG_NONE)); |
(...skipping 85 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
378 EXPECT_EQ(MOJO_RESULT_OK, | 301 EXPECT_EQ(MOJO_RESULT_OK, |
379 MojoReadMessage(h0, buffer, &num_bytes, nullptr, nullptr, | 302 MojoReadMessage(h0, buffer, &num_bytes, nullptr, nullptr, |
380 MOJO_READ_MESSAGE_FLAG_NONE)); | 303 MOJO_READ_MESSAGE_FLAG_NONE)); |
381 EXPECT_EQ(sizeof(kBarBaz), num_bytes); | 304 EXPECT_EQ(sizeof(kBarBaz), num_bytes); |
382 EXPECT_STREQ(kBarBaz, buffer); | 305 EXPECT_STREQ(kBarBaz, buffer); |
383 | 306 |
384 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(server_mp)); | 307 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(server_mp)); |
385 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(client_mp)); | 308 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(client_mp)); |
386 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(h0)); | 309 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(h0)); |
387 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(h1)); | 310 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(h1)); |
388 | |
389 server_channel.WaitForChannelCreationCompletion(); | |
390 client_channel.WaitForChannelCreationCompletion(); | |
391 EXPECT_TRUE(server_channel.channel_info()); | |
392 EXPECT_TRUE(client_channel.channel_info()); | |
393 } | |
394 | |
395 #if defined(OS_ANDROID) | |
396 // Android multi-process tests are not executing the new process. This is flaky. | |
397 // TODO(vtl): I'm guessing this is true of this test too? | |
398 #define MAYBE_MultiprocessMasterSlave DISABLED_MultiprocessMasterSlave | |
399 #else | |
400 #define MAYBE_MultiprocessMasterSlave MultiprocessMasterSlave | |
401 #endif // defined(OS_ANDROID) | |
402 TEST_F(EmbedderTest, MAYBE_MultiprocessMasterSlave) { | |
403 mojo::test::ScopedMasterIPCSupport ipc_support(test_io_task_runner()); | |
404 | |
405 mojo::test::MultiprocessTestHelper multiprocess_test_helper; | |
406 std::string connection_id; | |
407 base::WaitableEvent event(true, false); | |
408 ChannelInfo* channel_info = nullptr; | |
409 ScopedMessagePipeHandle mp = ConnectToSlave( | |
410 nullptr, multiprocess_test_helper.server_platform_handle.Pass(), | |
411 base::Bind(&base::WaitableEvent::Signal, base::Unretained(&event)), | |
412 nullptr, &connection_id, &channel_info); | |
413 ASSERT_TRUE(mp.is_valid()); | |
414 EXPECT_TRUE(channel_info); | |
415 ASSERT_FALSE(connection_id.empty()); | |
416 | |
417 multiprocess_test_helper.StartChildWithExtraSwitch( | |
418 "MultiprocessMasterSlave", kConnectionIdFlag, connection_id); | |
419 | |
420 // Send a message saying "hello". | |
421 EXPECT_EQ(MOJO_RESULT_OK, WriteMessageRaw(mp.get(), "hello", 5, nullptr, 0, | |
422 MOJO_WRITE_MESSAGE_FLAG_NONE)); | |
423 | |
424 // Wait for a response. | |
425 EXPECT_EQ(MOJO_RESULT_OK, | |
426 Wait(mp.get(), MOJO_HANDLE_SIGNAL_READABLE, | |
427 mojo::system::test::ActionDeadline(), nullptr)); | |
428 | |
429 // The response message should say "world". | |
430 char buffer[100]; | |
431 uint32_t num_bytes = static_cast<uint32_t>(sizeof(buffer)); | |
432 EXPECT_EQ(MOJO_RESULT_OK, | |
433 ReadMessageRaw(mp.get(), buffer, &num_bytes, nullptr, nullptr, | |
434 MOJO_READ_MESSAGE_FLAG_NONE)); | |
435 EXPECT_EQ(5u, num_bytes); | |
436 EXPECT_EQ(0, memcmp(buffer, "world", 5)); | |
437 | |
438 mp.reset(); | |
439 | |
440 EXPECT_TRUE(multiprocess_test_helper.WaitForChildTestShutdown()); | |
441 | |
442 EXPECT_TRUE(event.TimedWait(TestTimeouts::action_timeout())); | |
443 test_io_thread().PostTaskAndWait( | |
444 FROM_HERE, | |
445 base::Bind(&DestroyChannelOnIOThread, base::Unretained(channel_info))); | |
446 } | |
447 | |
448 TEST_F(EmbedderTest, ChannelShutdownRace_MessagePipeClose) { | |
449 const size_t kIterations = 1000; | |
450 mojo::test::ScopedIPCSupport ipc_support(test_io_task_runner()); | |
451 | |
452 for (size_t i = 0; i < kIterations; i++) { | |
453 PlatformChannelPair channel_pair; | |
454 scoped_ptr<ScopedTestChannel> server_channel( | |
455 new ScopedTestChannel(channel_pair.PassServerHandle())); | |
456 server_channel->WaitForChannelCreationCompletion(); | |
457 server_channel->NoWaitOnShutdown(); | |
458 | |
459 MojoHandle server_mp = server_channel->bootstrap_message_pipe(); | |
460 EXPECT_NE(server_mp, MOJO_HANDLE_INVALID); | |
461 | |
462 // Race between channel shutdown and closing a message pipe. The message | |
463 // pipe doesn't have to be the bootstrap pipe. It just has to be bound to | |
464 // the channel. | |
465 server_channel.reset(); | |
466 MojoClose(server_mp); | |
467 } | |
468 } | |
469 | |
470 TEST_F(EmbedderTest, ChannelShutdownRace_MessagePipePassing) { | |
471 const size_t kIterations = 1000; | |
472 mojo::test::ScopedIPCSupport ipc_support(test_io_task_runner()); | |
473 | |
474 for (size_t i = 0; i < kIterations; i++) { | |
475 PlatformChannelPair channel_pair; | |
476 scoped_ptr<ScopedTestChannel> server_channel( | |
477 new ScopedTestChannel(channel_pair.PassServerHandle())); | |
478 server_channel->WaitForChannelCreationCompletion(); | |
479 server_channel->NoWaitOnShutdown(); | |
480 | |
481 MojoHandle server_mp = server_channel->bootstrap_message_pipe(); | |
482 EXPECT_NE(server_mp, MOJO_HANDLE_INVALID); | |
483 | |
484 MessagePipe test_pipe; | |
485 MojoHandle passing_handle = test_pipe.handle0.release().value(); | |
486 | |
487 // Race between channel shutdown and passing a message pipe. | |
488 server_channel.reset(); | |
489 MojoWriteMessage(server_mp, nullptr, 0, &passing_handle, 1, | |
490 MOJO_WRITE_MESSAGE_FLAG_NONE); | |
491 MojoClose(server_mp); | |
492 MojoClose(passing_handle); | |
493 } | |
494 } | |
495 | |
496 MOJO_MULTIPROCESS_TEST_CHILD_TEST(MultiprocessMasterSlave) { | |
497 ScopedPlatformHandle client_platform_handle = | |
498 mojo::test::MultiprocessTestHelper::client_platform_handle.Pass(); | |
499 EXPECT_TRUE(client_platform_handle.is_valid()); | |
500 | |
501 base::TestIOThread test_io_thread(base::TestIOThread::kAutoStart); | |
502 test::InitWithSimplePlatformSupport(); | |
503 | |
504 { | |
505 mojo::test::ScopedSlaveIPCSupport ipc_support( | |
506 test_io_thread.task_runner(), client_platform_handle.Pass()); | |
507 | |
508 const base::CommandLine& command_line = | |
509 *base::CommandLine::ForCurrentProcess(); | |
510 ASSERT_TRUE(command_line.HasSwitch(kConnectionIdFlag)); | |
511 std::string connection_id = | |
512 command_line.GetSwitchValueASCII(kConnectionIdFlag); | |
513 ASSERT_FALSE(connection_id.empty()); | |
514 base::WaitableEvent event(true, false); | |
515 ChannelInfo* channel_info = nullptr; | |
516 ScopedMessagePipeHandle mp = ConnectToMaster( | |
517 connection_id, | |
518 base::Bind(&base::WaitableEvent::Signal, base::Unretained(&event)), | |
519 nullptr, &channel_info); | |
520 ASSERT_TRUE(mp.is_valid()); | |
521 EXPECT_TRUE(channel_info); | |
522 | |
523 // Wait for the master to send us a message. | |
524 EXPECT_EQ(MOJO_RESULT_OK, | |
525 Wait(mp.get(), MOJO_HANDLE_SIGNAL_READABLE, | |
526 mojo::system::test::ActionDeadline(), nullptr)); | |
527 | |
528 // It should say "hello". | |
529 char buffer[100]; | |
530 uint32_t num_bytes = static_cast<uint32_t>(sizeof(buffer)); | |
531 EXPECT_EQ(MOJO_RESULT_OK, | |
532 ReadMessageRaw(mp.get(), buffer, &num_bytes, nullptr, nullptr, | |
533 MOJO_READ_MESSAGE_FLAG_NONE)); | |
534 EXPECT_EQ(5u, num_bytes); | |
535 EXPECT_EQ(0, memcmp(buffer, "hello", 5)); | |
536 | |
537 // In response send a message saying "world". | |
538 EXPECT_EQ(MOJO_RESULT_OK, WriteMessageRaw(mp.get(), "world", 5, nullptr, 0, | |
539 MOJO_WRITE_MESSAGE_FLAG_NONE)); | |
540 | |
541 mp.reset(); | |
542 | |
543 EXPECT_TRUE(event.TimedWait(TestTimeouts::action_timeout())); | |
544 test_io_thread.PostTaskAndWait( | |
545 FROM_HERE, | |
546 base::Bind(&DestroyChannelOnIOThread, base::Unretained(channel_info))); | |
547 } | |
548 | |
549 EXPECT_TRUE(test::Shutdown()); | |
550 } | 311 } |
551 | 312 |
552 // The sequence of messages sent is: | 313 // The sequence of messages sent is: |
553 // server_mp client_mp mp0 mp1 mp2 mp3 | 314 // server_mp client_mp mp0 mp1 mp2 mp3 |
554 // 1. "hello" | 315 // 1. "hello" |
555 // 2. "world!" | 316 // 2. "world!" |
556 // 3. "FOO" | 317 // 3. "FOO" |
557 // 4. "Bar"+mp1 | 318 // 4. "Bar"+mp1 |
558 // 5. (close) | 319 // 5. (close) |
559 // 6. (close) | 320 // 6. (close) |
560 // 7. "baz" | 321 // 7. "baz" |
561 // 8. (closed) | 322 // 8. (closed) |
562 // 9. "quux"+mp2 | 323 // 9. "quux"+mp2 |
563 // 10. (close) | 324 // 10. (close) |
564 // 11. (wait/cl.) | 325 // 11. (wait/cl.) |
565 // 12. (wait/cl.) | 326 // 12. (wait/cl.) |
566 | 327 |
567 #if defined(OS_ANDROID) | 328 #if defined(OS_ANDROID) |
568 // Android multi-process tests are not executing the new process. This is flaky. | 329 // Android multi-process tests are not executing the new process. This is flaky. |
569 #define MAYBE_MultiprocessChannels DISABLED_MultiprocessChannels | 330 #define MAYBE_MultiprocessChannels DISABLED_MultiprocessChannels |
570 #else | 331 #else |
571 #define MAYBE_MultiprocessChannels MultiprocessChannels | 332 #define MAYBE_MultiprocessChannels MultiprocessChannels |
572 #endif // defined(OS_ANDROID) | 333 #endif // defined(OS_ANDROID) |
573 TEST_F(EmbedderTest, MAYBE_MultiprocessChannels) { | 334 TEST_F(EmbedderTest, MAYBE_MultiprocessChannels) { |
574 // TODO(vtl): This should eventually initialize a master process instead, | 335 test::MultiprocessTestHelper multiprocess_test_helper; |
575 // probably. | |
576 mojo::test::ScopedIPCSupport ipc_support(test_io_task_runner()); | |
577 | |
578 mojo::test::MultiprocessTestHelper multiprocess_test_helper; | |
579 multiprocess_test_helper.StartChild("MultiprocessChannelsClient"); | 336 multiprocess_test_helper.StartChild("MultiprocessChannelsClient"); |
580 | 337 |
581 { | 338 { |
582 ScopedTestChannel server_channel( | 339 MojoHandle server_mp = CreateMessagePipe( |
583 multiprocess_test_helper.server_platform_handle.Pass()); | 340 multiprocess_test_helper.server_platform_handle.Pass()).release(). |
584 MojoHandle server_mp = server_channel.bootstrap_message_pipe(); | 341 value(); |
585 EXPECT_NE(server_mp, MOJO_HANDLE_INVALID); | |
586 server_channel.WaitForChannelCreationCompletion(); | |
587 EXPECT_TRUE(server_channel.channel_info()); | |
588 | 342 |
589 // 1. Write a message to |server_mp| (attaching nothing). | 343 // 1. Write a message to |server_mp| (attaching nothing). |
590 const char kHello[] = "hello"; | 344 const char kHello[] = "hello"; |
591 EXPECT_EQ(MOJO_RESULT_OK, | 345 EXPECT_EQ(MOJO_RESULT_OK, |
592 MojoWriteMessage(server_mp, kHello, | 346 MojoWriteMessage(server_mp, kHello, |
593 static_cast<uint32_t>(sizeof(kHello)), nullptr, | 347 static_cast<uint32_t>(sizeof(kHello)), nullptr, |
594 0, MOJO_WRITE_MESSAGE_FLAG_NONE)); | 348 0, MOJO_WRITE_MESSAGE_FLAG_NONE)); |
595 | 349 |
596 // TODO(vtl): If the scope were ended immediately here (maybe after closing | 350 // TODO(vtl): If the scope were ended immediately here (maybe after closing |
597 // |server_mp|), we die with a fatal error in |Channel::HandleLocalError()|. | 351 // |server_mp|), we die with a fatal error in |Channel::HandleLocalError()|. |
(...skipping 48 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
646 EXPECT_EQ(MOJO_RESULT_OK, | 400 EXPECT_EQ(MOJO_RESULT_OK, |
647 MojoReadMessage(mp0, buffer, &num_bytes, &mp2, &num_handles, | 401 MojoReadMessage(mp0, buffer, &num_bytes, &mp2, &num_handles, |
648 MOJO_READ_MESSAGE_FLAG_NONE)); | 402 MOJO_READ_MESSAGE_FLAG_NONE)); |
649 const char kQuux[] = "quux"; | 403 const char kQuux[] = "quux"; |
650 EXPECT_EQ(sizeof(kQuux), num_bytes); | 404 EXPECT_EQ(sizeof(kQuux), num_bytes); |
651 EXPECT_STREQ(kQuux, buffer); | 405 EXPECT_STREQ(kQuux, buffer); |
652 EXPECT_EQ(1u, num_handles); | 406 EXPECT_EQ(1u, num_handles); |
653 EXPECT_NE(mp2, MOJO_HANDLE_INVALID); | 407 EXPECT_NE(mp2, MOJO_HANDLE_INVALID); |
654 | 408 |
655 // 7. Read a message from |mp2|. | 409 // 7. Read a message from |mp2|. |
656 EXPECT_EQ(MOJO_RESULT_OK, MojoWait(mp2, MOJO_HANDLE_SIGNAL_READABLE, | 410 EXPECT_EQ(MOJO_RESULT_OK, MojoWait(mp2, MOJO_HANDLE_SIGNAL_PEER_CLOSED, |
657 MOJO_DEADLINE_INDEFINITE, &state)); | 411 MOJO_DEADLINE_INDEFINITE, &state)); |
658 EXPECT_EQ(MOJO_HANDLE_SIGNAL_READABLE | MOJO_HANDLE_SIGNAL_PEER_CLOSED, | 412 EXPECT_EQ(MOJO_HANDLE_SIGNAL_READABLE | MOJO_HANDLE_SIGNAL_PEER_CLOSED, |
659 state.satisfied_signals); | 413 state.satisfied_signals); |
660 EXPECT_EQ(MOJO_HANDLE_SIGNAL_READABLE | MOJO_HANDLE_SIGNAL_PEER_CLOSED, | 414 EXPECT_EQ(MOJO_HANDLE_SIGNAL_READABLE | MOJO_HANDLE_SIGNAL_PEER_CLOSED, |
661 state.satisfiable_signals); | 415 state.satisfiable_signals); |
662 | 416 |
663 memset(buffer, 0, sizeof(buffer)); | 417 memset(buffer, 0, sizeof(buffer)); |
664 num_bytes = static_cast<uint32_t>(sizeof(buffer)); | 418 num_bytes = static_cast<uint32_t>(sizeof(buffer)); |
665 EXPECT_EQ(MOJO_RESULT_OK, | 419 EXPECT_EQ(MOJO_RESULT_OK, |
666 MojoReadMessage(mp2, buffer, &num_bytes, nullptr, nullptr, | 420 MojoReadMessage(mp2, buffer, &num_bytes, nullptr, nullptr, |
(...skipping 16 matching lines...) Expand all Loading... |
683 EXPECT_EQ(MOJO_HANDLE_SIGNAL_NONE, state.satisfiable_signals); | 437 EXPECT_EQ(MOJO_HANDLE_SIGNAL_NONE, state.satisfiable_signals); |
684 #endif | 438 #endif |
685 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(mp2)); | 439 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(mp2)); |
686 } | 440 } |
687 | 441 |
688 EXPECT_TRUE(multiprocess_test_helper.WaitForChildTestShutdown()); | 442 EXPECT_TRUE(multiprocess_test_helper.WaitForChildTestShutdown()); |
689 } | 443 } |
690 | 444 |
691 MOJO_MULTIPROCESS_TEST_CHILD_TEST(MultiprocessChannelsClient) { | 445 MOJO_MULTIPROCESS_TEST_CHILD_TEST(MultiprocessChannelsClient) { |
692 ScopedPlatformHandle client_platform_handle = | 446 ScopedPlatformHandle client_platform_handle = |
693 mojo::test::MultiprocessTestHelper::client_platform_handle.Pass(); | 447 test::MultiprocessTestHelper::client_platform_handle.Pass(); |
694 EXPECT_TRUE(client_platform_handle.is_valid()); | 448 EXPECT_TRUE(client_platform_handle.is_valid()); |
695 | 449 |
696 base::TestIOThread test_io_thread(base::TestIOThread::kAutoStart); | 450 base::TestIOThread test_io_thread(base::TestIOThread::kAutoStart); |
697 test::InitWithSimplePlatformSupport(); | |
698 | 451 |
699 { | 452 { |
700 // TODO(vtl): This should eventually initialize a slave process instead, | 453 test::ScopedIPCSupport ipc_support(test_io_thread.task_runner()); |
701 // probably. | 454 MojoHandle client_mp = CreateMessagePipe( |
702 mojo::test::ScopedIPCSupport ipc_support(test_io_thread.task_runner()); | 455 client_platform_handle.Pass()).release().value(); |
703 | |
704 ScopedTestChannel client_channel(client_platform_handle.Pass()); | |
705 MojoHandle client_mp = client_channel.bootstrap_message_pipe(); | |
706 EXPECT_NE(client_mp, MOJO_HANDLE_INVALID); | |
707 client_channel.WaitForChannelCreationCompletion(); | |
708 CHECK(client_channel.channel_info() != nullptr); | |
709 | 456 |
710 // 1. Read the first message from |client_mp|. | 457 // 1. Read the first message from |client_mp|. |
711 MojoHandleSignalsState state; | 458 MojoHandleSignalsState state; |
712 EXPECT_EQ(MOJO_RESULT_OK, MojoWait(client_mp, MOJO_HANDLE_SIGNAL_READABLE, | 459 EXPECT_EQ(MOJO_RESULT_OK, MojoWait(client_mp, MOJO_HANDLE_SIGNAL_READABLE, |
713 MOJO_DEADLINE_INDEFINITE, &state)); | 460 MOJO_DEADLINE_INDEFINITE, &state)); |
714 EXPECT_EQ(kSignalReadadableWritable, state.satisfied_signals); | 461 EXPECT_EQ(kSignalReadadableWritable, state.satisfied_signals); |
715 EXPECT_EQ(kSignalAll, state.satisfiable_signals); | 462 EXPECT_EQ(kSignalAll, state.satisfiable_signals); |
716 | 463 |
717 char buffer[1000] = {}; | 464 char buffer[1000] = {}; |
718 uint32_t num_bytes = static_cast<uint32_t>(sizeof(buffer)); | 465 uint32_t num_bytes = static_cast<uint32_t>(sizeof(buffer)); |
(...skipping 77 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
796 EXPECT_STREQ(kFoo, buffer); | 543 EXPECT_STREQ(kFoo, buffer); |
797 | 544 |
798 // 11. Wait on |mp1| (which should eventually fail) and then close it. | 545 // 11. Wait on |mp1| (which should eventually fail) and then close it. |
799 EXPECT_EQ(MOJO_RESULT_FAILED_PRECONDITION, | 546 EXPECT_EQ(MOJO_RESULT_FAILED_PRECONDITION, |
800 MojoWait(mp1, MOJO_HANDLE_SIGNAL_READABLE, | 547 MojoWait(mp1, MOJO_HANDLE_SIGNAL_READABLE, |
801 MOJO_DEADLINE_INDEFINITE, &state)); | 548 MOJO_DEADLINE_INDEFINITE, &state)); |
802 EXPECT_EQ(MOJO_HANDLE_SIGNAL_PEER_CLOSED, state.satisfied_signals); | 549 EXPECT_EQ(MOJO_HANDLE_SIGNAL_PEER_CLOSED, state.satisfied_signals); |
803 EXPECT_EQ(MOJO_HANDLE_SIGNAL_PEER_CLOSED, state.satisfiable_signals); | 550 EXPECT_EQ(MOJO_HANDLE_SIGNAL_PEER_CLOSED, state.satisfiable_signals); |
804 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(mp1)); | 551 EXPECT_EQ(MOJO_RESULT_OK, MojoClose(mp1)); |
805 } | 552 } |
806 | |
807 EXPECT_TRUE(test::Shutdown()); | |
808 } | 553 } |
809 | 554 |
810 // TODO(vtl): Test immediate write & close. | 555 // TODO(vtl): Test immediate write & close. |
811 // TODO(vtl): Test broken-connection cases. | 556 // TODO(vtl): Test broken-connection cases. |
812 | 557 |
813 } // namespace | 558 } // namespace |
814 } // namespace embedder | 559 } // namespace edk |
815 } // namespace mojo | 560 } // namespace mojo |
OLD | NEW |