Brian Silverman | 801d49c | 2016-03-20 15:50:22 -0700 | [diff] [blame] | 1 | #include "aos/vision/events/epoll_events.h" |
| 2 | |
Brian Silverman | 801d49c | 2016-03-20 15:50:22 -0700 | [diff] [blame] | 3 | #include <fcntl.h> |
Parker Schuh | b59bf5e | 2016-12-28 21:09:36 -0800 | [diff] [blame] | 4 | #include <string.h> |
Brian Silverman | 801d49c | 2016-03-20 15:50:22 -0700 | [diff] [blame] | 5 | #include <sys/epoll.h> |
Parker Schuh | b59bf5e | 2016-12-28 21:09:36 -0800 | [diff] [blame] | 6 | #include <sys/socket.h> |
| 7 | #include <sys/types.h> |
Brian Silverman | 801d49c | 2016-03-20 15:50:22 -0700 | [diff] [blame] | 8 | #include <vector> |
| 9 | |
Brian Silverman | 801d49c | 2016-03-20 15:50:22 -0700 | [diff] [blame] | 10 | #include "aos/common/logging/logging.h" |
| 11 | |
| 12 | namespace aos { |
| 13 | namespace events { |
| 14 | |
Parker Schuh | b59bf5e | 2016-12-28 21:09:36 -0800 | [diff] [blame] | 15 | EpollLoop::EpollLoop() : epoll_fd_(PCHECK(epoll_create1(0))) {} |
Brian Silverman | 801d49c | 2016-03-20 15:50:22 -0700 | [diff] [blame] | 16 | |
Parker Schuh | b59bf5e | 2016-12-28 21:09:36 -0800 | [diff] [blame] | 17 | void EpollLoop::Add(EpollEvent *event) { |
| 18 | event->loop_ = this; |
| 19 | struct epoll_event temp_event; |
| 20 | temp_event.data.ptr = static_cast<void *>(event); |
Parker Schuh | c1975fc | 2018-04-07 15:27:07 -0700 | [diff] [blame] | 21 | temp_event.events = event->events(); |
Parker Schuh | b59bf5e | 2016-12-28 21:09:36 -0800 | [diff] [blame] | 22 | PCHECK(epoll_ctl(epoll_fd(), EPOLL_CTL_ADD, event->fd(), &temp_event)); |
| 23 | } |
Brian Silverman | 801d49c | 2016-03-20 15:50:22 -0700 | [diff] [blame] | 24 | |
Parker Schuh | b59bf5e | 2016-12-28 21:09:36 -0800 | [diff] [blame] | 25 | void EpollLoop::Delete(EpollEvent *event) { |
Parker Schuh | c1975fc | 2018-04-07 15:27:07 -0700 | [diff] [blame] | 26 | event->loop_ = nullptr; |
Parker Schuh | b59bf5e | 2016-12-28 21:09:36 -0800 | [diff] [blame] | 27 | PCHECK(epoll_ctl(epoll_fd(), EPOLL_CTL_DEL, event->fd(), NULL)); |
| 28 | } |
Brian Silverman | 801d49c | 2016-03-20 15:50:22 -0700 | [diff] [blame] | 29 | |
Parker Schuh | b59bf5e | 2016-12-28 21:09:36 -0800 | [diff] [blame] | 30 | void EpollLoop::Run() { |
| 31 | while (true) { |
| 32 | const int timeout = CalculateTimeout(); |
| 33 | static constexpr size_t kNumberOfEvents = 64; |
| 34 | epoll_event events[kNumberOfEvents]; |
| 35 | const int number_events = |
| 36 | PCHECK(epoll_wait(epoll_fd(), events, kNumberOfEvents, timeout)); |
| 37 | |
| 38 | for (int i = 0; i < number_events; i++) { |
Parker Schuh | c1975fc | 2018-04-07 15:27:07 -0700 | [diff] [blame] | 39 | static_cast<EpollEvent *>(events[i].data.ptr)->DirectEvent(events[i].events); |
Parker Schuh | b59bf5e | 2016-12-28 21:09:36 -0800 | [diff] [blame] | 40 | } |
Brian Silverman | 801d49c | 2016-03-20 15:50:22 -0700 | [diff] [blame] | 41 | } |
Parker Schuh | b59bf5e | 2016-12-28 21:09:36 -0800 | [diff] [blame] | 42 | } |
Brian Silverman | 801d49c | 2016-03-20 15:50:22 -0700 | [diff] [blame] | 43 | |
Parker Schuh | b59bf5e | 2016-12-28 21:09:36 -0800 | [diff] [blame] | 44 | void EpollLoop::AddWait(EpollWait *wait) { waits_.push_back(wait); } |
Parker Schuh | b59bf5e | 2016-12-28 21:09:36 -0800 | [diff] [blame] | 45 | |
| 46 | // Calculates the new timeout value to pass to epoll_wait. |
| 47 | int EpollLoop::CalculateTimeout() { |
| 48 | const monotonic_clock::time_point monotonic_now = monotonic_clock::now(); |
| 49 | int r = -1; |
| 50 | for (EpollWait *c : waits_) { |
| 51 | const int new_timeout = c->Recalculate(monotonic_now); |
| 52 | if (new_timeout >= 0) { |
| 53 | if (r < 0 || new_timeout < r) { |
| 54 | r = new_timeout; |
| 55 | } |
| 56 | } |
| 57 | } |
| 58 | return r; |
Brian Silverman | 801d49c | 2016-03-20 15:50:22 -0700 | [diff] [blame] | 59 | } |
| 60 | |
| 61 | } // namespace events |
| 62 | } // namespace aos |