Introduce a RunLoop class that supports the TaskQueue interface
on the current thread.
This simplifies writing async tests that use TaskQueue and doesn't
require spinning up a new thread for simple things. The implementation
is currently based on rtc::Thread, which could also be useful in
some circumstances while migrating code over to TQ.
Remove PressEnterToContinue from the test_common files since
it's very specific and only used from one file.
Bug: none
Change-Id: I8b2c6c40809271a109ec17cf7e1120847645d58a
Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/174260
Reviewed-by: Mirko Bonadei <mbonadei@webrtc.org>
Commit-Queue: Tommi <tommi@webrtc.org>
Cr-Commit-Position: refs/heads/master@{#31160}
diff --git a/test/BUILD.gn b/test/BUILD.gn
index 58129bb..34da889 100644
--- a/test/BUILD.gn
+++ b/test/BUILD.gn
@@ -458,6 +458,7 @@
":perf_test",
":rtc_expect_death",
":rtp_test_utils",
+ ":test_common",
":test_main",
":test_support",
":test_support_test_artifacts",
@@ -487,7 +488,9 @@
"../modules/video_coding:webrtc_vp9",
"../rtc_base:criticalsection",
"../rtc_base:rtc_event",
+ "../rtc_base:rtc_task_queue",
"../rtc_base/system:file_wrapper",
+ "../rtc_base/task_utils:to_queued_task",
"pc/e2e:e2e_unittests",
"peer_scenario/tests",
"scenario:scenario_unittests",
@@ -505,6 +508,7 @@
"frame_generator_unittest.cc",
"rtp_file_reader_unittest.cc",
"rtp_file_writer_unittest.cc",
+ "run_loop_unittest.cc",
"testsupport/ivf_video_frame_generator_unittest.cc",
"testsupport/perf_test_unittest.cc",
"testsupport/test_artifacts_unittest.cc",
@@ -784,22 +788,11 @@
"layer_filtering_transport.cc",
"layer_filtering_transport.h",
"rtp_rtcp_observer.h",
+ "run_loop.cc",
+ "run_loop.h",
"video_decoder_proxy_factory.h",
"video_encoder_proxy_factory.h",
]
- if (current_os != "winuwp") {
- # The filtering of *_win.cc is not done for WinUWP (intentionally) as
- # most _win.cc files are compatible with WinUWP. However, the
- # peek/dispatch Win32 runloops are entirely WinUWP incompatible thus
- # WinUWP uses the generic runloop as defined for non-Windows targets.
- sources += [ "win/run_loop_win.cc" ]
- }
- if (!is_win || current_os == "winuwp") {
- sources += [
- "run_loop.cc",
- "run_loop.h",
- ]
- }
deps = [
":direct_transport",
@@ -840,8 +833,10 @@
"../modules/video_coding:codec_globals_headers",
"../rtc_base:checks",
"../rtc_base:criticalsection",
+ "../rtc_base:rtc_base",
"../rtc_base:rtc_event",
"../rtc_base:task_queue_for_test",
+ "../rtc_base/task_utils:to_queued_task",
"../system_wrappers",
"../system_wrappers:field_trial",
"//third_party/abseil-cpp/absl/types:optional",
diff --git a/test/run_loop.cc b/test/run_loop.cc
index 1fc200f..643da5d 100644
--- a/test/run_loop.cc
+++ b/test/run_loop.cc
@@ -9,15 +9,65 @@
*/
#include "test/run_loop.h"
-#include <stdio.h>
+#include "rtc_base/task_utils/to_queued_task.h"
namespace webrtc {
namespace test {
-void PressEnterToContinue(TaskQueueBase* /*task_queue*/) {
- puts(">> Press ENTER to continue...");
- while (getc(stdin) != '\n' && !feof(stdin))
- ;
+RunLoop::RunLoop() {
+ worker_thread_.WrapCurrent();
}
+
+RunLoop::~RunLoop() {
+ worker_thread_.UnwrapCurrent();
+}
+
+TaskQueueBase* RunLoop::task_queue() {
+ return &worker_thread_;
+}
+
+void RunLoop::Run() {
+ worker_thread_.ProcessMessages(WorkerThread::kForever);
+}
+
+void RunLoop::Quit() {
+ socket_server_.FailNextWait();
+}
+
+void RunLoop::Flush() {
+ worker_thread_.PostTask(
+ ToQueuedTask([this]() { socket_server_.FailNextWait(); }));
+ worker_thread_.ProcessMessages(1000);
+}
+
+RunLoop::FakeSocketServer::FakeSocketServer() = default;
+RunLoop::FakeSocketServer::~FakeSocketServer() = default;
+
+void RunLoop::FakeSocketServer::FailNextWait() {
+ fail_next_wait_ = true;
+}
+
+bool RunLoop::FakeSocketServer::Wait(int cms, bool process_io) {
+ if (fail_next_wait_) {
+ fail_next_wait_ = false;
+ return false;
+ }
+ return true;
+}
+
+void RunLoop::FakeSocketServer::WakeUp() {}
+
+rtc::Socket* RunLoop::FakeSocketServer::CreateSocket(int family, int type) {
+ return nullptr;
+}
+
+rtc::AsyncSocket* RunLoop::FakeSocketServer::CreateAsyncSocket(int family,
+ int type) {
+ return nullptr;
+}
+
+RunLoop::WorkerThread::WorkerThread(rtc::SocketServer* ss)
+ : rtc::Thread(ss), tq_setter_(this) {}
+
} // namespace test
} // namespace webrtc
diff --git a/test/run_loop.h b/test/run_loop.h
index 414e72c..f350b2c 100644
--- a/test/run_loop.h
+++ b/test/run_loop.h
@@ -10,13 +10,69 @@
#ifndef TEST_RUN_LOOP_H_
#define TEST_RUN_LOOP_H_
-#include "api/task_queue/task_queue_base.h"
+#include "rtc_base/task_utils/to_queued_task.h"
+#include "rtc_base/thread.h"
namespace webrtc {
namespace test {
-// Blocks until the user presses enter.
-void PressEnterToContinue(TaskQueueBase* task_queue);
+// This utility class allows you to run a TaskQueue supported interface on the
+// main test thread, call Run() while doing things asynchonously and break
+// the loop (from the same thread) from a callback by calling Quit().
+class RunLoop {
+ public:
+ RunLoop();
+ ~RunLoop();
+
+ TaskQueueBase* task_queue();
+
+ void Run();
+ void Quit();
+
+ void Flush();
+
+ // Convenience methods since TaskQueueBase doesn't support this sort of magic.
+ template <typename Closure>
+ void PostTask(Closure&& task) {
+ task_queue()->PostTask(ToQueuedTask(std::forward<Closure>(task)));
+ }
+
+ template <typename Closure>
+ void PostDelayedTask(Closure&& task, uint32_t milliseconds) {
+ task_queue()->PostDelayedTask(ToQueuedTask(std::forward<Closure>(task)),
+ milliseconds);
+ }
+
+ private:
+ class FakeSocketServer : public rtc::SocketServer {
+ public:
+ FakeSocketServer();
+ ~FakeSocketServer();
+
+ void FailNextWait();
+
+ private:
+ bool Wait(int cms, bool process_io) override;
+ void WakeUp() override;
+
+ rtc::Socket* CreateSocket(int family, int type) override;
+ rtc::AsyncSocket* CreateAsyncSocket(int family, int type) override;
+
+ private:
+ bool fail_next_wait_ = false;
+ };
+
+ class WorkerThread : public rtc::Thread {
+ public:
+ explicit WorkerThread(rtc::SocketServer* ss);
+
+ private:
+ CurrentTaskQueueSetter tq_setter_;
+ };
+
+ FakeSocketServer socket_server_;
+ WorkerThread worker_thread_{&socket_server_};
+};
} // namespace test
} // namespace webrtc
diff --git a/test/run_loop_unittest.cc b/test/run_loop_unittest.cc
new file mode 100644
index 0000000..a356cc2
--- /dev/null
+++ b/test/run_loop_unittest.cc
@@ -0,0 +1,61 @@
+/*
+ * Copyright (c) 2020 The WebRTC project authors. All Rights Reserved.
+ *
+ * Use of this source code is governed by a BSD-style license
+ * that can be found in the LICENSE file in the root of the source
+ * tree. An additional intellectual property rights grant can be found
+ * in the file PATENTS. All contributing project authors may
+ * be found in the AUTHORS file in the root of the source tree.
+ */
+
+#include "test/run_loop.h"
+
+#include "rtc_base/task_queue.h"
+#include "rtc_base/task_utils/to_queued_task.h"
+#include "test/gtest.h"
+
+namespace webrtc {
+
+TEST(RunLoopTest, TaskQueueOnThread) {
+ EXPECT_EQ(TaskQueueBase::Current(), nullptr);
+ test::RunLoop loop;
+ EXPECT_EQ(TaskQueueBase::Current(), loop.task_queue());
+ EXPECT_TRUE(loop.task_queue()->IsCurrent());
+}
+
+TEST(RunLoopTest, Flush) {
+ test::RunLoop loop;
+ int counter = 0;
+ loop.PostTask([&counter]() { ++counter; });
+ EXPECT_EQ(counter, 0);
+ loop.Flush();
+ EXPECT_EQ(counter, 1);
+}
+
+TEST(RunLoopTest, Delayed) {
+ test::RunLoop loop;
+ bool ran = false;
+ loop.PostDelayedTask(
+ [&ran, &loop]() {
+ ran = true;
+ loop.Quit();
+ },
+ 100);
+ loop.Flush();
+ EXPECT_FALSE(ran);
+ loop.Run();
+ EXPECT_TRUE(ran);
+}
+
+TEST(RunLoopTest, PostAndQuit) {
+ test::RunLoop loop;
+ bool ran = false;
+ loop.PostTask([&ran, &loop]() {
+ ran = true;
+ loop.Quit();
+ });
+ loop.Run();
+ EXPECT_TRUE(ran);
+}
+
+} // namespace webrtc
diff --git a/test/win/run_loop_win.cc b/test/win/run_loop_win.cc
deleted file mode 100644
index 95de16bf..0000000
--- a/test/win/run_loop_win.cc
+++ /dev/null
@@ -1,36 +0,0 @@
-/*
- * Copyright (c) 2013 The WebRTC project authors. All Rights Reserved.
- *
- * Use of this source code is governed by a BSD-style license
- * that can be found in the LICENSE file in the root of the source
- * tree. An additional intellectual property rights grant can be found
- * in the file PATENTS. All contributing project authors may
- * be found in the AUTHORS file in the root of the source tree.
- */
-#include <Windows.h>
-#include <assert.h>
-#include <conio.h>
-#include <stdio.h>
-
-#include "rtc_base/task_queue_for_test.h"
-#include "test/run_loop.h"
-
-namespace webrtc {
-namespace test {
-
-void PressEnterToContinue(TaskQueueBase* task_queue) {
- puts(">> Press ENTER to continue...");
-
- while (!_kbhit() || _getch() != '\r') {
- // Drive the message loop for the thread running the task_queue
- SendTask(RTC_FROM_HERE, task_queue, [&]() {
- MSG msg;
- if (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) {
- TranslateMessage(&msg);
- DispatchMessage(&msg);
- }
- });
- }
-}
-} // namespace test
-} // namespace webrtc