Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(46)

Side by Side Diff: base/threading/thread_checker_unittest.cc

Issue 2165663003: TaskScheduler: Add SequenceToken and ScopedSetSequenceTokenForCurrentThread (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: improve comment Created 4 years, 5 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
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
OLDNEW
« base/threading/thread_checker.h ('K') | « base/threading/thread_checker_impl.cc ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698