Identify UDPInput error messages by the stream, too.
[cubemap] / acceptor.cpp
1 #include <assert.h>
2 #include <errno.h>
3 #include <netinet/in.h>
4 #include <poll.h>
5 #include <stdio.h>
6 #include <stdlib.h>
7 #include <string.h>
8 #include <sys/ioctl.h>
9 #include <sys/socket.h>
10 #include <unistd.h>
11
12 #include "acceptor.h"
13 #include "log.h"
14 #include "serverpool.h"
15 #include "state.pb.h"
16
17 using namespace std;
18
19 extern ServerPool *servers;
20 extern volatile bool hupped;
21
22 int create_server_socket(int port, SocketType socket_type)
23 {
24         int server_sock;
25         if (socket_type == TCP_SOCKET) {
26                 server_sock = socket(PF_INET6, SOCK_STREAM, IPPROTO_TCP);
27         } else {
28                 assert(socket_type == UDP_SOCKET);
29                 server_sock = socket(PF_INET6, SOCK_DGRAM, IPPROTO_UDP);
30         }
31         if (server_sock == -1) {
32                 log_perror("socket");
33                 exit(1);
34         }
35
36         int one = 1;
37         if (setsockopt(server_sock, SOL_SOCKET, SO_REUSEADDR, &one, sizeof(one)) == -1) {
38                 log_perror("setsockopt(SO_REUSEADDR)");
39                 exit(1);
40         }
41
42         // We want dual-stack sockets. (Sorry, OpenBSD and Windows XP...)
43         int zero = 0;
44         if (setsockopt(server_sock, IPPROTO_IPV6, IPV6_V6ONLY, &zero, sizeof(zero)) == -1) {
45                 log_perror("setsockopt(IPV6_V6ONLY)");
46                 exit(1);
47         }
48
49         // Set as non-blocking, so the acceptor thread can notice that we want to shut it down.
50         if (ioctl(server_sock, FIONBIO, &one) == -1) {
51                 log_perror("ioctl(FIONBIO)");
52                 exit(1);
53         }
54
55         sockaddr_in6 addr;
56         memset(&addr, 0, sizeof(addr));
57         addr.sin6_family = AF_INET6;
58         addr.sin6_port = htons(port);
59
60         if (bind(server_sock, reinterpret_cast<sockaddr *>(&addr), sizeof(addr)) == -1) {
61                 log_perror("bind");
62                 exit(1);
63         }
64
65         if (socket_type == TCP_SOCKET) {
66                 if (listen(server_sock, 128) == -1) {
67                         log_perror("listen");
68                         exit(1);
69                 }
70         }
71
72         return server_sock;
73 }
74         
75 Acceptor::Acceptor(int server_sock, int port)
76         : server_sock(server_sock),
77           port(port)
78 {
79 }
80
81 Acceptor::Acceptor(const AcceptorProto &serialized)
82         : server_sock(serialized.server_sock()),
83           port(serialized.port())
84 {
85 }
86
87 AcceptorProto Acceptor::serialize() const
88 {
89         AcceptorProto serialized;
90         serialized.set_server_sock(server_sock);
91         serialized.set_port(port);
92         return serialized;
93 }
94
95 void Acceptor::close_socket()
96 {
97         int ret;
98         do {
99                 ret = close(server_sock);
100         } while (ret == -1 && errno == EINTR);
101
102         if (ret == -1) {
103                 log_perror("close");
104         }
105 }
106
107 void Acceptor::do_work()
108 {
109         while (!hupped) {
110                 // Since we are non-blocking, we need to wait for the right state first.
111                 // Wait up to 50 ms, then check hupped.
112                 pollfd pfd;
113                 pfd.fd = server_sock;
114                 pfd.events = POLLIN;
115
116                 int nfds = poll(&pfd, 1, 50);
117                 if (nfds == 0 || (nfds == -1 && errno == EINTR)) {
118                         continue;
119                 }
120                 if (nfds == -1) {
121                         log_perror("poll");
122                         usleep(100000);
123                         continue;
124                 }
125
126                 sockaddr_in6 addr;
127                 socklen_t addrlen = sizeof(addr);
128
129                 // Get a new socket.
130                 int sock = accept(server_sock, reinterpret_cast<sockaddr *>(&addr), &addrlen);
131                 if (sock == -1 && errno == EINTR) {
132                         continue;
133                 }
134                 if (sock == -1) {
135                         log_perror("accept");
136                         usleep(100000);
137                         continue;
138                 }
139
140                 // Set the socket as nonblocking.
141                 int one = 1;
142                 if (ioctl(sock, FIONBIO, &one) == -1) {
143                         log_perror("FIONBIO");
144                         exit(1);
145                 }
146
147                 // Pick a server, round-robin, and hand over the socket to it.
148                 servers->add_client(sock);
149         }
150 }