OLD | NEW |
1 // Copyright 2013 The Chromium Authors. All rights reserved. | 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 | 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 <stdint.h> | 5 #include <stdint.h> |
6 #include <utility> | 6 #include <utility> |
7 | 7 |
8 #include "base/message_loop/message_loop.h" | 8 #include "base/message_loop/message_loop.h" |
| 9 #include "base/run_loop.h" |
9 #include "mojo/message_pump/message_pump_mojo.h" | 10 #include "mojo/message_pump/message_pump_mojo.h" |
10 #include "mojo/public/cpp/bindings/binding.h" | 11 #include "mojo/public/cpp/bindings/binding.h" |
11 #include "mojo/public/cpp/bindings/strong_binding.h" | 12 #include "mojo/public/cpp/bindings/strong_binding.h" |
12 #include "mojo/public/cpp/test_support/test_utils.h" | 13 #include "mojo/public/cpp/test_support/test_utils.h" |
13 #include "mojo/public/interfaces/bindings/tests/sample_factory.mojom.h" | 14 #include "mojo/public/interfaces/bindings/tests/sample_factory.mojom.h" |
14 #include "testing/gtest/include/gtest/gtest.h" | 15 #include "testing/gtest/include/gtest/gtest.h" |
15 | 16 |
16 namespace mojo { | 17 namespace mojo { |
17 namespace test { | 18 namespace test { |
18 namespace { | 19 namespace { |
19 | 20 |
20 const char kText1[] = "hello"; | 21 const char kText1[] = "hello"; |
21 const char kText2[] = "world"; | 22 const char kText2[] = "world"; |
22 | 23 |
23 class StringRecorder { | 24 class StringRecorder { |
24 public: | 25 public: |
25 explicit StringRecorder(std::string* buf) : buf_(buf) {} | 26 StringRecorder(std::string* buf, const base::Closure& closure) |
26 void Run(const String& a) const { *buf_ = a.To<std::string>(); } | 27 : buf_(buf), closure_(closure) {} |
| 28 void Run(const String& a) const { |
| 29 *buf_ = a.To<std::string>(); |
| 30 closure_.Run(); |
| 31 } |
27 | 32 |
28 private: | 33 private: |
29 std::string* buf_; | 34 std::string* buf_; |
| 35 base::Closure closure_; |
30 }; | 36 }; |
31 | 37 |
32 class ImportedInterfaceImpl : public imported::ImportedInterface { | 38 class ImportedInterfaceImpl : public imported::ImportedInterface { |
33 public: | 39 public: |
34 explicit ImportedInterfaceImpl( | 40 ImportedInterfaceImpl( |
35 InterfaceRequest<imported::ImportedInterface> request) | 41 InterfaceRequest<imported::ImportedInterface> request, |
36 : binding_(this, std::move(request)) {} | 42 const base::Closure& closure) |
| 43 : binding_(this, std::move(request)), closure_(closure) {} |
37 | 44 |
38 void DoSomething() override { do_something_count_++; } | 45 void DoSomething() override { |
| 46 do_something_count_++; |
| 47 closure_.Run(); |
| 48 } |
39 | 49 |
40 static int do_something_count() { return do_something_count_; } | 50 static int do_something_count() { return do_something_count_; } |
41 | 51 |
42 private: | 52 private: |
43 static int do_something_count_; | 53 static int do_something_count_; |
44 Binding<ImportedInterface> binding_; | 54 Binding<ImportedInterface> binding_; |
| 55 base::Closure closure_; |
45 }; | 56 }; |
46 int ImportedInterfaceImpl::do_something_count_ = 0; | 57 int ImportedInterfaceImpl::do_something_count_ = 0; |
47 | 58 |
48 class SampleNamedObjectImpl : public sample::NamedObject { | 59 class SampleNamedObjectImpl : public sample::NamedObject { |
49 public: | 60 public: |
50 explicit SampleNamedObjectImpl(InterfaceRequest<sample::NamedObject> request) | 61 explicit SampleNamedObjectImpl(InterfaceRequest<sample::NamedObject> request) |
51 : binding_(this, std::move(request)) {} | 62 : binding_(this, std::move(request)) {} |
52 void SetName(const mojo::String& name) override { name_ = name; } | 63 void SetName(const mojo::String& name) override { name_ = name; } |
53 | 64 |
54 void GetName(const mojo::Callback<void(mojo::String)>& callback) override { | 65 void GetName(const mojo::Callback<void(mojo::String)>& callback) override { |
(...skipping 39 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
94 if (request->obj) | 105 if (request->obj) |
95 request->obj->DoSomething(); | 106 request->obj->DoSomething(); |
96 } | 107 } |
97 | 108 |
98 void DoStuff2(ScopedDataPipeConsumerHandle pipe, | 109 void DoStuff2(ScopedDataPipeConsumerHandle pipe, |
99 const DoStuff2Callback& callback) override { | 110 const DoStuff2Callback& callback) override { |
100 // Read the data from the pipe, writing the response (as a string) to | 111 // Read the data from the pipe, writing the response (as a string) to |
101 // DidStuff2(). | 112 // DidStuff2(). |
102 ASSERT_TRUE(pipe.is_valid()); | 113 ASSERT_TRUE(pipe.is_valid()); |
103 uint32_t data_size = 0; | 114 uint32_t data_size = 0; |
| 115 |
| 116 MojoHandleSignalsState state; |
| 117 ASSERT_EQ(MOJO_RESULT_OK, |
| 118 MojoWait(pipe.get().value(), MOJO_HANDLE_SIGNAL_READABLE, |
| 119 MOJO_DEADLINE_INDEFINITE, &state)); |
| 120 ASSERT_EQ(MOJO_HANDLE_SIGNAL_READABLE, state.satisfied_signals); |
104 ASSERT_EQ(MOJO_RESULT_OK, | 121 ASSERT_EQ(MOJO_RESULT_OK, |
105 ReadDataRaw( | 122 ReadDataRaw( |
106 pipe.get(), nullptr, &data_size, MOJO_READ_DATA_FLAG_QUERY)); | 123 pipe.get(), nullptr, &data_size, MOJO_READ_DATA_FLAG_QUERY)); |
107 ASSERT_NE(0, static_cast<int>(data_size)); | 124 ASSERT_NE(0, static_cast<int>(data_size)); |
108 char data[64]; | 125 char data[64]; |
109 ASSERT_LT(static_cast<int>(data_size), 64); | 126 ASSERT_LT(static_cast<int>(data_size), 64); |
110 ASSERT_EQ( | 127 ASSERT_EQ( |
111 MOJO_RESULT_OK, | 128 MOJO_RESULT_OK, |
112 ReadDataRaw( | 129 ReadDataRaw( |
113 pipe.get(), data, &data_size, MOJO_READ_DATA_FLAG_ALL_OR_NONE)); | 130 pipe.get(), data, &data_size, MOJO_READ_DATA_FLAG_ALL_OR_NONE)); |
(...skipping 30 matching lines...) Expand all Loading... |
144 | 161 |
145 void TearDown() override { PumpMessages(); } | 162 void TearDown() override { PumpMessages(); } |
146 | 163 |
147 void PumpMessages() { loop_.RunUntilIdle(); } | 164 void PumpMessages() { loop_.RunUntilIdle(); } |
148 | 165 |
149 private: | 166 private: |
150 base::MessageLoop loop_; | 167 base::MessageLoop loop_; |
151 }; | 168 }; |
152 | 169 |
153 struct DoStuffCallback { | 170 struct DoStuffCallback { |
154 DoStuffCallback(bool* got_response, std::string* got_text_reply) | 171 DoStuffCallback(bool* got_response, |
155 : got_response(got_response), got_text_reply(got_text_reply) {} | 172 std::string* got_text_reply, |
| 173 const base::Closure& closure) |
| 174 : got_response(got_response), |
| 175 got_text_reply(got_text_reply), |
| 176 closure(closure) {} |
156 | 177 |
157 void Run(sample::ResponsePtr response, const String& text_reply) const { | 178 void Run(sample::ResponsePtr response, const String& text_reply) const { |
158 *got_text_reply = text_reply; | 179 *got_text_reply = text_reply; |
159 | 180 |
160 if (response->pipe.is_valid()) { | 181 if (response->pipe.is_valid()) { |
161 std::string text2; | 182 std::string text2; |
162 EXPECT_TRUE(ReadTextMessage(response->pipe.get(), &text2)); | 183 EXPECT_TRUE(ReadTextMessage(response->pipe.get(), &text2)); |
163 | 184 |
164 // Ensure that simply accessing response.pipe does not close it. | 185 // Ensure that simply accessing response.pipe does not close it. |
165 EXPECT_TRUE(response->pipe.is_valid()); | 186 EXPECT_TRUE(response->pipe.is_valid()); |
166 | 187 |
167 EXPECT_EQ(std::string(kText2), text2); | 188 EXPECT_EQ(std::string(kText2), text2); |
168 | 189 |
169 // Do some more tests of handle passing: | 190 // Do some more tests of handle passing: |
170 ScopedMessagePipeHandle p = std::move(response->pipe); | 191 ScopedMessagePipeHandle p = std::move(response->pipe); |
171 EXPECT_TRUE(p.is_valid()); | 192 EXPECT_TRUE(p.is_valid()); |
172 EXPECT_FALSE(response->pipe.is_valid()); | 193 EXPECT_FALSE(response->pipe.is_valid()); |
173 } | 194 } |
174 | 195 |
175 *got_response = true; | 196 *got_response = true; |
| 197 closure.Run(); |
176 } | 198 } |
177 | 199 |
178 bool* got_response; | 200 bool* got_response; |
179 std::string* got_text_reply; | 201 std::string* got_text_reply; |
| 202 base::Closure closure; |
180 }; | 203 }; |
181 | 204 |
182 TEST_F(HandlePassingTest, Basic) { | 205 TEST_F(HandlePassingTest, Basic) { |
183 sample::FactoryPtr factory; | 206 sample::FactoryPtr factory; |
184 SampleFactoryImpl factory_impl(GetProxy(&factory)); | 207 SampleFactoryImpl factory_impl(GetProxy(&factory)); |
185 | 208 |
186 MessagePipe pipe0; | 209 MessagePipe pipe0; |
187 EXPECT_TRUE(WriteTextMessage(pipe0.handle1.get(), kText1)); | 210 EXPECT_TRUE(WriteTextMessage(pipe0.handle1.get(), kText1)); |
188 | 211 |
189 MessagePipe pipe1; | 212 MessagePipe pipe1; |
190 EXPECT_TRUE(WriteTextMessage(pipe1.handle1.get(), kText2)); | 213 EXPECT_TRUE(WriteTextMessage(pipe1.handle1.get(), kText2)); |
191 | 214 |
192 imported::ImportedInterfacePtr imported; | 215 imported::ImportedInterfacePtr imported; |
193 ImportedInterfaceImpl imported_impl(GetProxy(&imported)); | 216 base::RunLoop run_loop; |
| 217 ImportedInterfaceImpl imported_impl(GetProxy(&imported), |
| 218 run_loop.QuitClosure()); |
194 | 219 |
195 sample::RequestPtr request(sample::Request::New()); | 220 sample::RequestPtr request(sample::Request::New()); |
196 request->x = 1; | 221 request->x = 1; |
197 request->pipe = std::move(pipe1.handle0); | 222 request->pipe = std::move(pipe1.handle0); |
198 request->obj = std::move(imported); | 223 request->obj = std::move(imported); |
199 bool got_response = false; | 224 bool got_response = false; |
200 std::string got_text_reply; | 225 std::string got_text_reply; |
201 DoStuffCallback cb(&got_response, &got_text_reply); | 226 base::RunLoop run_loop2; |
| 227 DoStuffCallback cb(&got_response, &got_text_reply, run_loop2.QuitClosure()); |
202 factory->DoStuff(std::move(request), std::move(pipe0.handle0), cb); | 228 factory->DoStuff(std::move(request), std::move(pipe0.handle0), cb); |
203 | 229 |
204 EXPECT_FALSE(*cb.got_response); | 230 EXPECT_FALSE(*cb.got_response); |
205 int count_before = ImportedInterfaceImpl::do_something_count(); | 231 int count_before = ImportedInterfaceImpl::do_something_count(); |
206 | 232 |
207 PumpMessages(); | 233 run_loop.Run(); |
| 234 run_loop2.Run(); |
208 | 235 |
209 EXPECT_TRUE(*cb.got_response); | 236 EXPECT_TRUE(*cb.got_response); |
210 EXPECT_EQ(kText1, *cb.got_text_reply); | 237 EXPECT_EQ(kText1, *cb.got_text_reply); |
211 EXPECT_EQ(1, ImportedInterfaceImpl::do_something_count() - count_before); | 238 EXPECT_EQ(1, ImportedInterfaceImpl::do_something_count() - count_before); |
212 } | 239 } |
213 | 240 |
214 TEST_F(HandlePassingTest, PassInvalid) { | 241 TEST_F(HandlePassingTest, PassInvalid) { |
215 sample::FactoryPtr factory; | 242 sample::FactoryPtr factory; |
216 SampleFactoryImpl factory_impl(GetProxy(&factory)); | 243 SampleFactoryImpl factory_impl(GetProxy(&factory)); |
217 | 244 |
218 sample::RequestPtr request(sample::Request::New()); | 245 sample::RequestPtr request(sample::Request::New()); |
219 request->x = 1; | 246 request->x = 1; |
220 bool got_response = false; | 247 bool got_response = false; |
221 std::string got_text_reply; | 248 std::string got_text_reply; |
222 DoStuffCallback cb(&got_response, &got_text_reply); | 249 base::RunLoop run_loop; |
| 250 DoStuffCallback cb(&got_response, &got_text_reply, run_loop.QuitClosure()); |
223 factory->DoStuff(std::move(request), ScopedMessagePipeHandle(), cb); | 251 factory->DoStuff(std::move(request), ScopedMessagePipeHandle(), cb); |
224 | 252 |
225 EXPECT_FALSE(*cb.got_response); | 253 EXPECT_FALSE(*cb.got_response); |
226 | 254 |
227 PumpMessages(); | 255 run_loop.Run(); |
228 | 256 |
229 EXPECT_TRUE(*cb.got_response); | 257 EXPECT_TRUE(*cb.got_response); |
230 } | 258 } |
231 | 259 |
232 struct DoStuff2Callback { | 260 struct DoStuff2Callback { |
233 DoStuff2Callback(bool* got_response, std::string* got_text_reply) | 261 DoStuff2Callback(bool* got_response, |
234 : got_response(got_response), got_text_reply(got_text_reply) {} | 262 std::string* got_text_reply, |
| 263 const base::Closure& closure) |
| 264 : got_response(got_response), |
| 265 got_text_reply(got_text_reply), |
| 266 closure(closure) {} |
235 | 267 |
236 void Run(const String& text_reply) const { | 268 void Run(const String& text_reply) const { |
237 *got_response = true; | 269 *got_response = true; |
238 *got_text_reply = text_reply; | 270 *got_text_reply = text_reply; |
| 271 closure.Run(); |
239 } | 272 } |
240 | 273 |
241 bool* got_response; | 274 bool* got_response; |
242 std::string* got_text_reply; | 275 std::string* got_text_reply; |
| 276 base::Closure closure; |
243 }; | 277 }; |
244 | 278 |
245 // Verifies DataPipeConsumer can be passed and read from. | 279 // Verifies DataPipeConsumer can be passed and read from. |
246 TEST_F(HandlePassingTest, DataPipe) { | 280 TEST_F(HandlePassingTest, DataPipe) { |
247 sample::FactoryPtr factory; | 281 sample::FactoryPtr factory; |
248 SampleFactoryImpl factory_impl(GetProxy(&factory)); | 282 SampleFactoryImpl factory_impl(GetProxy(&factory)); |
249 | 283 |
250 // Writes a string to a data pipe and passes the data pipe (consumer) to the | 284 // Writes a string to a data pipe and passes the data pipe (consumer) to the |
251 // factory. | 285 // factory. |
252 ScopedDataPipeProducerHandle producer_handle; | 286 ScopedDataPipeProducerHandle producer_handle; |
253 ScopedDataPipeConsumerHandle consumer_handle; | 287 ScopedDataPipeConsumerHandle consumer_handle; |
254 MojoCreateDataPipeOptions options = {sizeof(MojoCreateDataPipeOptions), | 288 MojoCreateDataPipeOptions options = {sizeof(MojoCreateDataPipeOptions), |
255 MOJO_CREATE_DATA_PIPE_OPTIONS_FLAG_NONE, | 289 MOJO_CREATE_DATA_PIPE_OPTIONS_FLAG_NONE, |
256 1, | 290 1, |
257 1024}; | 291 1024}; |
258 ASSERT_EQ(MOJO_RESULT_OK, | 292 ASSERT_EQ(MOJO_RESULT_OK, |
259 CreateDataPipe(&options, &producer_handle, &consumer_handle)); | 293 CreateDataPipe(&options, &producer_handle, &consumer_handle)); |
260 std::string expected_text_reply = "got it"; | 294 std::string expected_text_reply = "got it"; |
261 // +1 for \0. | 295 // +1 for \0. |
262 uint32_t data_size = static_cast<uint32_t>(expected_text_reply.size() + 1); | 296 uint32_t data_size = static_cast<uint32_t>(expected_text_reply.size() + 1); |
263 ASSERT_EQ(MOJO_RESULT_OK, | 297 ASSERT_EQ(MOJO_RESULT_OK, |
264 WriteDataRaw(producer_handle.get(), | 298 WriteDataRaw(producer_handle.get(), |
265 expected_text_reply.c_str(), | 299 expected_text_reply.c_str(), |
266 &data_size, | 300 &data_size, |
267 MOJO_WRITE_DATA_FLAG_ALL_OR_NONE)); | 301 MOJO_WRITE_DATA_FLAG_ALL_OR_NONE)); |
268 | 302 |
269 bool got_response = false; | 303 bool got_response = false; |
270 std::string got_text_reply; | 304 std::string got_text_reply; |
271 DoStuff2Callback cb(&got_response, &got_text_reply); | 305 base::RunLoop run_loop; |
| 306 DoStuff2Callback cb(&got_response, &got_text_reply, run_loop.QuitClosure()); |
272 factory->DoStuff2(std::move(consumer_handle), cb); | 307 factory->DoStuff2(std::move(consumer_handle), cb); |
273 | 308 |
274 EXPECT_FALSE(*cb.got_response); | 309 EXPECT_FALSE(*cb.got_response); |
275 | 310 |
276 PumpMessages(); | 311 run_loop.Run(); |
277 | 312 |
278 EXPECT_TRUE(*cb.got_response); | 313 EXPECT_TRUE(*cb.got_response); |
279 EXPECT_EQ(expected_text_reply, *cb.got_text_reply); | 314 EXPECT_EQ(expected_text_reply, *cb.got_text_reply); |
280 } | 315 } |
281 | 316 |
282 TEST_F(HandlePassingTest, PipesAreClosed) { | 317 TEST_F(HandlePassingTest, PipesAreClosed) { |
283 sample::FactoryPtr factory; | 318 sample::FactoryPtr factory; |
284 SampleFactoryImpl factory_impl(GetProxy(&factory)); | 319 SampleFactoryImpl factory_impl(GetProxy(&factory)); |
285 | 320 |
286 MessagePipe extra_pipe; | 321 MessagePipe extra_pipe; |
(...skipping 45 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
332 factory->CreateNamedObject(std::move(object1_request)); | 367 factory->CreateNamedObject(std::move(object1_request)); |
333 EXPECT_FALSE(object1_request.is_pending()); // We've passed the request. | 368 EXPECT_FALSE(object1_request.is_pending()); // We've passed the request. |
334 | 369 |
335 ASSERT_TRUE(object1); | 370 ASSERT_TRUE(object1); |
336 object1->SetName("object1"); | 371 object1->SetName("object1"); |
337 | 372 |
338 sample::NamedObjectPtr object2; | 373 sample::NamedObjectPtr object2; |
339 factory->CreateNamedObject(GetProxy(&object2)); | 374 factory->CreateNamedObject(GetProxy(&object2)); |
340 object2->SetName("object2"); | 375 object2->SetName("object2"); |
341 | 376 |
| 377 base::RunLoop run_loop, run_loop2; |
342 std::string name1; | 378 std::string name1; |
343 object1->GetName(StringRecorder(&name1)); | 379 object1->GetName(StringRecorder(&name1, run_loop.QuitClosure())); |
344 | 380 |
345 std::string name2; | 381 std::string name2; |
346 object2->GetName(StringRecorder(&name2)); | 382 object2->GetName(StringRecorder(&name2, run_loop2.QuitClosure())); |
347 | 383 |
348 PumpMessages(); // Yield for results. | 384 run_loop.Run(); |
| 385 run_loop2.Run(); |
349 | 386 |
350 EXPECT_EQ(std::string("object1"), name1); | 387 EXPECT_EQ(std::string("object1"), name1); |
351 EXPECT_EQ(std::string("object2"), name2); | 388 EXPECT_EQ(std::string("object2"), name2); |
352 } | 389 } |
353 | 390 |
354 } // namespace | 391 } // namespace |
355 } // namespace test | 392 } // namespace test |
356 } // namespace mojo | 393 } // namespace mojo |
OLD | NEW |