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

Side by Side Diff: base/test/test_mock_time_task_runner.cc

Issue 899863002: Add support in TestMockTimeTaskRunner for vending out mock Time and mock Clocks. (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: Fix ash_unittests missing header. Created 5 years, 10 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 2015 The Chromium Authors. All rights reserved. 1 // Copyright 2015 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/test/test_mock_time_task_runner.h" 5 #include "base/test/test_mock_time_task_runner.h"
6 6
7 #include "base/logging.h" 7 #include "base/logging.h"
8 #include "base/memory/ref_counted.h" 8 #include "base/memory/ref_counted.h"
9 #include "base/time/clock.h"
10 #include "base/time/tick_clock.h"
9 11
10 namespace base { 12 namespace base {
11 13
12 namespace { 14 namespace {
13 15
14 // TickClock that always returns the then-current mock time of |task_runner| as 16 // MockTickClock --------------------------------------------------------------
15 // the current time. 17
18 // TickClock that always returns the then-current mock time ticks of
19 // |task_runner| as the current time ticks.
16 class MockTickClock : public TickClock { 20 class MockTickClock : public TickClock {
17 public: 21 public:
18 explicit MockTickClock( 22 explicit MockTickClock(
19 scoped_refptr<const TestMockTimeTaskRunner> task_runner); 23 scoped_refptr<const TestMockTimeTaskRunner> task_runner);
20 ~MockTickClock() override; 24 ~MockTickClock() override;
21 25
22 // TickClock: 26 // TickClock:
23 TimeTicks NowTicks() override; 27 TimeTicks NowTicks() override;
24 28
25 private: 29 private:
26 scoped_refptr<const TestMockTimeTaskRunner> task_runner_; 30 scoped_refptr<const TestMockTimeTaskRunner> task_runner_;
27 31
28 DISALLOW_COPY_AND_ASSIGN(MockTickClock); 32 DISALLOW_COPY_AND_ASSIGN(MockTickClock);
29 }; 33 };
30 34
31 MockTickClock::MockTickClock( 35 MockTickClock::MockTickClock(
32 scoped_refptr<const TestMockTimeTaskRunner> task_runner) 36 scoped_refptr<const TestMockTimeTaskRunner> task_runner)
33 : task_runner_(task_runner) { 37 : task_runner_(task_runner) {
34 } 38 }
35 39
36 MockTickClock::~MockTickClock() { 40 MockTickClock::~MockTickClock() {
37 } 41 }
38 42
39 TimeTicks MockTickClock::NowTicks() { 43 TimeTicks MockTickClock::NowTicks() {
40 return task_runner_->GetCurrentMockTime(); 44 return task_runner_->NowTicks();
45 }
46
47 // MockClock ------------------------------------------------------------------
48
49 // Clock that always returns the then-current mock time of |task_runner| as the
50 // current time.
51 class MockClock : public Clock {
52 public:
53 explicit MockClock(scoped_refptr<const TestMockTimeTaskRunner> task_runner);
54 ~MockClock() override;
55
56 // Clock:
57 Time Now() override;
58
59 private:
60 scoped_refptr<const TestMockTimeTaskRunner> task_runner_;
61
62 DISALLOW_COPY_AND_ASSIGN(MockClock);
63 };
64
65 MockClock::MockClock(scoped_refptr<const TestMockTimeTaskRunner> task_runner)
66 : task_runner_(task_runner) {
67 }
68
69 MockClock::~MockClock() {
70 }
71
72 Time MockClock::Now() {
73 return task_runner_->Now();
41 } 74 }
42 75
43 } // namespace 76 } // namespace
44 77
78 // TestMockTimeTaskRunner -----------------------------------------------------
79
45 bool TestMockTimeTaskRunner::TemporalOrder::operator()( 80 bool TestMockTimeTaskRunner::TemporalOrder::operator()(
46 const TestPendingTask& first_task, 81 const TestPendingTask& first_task,
47 const TestPendingTask& second_task) const { 82 const TestPendingTask& second_task) const {
48 return first_task.GetTimeToRun() > second_task.GetTimeToRun(); 83 return first_task.GetTimeToRun() > second_task.GetTimeToRun();
49 } 84 }
50 85
51 TestMockTimeTaskRunner::TestMockTimeTaskRunner() { 86 TestMockTimeTaskRunner::TestMockTimeTaskRunner() : now_(Time::UnixEpoch()) {
87 }
88
89 TestMockTimeTaskRunner::TestMockTimeTaskRunner(Time intial_time)
Lei Zhang 2015/02/04 20:27:37 typo
engedy 2015/02/05 10:22:57 Removed ctor, thanks for spotting the typo, though
90 : now_(intial_time) {
52 } 91 }
53 92
54 TestMockTimeTaskRunner::~TestMockTimeTaskRunner() { 93 TestMockTimeTaskRunner::~TestMockTimeTaskRunner() {
55 } 94 }
56 95
57 void TestMockTimeTaskRunner::FastForwardBy(TimeDelta delta) { 96 void TestMockTimeTaskRunner::FastForwardBy(TimeDelta delta) {
58 DCHECK(thread_checker_.CalledOnValidThread()); 97 DCHECK(thread_checker_.CalledOnValidThread());
59 98
60 OnBeforeSelectingTask(); 99 OnBeforeSelectingTask();
61 100
62 const base::TimeTicks original_now = now_; 101 const TimeTicks original_now_ticks = now_ticks_;
63 TestPendingTask task_info; 102 TestPendingTask task_info;
64 while (DequeueNextTask(original_now, delta, &task_info)) { 103 while (DequeueNextTask(original_now_ticks, delta, &task_info)) {
65 if (task_info.GetTimeToRun() - now_ > base::TimeDelta()) { 104 ForwardClocksUntilTickTime(task_info.GetTimeToRun());
66 now_ = task_info.GetTimeToRun();
67 OnAfterTimePassed();
68 }
69
70 task_info.task.Run(); 105 task_info.task.Run();
71
72 OnAfterTaskRun(); 106 OnAfterTaskRun();
73 OnBeforeSelectingTask(); 107 OnBeforeSelectingTask();
74 } 108 }
75 109
76 if (!delta.is_max() && now_ - original_now < delta) { 110 if (!delta.is_max())
77 now_ = original_now + delta; 111 ForwardClocksUntilTickTime(original_now_ticks + delta);
Lei Zhang 2015/02/04 20:27:37 Just trying to understand the original code here..
engedy 2015/02/05 10:22:57 You are correct: it has been, and it is still poss
Lei Zhang 2015/02/05 10:52:59 Is is possible to have both FastForwardBy() and Fa
engedy 2015/02/05 12:41:23 I went with a slightly less evil version of the ol
78 OnAfterTimePassed();
79 }
80 } 112 }
81 113
82 void TestMockTimeTaskRunner::RunUntilIdle() { 114 void TestMockTimeTaskRunner::RunUntilIdle() {
83 FastForwardBy(TimeDelta()); 115 FastForwardBy(TimeDelta());
84 } 116 }
85 117
86 void TestMockTimeTaskRunner::FastForwardUntilNoTasksRemain() { 118 void TestMockTimeTaskRunner::FastForwardUntilNoTasksRemain() {
87 FastForwardBy(TimeDelta::Max()); 119 FastForwardBy(TimeDelta::Max());
88 } 120 }
89 121
90 TimeTicks TestMockTimeTaskRunner::GetCurrentMockTime() const { 122 Time TestMockTimeTaskRunner::Now() const {
91 DCHECK(thread_checker_.CalledOnValidThread()); 123 DCHECK(thread_checker_.CalledOnValidThread());
92 return now_; 124 return now_;
93 } 125 }
94 126
127 TimeTicks TestMockTimeTaskRunner::NowTicks() const {
128 DCHECK(thread_checker_.CalledOnValidThread());
129 return now_ticks_;
130 }
131
132 scoped_ptr<Clock> TestMockTimeTaskRunner::GetMockClock() const {
133 DCHECK(thread_checker_.CalledOnValidThread());
134 return make_scoped_ptr(new MockClock(this));
135 }
136
95 scoped_ptr<TickClock> TestMockTimeTaskRunner::GetMockTickClock() const { 137 scoped_ptr<TickClock> TestMockTimeTaskRunner::GetMockTickClock() const {
96 DCHECK(thread_checker_.CalledOnValidThread()); 138 DCHECK(thread_checker_.CalledOnValidThread());
97 return make_scoped_ptr(new MockTickClock(this)); 139 return make_scoped_ptr(new MockTickClock(this));
98 } 140 }
99 141
100 bool TestMockTimeTaskRunner::HasPendingTask() const { 142 bool TestMockTimeTaskRunner::HasPendingTask() const {
101 DCHECK(thread_checker_.CalledOnValidThread()); 143 DCHECK(thread_checker_.CalledOnValidThread());
102 return !tasks_.empty(); 144 return !tasks_.empty();
103 } 145 }
104 146
105 size_t TestMockTimeTaskRunner::GetPendingTaskCount() const { 147 size_t TestMockTimeTaskRunner::GetPendingTaskCount() const {
106 DCHECK(thread_checker_.CalledOnValidThread()); 148 DCHECK(thread_checker_.CalledOnValidThread());
107 return tasks_.size(); 149 return tasks_.size();
108 } 150 }
109 151
110 TimeDelta TestMockTimeTaskRunner::NextPendingTaskDelay() const { 152 TimeDelta TestMockTimeTaskRunner::NextPendingTaskDelay() const {
111 DCHECK(thread_checker_.CalledOnValidThread()); 153 DCHECK(thread_checker_.CalledOnValidThread());
112 return tasks_.empty() ? TimeDelta::Max() : tasks_.top().GetTimeToRun() - now_; 154 return tasks_.empty() ? TimeDelta::Max()
155 : tasks_.top().GetTimeToRun() - now_ticks_;
113 } 156 }
114 157
115 bool TestMockTimeTaskRunner::RunsTasksOnCurrentThread() const { 158 bool TestMockTimeTaskRunner::RunsTasksOnCurrentThread() const {
116 return thread_checker_.CalledOnValidThread(); 159 return thread_checker_.CalledOnValidThread();
117 } 160 }
118 161
119 bool TestMockTimeTaskRunner::PostDelayedTask( 162 bool TestMockTimeTaskRunner::PostDelayedTask(
120 const tracked_objects::Location& from_here, 163 const tracked_objects::Location& from_here,
121 const Closure& task, 164 const Closure& task,
122 TimeDelta delay) { 165 TimeDelta delay) {
123 base::AutoLock scoped_lock(tasks_lock_); 166 AutoLock scoped_lock(tasks_lock_);
124 tasks_.push( 167 tasks_.push(TestPendingTask(from_here, task, now_ticks_, delay,
125 TestPendingTask(from_here, task, now_, delay, TestPendingTask::NESTABLE)); 168 TestPendingTask::NESTABLE));
126 return true; 169 return true;
127 } 170 }
128 171
129 bool TestMockTimeTaskRunner::PostNonNestableDelayedTask( 172 bool TestMockTimeTaskRunner::PostNonNestableDelayedTask(
130 const tracked_objects::Location& from_here, 173 const tracked_objects::Location& from_here,
131 const Closure& task, 174 const Closure& task,
132 TimeDelta delay) { 175 TimeDelta delay) {
133 NOTREACHED(); 176 NOTREACHED();
134 return false; 177 return false;
135 } 178 }
136 179
137 void TestMockTimeTaskRunner::OnBeforeSelectingTask() { 180 void TestMockTimeTaskRunner::OnBeforeSelectingTask() {
138 // Empty default implementation. 181 // Empty default implementation.
139 } 182 }
140 183
141 void TestMockTimeTaskRunner::OnAfterTimePassed() { 184 void TestMockTimeTaskRunner::OnAfterTimePassed() {
142 // Empty default implementation. 185 // Empty default implementation.
143 } 186 }
144 187
145 void TestMockTimeTaskRunner::OnAfterTaskRun() { 188 void TestMockTimeTaskRunner::OnAfterTaskRun() {
146 // Empty default implementation. 189 // Empty default implementation.
147 } 190 }
148 191
149 bool TestMockTimeTaskRunner::DequeueNextTask(const base::TimeTicks& reference, 192 void TestMockTimeTaskRunner::ForwardClocksUntilTickTime(TimeTicks later_ticks) {
150 const base::TimeDelta& max_delta, 193 now_ += later_ticks - now_ticks_;
Lei Zhang 2015/02/04 20:27:37 Can this make time move backwards? In the original
engedy 2015/02/05 10:22:57 Hmm, yes, if the consumer of this class posts task
194 if (now_ticks_ < later_ticks) {
195 now_ticks_ = later_ticks;
196 OnAfterTimePassed();
197 }
198 }
199
200 bool TestMockTimeTaskRunner::DequeueNextTask(const TimeTicks& reference,
201 const TimeDelta& max_delta,
151 TestPendingTask* next_task) { 202 TestPendingTask* next_task) {
152 base::AutoLock scoped_lock(tasks_lock_); 203 AutoLock scoped_lock(tasks_lock_);
153 if (!tasks_.empty() && 204 if (!tasks_.empty() &&
154 (tasks_.top().GetTimeToRun() - reference) <= max_delta) { 205 (tasks_.top().GetTimeToRun() - reference) <= max_delta) {
155 *next_task = tasks_.top(); 206 *next_task = tasks_.top();
156 tasks_.pop(); 207 tasks_.pop();
157 return true; 208 return true;
158 } 209 }
159 return false; 210 return false;
160 } 211 }
161 212
162 } // namespace base 213 } // namespace base
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698