TDME2 1.9.121
UDPSocket.cpp
Go to the documentation of this file.
1#include <errno.h>
2#include <string.h>
3#if defined(_WIN32)
4 #include <winsock2.h>
5 #include <ws2tcpip.h>
6 #define socklen_t int
7 #define BUF_CAST(buf) ((char*)buf)
8#else
9 #include <sys/socket.h>
10 #include <netinet/in.h>
11 #include <arpa/inet.h>
12 #define BUF_CAST(buf) ((void*)buf)
13#endif
14
15#include <string>
16
17#include <tdme/tdme.h>
19
20using std::string;
21using std::to_string;
22
24
25// determine which SO_REUSE option to use
26#if defined(_WIN32)
27 #define SO_REUSEOPTION SO_REUSEADDR
28#elif defined(__linux__)
29 // on linux < 3.9 we need to to use "addr". it behaves like "port" on BSD for UDP sockets
30 #define SO_REUSEOPTION SO_REUSEADDR
31#else
32 // standard is "port" as it exactly does what we want
33 #define SO_REUSEOPTION SO_REUSEPORT
34#endif
35
36UDPSocket::~UDPSocket() {
37}
38
39ssize_t UDPSocket::read(string& from, unsigned int& port, void* buf, const size_t bytes) {
40 // socket address in setup
41 socklen_t sinLen = 0;
42 void* sin;
43 sockaddr_in sinIPV4;
44 sockaddr_in6 sinIPV6;
45 switch(ipVersion) {
46 case IPV4:
47 {
48 sin = &sinIPV4;
49 sinLen = sizeof(sinIPV4);
50 }
51 break;
52 case IPV6:
53 {
54
55 sin = &sinIPV6;
56 sinLen = sizeof(sinIPV6);
57 }
58 }
59
60 ssize_t bytesRead = ::recvfrom(descriptor, BUF_CAST(buf), bytes, 0, (struct sockaddr *)sin, &sinLen);
61 if (bytesRead == -1) {
62 #if defined(_WIN32)
63 auto wsaError = WSAGetLastError();
64 if (wsaError == WSAEWOULDBLOCK ||
65 wsaError == WSAECONNRESET) {
66 return -1;
67 } else {
68 string msg = "error while reading from socket: ";
69 msg+= to_string(wsaError);
70 throw NetworkIOException(msg);
71 }
72 #else
73 // nope throw an exception
74 if (errno == EAGAIN) {
75 return -1;
76 } else {
77 string msg = "error while reading from socket: ";
78 msg+= strerror(errno);
79 throw NetworkIOException(msg);
80 }
81 #endif
82 }
83
84 // set up senders ip + port
85 switch(ipVersion) {
86 case IPV4:
87 {
88 char ipv4AddressString[INET_ADDRSTRLEN];
89 from = inet_ntop(AF_INET, &sinIPV4.sin_addr, ipv4AddressString, INET_ADDRSTRLEN) == NULL?"127.0.0.1":ipv4AddressString;
90 port = ntohs(sinIPV4.sin_port);
91 }
92 break;
93 case IPV6:
94 {
95 char ipv6AddressString[INET6_ADDRSTRLEN];
96 from = inet_ntop(AF_INET6, &sinIPV6.sin6_addr, ipv6AddressString, INET6_ADDRSTRLEN) == NULL?"::1":ipv6AddressString;
97 port = ntohs(sinIPV6.sin6_port);
98 }
99 }
100
101 // return bytes read
102 return bytesRead;
103}
104
105ssize_t UDPSocket::write(const string& to, const unsigned int port, void* buf, const size_t bytes) {
106 // receiver address in setup
107 socklen_t sinLen = 0;
108 void* sin;
109 sockaddr_in sinIPV4;
110 sockaddr_in6 sinIPV6;
111 switch(ipVersion) {
112 case IPV4:
113 {
114 sinLen = sizeof(sinIPV4);
115 memset(&sinIPV4, 0, sinLen);
116 sinIPV4.sin_family = AF_INET;
117 sinIPV4.sin_port = htons(port);
118 sinIPV4.sin_addr.s_addr = inet_addr(to.c_str());
119 sin = &sinIPV4;
120 }
121 break;
122 case IPV6:
123 {
124 sinLen = sizeof(sinIPV6);
125 memset(&sinIPV6, 0, sinLen);
126 sinIPV6.sin6_family = AF_INET6;
127 sinIPV6.sin6_port = htons(port);
128 inet_pton(AF_INET6, to.c_str(), &sinIPV6.sin6_addr);
129 sin = &sinIPV6;
130 }
131 break;
132 }
133
134 // go
135 #if defined(__APPLE__) || defined(_WIN32)
136 ssize_t bytesWritten = ::sendto(descriptor, BUF_CAST(buf), bytes, 0, (const struct sockaddr*)sin, sinLen);
137 #else
138 ssize_t bytesWritten = ::sendto(descriptor, BUF_CAST(buf), bytes, MSG_NOSIGNAL, (const struct sockaddr*)sin, sinLen);
139 #endif
140
141 // send successful?
142 if (bytesWritten == -1) {
143 #if defined(_WIN32)
144 auto wsaError = WSAGetLastError();
145 if (wsaError == WSAEWOULDBLOCK) {
146 return -1;
147 } else {
148 string msg = "error while writing to socket: ";
149 msg+= to_string(wsaError);
150 throw NetworkIOException(msg);
151 }
152 #else
153 // nope throw an exception
154 if (errno == EAGAIN) {
155 return -1;
156 } else {
157 string msg = "error while writing to socket: ";
158 msg+= strerror(errno);
159 throw NetworkIOException(msg);
160 }
161 #endif
162 }
163
164 // return bytes written
165 return bytesWritten;
166}
167
168void UDPSocket::create(UDPSocket& socket, IpVersion ipVersion) {
169 socket.ipVersion = ipVersion;
170 socket.descriptor = ::socket(ipVersion == IPV6?AF_INET6:AF_INET, SOCK_DGRAM, IPPROTO_UDP);
171 if (socket.descriptor == -1) {
172 string msg = "Could not create socket: ";
173 msg+= strerror(errno);
174 throw NetworkSocketException(msg);
175 }
176 #ifdef __APPLE__
177 int flag = 1;
178 if (setsockopt(socket.descriptor, SOL_SOCKET, SO_NOSIGPIPE, (void*)&flag, sizeof(flag)) == -1) {
179 string msg = "Could not set no sig pipe on socket: ";
180 msg+= strerror(errno);
181 throw NetworkSocketException(msg);
182 }
183 #endif
184}
185
186void UDPSocket::createServerSocket(UDPSocket& socket, const string& ip, const unsigned int port) {
187 // create socket
189
190 try {
191 // set non blocked
192 socket.setNonBlocked();
193
194 // enable socket reuse port
195 int flag = 1;
196 if (setsockopt(socket.descriptor, SOL_SOCKET, SO_REUSEOPTION, BUF_CAST(&flag), sizeof(flag)) == -1) {
197 string msg = "Could not set reuse port on socket: ";
198 msg+= strerror(errno);
199 throw NetworkSocketException(msg);
200 }
201
202 // bind socket to host
203 socket.bind(ip, port);
204
205 } catch (NetworkSocketException &exception) {
206 socket.close();
207 throw exception;
208 }
209}
210
212 // create socket
214
215 try {
216 // set non blocked
217 socket.setNonBlocked();
218 } catch (NetworkSocketException &exception) {
219 socket.close();
220 throw exception;
221 }
222}
#define SO_REUSEOPTION
Definition: UDPSocket.cpp:33
#define BUF_CAST(buf)
Definition: UDPSocket.cpp:12
Base exception class for network IO exceptions.
void bind(const string &ip, const unsigned int port)
Binds a socket to local ip and port.
void close()
Closes the socket.
static IpVersion determineIpVersion(const string &ip)
Determine IP version.
void setNonBlocked()
sets the socket non blocked
Class representing a UDP socket.
Definition: UDPSocket.h:27
static void createClientSocket(UDPSocket &socket, IpVersion ipVersion)
creates a udp client socket
Definition: UDPSocket.cpp:211
ssize_t read(string &from, unsigned int &port, void *buf, const size_t bytes)
reads a datagram from socket
Definition: UDPSocket.cpp:39
ssize_t write(const string &to, const unsigned int port, void *buf, const size_t bytes)
writes up to "bytes" bytes to socket
Definition: UDPSocket.cpp:105
static void create(UDPSocket &socket, IpVersion ipVersion)
creates a udp socket
Definition: UDPSocket.cpp:168
static void createServerSocket(UDPSocket &socket, const std::string &ip, const unsigned int port)
creates a udp server socket
Definition: UDPSocket.cpp:186