memgraph/include/threading/task.hpp

79 lines
1.6 KiB
C++
Raw Normal View History

#pragma once
#include <iostream>
#include "pool.hpp"
#include "io/uv/uvloop.hpp"
#include "utils/placeholder.hpp"
class Task
{
template <class T>
using work_t = std::function<T(void)>;
template <class T>
using after_work_t = std::function<void(T)>;
template <class T>
struct Work
{
2016-03-19 19:39:30 +08:00
Work(work_t<T> work, after_work_t<T> after_work)
: work(std::move(work)), after_work(std::move(after_work))
{
2016-03-19 19:39:30 +08:00
req.data = static_cast<void*>(this);
}
void operator()()
{
result.set(std::move(work()));
}
work_t<T> work;
after_work_t<T> after_work;
Placeholder<T> result;
2016-03-19 19:39:30 +08:00
uv_work_t req;
2016-03-19 19:39:30 +08:00
static void work_cb(uv_work_t* req)
{
2016-03-19 19:39:30 +08:00
auto& work = *static_cast<Work<T>*>(req->data);
work();
}
2016-03-19 19:39:30 +08:00
static void after_work_cb(uv_work_t* req, int)
{
auto work = static_cast<Work<T>*>(req->data);
2016-03-19 19:39:30 +08:00
work->after_work(std::move(work->result.get()));
2016-03-19 19:39:30 +08:00
delete work;
}
};
public:
using sptr = std::shared_ptr<Task>;
2016-03-19 19:39:30 +08:00
Task(uv::UvLoop::sptr loop) : loop(loop) {}
Task(Task&) = delete;
Task(Task&&) = delete;
template <class F1, class F2>
void run(F1&& work, F2&& after_work)
{
using T = decltype(work());
2016-03-19 19:39:30 +08:00
auto w = new Work<T>(std::forward<F1>(work),
std::forward<F2>(after_work));
2016-03-19 19:39:30 +08:00
uv_queue_work(*loop, &w->req,
Work<T>::work_cb,
Work<T>::after_work_cb);
}
private:
uv::UvLoop::sptr loop;
};