• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

PowerDNS / pdns / 19741624072

27 Nov 2025 03:45PM UTC coverage: 73.086% (+0.02%) from 73.065%
19741624072

Pull #16570

github

web-flow
Merge 08a2cdb1d into f94a3f63f
Pull Request #16570: rec: rewrite all unwrap calls in web.rs

38523 of 63408 branches covered (60.75%)

Branch coverage included in aggregate %.

128044 of 164496 relevant lines covered (77.84%)

6531485.83 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

86.34
/pdns/epollmplexer.cc
1
/*
2
 * This file is part of PowerDNS or dnsdist.
3
 * Copyright -- PowerDNS.COM B.V. and its contributors
4
 *
5
 * This program is free software; you can redistribute it and/or modify
6
 * it under the terms of version 2 of the GNU General Public License as
7
 * published by the Free Software Foundation.
8
 *
9
 * In addition, for the avoidance of any doubt, permission is granted to
10
 * link this program with OpenSSL and to (re)distribute the binaries
11
 * produced as the result of such linking.
12
 *
13
 * This program is distributed in the hope that it will be useful,
14
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16
 * GNU General Public License for more details.
17
 *
18
 * You should have received a copy of the GNU General Public License
19
 * along with this program; if not, write to the Free Software
20
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
21
 */
22
#ifdef HAVE_CONFIG_H
23
#include "config.h"
24
#endif
25
#include "mplexer.hh"
26
#include "sstuff.hh"
27
#include <iostream>
28
#include <unistd.h>
29
#include "misc.hh"
30
#ifdef __linux__
31
#include <sys/epoll.h>
32
#endif
33

34
#include "namespaces.hh"
35

36
class EpollFDMultiplexer : public FDMultiplexer
37
{
38
public:
39
  EpollFDMultiplexer(unsigned int maxEventsHint);
40
  ~EpollFDMultiplexer() override
41
  {
3,268✔
42
    if (d_epollfd >= 0) {
3,268!
43
      close(d_epollfd);
3,268✔
44
    }
3,268✔
45
  }
3,268✔
46

47
  int run(struct timeval* tv, int timeout = 500) override;
48
  void getAvailableFDs(std::vector<int>& fds, int timeout) override;
49

50
  void addFD(int fd, FDMultiplexer::EventKind kind) override;
51
  void removeFD(int fd, FDMultiplexer::EventKind kind) override;
52
  void alterFD(int fd, FDMultiplexer::EventKind from, FDMultiplexer::EventKind to) override;
53

54
  string getName() const override
55
  {
185✔
56
    return "epoll";
185✔
57
  }
185✔
58

59
private:
60
  int d_epollfd;
61
  std::vector<epoll_event> d_eevents;
62
};
63

64
static FDMultiplexer* makeEpoll(unsigned int maxEventsHint)
65
{
7,526✔
66
  return new EpollFDMultiplexer(maxEventsHint);
7,526✔
67
}
7,526✔
68

69
static struct EpollRegisterOurselves
70
{
71
  EpollRegisterOurselves()
72
  {
1,008✔
73
    FDMultiplexer::getMultiplexerMap().emplace(0, &makeEpoll); // priority 0!
1,008✔
74
  }
1,008✔
75
} doItEpoll;
76

77
EpollFDMultiplexer::EpollFDMultiplexer(unsigned int maxEventsHint) :
78
  d_eevents(maxEventsHint)
6,555✔
79
{
7,525✔
80
  d_epollfd = epoll_create(static_cast<int>(maxEventsHint)); // not hard max, just a hint that is actually ignored since Linux 2.6.8
7,525✔
81
  if (d_epollfd < 0) {
7,525✔
82
    throw FDMultiplexerException("Setting up epoll: " + stringerror());
7✔
83
  }
7✔
84
  int fd = socket(AF_INET, SOCK_DGRAM, 0); // for self-test
7,518✔
85

86
  if (fd < 0) {
7,518!
87
    return;
×
88
  }
×
89

90
  try {
7,518✔
91
    addReadFD(fd, 0);
7,518✔
92
    removeReadFD(fd);
7,518✔
93
    close(fd);
7,518✔
94
    return;
7,518✔
95
  }
7,518✔
96
  catch (const FDMultiplexerException& fe) {
7,518✔
97
    close(fd);
×
98
    close(d_epollfd);
×
99
    throw FDMultiplexerException("epoll multiplexer failed self-test: " + string(fe.what()));
×
100
  }
×
101
}
7,518✔
102

103
static uint32_t convertEventKind(FDMultiplexer::EventKind kind)
104
{
51,083✔
105
  switch (kind) {
51,083✔
106
  case FDMultiplexer::EventKind::Read:
49,063✔
107
    return EPOLLIN;
49,063✔
108
  case FDMultiplexer::EventKind::Write:
1,903✔
109
    return EPOLLOUT;
1,903✔
110
  case FDMultiplexer::EventKind::Both:
5✔
111
    return EPOLLIN | EPOLLOUT;
5✔
112
  }
51,083✔
113

114
  throw std::runtime_error("Unhandled event kind in the epoll multiplexer");
×
115
}
51,083✔
116

