2015-12-08 04:51:55 +08:00
|
|
|
#pragma once
|
2015-10-12 02:59:27 +08:00
|
|
|
|
|
|
|
#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)
|
2015-10-12 02:59:27 +08:00
|
|
|
: work(std::move(work)), after_work(std::move(after_work))
|
|
|
|
{
|
2016-03-19 19:39:30 +08:00
|
|
|
req.data = static_cast<void*>(this);
|
2015-10-12 02:59:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
2015-10-12 02:59:27 +08:00
|
|
|
|
2016-03-19 19:39:30 +08:00
|
|
|
static void work_cb(uv_work_t* req)
|
2015-10-12 02:59:27 +08:00
|
|
|
{
|
2016-03-19 19:39:30 +08:00
|
|
|
auto& work = *static_cast<Work<T>*>(req->data);
|
|
|
|
work();
|
|
|
|
}
|
2015-10-12 02:59:27 +08:00
|
|
|
|
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);
|
2015-10-12 02:59:27 +08:00
|
|
|
|
2016-03-19 19:39:30 +08:00
|
|
|
work->after_work(std::move(work->result.get()));
|
2015-10-12 02:59:27 +08:00
|
|
|
|
2016-03-19 19:39:30 +08:00
|
|
|
delete work;
|
2015-10-12 02:59:27 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
public:
|
|
|
|
using sptr = std::shared_ptr<Task>;
|
|
|
|
|
2016-03-19 19:39:30 +08:00
|
|
|
Task(uv::UvLoop::sptr loop) : loop(loop) {}
|
2015-10-12 02:59:27 +08:00
|
|
|
|
|
|
|
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),
|
2015-10-12 02:59:27 +08:00
|
|
|
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);
|
2015-10-12 02:59:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
uv::UvLoop::sptr loop;
|
|
|
|
};
|