| // Copyright 2015 The Weave Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifndef LIBWEAVE_EXAMPLES_PROVIDER_EVENT_TASK_RUNNER_H_ |
| #define LIBWEAVE_EXAMPLES_PROVIDER_EVENT_TASK_RUNNER_H_ |
| |
| #include <queue> |
| #include <utility> |
| #include <vector> |
| |
| #include <event2/event.h> |
| #include <weave/provider/task_runner.h> |
| |
| #include "examples/provider/event_deleter.h" |
| |
| namespace weave { |
| namespace examples { |
| |
| // Simple task runner implemented with libevent message loop. |
| class EventTaskRunner : public provider::TaskRunner { |
| public: |
| void PostDelayedTask(const tracked_objects::Location& from_here, |
| const base::Closure& task, |
| base::TimeDelta delay) override; |
| |
| // Defines the types of I/O completion events that the |
| // application can register to receive on a file descriptor. |
| enum IOEvent : int16_t { |
| kReadable = 0x01, |
| kWriteable = 0x02, |
| kClosed = 0x04, |
| kReadableWriteable = kReadable | kWriteable, |
| kReadableOrClosed = kReadable | kClosed, |
| kAll = kReadableOrClosed | kWriteable, |
| }; |
| |
| // Callback type for I/O completion events. |
| // Arguments: |
| // fd - file descriptor that triggered the event |
| // what - combination of IOEvent flags indicating |
| // which event(s) occurred |
| // sender - reference to the EventTaskRunner that |
| // called the IoCompletionCallback |
| using IoCompletionCallback = |
| base::Callback<void(int fd, int16_t what, EventTaskRunner* sender)>; |
| |
| // Adds a handler for the specified IO completion events on a file |
| // descriptor. The 'what' parameter is a combination of IOEvent flags. |
| // Only one callback is allowed per file descriptor; calling this function |
| // with an fd that has already been registered will replace the previous |
| // callback with the new one. |
| void AddIoCompletionTask(int fd, |
| int16_t what, |
| const IoCompletionCallback& task); |
| |
| // Remove the callback associated with this fd and stop listening for |
| // events related to it. |
| void RemoveIoCompletionTask(int fd); |
| |
| event_base* GetEventBase() const { return base_.get(); } |
| |
| void Run(); |
| |
| private: |
| void ReScheduleEvent(base::TimeDelta delay); |
| static void EventHandler(int, int16_t, void* runner); |
| static void FreeEvent(event* evnt); |
| void Process(); |
| |
| static void FdEventHandler(int fd, int16_t what, void* runner); |
| void ProcessFd(int fd, int16_t what); |
| |
| using QueueItem = std::pair<std::pair<base::Time, size_t>, base::Closure>; |
| |
| struct Greater { |
| bool operator()(const QueueItem& a, const QueueItem& b) const { |
| return a.first > b.first; |
| } |
| }; |
| |
| size_t counter_{0}; // Keeps order of tasks with the same time. |
| |
| std::priority_queue<QueueItem, |
| std::vector<QueueItem>, |
| EventTaskRunner::Greater> |
| queue_; |
| |
| EventPtr<event_base> base_{event_base_new()}; |
| |
| EventPtr<event> task_event_{ |
| event_new(base_.get(), -1, EV_TIMEOUT, &EventHandler, this)}; |
| |
| std::map<int, std::pair<EventPtr<event>, IoCompletionCallback>> fd_task_map_; |
| }; |
| |
| } // namespace examples |
| } // namespace weave |
| |
| #endif // LIBWEAVE_EXAMPLES_PROVIDER_EVENT_TASK_RUNNER_H_ |