| 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 "sandbox/linux/services/thread_helpers.h" | 5 #include "sandbox/linux/services/thread_helpers.h" |
| 6 | 6 |
| 7 #include <errno.h> | 7 #include <errno.h> |
| 8 #include <fcntl.h> | 8 #include <fcntl.h> |
| 9 #include <sys/stat.h> | 9 #include <sys/stat.h> |
| 10 #include <sys/types.h> | 10 #include <sys/types.h> |
| 11 #include <unistd.h> | 11 #include <unistd.h> |
| 12 | 12 |
| 13 #include "base/basictypes.h" | 13 #include "base/basictypes.h" |
| 14 #include "base/logging.h" | 14 #include "base/logging.h" |
| 15 #include "base/memory/scoped_ptr.h" | 15 #include "base/memory/scoped_ptr.h" |
| 16 #include "base/posix/eintr_wrapper.h" | 16 #include "base/posix/eintr_wrapper.h" |
| 17 #include "base/process/process_metrics.h" | 17 #include "base/process/process_metrics.h" |
| 18 #include "base/threading/platform_thread.h" | 18 #include "base/threading/platform_thread.h" |
| 19 #include "base/threading/thread.h" | 19 #include "base/threading/thread.h" |
| 20 #include "build/build_config.h" | 20 #include "build/build_config.h" |
| 21 #include "sandbox/linux/tests/unit_tests.h" | 21 #include "sandbox/linux/tests/unit_tests.h" |
| 22 #include "testing/gtest/include/gtest/gtest.h" | 22 #include "testing/gtest/include/gtest/gtest.h" |
| 23 | 23 |
| 24 using base::PlatformThread; | 24 using base::PlatformThread; |
| 25 | 25 |
| 26 namespace sandbox { | 26 namespace sandbox { |
| 27 | 27 |
| 28 namespace { | 28 namespace { |
| 29 | 29 |
| 30 // These tests fail under ThreadSanitizer, see http://crbug.com/342305 |
| 31 #if !defined(THREAD_SANITIZER) |
| 32 |
| 30 int GetRaceTestIterations() { | 33 int GetRaceTestIterations() { |
| 31 if (IsRunningOnValgrind()) { | 34 if (IsRunningOnValgrind()) { |
| 32 return 2; | 35 return 2; |
| 33 } else { | 36 } else { |
| 34 return 1000; | 37 return 1000; |
| 35 } | 38 } |
| 36 } | 39 } |
| 37 | 40 |
| 38 class ScopedProcSelfTask { | 41 class ScopedProcSelfTask { |
| 39 public: | 42 public: |
| 40 ScopedProcSelfTask() : fd_(-1) { | 43 ScopedProcSelfTask() : fd_(-1) { |
| 41 fd_ = open("/proc/self/task/", O_RDONLY | O_DIRECTORY); | 44 fd_ = open("/proc/self/task/", O_RDONLY | O_DIRECTORY); |
| 42 CHECK_LE(0, fd_); | 45 CHECK_LE(0, fd_); |
| 43 } | 46 } |
| 44 | 47 |
| 45 ~ScopedProcSelfTask() { PCHECK(0 == IGNORE_EINTR(close(fd_))); } | 48 ~ScopedProcSelfTask() { PCHECK(0 == IGNORE_EINTR(close(fd_))); } |
| 46 | 49 |
| 47 int fd() { return fd_; } | 50 int fd() { return fd_; } |
| 48 | 51 |
| 49 private: | 52 private: |
| 50 int fd_; | 53 int fd_; |
| 51 DISALLOW_COPY_AND_ASSIGN(ScopedProcSelfTask); | 54 DISALLOW_COPY_AND_ASSIGN(ScopedProcSelfTask); |
| 52 }; | 55 }; |
| 53 | 56 |
| 54 #if defined(THREAD_SANITIZER) | 57 TEST(ThreadHelpers, IsSingleThreadedBasic) { |
| 55 // These tests fail under ThreadSanitizer, see http://crbug.com/342305 | |
| 56 #define MAYBE_IsSingleThreadedBasic DISABLED_IsSingleThreadedBasic | |
| 57 #define MAYBE_IsSingleThreadedIterated DISABLED_IsSingleThreadedIterated | |
| 58 #define MAYBE_IsSingleThreadedStartAndStop DISABLED_IsSingleThreadedStartAndStop | |
| 59 #else | |
| 60 #define MAYBE_IsSingleThreadedBasic IsSingleThreadedBasic | |
| 61 #define MAYBE_IsSingleThreadedIterated IsSingleThreadedIterated | |
| 62 #define MAYBE_IsSingleThreadedStartAndStop IsSingleThreadedStartAndStop | |
| 63 #endif | |
| 64 | |
| 65 TEST(ThreadHelpers, MAYBE_IsSingleThreadedBasic) { | |
| 66 ScopedProcSelfTask task; | 58 ScopedProcSelfTask task; |
| 67 ASSERT_TRUE(ThreadHelpers::IsSingleThreaded(task.fd())); | 59 ASSERT_TRUE(ThreadHelpers::IsSingleThreaded(task.fd())); |
| 68 ASSERT_TRUE(ThreadHelpers::IsSingleThreaded(-1)); | 60 ASSERT_TRUE(ThreadHelpers::IsSingleThreaded()); |
| 69 | 61 |
| 70 base::Thread thread("sandbox_tests"); | 62 base::Thread thread("sandbox_tests"); |
| 71 ASSERT_TRUE(thread.Start()); | 63 ASSERT_TRUE(thread.Start()); |
| 72 ASSERT_FALSE(ThreadHelpers::IsSingleThreaded(task.fd())); | 64 ASSERT_FALSE(ThreadHelpers::IsSingleThreaded(task.fd())); |
| 73 ASSERT_FALSE(ThreadHelpers::IsSingleThreaded(-1)); | 65 ASSERT_FALSE(ThreadHelpers::IsSingleThreaded()); |
| 74 // Explicitly stop the thread here to not pollute the next test. | 66 // Explicitly stop the thread here to not pollute the next test. |
| 75 ASSERT_TRUE(ThreadHelpers::StopThreadAndWatchProcFS(task.fd(), &thread)); | 67 ASSERT_TRUE(ThreadHelpers::StopThreadAndWatchProcFS(task.fd(), &thread)); |
| 76 } | 68 } |
| 77 | 69 |
| 78 TEST(ThreadHelpers, MAYBE_IsSingleThreadedIterated) { | 70 SANDBOX_TEST(ThreadHelpers, AssertSingleThreaded) { |
| 71 ScopedProcSelfTask task; |
| 72 SANDBOX_ASSERT(ThreadHelpers::IsSingleThreaded(task.fd())); |
| 73 SANDBOX_ASSERT(ThreadHelpers::IsSingleThreaded()); |
| 74 |
| 75 ThreadHelpers::AssertSingleThreaded(task.fd()); |
| 76 ThreadHelpers::AssertSingleThreaded(); |
| 77 } |
| 78 |
| 79 TEST(ThreadHelpers, IsSingleThreadedIterated) { |
| 79 ScopedProcSelfTask task; | 80 ScopedProcSelfTask task; |
| 80 ASSERT_TRUE(ThreadHelpers::IsSingleThreaded(task.fd())); | 81 ASSERT_TRUE(ThreadHelpers::IsSingleThreaded(task.fd())); |
| 81 | 82 |
| 82 // Iterate to check for race conditions. | 83 // Iterate to check for race conditions. |
| 83 for (int i = 0; i < GetRaceTestIterations(); ++i) { | 84 for (int i = 0; i < GetRaceTestIterations(); ++i) { |
| 84 base::Thread thread("sandbox_tests"); | 85 base::Thread thread("sandbox_tests"); |
| 85 ASSERT_TRUE(thread.Start()); | 86 ASSERT_TRUE(thread.Start()); |
| 86 ASSERT_FALSE(ThreadHelpers::IsSingleThreaded(task.fd())); | 87 ASSERT_FALSE(ThreadHelpers::IsSingleThreaded(task.fd())); |
| 87 // Explicitly stop the thread here to not pollute the next test. | 88 // Explicitly stop the thread here to not pollute the next test. |
| 88 ASSERT_TRUE(ThreadHelpers::StopThreadAndWatchProcFS(task.fd(), &thread)); | 89 ASSERT_TRUE(ThreadHelpers::StopThreadAndWatchProcFS(task.fd(), &thread)); |
| 89 } | 90 } |
| 90 } | 91 } |
| 91 | 92 |
| 92 TEST(ThreadHelpers, MAYBE_IsSingleThreadedStartAndStop) { | 93 TEST(ThreadHelpers, IsSingleThreadedStartAndStop) { |
| 93 ScopedProcSelfTask task; | 94 ScopedProcSelfTask task; |
| 94 ASSERT_TRUE(ThreadHelpers::IsSingleThreaded(task.fd())); | 95 ASSERT_TRUE(ThreadHelpers::IsSingleThreaded(task.fd())); |
| 95 | 96 |
| 96 base::Thread thread("sandbox_tests"); | 97 base::Thread thread("sandbox_tests"); |
| 97 // This is testing for a race condition, so iterate. | 98 // This is testing for a race condition, so iterate. |
| 98 // Manually, this has been tested with more that 1M iterations. | 99 // Manually, this has been tested with more that 1M iterations. |
| 99 for (int i = 0; i < GetRaceTestIterations(); ++i) { | 100 for (int i = 0; i < GetRaceTestIterations(); ++i) { |
| 100 ASSERT_TRUE(thread.Start()); | 101 ASSERT_TRUE(thread.Start()); |
| 101 ASSERT_FALSE(ThreadHelpers::IsSingleThreaded(task.fd())); | 102 ASSERT_FALSE(ThreadHelpers::IsSingleThreaded(task.fd())); |
| 102 | 103 |
| 103 ASSERT_TRUE(ThreadHelpers::StopThreadAndWatchProcFS(task.fd(), &thread)); | 104 ASSERT_TRUE(ThreadHelpers::StopThreadAndWatchProcFS(task.fd(), &thread)); |
| 104 ASSERT_TRUE(ThreadHelpers::IsSingleThreaded(task.fd())); | 105 ASSERT_TRUE(ThreadHelpers::IsSingleThreaded(task.fd())); |
| 105 ASSERT_EQ(1, base::GetNumberOfThreads(base::GetCurrentProcessHandle())); | 106 ASSERT_EQ(1, base::GetNumberOfThreads(base::GetCurrentProcessHandle())); |
| 106 } | 107 } |
| 107 } | 108 } |
| 108 | 109 |
| 110 SANDBOX_TEST(ThreadHelpers, AssertSingleThreadedAfterThreadStopped) { |
| 111 SANDBOX_ASSERT(ThreadHelpers::IsSingleThreaded()); |
| 112 |
| 113 base::Thread thread1("sandbox_tests"); |
| 114 base::Thread thread2("sandbox_tests"); |
| 115 |
| 116 for (int i = 0; i < GetRaceTestIterations(); ++i) { |
| 117 SANDBOX_ASSERT(thread1.Start()); |
| 118 SANDBOX_ASSERT(thread2.Start()); |
| 119 SANDBOX_ASSERT(!ThreadHelpers::IsSingleThreaded()); |
| 120 |
| 121 thread1.Stop(); |
| 122 thread2.Stop(); |
| 123 // This will wait on /proc/ to reflect the state of threads in the |
| 124 // process. |
| 125 ThreadHelpers::AssertSingleThreaded(); |
| 126 SANDBOX_ASSERT(ThreadHelpers::IsSingleThreaded()); |
| 127 } |
| 128 } |
| 129 |
| 130 // Only run this test in Debug mode, where AssertSingleThreaded() will return |
| 131 // in less than 64ms. |
| 132 #if !defined(NDEBUG) |
| 133 SANDBOX_DEATH_TEST( |
| 134 ThreadHelpers, |
| 135 AssertSingleThreadedDies, |
| 136 DEATH_MESSAGE( |
| 137 ThreadHelpers::GetAssertSingleThreadedErrorMessageForTests())) { |
| 138 base::Thread thread1("sandbox_tests"); |
| 139 SANDBOX_ASSERT(thread1.Start()); |
| 140 ThreadHelpers::AssertSingleThreaded(); |
| 141 } |
| 142 #endif // !defined(NDEBUG) |
| 143 |
| 144 #endif // !defined(THREAD_SANITIZER) |
| 145 |
| 109 } // namespace | 146 } // namespace |
| 110 | 147 |
| 111 } // namespace sandbox | 148 } // namespace sandbox |
| OLD | NEW |