anope- supernets anope source code & configuration |
git clone git://git.acid.vegas/anope.git |
Log | Files | Refs | Archive | README |
socketengine_epoll.cpp (2640B)
1 /* 2 * 3 * (C) 2003-2022 Anope Team 4 * Contact us at team@anope.org 5 * 6 * Please read COPYING and README for further details. 7 * 8 * Based on the original code of Epona by Lara. 9 * Based on the original code of Services by Andy Church. 10 */ 11 12 #include "services.h" 13 #include "anope.h" 14 #include "sockets.h" 15 #include "socketengine.h" 16 #include "config.h" 17 18 #include <sys/epoll.h> 19 #include <ulimit.h> 20 #include <errno.h> 21 22 static int EngineHandle; 23 static std::vector<epoll_event> events; 24 25 void SocketEngine::Init() 26 { 27 EngineHandle = epoll_create(4); 28 29 if (EngineHandle == -1) 30 throw SocketException("Could not initialize epoll socket engine: " + Anope::LastError()); 31 32 events.resize(DefaultSize); 33 } 34 35 void SocketEngine::Shutdown() 36 { 37 while (!Sockets.empty()) 38 delete Sockets.begin()->second; 39 } 40 41 void SocketEngine::Change(Socket *s, bool set, SocketFlag flag) 42 { 43 if (set == s->flags[flag]) 44 return; 45 46 bool before_registered = s->flags[SF_READABLE] || s->flags[SF_WRITABLE]; 47 48 s->flags[flag] = set; 49 50 bool now_registered = s->flags[SF_READABLE] || s->flags[SF_WRITABLE]; 51 52 epoll_event ev; 53 54 memset(&ev, 0, sizeof(ev)); 55 56 ev.events = (s->flags[SF_READABLE] ? EPOLLIN : 0) | (s->flags[SF_WRITABLE] ? EPOLLOUT : 0); 57 ev.data.fd = s->GetFD(); 58 59 int mod; 60 if (!before_registered && now_registered) 61 mod = EPOLL_CTL_ADD; 62 else if (before_registered && !now_registered) 63 mod = EPOLL_CTL_DEL; 64 else if (before_registered && now_registered) 65 mod = EPOLL_CTL_MOD; 66 else 67 return; 68 69 if (epoll_ctl(EngineHandle, mod, ev.data.fd, &ev) == -1) 70 throw SocketException("Unable to epoll_ctl() fd " + stringify(ev.data.fd) + " to epoll: " + Anope::LastError()); 71 } 72 73 void SocketEngine::Process() 74 { 75 if (Sockets.size() > events.size()) 76 events.resize(events.size() * 2); 77 78 int total = epoll_wait(EngineHandle, &events.front(), events.size(), Config->ReadTimeout * 1000); 79 Anope::CurTime = time(NULL); 80 81 /* EINTR can be given if the read timeout expires */ 82 if (total == -1) 83 { 84 if (errno != EINTR) 85 Log() << "SockEngine::Process(): error: " << Anope::LastError(); 86 return; 87 } 88 89 for (int i = 0; i < total; ++i) 90 { 91 epoll_event &ev = events[i]; 92 93 std::map<int, Socket *>::iterator it = Sockets.find(ev.data.fd); 94 if (it == Sockets.end()) 95 continue; 96 Socket *s = it->second; 97 98 if (ev.events & (EPOLLHUP | EPOLLERR)) 99 { 100 s->ProcessError(); 101 delete s; 102 continue; 103 } 104 105 if (!s->Process()) 106 { 107 if (s->flags[SF_DEAD]) 108 delete s; 109 continue; 110 } 111 112 if ((ev.events & EPOLLIN) && !s->ProcessRead()) 113 s->flags[SF_DEAD] = true; 114 115 if ((ev.events & EPOLLOUT) && !s->ProcessWrite()) 116 s->flags[SF_DEAD] = true; 117 118 if (s->flags[SF_DEAD]) 119 delete s; 120 } 121 }