/* * Copyright (c) 2011 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. */ #ifndef MODULES_UTILITY_SOURCE_PROCESS_THREAD_IMPL_H_ #define MODULES_UTILITY_SOURCE_PROCESS_THREAD_IMPL_H_ #include #include #include #include #include "api/sequence_checker.h" #include "api/task_queue/queued_task.h" #include "modules/include/module.h" #include "modules/utility/include/process_thread.h" #include "rtc_base/event.h" #include "rtc_base/location.h" #include "rtc_base/platform_thread.h" namespace webrtc { class ProcessThreadImpl : public ProcessThread { public: explicit ProcessThreadImpl(const char* thread_name); ~ProcessThreadImpl() override; void Start() override; void Stop() override; void WakeUp(Module* module) override; void PostTask(std::unique_ptr task) override; void PostDelayedTask(std::unique_ptr task, uint32_t milliseconds) override; void RegisterModule(Module* module, const rtc::Location& from) override; void DeRegisterModule(Module* module) override; protected: bool Process(); private: struct ModuleCallback { ModuleCallback() = delete; ModuleCallback(ModuleCallback&& cb) = default; ModuleCallback(const ModuleCallback& cb) = default; ModuleCallback(Module* module, const rtc::Location& location) : module(module), location(location) {} bool operator==(const ModuleCallback& cb) const { return cb.module == module; } Module* const module; int64_t next_callback = 0; // Absolute timestamp. const rtc::Location location; private: ModuleCallback& operator=(ModuleCallback&); }; struct DelayedTask { DelayedTask(int64_t run_at_ms, uint64_t sequence_id, std::unique_ptr task) : run_at_ms(run_at_ms), sequence_id_(sequence_id), task(task.release()) {} friend bool operator<(const DelayedTask& lhs, const DelayedTask& rhs) { // Earliest DelayedTask should be at the top of the priority queue. if (lhs.run_at_ms != rhs.run_at_ms) { return lhs.run_at_ms > rhs.run_at_ms; } return lhs.sequence_id_ > rhs.sequence_id_; } int64_t run_at_ms; uint64_t sequence_id_; // DelayedTask owns the `task`, but some delayed tasks must be removed from // the std::priority_queue, but mustn't be deleted. std::priority_queue does // not give non-const access to the values, so storing unique_ptr would // delete the task as soon as it is remove from the priority queue. // Thus lifetime of the `task` is managed manually. QueuedTask* task; }; typedef std::list ModuleList; void Delete() override; // The part of Stop processing that doesn't need any locking. void StopNoLocks(); void WakeUpNoLocks(Module* module); void WakeUpInternal(Module* module) RTC_EXCLUSIVE_LOCKS_REQUIRED(mutex_); // Members protected by this mutex are accessed on the constructor thread and // on the spawned process thread, and locking is needed only while the process // thread is running. Mutex mutex_; SequenceChecker thread_checker_; rtc::Event wake_up_; rtc::PlatformThread thread_; ModuleList modules_ RTC_GUARDED_BY(mutex_); // Set to true when calling Process, to allow reentrant calls to WakeUp. bool holds_mutex_ RTC_GUARDED_BY(this) = false; std::queue queue_; // `std::priority_queue` does not guarantee stable sort. For delayed tasks // with the same wakeup time, use `sequence_id_` to ensure FIFO ordering. std::priority_queue delayed_tasks_ RTC_GUARDED_BY(mutex_); uint64_t sequence_id_ RTC_GUARDED_BY(mutex_) = 0; // The `stop_` flag is modified only by the construction thread, protected by // `thread_checker_`. It is read also by the spawned `thread_`. The latter // thread must take `mutex_` before access, and for thread safety, the // constructor thread needs to take `mutex_` when it modifies `stop_` and // `thread_` is running. Annotations like RTC_GUARDED_BY doesn't support this // usage pattern. bool stop_ RTC_GUARDED_BY(mutex_); const char* thread_name_; }; } // namespace webrtc #endif // MODULES_UTILITY_SOURCE_PROCESS_THREAD_IMPL_H_