mirror of https://github.com/auygun/kaliber.git
89 lines
2.4 KiB
C++
89 lines
2.4 KiB
C++
#ifndef TASK_RUNNER_H
|
|
#define TASK_RUNNER_H
|
|
|
|
#include <atomic>
|
|
#include <condition_variable>
|
|
#include <deque>
|
|
#include <memory>
|
|
#include <mutex>
|
|
#include <tuple>
|
|
|
|
#include "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 <typename ReturnType>
|
|
void ReturnAsParamAdapter(std::function<ReturnType()> func,
|
|
ReturnType* result) {
|
|
*result = func();
|
|
}
|
|
|
|
// Adapts a ReturnType* result to a callblack that expects a ReturnType.
|
|
template <typename ReturnType>
|
|
void ReplyAdapter(std::function<void(ReturnType)> 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 <typename ReturnType>
|
|
void PostTaskAndReplyWithResult(const Location& from,
|
|
std::function<ReturnType()> task,
|
|
std::function<void(ReturnType)> reply) {
|
|
auto* result = new ReturnType;
|
|
return PostTaskAndReply(
|
|
from,
|
|
std::bind(internal::ReturnAsParamAdapter<ReturnType>, std::move(task),
|
|
result),
|
|
std::bind(internal::ReplyAdapter<ReturnType>, std::move(reply),
|
|
result));
|
|
}
|
|
|
|
void MultiConsumerRun();
|
|
|
|
void SingleConsumerRun();
|
|
|
|
void WaitForCompletion();
|
|
|
|
static void CreateThreadLocalTaskRunner();
|
|
static TaskRunner* GetThreadLocalTaskRunner();
|
|
|
|
private:
|
|
using Task = std::tuple<Location, Closure>;
|
|
|
|
std::deque<Task> queue_;
|
|
mutable std::mutex lock_;
|
|
std::condition_variable cv_;
|
|
std::atomic<size_t> task_count_{0};
|
|
|
|
static thread_local std::unique_ptr<TaskRunner> thread_local_task_runner;
|
|
|
|
TaskRunner(TaskRunner const&) = delete;
|
|
TaskRunner& operator=(TaskRunner const&) = delete;
|
|
};
|
|
|
|
} // namespace base
|
|
|
|
#endif // TASK_RUNNER_H
|