Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 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 "base/threading/thread_checker.h" | |
| 6 | |
| 7 #include <memory> | 5 #include <memory> |
| 8 | 6 |
| 9 #include "base/logging.h" | 7 #include "base/bind.h" |
| 8 #include "base/bind_helpers.h" | |
| 10 #include "base/macros.h" | 9 #include "base/macros.h" |
| 10 #include "base/memory/ref_counted.h" | |
| 11 #include "base/sequence_token.h" | |
| 12 #include "base/test/test_simple_task_runner.h" | |
| 11 #include "base/threading/simple_thread.h" | 13 #include "base/threading/simple_thread.h" |
| 14 #include "base/threading/thread_checker_impl.h" | |
| 15 #include "base/threading/thread_task_runner_handle.h" | |
| 12 #include "testing/gtest/include/gtest/gtest.h" | 16 #include "testing/gtest/include/gtest/gtest.h" |
| 13 | 17 |
| 14 // Duplicated from base/threading/thread_checker.h so that we can be | |
| 15 // good citizens there and undef the macro. | |
| 16 #if !defined(NDEBUG) || defined(DCHECK_ALWAYS_ON) | |
| 17 #define ENABLE_THREAD_CHECKER 1 | |
| 18 #else | |
| 19 #define ENABLE_THREAD_CHECKER 0 | |
| 20 #endif | |
| 21 | |
| 22 namespace base { | 18 namespace base { |
| 23 | |
| 24 namespace { | 19 namespace { |
| 25 | 20 |
| 26 // Simple class to exercise the basics of ThreadChecker. | 21 // Runs a callback on another thread. |
| 27 // Both the destructor and DoStuff should verify that they were | 22 class RunCallbackThread : public SimpleThread { |
| 28 // called on the same thread as the constructor. | |
| 29 class ThreadCheckerClass : public ThreadChecker { | |
| 30 public: | 23 public: |
| 31 ThreadCheckerClass() {} | 24 explicit RunCallbackThread(const Closure& callback) |
| 32 | 25 : SimpleThread("RunCallbackThread"), callback_(callback) { |
| 33 // Verifies that it was called on the same thread as the constructor. | 26 Start(); |
| 34 void DoStuff() { | 27 Join(); |
| 35 DCHECK(CalledOnValidThread()); | |
| 36 } | 28 } |
| 37 | 29 |
| 38 void DetachFromThread() { | 30 private: |
| 39 ThreadChecker::DetachFromThread(); | 31 // SimpleThread: |
| 40 } | 32 void Run() override { callback_.Run(); } |
| 41 | 33 |
| 42 static void MethodOnDifferentThreadImpl(); | 34 const Closure callback_; |
| 43 static void DetachThenCallFromDifferentThreadImpl(); | |
| 44 | 35 |
| 45 private: | 36 DISALLOW_COPY_AND_ASSIGN(RunCallbackThread); |
| 46 DISALLOW_COPY_AND_ASSIGN(ThreadCheckerClass); | |
| 47 }; | 37 }; |
| 48 | 38 |
| 49 // Calls ThreadCheckerClass::DoStuff on another thread. | 39 void ExpectCalledOnValidThread(ThreadCheckerImpl* thread_checker) { |
| 50 class CallDoStuffOnThread : public base::SimpleThread { | 40 ASSERT_TRUE(thread_checker); |
| 51 public: | |
| 52 explicit CallDoStuffOnThread(ThreadCheckerClass* thread_checker_class) | |
| 53 : SimpleThread("call_do_stuff_on_thread"), | |
| 54 thread_checker_class_(thread_checker_class) { | |
| 55 } | |
| 56 | 41 |
| 57 void Run() override { thread_checker_class_->DoStuff(); } | 42 // This should bind |thread_checker| to the current thread if it wasn't |
| 43 // already bound to a thread. | |
| 44 EXPECT_TRUE(thread_checker->CalledOnValidThread()); | |
| 58 | 45 |
| 59 private: | 46 // Since |thread_checker| is now bound to the current thread, another call to |
| 60 ThreadCheckerClass* thread_checker_class_; | 47 // CalledOnValidThread() should return true. |
| 48 EXPECT_TRUE(thread_checker->CalledOnValidThread()); | |
| 49 } | |
| 61 | 50 |
| 62 DISALLOW_COPY_AND_ASSIGN(CallDoStuffOnThread); | 51 void ExpectNotCalledOnValidThread(ThreadCheckerImpl* thread_checker) { |
| 63 }; | 52 ASSERT_TRUE(thread_checker); |
| 53 EXPECT_FALSE(thread_checker->CalledOnValidThread()); | |
| 54 } | |
| 64 | 55 |
| 65 // Deletes ThreadCheckerClass on a different thread. | 56 void ExpectNotCalledOnValidThreadWithSequenceTokenAndThreadTaskRunnerHandle( |
| 66 class DeleteThreadCheckerClassOnThread : public base::SimpleThread { | 57 ThreadCheckerImpl* thread_checker, |
| 67 public: | 58 internal::SequenceToken sequence_token) { |
| 68 explicit DeleteThreadCheckerClassOnThread( | 59 ThreadTaskRunnerHandle thread_task_runner_handle( |
| 69 ThreadCheckerClass* thread_checker_class) | 60 make_scoped_refptr(new TestSimpleTaskRunner)); |
| 70 : SimpleThread("delete_thread_checker_class_on_thread"), | 61 internal::ScopedSetCurrentSequenceToken scoped_set_current_sequence_token( |
| 71 thread_checker_class_(thread_checker_class) { | 62 sequence_token); |
| 72 } | 63 ExpectNotCalledOnValidThread(thread_checker); |
| 73 | 64 } |
| 74 void Run() override { thread_checker_class_.reset(); } | |
| 75 | |
| 76 private: | |
| 77 std::unique_ptr<ThreadCheckerClass> thread_checker_class_; | |
| 78 | |
| 79 DISALLOW_COPY_AND_ASSIGN(DeleteThreadCheckerClassOnThread); | |
| 80 }; | |
| 81 | 65 |
| 82 } // namespace | 66 } // namespace |
| 83 | 67 |
| 84 TEST(ThreadCheckerTest, CallsAllowedOnSameThread) { | 68 TEST(ThreadCheckerTest, CallsAllowedSameThreadNoSequenceToken) { |
| 85 std::unique_ptr<ThreadCheckerClass> thread_checker_class( | 69 ThreadCheckerImpl thread_checker; |
| 86 new ThreadCheckerClass); | 70 EXPECT_TRUE(thread_checker.CalledOnValidThread()); |
| 87 | |
| 88 // Verify that DoStuff doesn't assert. | |
| 89 thread_checker_class->DoStuff(); | |
| 90 | |
| 91 // Verify that the destructor doesn't assert. | |
| 92 thread_checker_class.reset(); | |
| 93 } | 71 } |
| 94 | 72 |
| 95 TEST(ThreadCheckerTest, DestructorAllowedOnDifferentThread) { | 73 TEST(ThreadCheckerTest, |
| 96 std::unique_ptr<ThreadCheckerClass> thread_checker_class( | 74 CallsAllowedSameThreadSameSequenceTokenWithThreadTaskRunnerHandle) { |
| 97 new ThreadCheckerClass); | 75 ThreadTaskRunnerHandle thread_task_runner_handle( |
| 76 make_scoped_refptr(new TestSimpleTaskRunner)); | |
| 77 internal::ScopedSetCurrentSequenceToken scoped_set_current_sequence_token( | |
| 78 internal::SequenceToken::Create()); | |
| 79 ThreadCheckerImpl thread_checker; | |
| 80 EXPECT_TRUE(thread_checker.CalledOnValidThread()); | |
| 81 } | |
| 98 | 82 |
| 99 // Verify that the destructor doesn't assert | 83 TEST(ThreadCheckerTest, |
| 100 // when called on a different thread. | 84 CallsDisallowedSameThreadSameSequenceTokenNoThreadTaskRunnerHandle) { |
| 101 DeleteThreadCheckerClassOnThread delete_on_thread( | 85 internal::ScopedSetCurrentSequenceToken scoped_set_current_sequence_token( |
| 102 thread_checker_class.release()); | 86 internal::SequenceToken::Create()); |
| 87 ThreadCheckerImpl thread_checker; | |
| 88 EXPECT_FALSE(thread_checker.CalledOnValidThread()); | |
| 89 } | |
| 103 | 90 |
| 104 delete_on_thread.Start(); | 91 TEST(ThreadCheckerTest, CallsDisallowedOnDifferentThreadsNoSequenceToken) { |
| 105 delete_on_thread.Join(); | 92 ThreadCheckerImpl thread_checker; |
| 93 RunCallbackThread thread( | |
| 94 Bind(&ExpectNotCalledOnValidThread, Unretained(&thread_checker))); | |
| 95 } | |
| 96 | |
| 97 TEST(ThreadCheckerTest, CallsDisallowedOnDifferentThreadsSameSequenceToken) { | |
| 98 ThreadTaskRunnerHandle thread_task_runner_handle( | |
| 99 make_scoped_refptr(new TestSimpleTaskRunner)); | |
| 100 const internal::SequenceToken sequence_token( | |
| 101 internal::SequenceToken::Create()); | |
| 102 | |
| 103 internal::ScopedSetCurrentSequenceToken scoped_set_current_sequence_token( | |
| 104 sequence_token); | |
| 105 ThreadCheckerImpl thread_checker; | |
|
gab
2016/07/21 21:14:24
EXPECT_TRUE on this thread to drive essence of tes
fdoray
2016/07/25 13:24:28
Done.
| |
| 106 | |
| 107 RunCallbackThread thread(Bind( | |
| 108 &ExpectNotCalledOnValidThreadWithSequenceTokenAndThreadTaskRunnerHandle, | |
| 109 Unretained(&thread_checker), sequence_token)); | |
| 110 } | |
| 111 | |
| 112 TEST(ThreadCheckerTest, CallsDisallowedSameThreadDifferentSequenceToken) { | |
| 113 std::unique_ptr<ThreadCheckerImpl> thread_checker; | |
| 114 | |
| 115 ThreadTaskRunnerHandle thread_task_runner_handle( | |
| 116 make_scoped_refptr(new TestSimpleTaskRunner)); | |
| 117 | |
| 118 { | |
| 119 internal::ScopedSetCurrentSequenceToken scoped_set_current_sequence_token( | |
| 120 internal::SequenceToken::Create()); | |
| 121 thread_checker.reset(new ThreadCheckerImpl); | |
| 122 } | |
| 123 | |
| 124 { | |
| 125 // Different SequenceToken. | |
| 126 internal::ScopedSetCurrentSequenceToken scoped_set_current_sequence_token( | |
| 127 internal::SequenceToken::Create()); | |
| 128 EXPECT_FALSE(thread_checker->CalledOnValidThread()); | |
| 129 } | |
| 130 | |
| 131 // No SequenceToken. | |
| 132 EXPECT_FALSE(thread_checker->CalledOnValidThread()); | |
| 106 } | 133 } |
| 107 | 134 |
| 108 TEST(ThreadCheckerTest, DetachFromThread) { | 135 TEST(ThreadCheckerTest, DetachFromThread) { |
| 109 std::unique_ptr<ThreadCheckerClass> thread_checker_class( | 136 ThreadCheckerImpl thread_checker; |
| 110 new ThreadCheckerClass); | 137 thread_checker.DetachFromThread(); |
| 111 | 138 |
| 112 // Verify that DoStuff doesn't assert when called on a different thread after | 139 // Verify that CalledOnValidThread() returns true when called on a different |
| 113 // a call to DetachFromThread. | 140 // thread after a call to DetachFromThread(). |
| 114 thread_checker_class->DetachFromThread(); | 141 RunCallbackThread thread( |
| 115 CallDoStuffOnThread call_on_thread(thread_checker_class.get()); | 142 Bind(&ExpectCalledOnValidThread, Unretained(&thread_checker))); |
| 116 | 143 |
| 117 call_on_thread.Start(); | 144 EXPECT_FALSE(thread_checker.CalledOnValidThread()); |
| 118 call_on_thread.Join(); | |
| 119 } | 145 } |
| 120 | 146 |
| 121 #if GTEST_HAS_DEATH_TEST || !ENABLE_THREAD_CHECKER | 147 TEST(ThreadCheckerTest, DetachFromThreadWithSequenceToken) { |
| 148 ThreadTaskRunnerHandle thread_task_runner_handle( | |
| 149 make_scoped_refptr(new TestSimpleTaskRunner)); | |
| 150 internal::ScopedSetCurrentSequenceToken scoped_set_current_sequence_token( | |
| 151 internal::SequenceToken::Create()); | |
| 152 ThreadCheckerImpl thread_checker; | |
| 153 thread_checker.DetachFromThread(); | |
| 122 | 154 |
| 123 void ThreadCheckerClass::MethodOnDifferentThreadImpl() { | 155 // Verify that CalledOnValidThread() returns true when called on a different |
| 124 std::unique_ptr<ThreadCheckerClass> thread_checker_class( | 156 // thread after a call to DetachFromThread(). |
| 125 new ThreadCheckerClass); | 157 RunCallbackThread thread( |
| 158 Bind(&ExpectCalledOnValidThread, Unretained(&thread_checker))); | |
| 126 | 159 |
| 127 // DoStuff should assert in debug builds only when called on a | 160 EXPECT_FALSE(thread_checker.CalledOnValidThread()); |
| 128 // different thread. | |
| 129 CallDoStuffOnThread call_on_thread(thread_checker_class.get()); | |
| 130 | |
| 131 call_on_thread.Start(); | |
| 132 call_on_thread.Join(); | |
| 133 } | 161 } |
| 134 | 162 |
| 135 #if ENABLE_THREAD_CHECKER | |
| 136 TEST(ThreadCheckerDeathTest, MethodNotAllowedOnDifferentThreadInDebug) { | |
| 137 ASSERT_DEATH({ | |
| 138 ThreadCheckerClass::MethodOnDifferentThreadImpl(); | |
| 139 }, ""); | |
| 140 } | |
| 141 #else | |
| 142 TEST(ThreadCheckerTest, MethodAllowedOnDifferentThreadInRelease) { | |
| 143 ThreadCheckerClass::MethodOnDifferentThreadImpl(); | |
| 144 } | |
| 145 #endif // ENABLE_THREAD_CHECKER | |
| 146 | |
| 147 void ThreadCheckerClass::DetachThenCallFromDifferentThreadImpl() { | |
| 148 std::unique_ptr<ThreadCheckerClass> thread_checker_class( | |
| 149 new ThreadCheckerClass); | |
| 150 | |
| 151 // DoStuff doesn't assert when called on a different thread | |
| 152 // after a call to DetachFromThread. | |
| 153 thread_checker_class->DetachFromThread(); | |
| 154 CallDoStuffOnThread call_on_thread(thread_checker_class.get()); | |
| 155 | |
| 156 call_on_thread.Start(); | |
| 157 call_on_thread.Join(); | |
| 158 | |
| 159 // DoStuff should assert in debug builds only after moving to | |
| 160 // another thread. | |
| 161 thread_checker_class->DoStuff(); | |
| 162 } | |
| 163 | |
| 164 #if ENABLE_THREAD_CHECKER | |
| 165 TEST(ThreadCheckerDeathTest, DetachFromThreadInDebug) { | |
| 166 ASSERT_DEATH({ | |
| 167 ThreadCheckerClass::DetachThenCallFromDifferentThreadImpl(); | |
| 168 }, ""); | |
| 169 } | |
| 170 #else | |
| 171 TEST(ThreadCheckerTest, DetachFromThreadInRelease) { | |
| 172 ThreadCheckerClass::DetachThenCallFromDifferentThreadImpl(); | |
| 173 } | |
| 174 #endif // ENABLE_THREAD_CHECKER | |
| 175 | |
| 176 #endif // GTEST_HAS_DEATH_TEST || !ENABLE_THREAD_CHECKER | |
| 177 | |
| 178 // Just in case we ever get lumped together with other compilation units. | |
| 179 #undef ENABLE_THREAD_CHECKER | |
| 180 | |
| 181 } // namespace base | 163 } // namespace base |
| OLD | NEW |