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

PowerDNS / pdns / 12595591960

03 Jan 2025 09:27AM UTC coverage: 62.774% (+2.5%) from 60.245%
12595591960

Pull #15008

github

web-flow
Merge c2a2749d3 into 788f396a7
Pull Request #15008: Do not follow CNAME records for ANY or CNAME queries

30393 of 78644 branches covered (38.65%)

Branch coverage included in aggregate %.

105822 of 138350 relevant lines covered (76.49%)

4613078.44 hits per line

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

83.75
/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
  {
1,261✔
42
    if (d_epollfd >= 0) {
1,261!
43
      close(d_epollfd);
1,261✔
44
    }
1,261✔
45
  }
1,261✔
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
  {
153✔
56
    return "epoll";
153✔
57
  }
153✔
58

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

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

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

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

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

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

103
static uint32_t convertEventKind(FDMultiplexer::EventKind kind)
104
{
41,393✔
105
  switch (kind) {
41,393✔
106
  case FDMultiplexer::EventKind::Read:
40,255✔
107
    return EPOLLIN;
40,255✔
108
  case FDMultiplexer::EventKind::Write:
1,146✔
109
    return EPOLLOUT;
1,146✔
110
  case FDMultiplexer::EventKind::Both:
5✔
111
    return EPOLLIN | EPOLLOUT;
5✔
112
  }
41,393✔
113

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

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

121
  eevent.events = convertEventKind(kind);
40,272✔
122

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

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

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

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

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

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

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

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

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

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

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

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

178
  if (ret < 1) { // thanks AB!
54,526✔
179
    return 0;
21,814✔
180
  }
21,814✔
181

182
  int count = 0;
32,712✔
183

184
  for (int n = 0; n < ret; ++n) {
64,277✔
185
    if ((d_eevents[n].events & EPOLLIN) || (d_eevents[n].events & EPOLLERR) || (d_eevents[n].events & EPOLLHUP)) {
31,574!
186
      const auto& iter = d_readCallbacks.find(d_eevents[n].data.fd);
31,544✔
187
      if (iter != d_readCallbacks.end()) {
31,552✔
188
        iter->d_callback(iter->d_fd, iter->d_parameter);
31,538✔
189
        count++;
31,538✔
190
      }
31,538✔
191
    }
31,544✔
192

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

202
  return count;
32,712✔
203
}
54,526✔
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