123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396 |
- #ifndef DASYNQ_EPOLL_H_
- #define DASYNQ_EPOLL_H_
- #include <system_error>
- #include <mutex>
- #include <type_traits>
- #include <unordered_map>
- #include <vector>
- #include <sys/epoll.h>
- #include <sys/signalfd.h>
- #include <sys/types.h>
- #include <sys/wait.h>
- #include <unistd.h>
- #include <csignal>
- namespace dasynq {
- inline namespace v2 {
- namespace dprivate {
- class proc_status;
- }
- template <class Base> class epoll_loop;
- class epoll_traits
- {
- template <class Base> friend class epoll_loop;
- public:
- class sigdata_t
- {
- template <class Base> friend class epoll_loop;
-
- struct signalfd_siginfo info;
-
- public:
-
- int get_signo() { return info.ssi_signo; }
- int get_sicode() { return info.ssi_code; }
- pid_t get_sipid() { return info.ssi_pid; }
- uid_t get_siuid() { return info.ssi_uid; }
- void * get_siaddr() { return reinterpret_cast<void *>(info.ssi_addr); }
- int get_sistatus() { return info.ssi_status; }
- int get_sival_int() { return info.ssi_int; }
- void * get_sival_ptr() { return reinterpret_cast<void *>(info.ssi_ptr); }
-
- int get_sierrno() { return info.ssi_errno; }
-
- int get_siband() { return info.ssi_band; }
-
- int32_t get_sifd() { return info.ssi_fd; }
- uint32_t get_sittimerid() { return info.ssi_tid; }
- uint32_t get_sioverrun() { return info.ssi_overrun; }
- uint32_t get_sitrapno() { return info.ssi_trapno; }
- uint32_t get_siutime() { return info.ssi_utime; }
- uint32_t get_sistime() { return info.ssi_stime; }
-
-
-
- void set_signo(int signo) { info.ssi_signo = signo; }
- };
- using proc_status_t = dprivate::proc_status;
- class fd_r;
-
-
- class fd_s {
- friend class fd_r;
-
-
-
- int fd;
- public:
- fd_s(int fd_p) noexcept : fd(fd_p) { }
- };
-
-
-
- class fd_r {
- public:
- int get_fd(fd_s ss)
- {
- return ss.fd;
- }
- };
-
- constexpr static bool has_bidi_fd_watch = true;
- constexpr static bool has_separate_rw_fd_watches = false;
- constexpr static bool interrupt_after_fd_add = false;
- constexpr static bool interrupt_after_signal_add = false;
- constexpr static bool supports_non_oneshot_fd = true;
- };
- template <class Base> class epoll_loop : public Base
- {
- int epfd = -1;
- int sigfd = -1;
- sigset_t sigmask;
- std::unordered_map<int, void *> sigdataMap;
-
-
-
-
-
-
- using sigdata_t = epoll_traits::sigdata_t;
- using fd_r = typename epoll_traits::fd_r;
-
- void process_events(epoll_event *events, int r)
- {
- std::lock_guard<decltype(Base::lock)> guard(Base::lock);
-
- for (int i = 0; i < r; i++) {
- void * ptr = events[i].data.ptr;
-
- if (ptr == &sigfd) {
-
- sigdata_t siginfo;
- while (true) {
- int r = read(sigfd, &siginfo.info, sizeof(siginfo.info));
- if (r == -1) break;
- auto iter = sigdataMap.find(siginfo.get_signo());
- if (iter != sigdataMap.end()) {
- void *userdata = (*iter).second;
- if (Base::receive_signal(*this, siginfo, userdata)) {
- sigdelset(&sigmask, siginfo.get_signo());
- }
- }
- }
- signalfd(sigfd, &sigmask, SFD_NONBLOCK | SFD_CLOEXEC);
- }
- else {
- int flags = 0;
- (events[i].events & EPOLLIN) && (flags |= IN_EVENTS);
- (events[i].events & EPOLLOUT) && (flags |= OUT_EVENTS);
-
-
- (events[i].events & EPOLLHUP) && (flags |= ERR_EVENTS);
- (events[i].events & EPOLLERR) && (flags |= ERR_EVENTS);
- auto r = Base::receive_fd_event(*this, fd_r(), ptr, flags);
- if (std::get<0>(r) != 0) {
- enable_fd_watch_nolock(fd_r().get_fd(std::get<1>(r)), ptr, std::get<0>(r));
- }
- }
- }
- }
-
- public:
-
-
- epoll_loop()
- {
- init();
- }
- epoll_loop(typename Base::delayed_init d) noexcept
- {
-
- }
- void init()
- {
- epfd = epoll_create1(EPOLL_CLOEXEC);
- if (epfd == -1) {
- throw std::system_error(errno, std::system_category());
- }
- sigemptyset(&sigmask);
- try {
- Base::init(this);
- }
- catch (...) {
- close(epfd);
- throw;
- }
- }
-
- ~epoll_loop() noexcept
- {
- if (epfd != -1) {
- Base::cleanup();
- close(epfd);
- if (sigfd != -1) {
- close(sigfd);
- }
- }
- }
-
-
-
-
-
-
-
-
- bool add_fd_watch(int fd, void *userdata, int flags, bool enabled = true, bool soft_fail = false)
- {
- struct epoll_event epevent;
-
- epevent.data.ptr = userdata;
- epevent.events = 0;
-
- if (flags & ONE_SHOT) {
- epevent.events = EPOLLONESHOT;
- }
- if ((flags & IN_EVENTS) && enabled) {
- epevent.events |= EPOLLIN;
- }
- if ((flags & OUT_EVENTS) && enabled) {
- epevent.events |= EPOLLOUT;
- }
- if (epoll_ctl(epfd, EPOLL_CTL_ADD, fd, &epevent) == -1) {
- if (soft_fail && errno == EPERM) {
- return false;
- }
- throw std::system_error(errno, std::system_category());
- }
- return true;
- }
-
- bool add_bidi_fd_watch(int fd, void *userdata, int flags, bool emulate)
- {
-
- throw std::system_error(std::make_error_code(std::errc::not_supported));
- }
-
-
-
- void remove_fd_watch(int fd, int flags) noexcept
- {
- epoll_ctl(epfd, EPOLL_CTL_DEL, fd, nullptr);
- }
-
- void remove_fd_watch_nolock(int fd, int flags) noexcept
- {
- remove_fd_watch(fd, flags);
- }
-
- void remove_bidi_fd_watch(int fd) noexcept
- {
-
- remove_fd_watch(fd, IN_EVENTS | OUT_EVENTS);
- }
-
-
-
- void enable_fd_watch(int fd, void *userdata, int flags) noexcept
- {
- struct epoll_event epevent;
-
- epevent.data.ptr = userdata;
- epevent.events = 0;
-
- if (flags & ONE_SHOT) {
- epevent.events = EPOLLONESHOT;
- }
- if (flags & IN_EVENTS) {
- epevent.events |= EPOLLIN;
- }
- if (flags & OUT_EVENTS) {
- epevent.events |= EPOLLOUT;
- }
-
- if (epoll_ctl(epfd, EPOLL_CTL_MOD, fd, &epevent) == -1) {
-
-
- }
- }
-
- void enable_fd_watch_nolock(int fd, void *userdata, int flags) noexcept
- {
- enable_fd_watch(fd, userdata, flags);
- }
-
- void disable_fd_watch(int fd, int flags) noexcept
- {
- struct epoll_event epevent;
-
- epevent.data.ptr = nullptr;
- epevent.events = 0;
-
-
-
-
- if (epoll_ctl(epfd, EPOLL_CTL_MOD, fd, &epevent) == -1) {
-
-
- }
- }
-
- void disable_fd_watch_nolock(int fd, int flags) noexcept
- {
- disable_fd_watch(fd, flags);
- }
-
- void add_signal_watch(int signo, void *userdata)
- {
- std::lock_guard<decltype(Base::lock)> guard(Base::lock);
- add_signal_watch_nolock(signo, userdata);
- }
-
- void add_signal_watch_nolock(int signo, void *userdata)
- {
- sigdataMap[signo] = userdata;
-
- bool was_no_sigfd = (sigfd == -1);
- sigaddset(&sigmask, signo);
- sigfd = signalfd(sigfd, &sigmask, SFD_NONBLOCK | SFD_CLOEXEC);
- if (sigfd == -1) {
- throw std::system_error(errno, std::system_category());
- }
-
- if (was_no_sigfd) {
-
- struct epoll_event epevent;
- epevent.data.ptr = &sigfd;
- epevent.events = EPOLLIN;
-
-
- if (epoll_ctl(epfd, EPOLL_CTL_ADD, sigfd, &epevent) == -1) {
- close(sigfd);
- sigfd = -1;
- throw std::system_error(errno, std::system_category());
- }
- }
- }
-
-
- void rearm_signal_watch_nolock(int signo, void *userdata) noexcept
- {
- sigaddset(&sigmask, signo);
- signalfd(sigfd, &sigmask, SFD_NONBLOCK | SFD_CLOEXEC);
- }
-
- void remove_signal_watch_nolock(int signo) noexcept
- {
- sigdelset(&sigmask, signo);
- signalfd(sigfd, &sigmask, 0);
- }
- void remove_signal_watch(int signo) noexcept
- {
- std::lock_guard<decltype(Base::lock)> guard(Base::lock);
- remove_signal_watch_nolock(signo);
- }
-
-
-
-
-
-
-
-
-
-
-
- void pull_events(bool do_wait)
- {
- epoll_event events[16];
- int r = epoll_wait(epfd, events, 16, do_wait ? -1 : 0);
- if (r == -1 || r == 0) {
-
- return;
- }
-
- do {
- process_events(events, r);
- r = epoll_wait(epfd, events, 16, 0);
- } while (r > 0);
- }
- };
- }
- }
- #endif
|