Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 1 | // Copyright 2019 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 | |
Jordan Bayles | a26582d | 2019-07-10 14:44:58 -0700 | [diff] [blame] | 5 | #include "platform/impl/task_runner.h" |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 6 | |
btolsch | d94fe62 | 2019-05-09 14:21:40 -0700 | [diff] [blame] | 7 | #include <atomic> |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 8 | #include <thread> // NOLINT |
| 9 | |
Ryan Keane | 32c88d0 | 2019-07-02 18:46:14 -0700 | [diff] [blame] | 10 | #include "gmock/gmock.h" |
Jordan Bayles | 2d01f17 | 2019-06-07 11:11:50 -0700 | [diff] [blame] | 11 | #include "gtest/gtest.h" |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 12 | #include "platform/api/time.h" |
Jordan Bayles | 2d01f17 | 2019-06-07 11:11:50 -0700 | [diff] [blame] | 13 | #include "platform/test/fake_clock.h" |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 14 | |
btolsch | d94fe62 | 2019-05-09 14:21:40 -0700 | [diff] [blame] | 15 | namespace openscreen { |
| 16 | namespace platform { |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 17 | namespace { |
btolsch | d94fe62 | 2019-05-09 14:21:40 -0700 | [diff] [blame] | 18 | |
Ryan Keane | 32c88d0 | 2019-07-02 18:46:14 -0700 | [diff] [blame] | 19 | using namespace ::testing; |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 20 | using std::chrono::milliseconds; |
Ryan Keane | 32c88d0 | 2019-07-02 18:46:14 -0700 | [diff] [blame] | 21 | using ::testing::_; |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 22 | |
| 23 | const auto kTaskRunnerSleepTime = milliseconds(1); |
btolsch | d94fe62 | 2019-05-09 14:21:40 -0700 | [diff] [blame] | 24 | constexpr Clock::duration kWaitTimeout = milliseconds(1000); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 25 | |
| 26 | void WaitUntilCondition(std::function<bool()> predicate) { |
| 27 | while (!predicate()) { |
| 28 | std::this_thread::sleep_for(kTaskRunnerSleepTime); |
| 29 | } |
| 30 | } |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 31 | |
btolsch | d94fe62 | 2019-05-09 14:21:40 -0700 | [diff] [blame] | 32 | class FakeTaskWaiter final : public TaskRunnerImpl::TaskWaiter { |
| 33 | public: |
| 34 | explicit FakeTaskWaiter(platform::ClockNowFunctionPtr now_function) |
| 35 | : now_function_(now_function) {} |
| 36 | ~FakeTaskWaiter() override = default; |
| 37 | |
| 38 | Error WaitForTaskToBePosted(Clock::duration timeout) override { |
| 39 | Clock::time_point start = now_function_(); |
| 40 | waiting_.store(true); |
| 41 | while (!has_event_.load() && (now_function_() - start) < timeout) { |
| 42 | ; |
| 43 | } |
| 44 | waiting_.store(false); |
| 45 | has_event_.store(false); |
| 46 | return Error::None(); |
| 47 | } |
| 48 | |
| 49 | void OnTaskPosted() override { has_event_.store(true); } |
| 50 | |
| 51 | void WakeUpAndStop() { |
| 52 | OnTaskPosted(); |
Yuri Wiitala | c05ada2 | 2019-09-24 14:25:19 -0700 | [diff] [blame^] | 53 | task_runner_->RequestStopSoon(); |
btolsch | d94fe62 | 2019-05-09 14:21:40 -0700 | [diff] [blame] | 54 | } |
| 55 | |
| 56 | bool IsWaiting() const { return waiting_.load(); } |
| 57 | |
| 58 | void SetTaskRunner(TaskRunnerImpl* task_runner) { |
| 59 | task_runner_ = task_runner; |
| 60 | } |
| 61 | |
| 62 | private: |
| 63 | const platform::ClockNowFunctionPtr now_function_; |
| 64 | TaskRunnerImpl* task_runner_; |
| 65 | std::atomic<bool> has_event_{false}; |
| 66 | std::atomic<bool> waiting_{false}; |
| 67 | }; |
| 68 | |
| 69 | class TaskRunnerWithWaiterFactory { |
| 70 | public: |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 71 | static std::unique_ptr<TaskRunnerImpl> Create( |
btolsch | d94fe62 | 2019-05-09 14:21:40 -0700 | [diff] [blame] | 72 | platform::ClockNowFunctionPtr now_function) { |
| 73 | fake_waiter = std::make_unique<FakeTaskWaiter>(now_function); |
| 74 | auto runner = std::make_unique<TaskRunnerImpl>( |
| 75 | now_function, fake_waiter.get(), std::chrono::hours(1)); |
| 76 | fake_waiter->SetTaskRunner(runner.get()); |
| 77 | return runner; |
| 78 | } |
| 79 | |
| 80 | static std::unique_ptr<FakeTaskWaiter> fake_waiter; |
| 81 | }; |
| 82 | |
| 83 | // static |
| 84 | std::unique_ptr<FakeTaskWaiter> TaskRunnerWithWaiterFactory::fake_waiter; |
| 85 | |
| 86 | } // anonymous namespace |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 87 | |
Yuri Wiitala | c05ada2 | 2019-09-24 14:25:19 -0700 | [diff] [blame^] | 88 | TEST(TaskRunnerImplTest, TaskRunnerExecutesTaskAndStops) { |
Jordan Bayles | 5d72bc2 | 2019-04-09 13:33:52 -0700 | [diff] [blame] | 89 | FakeClock fake_clock{platform::Clock::time_point(milliseconds(1337))}; |
Yuri Wiitala | c05ada2 | 2019-09-24 14:25:19 -0700 | [diff] [blame^] | 90 | TaskRunnerImpl runner(&fake_clock.now); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 91 | |
| 92 | std::string ran_tasks = ""; |
Yuri Wiitala | c05ada2 | 2019-09-24 14:25:19 -0700 | [diff] [blame^] | 93 | runner.PostTask([&ran_tasks] { ran_tasks += "1"; }); |
| 94 | runner.RequestStopSoon(); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 95 | |
Yuri Wiitala | c05ada2 | 2019-09-24 14:25:19 -0700 | [diff] [blame^] | 96 | runner.RunUntilStopped(); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 97 | EXPECT_EQ(ran_tasks, "1"); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 98 | } |
| 99 | |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 100 | TEST(TaskRunnerImplTest, TaskRunnerRunsDelayedTasksInOrder) { |
Jordan Bayles | 5d72bc2 | 2019-04-09 13:33:52 -0700 | [diff] [blame] | 101 | FakeClock fake_clock{platform::Clock::time_point(milliseconds(1337))}; |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 102 | TaskRunnerImpl runner(&fake_clock.now); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 103 | |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 104 | std::thread t([&runner] { runner.RunUntilStopped(); }); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 105 | |
| 106 | std::string ran_tasks = ""; |
| 107 | |
Jordan Bayles | 5d72bc2 | 2019-04-09 13:33:52 -0700 | [diff] [blame] | 108 | const auto kDelayTime = milliseconds(5); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 109 | const auto task_one = [&ran_tasks] { ran_tasks += "1"; }; |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 110 | runner.PostTaskWithDelay(task_one, kDelayTime); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 111 | |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 112 | const auto task_two = [&ran_tasks] { ran_tasks += "2"; }; |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 113 | runner.PostTaskWithDelay(task_two, kDelayTime * 2); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 114 | |
Jordan Bayles | 5d72bc2 | 2019-04-09 13:33:52 -0700 | [diff] [blame] | 115 | EXPECT_EQ(ran_tasks, ""); |
| 116 | fake_clock.Advance(kDelayTime); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 117 | WaitUntilCondition([&ran_tasks] { return ran_tasks == "1"; }); |
| 118 | EXPECT_EQ(ran_tasks, "1"); |
| 119 | |
Jordan Bayles | 5d72bc2 | 2019-04-09 13:33:52 -0700 | [diff] [blame] | 120 | fake_clock.Advance(kDelayTime); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 121 | WaitUntilCondition([&ran_tasks] { return ran_tasks == "12"; }); |
| 122 | EXPECT_EQ(ran_tasks, "12"); |
| 123 | |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 124 | runner.RequestStopSoon(); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 125 | t.join(); |
| 126 | } |
| 127 | |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 128 | TEST(TaskRunnerImplTest, SingleThreadedTaskRunnerRunsSequentially) { |
Jordan Bayles | 5d72bc2 | 2019-04-09 13:33:52 -0700 | [diff] [blame] | 129 | FakeClock fake_clock{platform::Clock::time_point(milliseconds(1337))}; |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 130 | TaskRunnerImpl runner(&fake_clock.now); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 131 | |
| 132 | std::string ran_tasks; |
| 133 | const auto task_one = [&ran_tasks] { ran_tasks += "1"; }; |
| 134 | const auto task_two = [&ran_tasks] { ran_tasks += "2"; }; |
| 135 | const auto task_three = [&ran_tasks] { ran_tasks += "3"; }; |
| 136 | const auto task_four = [&ran_tasks] { ran_tasks += "4"; }; |
| 137 | const auto task_five = [&ran_tasks] { ran_tasks += "5"; }; |
| 138 | |
| 139 | runner.PostTask(task_one); |
| 140 | runner.PostTask(task_two); |
| 141 | runner.PostTask(task_three); |
| 142 | runner.PostTask(task_four); |
| 143 | runner.PostTask(task_five); |
Yuri Wiitala | c05ada2 | 2019-09-24 14:25:19 -0700 | [diff] [blame^] | 144 | runner.RequestStopSoon(); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 145 | EXPECT_EQ(ran_tasks, ""); |
| 146 | |
Yuri Wiitala | c05ada2 | 2019-09-24 14:25:19 -0700 | [diff] [blame^] | 147 | runner.RunUntilStopped(); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 148 | EXPECT_EQ(ran_tasks, "12345"); |
| 149 | } |
| 150 | |
Yuri Wiitala | c05ada2 | 2019-09-24 14:25:19 -0700 | [diff] [blame^] | 151 | TEST(TaskRunnerImplTest, RunsAllImmediateTasksBeforeStopping) { |
Jordan Bayles | 5d72bc2 | 2019-04-09 13:33:52 -0700 | [diff] [blame] | 152 | FakeClock fake_clock{platform::Clock::time_point(milliseconds(1337))}; |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 153 | TaskRunnerImpl runner(&fake_clock.now); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 154 | |
Yuri Wiitala | c05ada2 | 2019-09-24 14:25:19 -0700 | [diff] [blame^] | 155 | std::string result; |
| 156 | runner.PostTask([&] { |
| 157 | result += "Alice"; |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 158 | |
Yuri Wiitala | c05ada2 | 2019-09-24 14:25:19 -0700 | [diff] [blame^] | 159 | // Post a task that runs just before the quit task. |
| 160 | runner.PostTask([&] { |
| 161 | result += " says goodbye"; |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 162 | |
Yuri Wiitala | c05ada2 | 2019-09-24 14:25:19 -0700 | [diff] [blame^] | 163 | // These tasks will enter the queue after the quit task *and* after the |
| 164 | // main loop breaks. They will be executed by the flushing phase. |
| 165 | runner.PostTask([&] { |
| 166 | result += " and is not"; |
| 167 | runner.PostTask([&] { result += " forgotten."; }); |
| 168 | }); |
| 169 | }); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 170 | |
Yuri Wiitala | c05ada2 | 2019-09-24 14:25:19 -0700 | [diff] [blame^] | 171 | // Post the quit task. |
| 172 | runner.RequestStopSoon(); |
| 173 | }); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 174 | |
Yuri Wiitala | c05ada2 | 2019-09-24 14:25:19 -0700 | [diff] [blame^] | 175 | EXPECT_EQ(result, ""); |
| 176 | runner.RunUntilStopped(); |
| 177 | // All posted tasks will execute because RequestStopSoon() guarantees all |
| 178 | // immediately-runnable tasks will run before exiting, even if new |
| 179 | // immediately-runnable tasks are posted in the meantime. |
| 180 | EXPECT_EQ(result, "Alice says goodbye and is not forgotten."); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 181 | } |
| 182 | |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 183 | TEST(TaskRunnerImplTest, TaskRunnerIsStableWithLotsOfTasks) { |
Jordan Bayles | 5d72bc2 | 2019-04-09 13:33:52 -0700 | [diff] [blame] | 184 | FakeClock fake_clock{platform::Clock::time_point(milliseconds(1337))}; |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 185 | TaskRunnerImpl runner(&fake_clock.now); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 186 | |
| 187 | const int kNumberOfTasks = 500; |
| 188 | std::string expected_ran_tasks; |
| 189 | expected_ran_tasks.append(kNumberOfTasks, '1'); |
| 190 | |
| 191 | std::string ran_tasks; |
| 192 | for (int i = 0; i < kNumberOfTasks; ++i) { |
| 193 | const auto task = [&ran_tasks] { ran_tasks += "1"; }; |
| 194 | runner.PostTask(task); |
| 195 | } |
| 196 | |
Yuri Wiitala | c05ada2 | 2019-09-24 14:25:19 -0700 | [diff] [blame^] | 197 | runner.RequestStopSoon(); |
| 198 | runner.RunUntilStopped(); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 199 | EXPECT_EQ(ran_tasks, expected_ran_tasks); |
| 200 | } |
Jordan Bayles | a8e9677 | 2019-04-08 10:53:54 -0700 | [diff] [blame] | 201 | |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 202 | TEST(TaskRunnerImplTest, TaskRunnerDelayedTasksDontBlockImmediateTasks) { |
Jordan Bayles | a8e9677 | 2019-04-08 10:53:54 -0700 | [diff] [blame] | 203 | TaskRunnerImpl runner(platform::Clock::now); |
| 204 | |
| 205 | std::string ran_tasks; |
| 206 | const auto task = [&ran_tasks] { ran_tasks += "1"; }; |
| 207 | const auto delayed_task = [&ran_tasks] { ran_tasks += "A"; }; |
| 208 | |
| 209 | runner.PostTaskWithDelay(delayed_task, milliseconds(10000)); |
| 210 | runner.PostTask(task); |
| 211 | |
Yuri Wiitala | c05ada2 | 2019-09-24 14:25:19 -0700 | [diff] [blame^] | 212 | runner.RequestStopSoon(); |
| 213 | runner.RunUntilStopped(); |
Jordan Bayles | a8e9677 | 2019-04-08 10:53:54 -0700 | [diff] [blame] | 214 | // The immediate task should have run, even though the delayed task |
| 215 | // was added first. |
| 216 | |
| 217 | EXPECT_EQ(ran_tasks, "1"); |
| 218 | } |
btolsch | d94fe62 | 2019-05-09 14:21:40 -0700 | [diff] [blame] | 219 | |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 220 | TEST(TaskRunnerImplTest, TaskRunnerUsesEventWaiter) { |
| 221 | std::unique_ptr<TaskRunnerImpl> runner = |
btolsch | d94fe62 | 2019-05-09 14:21:40 -0700 | [diff] [blame] | 222 | TaskRunnerWithWaiterFactory::Create(Clock::now); |
| 223 | |
btolsch | 691996a | 2019-07-30 18:52:43 -0700 | [diff] [blame] | 224 | std::atomic<int> x{0}; |
btolsch | d94fe62 | 2019-05-09 14:21:40 -0700 | [diff] [blame] | 225 | std::thread t([&runner, &x] { |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 226 | runner.get()->RunUntilStopped(); |
btolsch | d94fe62 | 2019-05-09 14:21:40 -0700 | [diff] [blame] | 227 | x = 1; |
| 228 | }); |
| 229 | |
| 230 | const Clock::time_point start1 = Clock::now(); |
| 231 | FakeTaskWaiter* fake_waiter = TaskRunnerWithWaiterFactory::fake_waiter.get(); |
| 232 | while ((Clock::now() - start1) < kWaitTimeout && !fake_waiter->IsWaiting()) { |
| 233 | std::this_thread::sleep_for(kTaskRunnerSleepTime); |
| 234 | } |
| 235 | ASSERT_TRUE(fake_waiter->IsWaiting()); |
| 236 | |
| 237 | fake_waiter->WakeUpAndStop(); |
| 238 | const Clock::time_point start2 = Clock::now(); |
| 239 | while ((Clock::now() - start2) < kWaitTimeout && x == 0) { |
| 240 | std::this_thread::sleep_for(kTaskRunnerSleepTime); |
| 241 | } |
| 242 | ASSERT_EQ(x, 1); |
| 243 | ASSERT_FALSE(fake_waiter->IsWaiting()); |
| 244 | t.join(); |
| 245 | } |
| 246 | |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 247 | TEST(TaskRunnerImplTest, WakesEventWaiterOnPostTask) { |
| 248 | std::unique_ptr<TaskRunnerImpl> runner = |
btolsch | d94fe62 | 2019-05-09 14:21:40 -0700 | [diff] [blame] | 249 | TaskRunnerWithWaiterFactory::Create(Clock::now); |
| 250 | |
btolsch | 691996a | 2019-07-30 18:52:43 -0700 | [diff] [blame] | 251 | std::atomic<int> x{0}; |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 252 | std::thread t([&runner] { runner.get()->RunUntilStopped(); }); |
btolsch | d94fe62 | 2019-05-09 14:21:40 -0700 | [diff] [blame] | 253 | |
| 254 | const Clock::time_point start1 = Clock::now(); |
| 255 | FakeTaskWaiter* fake_waiter = TaskRunnerWithWaiterFactory::fake_waiter.get(); |
| 256 | while ((Clock::now() - start1) < kWaitTimeout && !fake_waiter->IsWaiting()) { |
| 257 | std::this_thread::sleep_for(kTaskRunnerSleepTime); |
| 258 | } |
| 259 | ASSERT_TRUE(fake_waiter->IsWaiting()); |
| 260 | |
| 261 | runner->PostTask([&x]() { x = 1; }); |
| 262 | const Clock::time_point start2 = Clock::now(); |
| 263 | while ((Clock::now() - start2) < kWaitTimeout && x == 0) { |
| 264 | std::this_thread::sleep_for(kTaskRunnerSleepTime); |
| 265 | } |
| 266 | ASSERT_EQ(x, 1); |
| 267 | |
| 268 | fake_waiter->WakeUpAndStop(); |
| 269 | t.join(); |
| 270 | } |
| 271 | |
Ryan Keane | 32c88d0 | 2019-07-02 18:46:14 -0700 | [diff] [blame] | 272 | class RepeatedClass { |
| 273 | public: |
Ryan Keane | 32c88d0 | 2019-07-02 18:46:14 -0700 | [diff] [blame] | 274 | MOCK_METHOD0(Repeat, absl::optional<Clock::duration>()); |
| 275 | |
| 276 | absl::optional<Clock::duration> DoCall() { |
| 277 | auto result = Repeat(); |
| 278 | execution_count++; |
| 279 | return result; |
| 280 | } |
| 281 | |
btolsch | 691996a | 2019-07-30 18:52:43 -0700 | [diff] [blame] | 282 | std::atomic<int> execution_count{0}; |
Ryan Keane | 32c88d0 | 2019-07-02 18:46:14 -0700 | [diff] [blame] | 283 | }; |
| 284 | |
| 285 | TEST(TaskRunnerImplTest, RepeatingFunctionCalledRepeatedly) { |
| 286 | std::unique_ptr<TaskRunnerImpl> runner = |
| 287 | TaskRunnerWithWaiterFactory::Create(Clock::now); |
| 288 | |
| 289 | std::thread running_thread([&runner]() { runner.get()->RunUntilStopped(); }); |
| 290 | |
| 291 | RepeatedClass c; |
| 292 | EXPECT_CALL(c, Repeat()) |
| 293 | .Times(3) |
| 294 | .WillOnce(Return(Clock::duration(0))) |
| 295 | .WillOnce(Return(Clock::duration(1))) |
| 296 | .WillOnce(Return(absl::nullopt)); |
| 297 | |
| 298 | RepeatingFunction::Post(runner.get(), [&c]() { return c.DoCall(); }); |
| 299 | const Clock::time_point start2 = Clock::now(); |
| 300 | while ((Clock::now() - start2) < kWaitTimeout && c.execution_count < 3) { |
| 301 | std::this_thread::sleep_for(kTaskRunnerSleepTime); |
| 302 | } |
| 303 | ASSERT_EQ(c.execution_count, 3); |
| 304 | |
| 305 | runner->RequestStopSoon(); |
| 306 | running_thread.join(); |
| 307 | } |
| 308 | |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 309 | } // namespace platform |
| 310 | } // namespace openscreen |