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 |