| 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 "mojo/edk/embedder/embedder.h" | 5 #include "mojo/edk/embedder/embedder.h" |
| 6 | 6 |
| 7 #include <stddef.h> | 7 #include <stddef.h> |
| 8 #include <stdint.h> | 8 #include <stdint.h> |
| 9 #include <string.h> | 9 #include <string.h> |
| 10 | 10 |
| 11 #include <utility> | 11 #include <utility> |
| 12 | 12 |
| 13 #include "base/bind.h" | 13 #include "base/bind.h" |
| 14 #include "base/command_line.h" | 14 #include "base/command_line.h" |
| 15 #include "base/files/file.h" |
| 15 #include "base/logging.h" | 16 #include "base/logging.h" |
| 16 #include "base/macros.h" | 17 #include "base/macros.h" |
| 17 #include "base/memory/shared_memory.h" | 18 #include "base/memory/shared_memory.h" |
| 18 #include "base/message_loop/message_loop.h" | 19 #include "base/message_loop/message_loop.h" |
| 19 #include "base/synchronization/waitable_event.h" | 20 #include "base/synchronization/waitable_event.h" |
| 20 #include "base/test/test_timeouts.h" | 21 #include "base/test/test_timeouts.h" |
| 21 #include "mojo/edk/embedder/platform_channel_pair.h" | 22 #include "mojo/edk/embedder/platform_channel_pair.h" |
| 22 #include "mojo/edk/embedder/test_embedder.h" | 23 #include "mojo/edk/embedder/test_embedder.h" |
| 23 #include "mojo/edk/system/test_utils.h" | 24 #include "mojo/edk/system/test_utils.h" |
| 24 #include "mojo/edk/test/mojo_test_base.h" | 25 #include "mojo/edk/test/mojo_test_base.h" |
| (...skipping 390 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 415 // 6. Map the original base::SharedMemory and expect it contains the | 416 // 6. Map the original base::SharedMemory and expect it contains the |
| 416 // expected value. | 417 // expected value. |
| 417 ASSERT_TRUE(shared_memory.Map(123)); | 418 ASSERT_TRUE(shared_memory.Map(123)); |
| 418 EXPECT_EQ(kByeWorld, | 419 EXPECT_EQ(kByeWorld, |
| 419 std::string(static_cast<char*>(shared_memory.memory()))); | 420 std::string(static_cast<char*>(shared_memory.memory()))); |
| 420 | 421 |
| 421 ASSERT_EQ(MOJO_RESULT_OK, MojoClose(sb1)); | 422 ASSERT_EQ(MOJO_RESULT_OK, MojoClose(sb1)); |
| 422 END_CHILD() | 423 END_CHILD() |
| 423 } | 424 } |
| 424 | 425 |
| 425 const base::SharedMemoryHandle::Type kTestHandleTypes[] = { | 426 enum class HandleType { |
| 426 base::SharedMemoryHandle::MACH, | 427 POSIX, |
| 427 base::SharedMemoryHandle::POSIX, | 428 MACH, |
| 428 base::SharedMemoryHandle::POSIX, | 429 MACH_NULL, |
| 429 base::SharedMemoryHandle::MACH, | |
| 430 }; | 430 }; |
| 431 | 431 |
| 432 // Test that we can mix file descriptor and mach port handles. | 432 const HandleType kTestHandleTypes[] = { |
| 433 HandleType::MACH, |
| 434 HandleType::MACH_NULL, |
| 435 HandleType::POSIX, |
| 436 HandleType::POSIX, |
| 437 HandleType::MACH, |
| 438 }; |
| 439 |
| 440 // Test that we can mix file descriptors and mach port handles. |
| 433 TEST_F(EmbedderTest, MultiprocessMixMachAndFds) { | 441 TEST_F(EmbedderTest, MultiprocessMixMachAndFds) { |
| 434 const size_t kShmSize = 1234; | 442 const size_t kShmSize = 1234; |
| 435 RUN_CHILD_ON_PIPE(MultiprocessMixMachAndFdsClient, server_mp) | 443 RUN_CHILD_ON_PIPE(MultiprocessMixMachAndFdsClient, server_mp) |
| 436 // 1. Create the base::SharedMemory objects and mojo handles from them. | 444 // 1. Create fds or Mach objects and mojo handles from them. |
| 437 MojoHandle platform_handles[arraysize(kTestHandleTypes)]; | 445 MojoHandle platform_handles[arraysize(kTestHandleTypes)]; |
| 438 for (size_t i = 0; i < arraysize(kTestHandleTypes); i++) { | 446 for (size_t i = 0; i < arraysize(kTestHandleTypes); i++) { |
| 439 const auto type = kTestHandleTypes[i]; | 447 const auto type = kTestHandleTypes[i]; |
| 440 base::SharedMemoryCreateOptions options; | |
| 441 options.size = kShmSize; | |
| 442 options.type = type; | |
| 443 base::SharedMemory shared_memory; | |
| 444 ASSERT_TRUE(shared_memory.Create(options)); | |
| 445 base::SharedMemoryHandle shm_handle = base::SharedMemory::DuplicateHandle( | |
| 446 shared_memory.handle()); | |
| 447 ScopedPlatformHandle scoped_handle; | 448 ScopedPlatformHandle scoped_handle; |
| 448 if (type == base::SharedMemoryHandle::POSIX) | 449 if (type == HandleType::POSIX) { |
| 449 scoped_handle.reset(PlatformHandle(shm_handle.GetFileDescriptor().fd)); | 450 // The easiest source of fds is opening /dev/null. |
| 450 else | 451 base::File file(base::FilePath("/dev/null"), |
| 452 base::File::FLAG_OPEN | base::File::FLAG_WRITE); |
| 453 ASSERT_TRUE(file.IsValid()); |
| 454 scoped_handle.reset(PlatformHandle(file.TakePlatformFile())); |
| 455 EXPECT_EQ(PlatformHandle::Type::POSIX, scoped_handle.get().type); |
| 456 } else if (type == HandleType::MACH_NULL) { |
| 457 scoped_handle.reset(PlatformHandle( |
| 458 static_cast<mach_port_t>(MACH_PORT_NULL))); |
| 459 EXPECT_EQ(PlatformHandle::Type::MACH, scoped_handle.get().type); |
| 460 } else { |
| 461 base::SharedMemoryCreateOptions options; |
| 462 options.size = kShmSize; |
| 463 options.type = base::SharedMemoryHandle::MACH; |
| 464 base::SharedMemory shared_memory; |
| 465 ASSERT_TRUE(shared_memory.Create(options)); |
| 466 base::SharedMemoryHandle shm_handle = |
| 467 base::SharedMemory::DuplicateHandle(shared_memory.handle()); |
| 451 scoped_handle.reset(PlatformHandle(shm_handle.GetMemoryObject())); | 468 scoped_handle.reset(PlatformHandle(shm_handle.GetMemoryObject())); |
| 469 EXPECT_EQ(PlatformHandle::Type::MACH, scoped_handle.get().type); |
| 470 } |
| 452 ASSERT_EQ(MOJO_RESULT_OK, CreatePlatformHandleWrapper( | 471 ASSERT_EQ(MOJO_RESULT_OK, CreatePlatformHandleWrapper( |
| 453 std::move(scoped_handle), platform_handles + i)); | 472 std::move(scoped_handle), platform_handles + i)); |
| 454 | |
| 455 // Map the shared memory object and write the type into it. 'P' for POSIX, | |
| 456 // and 'M' for Mach. | |
| 457 ASSERT_TRUE(shared_memory.Map(kShmSize)); | |
| 458 static_cast<char*>(shared_memory.memory())[0] = | |
| 459 type == base::SharedMemoryHandle::POSIX ? 'P' : 'M'; | |
| 460 } | 473 } |
| 461 | 474 |
| 462 // 2. Send all the handles to the child. | 475 // 2. Send all the handles to the child. |
| 463 WriteMessageWithHandles(server_mp, "hello", platform_handles, | 476 WriteMessageWithHandles(server_mp, "hello", platform_handles, |
| 464 arraysize(kTestHandleTypes)); | 477 arraysize(kTestHandleTypes)); |
| 465 | 478 |
| 466 // 3. Read a message from |server_mp|. | 479 // 3. Read a message from |server_mp|. |
| 467 EXPECT_EQ("bye", ReadMessage(server_mp)); | 480 EXPECT_EQ("bye", ReadMessage(server_mp)); |
| 468 END_CHILD() | 481 END_CHILD() |
| 469 } | 482 } |
| 470 | 483 |
| 471 DEFINE_TEST_CLIENT_TEST_WITH_PIPE(MultiprocessMixMachAndFdsClient, EmbedderTest, | 484 DEFINE_TEST_CLIENT_TEST_WITH_PIPE(MultiprocessMixMachAndFdsClient, EmbedderTest, |
| 472 client_mp) { | 485 client_mp) { |
| 473 const int kNumHandles = 4; | 486 const int kNumHandles = arraysize(kTestHandleTypes); |
| 474 const size_t kShmSize = 1234; | |
| 475 MojoHandle platform_handles[kNumHandles]; | 487 MojoHandle platform_handles[kNumHandles]; |
| 476 | 488 |
| 477 // 1. Read from |client_mp|, which should have a message containing | 489 // 1. Read from |client_mp|, which should have a message containing |
| 478 // |kNumHandles| handles. | 490 // |kNumHandles| handles. |
| 479 EXPECT_EQ("hello", | 491 EXPECT_EQ("hello", |
| 480 ReadMessageWithHandles(client_mp, platform_handles, kNumHandles)); | 492 ReadMessageWithHandles(client_mp, platform_handles, kNumHandles)); |
| 481 | 493 |
| 482 // 2. Extract each handle, map it, and verify the type. | 494 // 2. Extract each handle, and verify the type. |
| 483 for (int i = 0; i < kNumHandles; i++) { | 495 for (int i = 0; i < kNumHandles; i++) { |
| 496 const auto type = kTestHandleTypes[i]; |
| 484 ScopedPlatformHandle scoped_handle; | 497 ScopedPlatformHandle scoped_handle; |
| 485 ASSERT_EQ(MOJO_RESULT_OK, | 498 ASSERT_EQ(MOJO_RESULT_OK, |
| 486 PassWrappedPlatformHandle(platform_handles[i], &scoped_handle)); | 499 PassWrappedPlatformHandle(platform_handles[i], &scoped_handle)); |
| 487 base::SharedMemoryHandle shm_handle; | 500 if (type == HandleType::POSIX) { |
| 488 char type = 0; | 501 EXPECT_NE(0, scoped_handle.get().handle); |
| 489 if (scoped_handle.get().type == PlatformHandle::Type::POSIX) { | 502 EXPECT_EQ(PlatformHandle::Type::POSIX, scoped_handle.get().type); |
| 490 shm_handle = base::SharedMemoryHandle(scoped_handle.release().handle, | 503 } else if (type == HandleType::MACH_NULL) { |
| 491 false); | 504 EXPECT_EQ(static_cast<mach_port_t>(MACH_PORT_NULL), |
| 492 type = 'P'; | 505 scoped_handle.get().port); |
| 506 EXPECT_EQ(PlatformHandle::Type::MACH, scoped_handle.get().type); |
| 493 } else { | 507 } else { |
| 494 shm_handle = base::SharedMemoryHandle(scoped_handle.release().port, | 508 EXPECT_NE(static_cast<mach_port_t>(MACH_PORT_NULL), |
| 495 kShmSize, base::GetCurrentProcId()); | 509 scoped_handle.get().port); |
| 496 type = 'M'; | 510 EXPECT_EQ(PlatformHandle::Type::MACH, scoped_handle.get().type); |
| 497 } | 511 } |
| 498 | |
| 499 // Verify the type order. | |
| 500 EXPECT_EQ(kTestHandleTypes[i], shm_handle.GetType()); | |
| 501 | |
| 502 base::SharedMemory shared_memory(shm_handle, false); | |
| 503 ASSERT_TRUE(shared_memory.Map(kShmSize)); | |
| 504 EXPECT_EQ(type, static_cast<char*>(shared_memory.memory())[0]); | |
| 505 } | 512 } |
| 506 | 513 |
| 507 // 3. Say bye! | 514 // 3. Say bye! |
| 508 WriteMessage(client_mp, "bye"); | 515 WriteMessage(client_mp, "bye"); |
| 509 } | 516 } |
| 510 | 517 |
| 511 #endif // defined(OS_MACOSX) && !defined(OS_IOS) | 518 #endif // defined(OS_MACOSX) && !defined(OS_IOS) |
| 512 | 519 |
| 513 // TODO(vtl): Test immediate write & close. | 520 // TODO(vtl): Test immediate write & close. |
| 514 // TODO(vtl): Test broken-connection cases. | 521 // TODO(vtl): Test broken-connection cases. |
| 515 | 522 |
| 516 #endif // !defined(OS_IOS) | 523 #endif // !defined(OS_IOS) |
| 517 | 524 |
| 518 } // namespace | 525 } // namespace |
| 519 } // namespace edk | 526 } // namespace edk |
| 520 } // namespace mojo | 527 } // namespace mojo |
| OLD | NEW |