]> git.netwichtig.de Git - user/henk/code/inspircd.git/blob - src/socketengines/socketengine_poll.cpp
Merge pull request #1162 from SaberUK/insp20+fix-deinstall
[user/henk/code/inspircd.git] / src / socketengines / socketengine_poll.cpp
1 /*
2  * InspIRCd -- Internet Relay Chat Daemon
3  *
4  *   Copyright (C) 2009 Daniel De Graaf <danieldg@inspircd.org>
5  *   Copyright (C) 2009 Uli Schlachter <psychon@znc.in>
6  *   Copyright (C) 2009 Craig Edwards <craigedwards@brainbox.cc>
7  *   Copyright (C) 2008 Robin Burchell <robin+git@viroteck.net>
8  *
9  * This file is part of InspIRCd.  InspIRCd is free software: you can
10  * redistribute it and/or modify it under the terms of the GNU General Public
11  * License as published by the Free Software Foundation, version 2.
12  *
13  * This program is distributed in the hope that it will be useful, but WITHOUT
14  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
15  * FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more
16  * details.
17  *
18  * You should have received a copy of the GNU General Public License
19  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
20  */
21
22
23 #include "inspircd.h"
24 #include "exitcodes.h"
25
26 #ifndef SOCKETENGINE_POLL
27 #define SOCKETENGINE_POLL
28
29 #include <iostream>
30 #include <vector>
31 #include <string>
32 #include <map>
33 #include "inspircd_config.h"
34 #include "inspircd.h"
35 #include "socketengine.h"
36
37 #ifndef _WIN32
38 # ifndef __USE_XOPEN
39 #  define __USE_XOPEN /* fuck every fucking OS ever made. needed by poll.h to work.*/
40 # endif
41 # include <poll.h>
42 # include <sys/poll.h>
43 # include <sys/resource.h>
44 #else
45 # define struct pollfd WSAPOLLFD
46 # define poll WSAPoll
47 #endif
48
49 class InspIRCd;
50
51 /** A specialisation of the SocketEngine class, designed to use poll().
52  */
53 class PollEngine : public SocketEngine
54 {
55 private:
56         /** These are used by poll() to hold socket events
57          */
58         struct pollfd *events;
59         /** This map maps fds to an index in the events array.
60          */
61         std::map<int, unsigned int> fd_mappings;
62 public:
63         /** Create a new PollEngine
64          */
65         PollEngine();
66         /** Delete a PollEngine
67          */
68         virtual ~PollEngine();
69         virtual bool AddFd(EventHandler* eh, int event_mask);
70         virtual void OnSetEvent(EventHandler* eh, int old_mask, int new_mask);
71         virtual EventHandler* GetRef(int fd);
72         virtual void DelFd(EventHandler* eh);
73         virtual int DispatchEvents();
74         virtual std::string GetName();
75 };
76
77 #endif
78
79 PollEngine::PollEngine()
80 {
81         CurrentSetSize = 0;
82         struct rlimit limits;
83         if (!getrlimit(RLIMIT_NOFILE, &limits))
84         {
85                 MAX_DESCRIPTORS = limits.rlim_cur;
86         }
87         else
88         {
89                 ServerInstance->Logs->Log("SOCKET", DEFAULT, "ERROR: Can't determine maximum number of open sockets: %s", strerror(errno));
90                 std::cout << "ERROR: Can't determine maximum number of open sockets: " << strerror(errno) << std::endl;
91                 ServerInstance->QuickExit(EXIT_STATUS_SOCKETENGINE);
92         }
93
94         ref = new EventHandler* [GetMaxFds()];
95         events = new struct pollfd[GetMaxFds()];
96
97         memset(events, 0, GetMaxFds() * sizeof(struct pollfd));
98         memset(ref, 0, GetMaxFds() * sizeof(EventHandler*));
99 }
100
101 PollEngine::~PollEngine()
102 {
103         // No destruction required, either.
104         delete[] ref;
105         delete[] events;
106 }
107
108 static int mask_to_poll(int event_mask)
109 {
110         int rv = 0;
111         if (event_mask & (FD_WANT_POLL_READ | FD_WANT_FAST_READ))
112                 rv |= POLLIN;
113         if (event_mask & (FD_WANT_POLL_WRITE | FD_WANT_FAST_WRITE | FD_WANT_SINGLE_WRITE))
114                 rv |= POLLOUT;
115         return rv;
116 }
117
118 bool PollEngine::AddFd(EventHandler* eh, int event_mask)
119 {
120         int fd = eh->GetFd();
121         if ((fd < 0) || (fd > GetMaxFds() - 1))
122         {
123                 ServerInstance->Logs->Log("SOCKET",DEBUG,"AddFd out of range: (fd: %d, max: %d)", fd, GetMaxFds());
124                 return false;
125         }
126
127         if (fd_mappings.find(fd) != fd_mappings.end())
128         {
129                 ServerInstance->Logs->Log("SOCKET",DEBUG,"Attempt to add duplicate fd: %d", fd);
130                 return false;
131         }
132
133         unsigned int index = CurrentSetSize;
134
135         fd_mappings[fd] = index;
136         ref[index] = eh;
137         events[index].fd = fd;
138         events[index].events = mask_to_poll(event_mask);
139
140         ServerInstance->Logs->Log("SOCKET", DEBUG,"New file descriptor: %d (%d; index %d)", fd, events[index].events, index);
141         SocketEngine::SetEventMask(eh, event_mask);
142         CurrentSetSize++;
143         return true;
144 }
145
146 EventHandler* PollEngine::GetRef(int fd)
147 {
148         std::map<int, unsigned int>::iterator it = fd_mappings.find(fd);
149         if (it == fd_mappings.end())
150                 return NULL;
151         return ref[it->second];
152 }
153
154 void PollEngine::OnSetEvent(EventHandler* eh, int old_mask, int new_mask)
155 {
156         std::map<int, unsigned int>::iterator it = fd_mappings.find(eh->GetFd());
157         if (it == fd_mappings.end())
158         {
159                 ServerInstance->Logs->Log("SOCKET",DEBUG,"SetEvents() on unknown fd: %d", eh->GetFd());
160                 return;
161         }
162
163         events[it->second].events = mask_to_poll(new_mask);
164 }
165
166 void PollEngine::DelFd(EventHandler* eh)
167 {
168         int fd = eh->GetFd();
169         if ((fd < 0) || (fd > MAX_DESCRIPTORS))
170         {
171                 ServerInstance->Logs->Log("SOCKET", DEBUG, "DelFd out of range: (fd: %d, max: %d)", fd, GetMaxFds());
172                 return;
173         }
174
175         std::map<int, unsigned int>::iterator it = fd_mappings.find(fd);
176         if (it == fd_mappings.end())
177         {
178                 ServerInstance->Logs->Log("SOCKET",DEBUG,"DelFd() on unknown fd: %d", fd);
179                 return;
180         }
181
182         unsigned int index = it->second;
183         unsigned int last_index = CurrentSetSize - 1;
184         int last_fd = events[last_index].fd;
185
186         if (index != last_index)
187         {
188                 // We need to move the last fd we got into this gap (gaps are evil!)
189
190                 // So update the mapping for the last fd to its new position
191                 fd_mappings[last_fd] = index;
192
193                 // move last_fd from last_index into index
194                 events[index].fd = last_fd;
195                 events[index].events = events[last_index].events;
196
197                 ref[index] = ref[last_index];
198         }
199
200         // Now remove all data for the last fd we got into out list.
201         // Above code made sure this always is right
202         fd_mappings.erase(it);
203         events[last_index].fd = 0;
204         events[last_index].events = 0;
205         ref[last_index] = NULL;
206
207         CurrentSetSize--;
208
209         ServerInstance->Logs->Log("SOCKET", DEBUG, "Remove file descriptor: %d (index: %d) "
210                         "(Filled gap with: %d (index: %d))", fd, index, last_fd, last_index);
211 }
212
213 int PollEngine::DispatchEvents()
214 {
215         int i = poll(events, CurrentSetSize, 1000);
216         int index;
217         socklen_t codesize = sizeof(int);
218         int errcode;
219         int processed = 0;
220         ServerInstance->UpdateTime();
221
222         if (i > 0)
223         {
224                 for (index = 0; index < CurrentSetSize && processed != i; index++)
225                 {
226                         if (events[index].revents)
227                                 processed++;
228                         EventHandler* eh = ref[index];
229                         if (!eh)
230                                 continue;
231
232                         if (events[index].revents & POLLHUP)
233                         {
234                                 eh->HandleEvent(EVENT_ERROR, 0);
235                                 continue;
236                         }
237
238                         if (events[index].revents & POLLERR)
239                         {
240                                 // Get fd
241                                 int fd = events[index].fd;
242
243                                 // Get error number
244                                 if (getsockopt(fd, SOL_SOCKET, SO_ERROR, &errcode, &codesize) < 0)
245                                         errcode = errno;
246                                 eh->HandleEvent(EVENT_ERROR, errcode);
247                                 continue;
248                         }
249
250                         if (events[index].revents & POLLIN)
251                         {
252                                 SetEventMask(eh, eh->GetEventMask() & ~FD_READ_WILL_BLOCK);
253                                 eh->HandleEvent(EVENT_READ);
254                                 if (eh != ref[index])
255                                         // whoops, deleted out from under us
256                                         continue;
257                         }
258                         
259                         if (events[index].revents & POLLOUT)
260                         {
261                                 int mask = eh->GetEventMask();
262                                 mask &= ~(FD_WRITE_WILL_BLOCK | FD_WANT_SINGLE_WRITE);
263                                 SetEventMask(eh, mask);
264                                 events[index].events = mask_to_poll(mask);
265                                 eh->HandleEvent(EVENT_WRITE);
266                         }
267                 }
268         }
269
270         return i;
271 }
272
273 std::string PollEngine::GetName()
274 {
275         return "poll";
276 }
277
278 SocketEngine* CreateSocketEngine()
279 {
280         return new PollEngine;
281 }