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
3 // found in the LICENSE file.
4
5 #include "base/test/sequenced_task_runner_test_template.h"
6
7 #include <ostream>
8
9 #include "base/location.h"
10
11 namespace base {
12
13 namespace internal {
14
TaskEvent(int i,Type type)15 TaskEvent::TaskEvent(int i, Type type)
16 : i(i), type(type) {
17 }
18
SequencedTaskTracker()19 SequencedTaskTracker::SequencedTaskTracker()
20 : next_post_i_(0),
21 task_end_count_(0),
22 task_end_cv_(&lock_) {
23 }
24
PostWrappedNonNestableTask(const scoped_refptr<SequencedTaskRunner> & task_runner,const Closure & task)25 void SequencedTaskTracker::PostWrappedNonNestableTask(
26 const scoped_refptr<SequencedTaskRunner>& task_runner,
27 const Closure& task) {
28 AutoLock event_lock(lock_);
29 const int post_i = next_post_i_++;
30 Closure wrapped_task = Bind(&SequencedTaskTracker::RunTask, this,
31 task, post_i);
32 task_runner->PostNonNestableTask(FROM_HERE, wrapped_task);
33 TaskPosted(post_i);
34 }
35
PostWrappedNestableTask(const scoped_refptr<SequencedTaskRunner> & task_runner,const Closure & task)36 void SequencedTaskTracker::PostWrappedNestableTask(
37 const scoped_refptr<SequencedTaskRunner>& task_runner,
38 const Closure& task) {
39 AutoLock event_lock(lock_);
40 const int post_i = next_post_i_++;
41 Closure wrapped_task = Bind(&SequencedTaskTracker::RunTask, this,
42 task, post_i);
43 task_runner->PostTask(FROM_HERE, wrapped_task);
44 TaskPosted(post_i);
45 }
46
PostWrappedDelayedNonNestableTask(const scoped_refptr<SequencedTaskRunner> & task_runner,const Closure & task,TimeDelta delay)47 void SequencedTaskTracker::PostWrappedDelayedNonNestableTask(
48 const scoped_refptr<SequencedTaskRunner>& task_runner,
49 const Closure& task,
50 TimeDelta delay) {
51 AutoLock event_lock(lock_);
52 const int post_i = next_post_i_++;
53 Closure wrapped_task = Bind(&SequencedTaskTracker::RunTask, this,
54 task, post_i);
55 task_runner->PostNonNestableDelayedTask(FROM_HERE, wrapped_task, delay);
56 TaskPosted(post_i);
57 }
58
PostNonNestableTasks(const scoped_refptr<SequencedTaskRunner> & task_runner,int task_count)59 void SequencedTaskTracker::PostNonNestableTasks(
60 const scoped_refptr<SequencedTaskRunner>& task_runner,
61 int task_count) {
62 for (int i = 0; i < task_count; ++i) {
63 PostWrappedNonNestableTask(task_runner, Closure());
64 }
65 }
66
RunTask(const Closure & task,int task_i)67 void SequencedTaskTracker::RunTask(const Closure& task, int task_i) {
68 TaskStarted(task_i);
69 if (!task.is_null())
70 task.Run();
71 TaskEnded(task_i);
72 }
73
TaskPosted(int i)74 void SequencedTaskTracker::TaskPosted(int i) {
75 // Caller must own |lock_|.
76 events_.push_back(TaskEvent(i, TaskEvent::POST));
77 }
78
TaskStarted(int i)79 void SequencedTaskTracker::TaskStarted(int i) {
80 AutoLock lock(lock_);
81 events_.push_back(TaskEvent(i, TaskEvent::START));
82 }
83
TaskEnded(int i)84 void SequencedTaskTracker::TaskEnded(int i) {
85 AutoLock lock(lock_);
86 events_.push_back(TaskEvent(i, TaskEvent::END));
87 ++task_end_count_;
88 task_end_cv_.Signal();
89 }
90
91 const std::vector<TaskEvent>&
GetTaskEvents() const92 SequencedTaskTracker::GetTaskEvents() const {
93 return events_;
94 }
95
WaitForCompletedTasks(int count)96 void SequencedTaskTracker::WaitForCompletedTasks(int count) {
97 AutoLock lock(lock_);
98 while (task_end_count_ < count)
99 task_end_cv_.Wait();
100 }
101
~SequencedTaskTracker()102 SequencedTaskTracker::~SequencedTaskTracker() {
103 }
104
PrintTo(const TaskEvent & event,std::ostream * os)105 void PrintTo(const TaskEvent& event, std::ostream* os) {
106 *os << "(i=" << event.i << ", type=";
107 switch (event.type) {
108 case TaskEvent::POST: *os << "POST"; break;
109 case TaskEvent::START: *os << "START"; break;
110 case TaskEvent::END: *os << "END"; break;
111 }
112 *os << ")";
113 }
114
115 namespace {
116
117 // Returns the task ordinals for the task event type |type| in the order that
118 // they were recorded.
GetEventTypeOrder(const std::vector<TaskEvent> & events,TaskEvent::Type type)119 std::vector<int> GetEventTypeOrder(const std::vector<TaskEvent>& events,
120 TaskEvent::Type type) {
121 std::vector<int> tasks;
122 std::vector<TaskEvent>::const_iterator event;
123 for (event = events.begin(); event != events.end(); ++event) {
124 if (event->type == type)
125 tasks.push_back(event->i);
126 }
127 return tasks;
128 }
129
130 // Returns all task events for task |task_i|.
GetEventsForTask(const std::vector<TaskEvent> & events,int task_i)131 std::vector<TaskEvent::Type> GetEventsForTask(
132 const std::vector<TaskEvent>& events,
133 int task_i) {
134 std::vector<TaskEvent::Type> task_event_orders;
135 std::vector<TaskEvent>::const_iterator event;
136 for (event = events.begin(); event != events.end(); ++event) {
137 if (event->i == task_i)
138 task_event_orders.push_back(event->type);
139 }
140 return task_event_orders;
141 }
142
143 // Checks that the task events for each task in |events| occur in the order
144 // {POST, START, END}, and that there is only one instance of each event type
145 // per task.
CheckEventOrdersForEachTask(const std::vector<TaskEvent> & events,int task_count)146 ::testing::AssertionResult CheckEventOrdersForEachTask(
147 const std::vector<TaskEvent>& events,
148 int task_count) {
149 std::vector<TaskEvent::Type> expected_order;
150 expected_order.push_back(TaskEvent::POST);
151 expected_order.push_back(TaskEvent::START);
152 expected_order.push_back(TaskEvent::END);
153
154 // This is O(n^2), but it runs fast enough currently so is not worth
155 // optimizing.
156 for (int i = 0; i < task_count; ++i) {
157 const std::vector<TaskEvent::Type> task_events =
158 GetEventsForTask(events, i);
159 if (task_events != expected_order) {
160 return ::testing::AssertionFailure()
161 << "Events for task " << i << " are out of order; expected: "
162 << ::testing::PrintToString(expected_order) << "; actual: "
163 << ::testing::PrintToString(task_events);
164 }
165 }
166 return ::testing::AssertionSuccess();
167 }
168
169 // Checks that no two tasks were running at the same time. I.e. the only
170 // events allowed between the START and END of a task are the POSTs of other
171 // tasks.
CheckNoTaskRunsOverlap(const std::vector<TaskEvent> & events)172 ::testing::AssertionResult CheckNoTaskRunsOverlap(
173 const std::vector<TaskEvent>& events) {
174 // If > -1, we're currently inside a START, END pair.
175 int current_task_i = -1;
176
177 std::vector<TaskEvent>::const_iterator event;
178 for (event = events.begin(); event != events.end(); ++event) {
179 bool spurious_event_found = false;
180
181 if (current_task_i == -1) { // Not inside a START, END pair.
182 switch (event->type) {
183 case TaskEvent::POST:
184 break;
185 case TaskEvent::START:
186 current_task_i = event->i;
187 break;
188 case TaskEvent::END:
189 spurious_event_found = true;
190 break;
191 }
192
193 } else { // Inside a START, END pair.
194 bool interleaved_task_detected = false;
195
196 switch (event->type) {
197 case TaskEvent::POST:
198 if (event->i == current_task_i)
199 spurious_event_found = true;
200 break;
201 case TaskEvent::START:
202 interleaved_task_detected = true;
203 break;
204 case TaskEvent::END:
205 if (event->i != current_task_i)
206 interleaved_task_detected = true;
207 else
208 current_task_i = -1;
209 break;
210 }
211
212 if (interleaved_task_detected) {
213 return ::testing::AssertionFailure()
214 << "Found event " << ::testing::PrintToString(*event)
215 << " between START and END events for task " << current_task_i
216 << "; event dump: " << ::testing::PrintToString(events);
217 }
218 }
219
220 if (spurious_event_found) {
221 const int event_i = event - events.begin();
222 return ::testing::AssertionFailure()
223 << "Spurious event " << ::testing::PrintToString(*event)
224 << " at position " << event_i << "; event dump: "
225 << ::testing::PrintToString(events);
226 }
227 }
228
229 return ::testing::AssertionSuccess();
230 }
231
232 } // namespace
233
CheckNonNestableInvariants(const std::vector<TaskEvent> & events,int task_count)234 ::testing::AssertionResult CheckNonNestableInvariants(
235 const std::vector<TaskEvent>& events,
236 int task_count) {
237 const std::vector<int> post_order =
238 GetEventTypeOrder(events, TaskEvent::POST);
239 const std::vector<int> start_order =
240 GetEventTypeOrder(events, TaskEvent::START);
241 const std::vector<int> end_order =
242 GetEventTypeOrder(events, TaskEvent::END);
243
244 if (start_order != post_order) {
245 return ::testing::AssertionFailure()
246 << "Expected START order (which equals actual POST order): \n"
247 << ::testing::PrintToString(post_order)
248 << "\n Actual START order:\n"
249 << ::testing::PrintToString(start_order);
250 }
251
252 if (end_order != post_order) {
253 return ::testing::AssertionFailure()
254 << "Expected END order (which equals actual POST order): \n"
255 << ::testing::PrintToString(post_order)
256 << "\n Actual END order:\n"
257 << ::testing::PrintToString(end_order);
258 }
259
260 const ::testing::AssertionResult result =
261 CheckEventOrdersForEachTask(events, task_count);
262 if (!result)
263 return result;
264
265 return CheckNoTaskRunsOverlap(events);
266 }
267
268 } // namespace internal
269
270 } // namespace base
271