117
void EpollFDMultiplexer::addFD(int fd, FDMultiplexer::EventKind kind)
118
{
49,156✔
119
  struct epoll_event eevent;
49,156✔
120

121
  eevent.events = convertEventKind(kind);
49,156✔
122

123
  eevent.data.u64 = 0; // placate valgrind (I love it so much)
49,156✔
124
  eevent.data.fd = fd;
49,156✔
125

126
  if (epoll_ctl(d_epollfd, EPOLL_CTL_ADD, fd, &eevent) < 0) {
49,156✔
127
    throw FDMultiplexerException("Adding fd to epoll set: " + stringerror());
5✔
128
  }
5✔
129
}
49,156✔
130

131
void EpollFDMultiplexer::removeFD(int fd, FDMultiplexer::EventKind)
132
{
34,353✔
133
  struct epoll_event dummy;
34,353✔
134
  dummy.events = 0;
34,353✔
135
  dummy.data.u64 = 0;
34,353✔
136

137
  if (epoll_ctl(d_epollfd, EPOLL_CTL_DEL, fd, &dummy) < 0) {
34,353!
138
    throw FDMultiplexerException("Removing fd from epoll set: " + stringerror());
×
139
  }
×
140
}
34,353✔
141

142
void EpollFDMultiplexer::alterFD(int fd, FDMultiplexer::EventKind, FDMultiplexer::EventKind to)
143
{
1,942✔
144
  struct epoll_event eevent;
1,942✔
145
  eevent.events = convertEventKind(to);
1,942✔
146
  eevent.data.u64 = 0; // placate valgrind (I love it so much)
1,942✔
147
  eevent.data.fd = fd;
1,942✔
148

149
  if (epoll_ctl(d_epollfd, EPOLL_CTL_MOD, fd, &eevent) < 0) {
1,942!
150
    throw FDMultiplexerException("Altering fd in epoll set: " + stringerror());
×
151
  }
×
152
}
1,942✔
153

154
void EpollFDMultiplexer::getAvailableFDs(std::vector<int>& fds, int timeout)
155
{
2,149✔
156
  int ret = epoll_wait(d_epollfd, d_eevents.data(), d_eevents.size(), timeout);
2,149✔
157

158
  if (ret < 0 && errno != EINTR) {
2,149!
159
    throw FDMultiplexerException("epoll returned error: " + stringerror());
×
160
  }
×
161

162
  for (int n = 0; n < ret; ++n) {
3,913✔
163
    fds.push_back(d_eevents[n].data.fd);
1,764✔
164
  }
1,764✔
165
}
2,149✔
166

167
int EpollFDMultiplexer::run(struct timeval* now, int timeout)
168
{
72,556✔
169
  InRun guard(d_inrun);
72,556✔
170

171
  int ret = epoll_wait(d_epollfd, d_eevents.data(), d_eevents.size(), timeout);
72,556✔
172
  gettimeofday(now, nullptr); // MANDATORY
72,556✔
173

174
  if (ret < 0 && errno != EINTR) {
72,556!
175
    throw FDMultiplexerException("epoll returned error: " + stringerror());
×
176
  }
×
177

178
  if (ret < 1) { // thanks AB!
72,556✔
179
    return 0;
36,520✔
180
  }
36,520✔
181

182
  int count = 0;
36,036✔
183

184
  for (int n = 0; n < ret; ++n) {
73,111✔
185
    if ((d_eevents[n].events & EPOLLIN) || (d_eevents[n].events & EPOLLERR) || (d_eevents[n].events & EPOLLHUP)) {
37,082!
186
      const auto& iter = d_readCallbacks.find(d_eevents[n].data.fd);
37,038✔
187
      if (iter != d_readCallbacks.end()) {
37,042✔
188
        iter->d_callback(iter->d_fd, iter->d_parameter);
37,032✔
189
        count++;
37,032✔
190
      }
37,032✔
191
    }
37,038✔
192

193
    if ((d_eevents[n].events & EPOLLOUT) || (d_eevents[n].events & EPOLLERR) || (d_eevents[n].events & EPOLLHUP)) {
37,089✔
194
      const auto& iter = d_writeCallbacks.find(d_eevents[n].data.fd);
441✔
195
      if (iter != d_writeCallbacks.end()) {
441✔
196
        iter->d_callback(iter->d_fd, iter->d_parameter);
42✔
197
        count++;
42✔
198
      }
42✔
199
    }
441✔
200
  }
37,075✔
201

202
  return count;
36,036✔
203
}
72,556✔
204

205
#if 0
206
void acceptData(int fd, funcparam_t& parameter)
207
{
208
  cout<<"Have data on fd "<<fd<<endl;
209
  Socket* sock=funcparam_t_cast<Socket*>(parameter);
210
  string packet;
211
  IPEndpoint rem;
212
  sock->recvFrom(packet, rem);
213
  cout<<"Received "<<packet.size()<<" bytes!\n";
214
}
215

216

217
int main()
218
{
219
  Socket s(AF_INET, SOCK_DGRAM);
220

221
  IPEndpoint loc("0.0.0.0", 2000);
222
  s.bind(loc);
223

224
  EpollFDMultiplexer sfm;
225

226
  sfm.addReadFD(s.getHandle(), &acceptData, &s);
227

228
  for(int n=0; n < 100 ; ++n) {
229
    sfm.run();
230
  }
231
  sfm.removeReadFD(s.getHandle());
232
  sfm.removeReadFD(s.getHandle());
233
}
234
#endif
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2025 Coveralls, Inc