#ifndef BASE_TASK_RUNNER_H #define BASE_TASK_RUNNER_H #include #include #include #include #include #include #include "base/closure.h" namespace base { namespace internal { // Adapted from Chromium project. // Adapts a function that produces a result via a return value to // one that returns via an output parameter. template void ReturnAsParamAdapter(std::function func, ReturnType* result) { *result = func(); } // Adapts a ReturnType* result to a callblack that expects a ReturnType. template void ReplyAdapter(std::function callback, ReturnType* result) { callback(*result); delete result; } } // namespace internal // Runs queued tasks (in the form of Closure objects). All methods are // thread-safe and can be called on any thread. // Tasks run in FIFO order. When consumed concurrently by multiple threads, it // doesn't guarantee whether tasks overlap, or whether they run on a particular // thread. class TaskRunner { public: TaskRunner() = default; ~TaskRunner() = default; void PostTask(const Location& from, Closure task); void PostTaskAndReply(const Location& from, Closure task, Closure reply); template void PostTaskAndReplyWithResult(const Location& from, std::function task, std::function reply) { auto* result = new ReturnType; return PostTaskAndReply( from, std::bind(internal::ReturnAsParamAdapter, std::move(task), result), std::bind(internal::ReplyAdapter, std::move(reply), result)); } void MultiConsumerRun(); void SingleConsumerRun(); void WaitForCompletion(); static void CreateThreadLocalTaskRunner(); static TaskRunner* GetThreadLocalTaskRunner(); private: using Task = std::tuple; std::deque queue_; mutable std::mutex lock_; std::condition_variable cv_; std::atomic task_count_{0}; static thread_local std::unique_ptr thread_local_task_runner; TaskRunner(TaskRunner const&) = delete; TaskRunner& operator=(TaskRunner const&) = delete; }; } // namespace base #endif // BASE_TASK_RUNNER_H