summaryrefslogtreecommitdiff
path: root/modules/socketengines/m_socketengine_poll.cpp
diff options
context:
space:
mode:
authorAdam <Adam@anope.org>2010-12-24 01:22:07 -0500
committerAdam <Adam@anope.org>2010-12-24 01:22:07 -0500
commitbefb4b364418a6e29d5ecb6bcefba565ab324978 (patch)
tree88ee5ad9d5a8492bb707a016402e6a8837c36eda /modules/socketengines/m_socketengine_poll.cpp
parenta6c8a6a9f2c3f6c0fda6e5bdfeecc3af8c9b5708 (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.cpp192
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)