diff options
author | Adam <Adam@anope.org> | 2010-12-24 01:22:07 -0500 |
---|---|---|
committer | Adam <Adam@anope.org> | 2010-12-24 01:22:07 -0500 |
commit | befb4b364418a6e29d5ecb6bcefba565ab324978 (patch) | |
tree | 88ee5ad9d5a8492bb707a016402e6a8837c36eda /modules/socketengines/m_socketengine_poll.cpp | |
parent | a6c8a6a9f2c3f6c0fda6e5bdfeecc3af8c9b5708 (diff) |
Made the default socket poller use poll() because it is supported on most platforms
Diffstat (limited to 'modules/socketengines/m_socketengine_poll.cpp')
-rw-r--r-- | modules/socketengines/m_socketengine_poll.cpp | 192 |
1 files changed, 192 insertions, 0 deletions
diff --git a/modules/socketengines/m_socketengine_poll.cpp b/modules/socketengines/m_socketengine_poll.cpp new file mode 100644 index 000000000..044a79c4f --- /dev/null +++ b/modules/socketengines/m_socketengine_poll.cpp @@ -0,0 +1,192 @@ +#include "module.h" + +#ifndef _WIN32 +# include <ulimit.h> +# include <sys/poll.h> +# include <poll.h> +# ifndef POLLRDHUP +# define POLLRDHUP 0 +# endif +#else +# define poll WSAPoll +# define POLLRDHUP POLLHUP +#endif + +class SocketEnginePoll : public SocketEngineBase +{ + private: + long max; + pollfd *events; + int SocketCount; + std::map<int, int> socket_positions; + + public: + SocketEnginePoll() + { + SocketCount = 0; +#ifndef _WIN32 + max = ulimit(4, 0); +#else + max = 1024; +#endif + + if (max <= 0) + { + Log() << "Can't determine maximum number of open sockets"; + throw ModuleException("Can't determine maximum number of open sockets"); + } + + events = new pollfd[max]; + } + + ~SocketEnginePoll() + { + delete [] events; + } + + void AddSocket(Socket *s) + { + if (SocketCount == max) + { + Log() << "Unable to add fd " << s->GetFD() << " to socketengine poll, engine is full"; + return; + } + + pollfd *ev = &this->events[SocketCount]; + ev->fd = s->GetFD(); + ev->events = POLLIN; + ev->revents = 0; + + Sockets.insert(std::make_pair(ev->fd, s)); + socket_positions.insert(std::make_pair(ev->fd, SocketCount)); + + ++SocketCount; + } + + void DelSocket(Socket *s) + { + std::map<int, int>::iterator pos = socket_positions.find(s->GetFD()); + if (pos == socket_positions.end()) + { + Log() << "Unable to delete unknown fd " << s->GetFD() << " from socketengine poll"; + return; + } + + if (pos->second != SocketCount) + { + pollfd *ev = &this->events[pos->second], + *last_ev = &this->events[SocketCount - 1]; + + ev->fd = last_ev->fd; + ev->events = last_ev->events; + ev->revents = last_ev->revents; + + socket_positions[ev->fd] = pos->second; + } + + Sockets.erase(s->GetFD()); + this->socket_positions.erase(pos); + + --SocketCount; + } + + void MarkWritable(Socket *s) + { + if (s->HasFlag(SF_WRITABLE)) + return; + + std::map<int, int>::iterator pos = socket_positions.find(s->GetFD()); + if (pos == socket_positions.end()) + { + Log() << "Unable to mark unknown fd " << s->GetFD() << " as writable"; + return; + } + + pollfd *ev = &this->events[pos->second]; + ev->events |= POLLOUT; + + s->SetFlag(SF_WRITABLE); + } + + void ClearWritable(Socket *s) + { + if (!s->HasFlag(SF_WRITABLE)) + return; + + std::map<int, int>::iterator pos = socket_positions.find(s->GetFD()); + if (pos == socket_positions.end()) + { + Log() << "Unable to mark unknown fd " << s->GetFD() << " as writable"; + return; + } + + pollfd *ev = &this->events[pos->second]; + ev->events &= ~POLLOUT; + + s->UnsetFlag(SF_WRITABLE); + } + + void Process() + { + int total = poll(this->events, this->SocketCount, Config->ReadTimeout * 1000); + Anope::CurTime = time(NULL); + + if (total == -1) + { + Log() << "SockEngine::Process(): error: " << Anope::LastError(); + return; + } + + for (int i = 0; i < total; ++i) + { + pollfd *ev = &this->events[i]; + Socket *s = Sockets[ev->fd]; + + if (s->HasFlag(SF_DEAD)) + continue; + if (ev->revents & (POLLERR | POLLRDHUP)) + { + s->ProcessError(); + s->SetFlag(SF_DEAD); + continue; + } + + if ((ev->revents & POLLIN) && !s->ProcessRead()) + s->SetFlag(SF_DEAD); + + if ((ev->revents & POLLOUT) && !s->ProcessWrite()) + s->SetFlag(SF_DEAD); + } + + for (int i = 0; i < total; ++i) + { + pollfd *ev = &this->events[i]; + Socket *s = Sockets[ev->fd]; + + if (s->HasFlag(SF_DEAD)) + delete s; + } + } +}; + +class ModuleSocketEnginePoll : public Module +{ + SocketEnginePoll engine; + + public: + ModuleSocketEnginePoll(const Anope::string &modname, const Anope::string &creator) : Module(modname, creator) + { + this->SetAuthor("Anope"); + this->SetPermanent(true); + this->SetType(SOCKETENGINE); + + SocketEngine = &engine; + } + + ~ModuleSocketEnginePoll() + { + SocketEngine = NULL; + } +}; + +MODULE_INIT(ModuleSocketEnginePoll) |