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(); |
| 53 | task_runner_->PostTask([this]() { task_runner_->RequestStopSoon(); }); |
| 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 | |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 88 | TEST(TaskRunnerImplTest, TaskRunnerExecutesTask) { |
Jordan Bayles | 5d72bc2 | 2019-04-09 13:33:52 -0700 | [diff] [blame] | 89 | FakeClock fake_clock{platform::Clock::time_point(milliseconds(1337))}; |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 90 | auto runner = std::make_unique<TaskRunnerImpl>(&fake_clock.now); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 91 | |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 92 | std::thread t([&runner] { runner.get()->RunUntilStopped(); }); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 93 | |
| 94 | std::string ran_tasks = ""; |
| 95 | const auto task = [&ran_tasks] { ran_tasks += "1"; }; |
| 96 | EXPECT_EQ(ran_tasks, ""); |
| 97 | |
| 98 | runner->PostTask(task); |
| 99 | |
| 100 | WaitUntilCondition([&ran_tasks] { return ran_tasks == "1"; }); |
| 101 | EXPECT_EQ(ran_tasks, "1"); |
| 102 | |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 103 | runner.get()->RequestStopSoon(); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 104 | t.join(); |
| 105 | } |
| 106 | |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 107 | TEST(TaskRunnerImplTest, TaskRunnerRunsDelayedTasksInOrder) { |
Jordan Bayles | 5d72bc2 | 2019-04-09 13:33:52 -0700 | [diff] [blame] | 108 | FakeClock fake_clock{platform::Clock::time_point(milliseconds(1337))}; |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 109 | TaskRunnerImpl runner(&fake_clock.now); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 110 | |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 111 | std::thread t([&runner] { runner.RunUntilStopped(); }); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 112 | |
| 113 | std::string ran_tasks = ""; |
| 114 | |
Jordan Bayles | 5d72bc2 | 2019-04-09 13:33:52 -0700 | [diff] [blame] | 115 | const auto kDelayTime = milliseconds(5); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 116 | const auto task_one = [&ran_tasks] { ran_tasks += "1"; }; |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 117 | runner.PostTaskWithDelay(task_one, kDelayTime); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 118 | |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 119 | const auto task_two = [&ran_tasks] { ran_tasks += "2"; }; |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 120 | runner.PostTaskWithDelay(task_two, kDelayTime * 2); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 121 | |
Jordan Bayles | 5d72bc2 | 2019-04-09 13:33:52 -0700 | [diff] [blame] | 122 | EXPECT_EQ(ran_tasks, ""); |
| 123 | fake_clock.Advance(kDelayTime); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 124 | WaitUntilCondition([&ran_tasks] { return ran_tasks == "1"; }); |
| 125 | EXPECT_EQ(ran_tasks, "1"); |
| 126 | |
Jordan Bayles | 5d72bc2 | 2019-04-09 13:33:52 -0700 | [diff] [blame] | 127 | fake_clock.Advance(kDelayTime); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 128 | WaitUntilCondition([&ran_tasks] { return ran_tasks == "12"; }); |
| 129 | EXPECT_EQ(ran_tasks, "12"); |
| 130 | |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 131 | runner.RequestStopSoon(); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 132 | t.join(); |
| 133 | } |
| 134 | |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 135 | TEST(TaskRunnerImplTest, SingleThreadedTaskRunnerRunsSequentially) { |
Jordan Bayles | 5d72bc2 | 2019-04-09 13:33:52 -0700 | [diff] [blame] | 136 | FakeClock fake_clock{platform::Clock::time_point(milliseconds(1337))}; |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 137 | TaskRunnerImpl runner(&fake_clock.now); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 138 | |
| 139 | std::string ran_tasks; |
| 140 | const auto task_one = [&ran_tasks] { ran_tasks += "1"; }; |
| 141 | const auto task_two = [&ran_tasks] { ran_tasks += "2"; }; |
| 142 | const auto task_three = [&ran_tasks] { ran_tasks += "3"; }; |
| 143 | const auto task_four = [&ran_tasks] { ran_tasks += "4"; }; |
| 144 | const auto task_five = [&ran_tasks] { ran_tasks += "5"; }; |
| 145 | |
| 146 | runner.PostTask(task_one); |
| 147 | runner.PostTask(task_two); |
| 148 | runner.PostTask(task_three); |
| 149 | runner.PostTask(task_four); |
| 150 | runner.PostTask(task_five); |
| 151 | EXPECT_EQ(ran_tasks, ""); |
| 152 | |
| 153 | runner.RunUntilIdleForTesting(); |
| 154 | EXPECT_EQ(ran_tasks, "12345"); |
| 155 | } |
| 156 | |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 157 | TEST(TaskRunnerImplTest, TaskRunnerCanStopRunning) { |
Jordan Bayles | 5d72bc2 | 2019-04-09 13:33:52 -0700 | [diff] [blame] | 158 | FakeClock fake_clock{platform::Clock::time_point(milliseconds(1337))}; |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 159 | TaskRunnerImpl runner(&fake_clock.now); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 160 | |
| 161 | std::string ran_tasks; |
| 162 | const auto task_one = [&ran_tasks] { ran_tasks += "1"; }; |
| 163 | const auto task_two = [&ran_tasks] { ran_tasks += "2"; }; |
| 164 | |
| 165 | runner.PostTask(task_one); |
| 166 | EXPECT_EQ(ran_tasks, ""); |
| 167 | |
| 168 | std::thread start_thread([&runner] { runner.RunUntilStopped(); }); |
| 169 | |
| 170 | WaitUntilCondition([&ran_tasks] { return !ran_tasks.empty(); }); |
| 171 | EXPECT_EQ(ran_tasks, "1"); |
| 172 | |
| 173 | // Since Stop is called first, and the single threaded task |
| 174 | // runner should honor the queue, we know the task runner is not running |
| 175 | // since task two doesn't get ran. |
| 176 | runner.RequestStopSoon(); |
| 177 | runner.PostTask(task_two); |
| 178 | EXPECT_EQ(ran_tasks, "1"); |
| 179 | |
| 180 | start_thread.join(); |
| 181 | } |
| 182 | |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 183 | TEST(TaskRunnerImplTest, StoppingDoesNotDeleteTasks) { |
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 | std::string ran_tasks; |
| 188 | const auto task_one = [&ran_tasks] { ran_tasks += "1"; }; |
| 189 | |
| 190 | runner.PostTask(task_one); |
| 191 | runner.RequestStopSoon(); |
| 192 | |
| 193 | EXPECT_EQ(ran_tasks, ""); |
| 194 | runner.RunUntilIdleForTesting(); |
| 195 | |
| 196 | EXPECT_EQ(ran_tasks, "1"); |
| 197 | } |
| 198 | |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 199 | TEST(TaskRunnerImplTest, TaskRunnerIsStableWithLotsOfTasks) { |
Jordan Bayles | 5d72bc2 | 2019-04-09 13:33:52 -0700 | [diff] [blame] | 200 | FakeClock fake_clock{platform::Clock::time_point(milliseconds(1337))}; |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 201 | TaskRunnerImpl runner(&fake_clock.now); |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 202 | |
| 203 | const int kNumberOfTasks = 500; |
| 204 | std::string expected_ran_tasks; |
| 205 | expected_ran_tasks.append(kNumberOfTasks, '1'); |
| 206 | |
| 207 | std::string ran_tasks; |
| 208 | for (int i = 0; i < kNumberOfTasks; ++i) { |
| 209 | const auto task = [&ran_tasks] { ran_tasks += "1"; }; |
| 210 | runner.PostTask(task); |
| 211 | } |
| 212 | |
| 213 | runner.RunUntilIdleForTesting(); |
| 214 | EXPECT_EQ(ran_tasks, expected_ran_tasks); |
| 215 | } |
Jordan Bayles | a8e9677 | 2019-04-08 10:53:54 -0700 | [diff] [blame] | 216 | |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 217 | TEST(TaskRunnerImplTest, TaskRunnerDelayedTasksDontBlockImmediateTasks) { |
Jordan Bayles | a8e9677 | 2019-04-08 10:53:54 -0700 | [diff] [blame] | 218 | TaskRunnerImpl runner(platform::Clock::now); |
| 219 | |
| 220 | std::string ran_tasks; |
| 221 | const auto task = [&ran_tasks] { ran_tasks += "1"; }; |
| 222 | const auto delayed_task = [&ran_tasks] { ran_tasks += "A"; }; |
| 223 | |
| 224 | runner.PostTaskWithDelay(delayed_task, milliseconds(10000)); |
| 225 | runner.PostTask(task); |
| 226 | |
| 227 | runner.RunUntilIdleForTesting(); |
| 228 | // The immediate task should have run, even though the delayed task |
| 229 | // was added first. |
| 230 | |
| 231 | EXPECT_EQ(ran_tasks, "1"); |
| 232 | } |
btolsch | d94fe62 | 2019-05-09 14:21:40 -0700 | [diff] [blame] | 233 | |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 234 | TEST(TaskRunnerImplTest, TaskRunnerUsesEventWaiter) { |
| 235 | std::unique_ptr<TaskRunnerImpl> runner = |
btolsch | d94fe62 | 2019-05-09 14:21:40 -0700 | [diff] [blame] | 236 | TaskRunnerWithWaiterFactory::Create(Clock::now); |
| 237 | |
btolsch | 691996a | 2019-07-30 18:52:43 -0700 | [diff] [blame] | 238 | std::atomic<int> x{0}; |
btolsch | d94fe62 | 2019-05-09 14:21:40 -0700 | [diff] [blame] | 239 | std::thread t([&runner, &x] { |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 240 | runner.get()->RunUntilStopped(); |
btolsch | d94fe62 | 2019-05-09 14:21:40 -0700 | [diff] [blame] | 241 | x = 1; |
| 242 | }); |
| 243 | |
| 244 | const Clock::time_point start1 = Clock::now(); |
| 245 | FakeTaskWaiter* fake_waiter = TaskRunnerWithWaiterFactory::fake_waiter.get(); |
| 246 | while ((Clock::now() - start1) < kWaitTimeout && !fake_waiter->IsWaiting()) { |
| 247 | std::this_thread::sleep_for(kTaskRunnerSleepTime); |
| 248 | } |
| 249 | ASSERT_TRUE(fake_waiter->IsWaiting()); |
| 250 | |
| 251 | fake_waiter->WakeUpAndStop(); |
| 252 | const Clock::time_point start2 = Clock::now(); |
| 253 | while ((Clock::now() - start2) < kWaitTimeout && x == 0) { |
| 254 | std::this_thread::sleep_for(kTaskRunnerSleepTime); |
| 255 | } |
| 256 | ASSERT_EQ(x, 1); |
| 257 | ASSERT_FALSE(fake_waiter->IsWaiting()); |
| 258 | t.join(); |
| 259 | } |
| 260 | |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 261 | TEST(TaskRunnerImplTest, WakesEventWaiterOnPostTask) { |
| 262 | std::unique_ptr<TaskRunnerImpl> runner = |
btolsch | d94fe62 | 2019-05-09 14:21:40 -0700 | [diff] [blame] | 263 | TaskRunnerWithWaiterFactory::Create(Clock::now); |
| 264 | |
btolsch | 691996a | 2019-07-30 18:52:43 -0700 | [diff] [blame] | 265 | std::atomic<int> x{0}; |
btolsch | 4051e72 | 2019-06-07 16:15:17 -0700 | [diff] [blame] | 266 | std::thread t([&runner] { runner.get()->RunUntilStopped(); }); |
btolsch | d94fe62 | 2019-05-09 14:21:40 -0700 | [diff] [blame] | 267 | |
| 268 | const Clock::time_point start1 = Clock::now(); |
| 269 | FakeTaskWaiter* fake_waiter = TaskRunnerWithWaiterFactory::fake_waiter.get(); |
| 270 | while ((Clock::now() - start1) < kWaitTimeout && !fake_waiter->IsWaiting()) { |
| 271 | std::this_thread::sleep_for(kTaskRunnerSleepTime); |
| 272 | } |
| 273 | ASSERT_TRUE(fake_waiter->IsWaiting()); |
| 274 | |
| 275 | runner->PostTask([&x]() { x = 1; }); |
| 276 | const Clock::time_point start2 = Clock::now(); |
| 277 | while ((Clock::now() - start2) < kWaitTimeout && x == 0) { |
| 278 | std::this_thread::sleep_for(kTaskRunnerSleepTime); |
| 279 | } |
| 280 | ASSERT_EQ(x, 1); |
| 281 | |
| 282 | fake_waiter->WakeUpAndStop(); |
| 283 | t.join(); |
| 284 | } |
| 285 | |
Ryan Keane | 32c88d0 | 2019-07-02 18:46:14 -0700 | [diff] [blame] | 286 | class RepeatedClass { |
| 287 | public: |
Ryan Keane | 32c88d0 | 2019-07-02 18:46:14 -0700 | [diff] [blame] | 288 | MOCK_METHOD0(Repeat, absl::optional<Clock::duration>()); |
| 289 | |
| 290 | absl::optional<Clock::duration> DoCall() { |
| 291 | auto result = Repeat(); |
| 292 | execution_count++; |
| 293 | return result; |
| 294 | } |
| 295 | |
btolsch | 691996a | 2019-07-30 18:52:43 -0700 | [diff] [blame] | 296 | std::atomic<int> execution_count{0}; |
Ryan Keane | 32c88d0 | 2019-07-02 18:46:14 -0700 | [diff] [blame] | 297 | }; |
| 298 | |
| 299 | TEST(TaskRunnerImplTest, RepeatingFunctionCalledRepeatedly) { |
| 300 | std::unique_ptr<TaskRunnerImpl> runner = |
| 301 | TaskRunnerWithWaiterFactory::Create(Clock::now); |
| 302 | |
| 303 | std::thread running_thread([&runner]() { runner.get()->RunUntilStopped(); }); |
| 304 | |
| 305 | RepeatedClass c; |
| 306 | EXPECT_CALL(c, Repeat()) |
| 307 | .Times(3) |
| 308 | .WillOnce(Return(Clock::duration(0))) |
| 309 | .WillOnce(Return(Clock::duration(1))) |
| 310 | .WillOnce(Return(absl::nullopt)); |
| 311 | |
| 312 | RepeatingFunction::Post(runner.get(), [&c]() { return c.DoCall(); }); |
| 313 | const Clock::time_point start2 = Clock::now(); |
| 314 | while ((Clock::now() - start2) < kWaitTimeout && c.execution_count < 3) { |
| 315 | std::this_thread::sleep_for(kTaskRunnerSleepTime); |
| 316 | } |
| 317 | ASSERT_EQ(c.execution_count, 3); |
| 318 | |
| 319 | runner->RequestStopSoon(); |
| 320 | running_thread.join(); |
| 321 | } |
| 322 | |
Jordan Bayles | b0c191e | 2019-03-26 15:49:57 -0700 | [diff] [blame] | 323 | } // namespace platform |
| 324 | } // namespace openscreen |