-
-
Notifications
You must be signed in to change notification settings - Fork 392
/
Copy pathpollinterrupter.cpp
125 lines (104 loc) · 2.86 KB
/
pollinterrupter.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
/**
* Copyright (c) 2020-2021 Paul-Louis Ageneau
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at https://mozilla.org/MPL/2.0/.
*/
#include "pollinterrupter.hpp"
#include "internals.hpp"
#if RTC_ENABLE_WEBSOCKET
#ifndef _WIN32
#include <fcntl.h>
#include <unistd.h>
#endif
namespace rtc::impl {
PollInterrupter::PollInterrupter() {
#ifdef _WIN32
struct addrinfo *ai = NULL;
struct addrinfo hints = {};
hints.ai_family = AF_UNSPEC;
hints.ai_socktype = SOCK_DGRAM;
hints.ai_protocol = IPPROTO_UDP;
hints.ai_flags = AI_PASSIVE | AI_NUMERICSERV;
if (getaddrinfo("localhost", "0", &hints, &ai) != 0)
throw std::runtime_error("Resolution failed for localhost address");
try {
mSock = ::socket(ai->ai_family, ai->ai_socktype, ai->ai_protocol);
if (mSock == INVALID_SOCKET)
throw std::runtime_error("UDP socket creation failed");
// Set non-blocking
ctl_t nbio = 1;
::ioctlsocket(mSock, FIONBIO, &nbio);
// Bind
if (::bind(mSock, ai->ai_addr, (socklen_t)ai->ai_addrlen) < 0)
throw std::runtime_error("Failed to bind UDP socket");
// Connect to self
struct sockaddr_storage addr;
socklen_t addrlen = sizeof(addr);
if (::getsockname(mSock, reinterpret_cast<struct sockaddr *>(&addr), &addrlen) < 0)
throw std::runtime_error("getsockname failed");
if (::connect(mSock, reinterpret_cast<struct sockaddr *>(&addr), addrlen) < 0)
throw std::runtime_error("Failed to connect UDP socket");
} catch (...) {
freeaddrinfo(ai);
if (mSock != INVALID_SOCKET)
::closesocket(mSock);
throw;
}
freeaddrinfo(ai);
#else
int pipefd[2];
if (::pipe(pipefd) != 0)
throw std::runtime_error("Failed to create pipe");
::fcntl(pipefd[0], F_SETFL, O_NONBLOCK);
::fcntl(pipefd[1], F_SETFL, O_NONBLOCK);
mPipeOut = pipefd[1]; // read
mPipeIn = pipefd[0]; // write
#endif
}
PollInterrupter::~PollInterrupter() {
#ifdef _WIN32
::closesocket(mSock);
#else
::close(mPipeIn);
::close(mPipeOut);
#endif
}
void PollInterrupter::prepare(struct pollfd &pfd) {
#ifdef _WIN32
pfd.fd = mSock;
#else
pfd.fd = mPipeIn;
#endif
pfd.events = POLLIN;
}
void PollInterrupter::process(struct pollfd &pfd) {
if (pfd.revents & POLLIN) {
#ifdef _WIN32
char dummy;
while (::recv(pfd.fd, &dummy, 1, 0) >= 0) {
// Ignore
}
#else
char dummy;
while (::read(pfd.fd, &dummy, 1) > 0) {
// Ignore
}
#endif
}
}
void PollInterrupter::interrupt() {
#ifdef _WIN32
if (::send(mSock, NULL, 0, 0) < 0 && sockerrno != SEAGAIN && sockerrno != SEWOULDBLOCK) {
PLOG_WARNING << "Writing to interrupter socket failed, errno=" << sockerrno;
}
#else
char dummy = 0;
if (::write(mPipeOut, &dummy, 1) < 0 && errno != EAGAIN && errno != EWOULDBLOCK) {
PLOG_WARNING << "Writing to interrupter pipe failed, errno=" << errno;
}
#endif
}
} // namespace rtc::impl
#endif