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 | |
John Park | 33858a3 | 2018-09-28 23:05:48 -0700 | [diff] [blame] | 10 | #include "aos/logging/logging.h" |
Brian Silverman | 801d49c | 2016-03-20 15:50:22 -0700 | [diff] [blame] | 11 | |
| 12 | namespace aos { |
| 13 | namespace events { |
| 14 | |
Brian Silverman | 58899fd | 2019-03-24 11:03:11 -0700 | [diff] [blame] | 15 | void EpollEvent::DirectEvent(uint32_t events) { |
| 16 | if ((events & ~(EPOLLIN | EPOLLPRI | EPOLLERR)) != 0) { |
| 17 | LOG(FATAL, "unexpected epoll events set in %x on %d\n", events, fd()); |
| 18 | } |
| 19 | ReadEvent(); |
| 20 | } |
| 21 | |
| 22 | void EpollEvent::SetEvents(uint32_t events) { |
| 23 | events_ |= events; |
| 24 | CHECK(!loop_); |
| 25 | } |
| 26 | |
Parker Schuh | b59bf5e | 2016-12-28 21:09:36 -0800 | [diff] [blame] | 27 | EpollLoop::EpollLoop() : epoll_fd_(PCHECK(epoll_create1(0))) {} |
Brian Silverman | 801d49c | 2016-03-20 15:50:22 -0700 | [diff] [blame] | 28 | |
Parker Schuh | b59bf5e | 2016-12-28 21:09:36 -0800 | [diff] [blame] | 29 | void EpollLoop::Add(EpollEvent *event) { |
| 30 | event->loop_ = this; |
| 31 | struct epoll_event temp_event; |
| 32 | temp_event.data.ptr = static_cast<void *>(event); |
Parker Schuh | c1975fc | 2018-04-07 15:27:07 -0700 | [diff] [blame] | 33 | temp_event.events = event->events(); |
Parker Schuh | b59bf5e | 2016-12-28 21:09:36 -0800 | [diff] [blame] | 34 | PCHECK(epoll_ctl(epoll_fd(), EPOLL_CTL_ADD, event->fd(), &temp_event)); |
| 35 | } |
Brian Silverman | 801d49c | 2016-03-20 15:50:22 -0700 | [diff] [blame] | 36 | |
Parker Schuh | b59bf5e | 2016-12-28 21:09:36 -0800 | [diff] [blame] | 37 | void EpollLoop::Delete(EpollEvent *event) { |
Parker Schuh | c1975fc | 2018-04-07 15:27:07 -0700 | [diff] [blame] | 38 | event->loop_ = nullptr; |
Parker Schuh | b59bf5e | 2016-12-28 21:09:36 -0800 | [diff] [blame] | 39 | PCHECK(epoll_ctl(epoll_fd(), EPOLL_CTL_DEL, event->fd(), NULL)); |
| 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 | void EpollLoop::Run() { |
| 43 | while (true) { |
| 44 | const int timeout = CalculateTimeout(); |
| 45 | static constexpr size_t kNumberOfEvents = 64; |
| 46 | epoll_event events[kNumberOfEvents]; |
| 47 | const int number_events = |
| 48 | PCHECK(epoll_wait(epoll_fd(), events, kNumberOfEvents, timeout)); |
| 49 | |
| 50 | for (int i = 0; i < number_events; i++) { |
Parker Schuh | c1975fc | 2018-04-07 15:27:07 -0700 | [diff] [blame] | 51 | static_cast<EpollEvent *>(events[i].data.ptr)->DirectEvent(events[i].events); |
Parker Schuh | b59bf5e | 2016-12-28 21:09:36 -0800 | [diff] [blame] | 52 | } |
Brian Silverman | 801d49c | 2016-03-20 15:50:22 -0700 | [diff] [blame] | 53 | } |
Parker Schuh | b59bf5e | 2016-12-28 21:09:36 -0800 | [diff] [blame] | 54 | } |
Brian Silverman | 801d49c | 2016-03-20 15:50:22 -0700 | [diff] [blame] | 55 | |
Parker Schuh | b59bf5e | 2016-12-28 21:09:36 -0800 | [diff] [blame] | 56 | void EpollLoop::AddWait(EpollWait *wait) { waits_.push_back(wait); } |
Parker Schuh | b59bf5e | 2016-12-28 21:09:36 -0800 | [diff] [blame] | 57 | |
| 58 | // Calculates the new timeout value to pass to epoll_wait. |
| 59 | int EpollLoop::CalculateTimeout() { |
| 60 | const monotonic_clock::time_point monotonic_now = monotonic_clock::now(); |
| 61 | int r = -1; |
| 62 | for (EpollWait *c : waits_) { |
| 63 | const int new_timeout = c->Recalculate(monotonic_now); |
| 64 | if (new_timeout >= 0) { |
| 65 | if (r < 0 || new_timeout < r) { |
| 66 | r = new_timeout; |
| 67 | } |
| 68 | } |
| 69 | } |
| 70 | return r; |
Brian Silverman | 801d49c | 2016-03-20 15:50:22 -0700 | [diff] [blame] | 71 | } |
| 72 | |
| 73 | } // namespace events |
| 74 | } // namespace aos |