blob: c3497a1b4b6e98329585aa5343add0c27b072377 [file] [log] [blame]
/*
* Copyright (c) 2022 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 "modules/utility/maybe_worker_thread.h"
#include <utility>
#include "api/task_queue/pending_task_safety_flag.h"
#include "api/task_queue/task_queue_base.h"
#include "rtc_base/checks.h"
#include "rtc_base/event.h"
#include "rtc_base/logging.h"
#include "rtc_base/task_queue.h"
namespace webrtc {
MaybeWorkerThread::MaybeWorkerThread(const FieldTrialsView& field_trials,
absl::string_view task_queue_name,
TaskQueueFactory* factory)
: owned_task_queue_(
field_trials.IsEnabled("WebRTC-SendPacketsOnWorkerThread")
? nullptr
: factory->CreateTaskQueue(task_queue_name,
rtc::TaskQueue::Priority::NORMAL)),
worker_thread_(TaskQueueBase::Current()) {
RTC_DCHECK(worker_thread_);
RTC_LOG(LS_INFO) << "WebRTC-SendPacketsOnWorkerThread"
<< (owned_task_queue_ ? " Disabled" : " Enabled");
}
MaybeWorkerThread::~MaybeWorkerThread() {
RTC_DCHECK_RUN_ON(&sequence_checker_);
if (owned_task_queue_) {
// owned_task_queue_ must be a valid pointer when the task queue is
// destroyed since there may be tasks that use this object that run when the
// task queue is deleted.
owned_task_queue_->Delete();
owned_task_queue_.release();
}
}
void MaybeWorkerThread::RunSynchronous(absl::AnyInvocable<void() &&> task) {
if (owned_task_queue_) {
rtc::Event thread_sync_event;
auto closure = [&thread_sync_event, task = std::move(task)]() mutable {
std::move(task)();
thread_sync_event.Set();
};
owned_task_queue_->PostTask(std::move(closure));
thread_sync_event.Wait(rtc::Event::kForever);
} else {
RTC_DCHECK_RUN_ON(&sequence_checker_);
std::move(task)();
}
}
void MaybeWorkerThread::RunOrPost(absl::AnyInvocable<void() &&> task) {
if (owned_task_queue_) {
owned_task_queue_->PostTask(std::move(task));
} else {
RTC_DCHECK_RUN_ON(&sequence_checker_);
std::move(task)();
}
}
TaskQueueBase* MaybeWorkerThread::TaskQueueForDelayedTasks() const {
RTC_DCHECK(IsCurrent());
return owned_task_queue_ ? owned_task_queue_.get() : worker_thread_;
}
TaskQueueBase* MaybeWorkerThread::TaskQueueForPost() const {
RTC_DLOG_IF(LS_WARNING, IsCurrent())
<< "TaskQueueForPost called on the current thread. Ok only in unit "
"tests.";
return owned_task_queue_ ? owned_task_queue_.get() : worker_thread_;
}
bool MaybeWorkerThread::IsCurrent() const {
if (owned_task_queue_) {
return owned_task_queue_->IsCurrent();
}
return worker_thread_->IsCurrent();
}
absl::AnyInvocable<void() &&> MaybeWorkerThread::MaybeSafeTask(
rtc::scoped_refptr<PendingTaskSafetyFlag> flag,
absl::AnyInvocable<void() &&> task) {
if (owned_task_queue_) {
return task;
} else {
return SafeTask(std::move(flag), std::move(task));
}
}
} // namespace webrtc