TDME2 1.9.121
NetworkSocket.cpp
Go to the documentation of this file.
1#include <string.h>
2
3#include <string>
4
5#include <tdme/tdme.h>
6
7#if defined(_WIN32)
8 #include <winsock2.h>
9 #include <ws2tcpip.h>
10 #define SHUT_RDWR SD_BOTH
11#else
12 #include <errno.h>
13 #include <fcntl.h>
14 #include <unistd.h>
15 #include <netinet/in.h>
16 #include <sys/socket.h>
17 #include <arpa/inet.h>
18#endif
19
21
22using std::to_string;
23
25
26NetworkSocket& NetworkSocket::operator=(NetworkSocket& socket) {
27 descriptor = socket.descriptor;
28 ip = socket.ip;
29 ipVersion = socket.ipVersion;
30 return *this;
31}
32
33NetworkSocket::NetworkSocket() : descriptor(-1), ip("0.0.0.0"), port(0), ipVersion(IpVersion::IPV4) {
34}
35
37}
38
39const std::string& NetworkSocket::getAddress() {
40 return ip;
41}
42
43const unsigned int NetworkSocket::getPort() {
44 return port;
45}
46
48 if (descriptor != -1) ::shutdown(descriptor, SHUT_RDWR);
49}
50
51void NetworkSocket::bind(const std::string& ip, const unsigned int port) {
52 // determine IP version
54
55 // socket address in setup
56 socklen_t sinLen = 0;
57 void* sin;
58 sockaddr_in sinIPV4;
59 sockaddr_in6 sinIPV6;
60 switch(ipVersion) {
61 case IPV4:
62 {
63 sinLen = sizeof(sinIPV4);
64 memset(&sinIPV4, 0, sinLen);
65 sinIPV4.sin_family = AF_INET;
66 sinIPV4.sin_port = htons(port);
67 sinIPV4.sin_addr.s_addr = inet_addr(ip.c_str());
68 sin = &sinIPV4;
69 }
70 break;
71 case IPV6:
72 {
73 sinLen = sizeof(sinIPV6);
74 memset(&sinIPV6, 0, sinLen);
75 sinIPV6.sin6_family = AF_INET6;
76 sinIPV6.sin6_port = htons(port);
77 inet_pton(AF_INET6, ip.c_str(), &sinIPV6.sin6_addr);
78 sin = &sinIPV6;
79 }
80 break;
81 }
82
83 // bind
84 if (::bind(descriptor, (const struct sockaddr*)sin, sinLen) == -1) {
85 std::string msg = "Could not bind socket: ";
86 #if defined(_WIN32)
87 msg+= to_string(WSAGetLastError());
88 #else
89 msg+= strerror(errno);
90 #endif
91 throw NetworkSocketException(msg);
92 }
93
94 // set address
95 this->ip = ip;
96 this->port = port;
97}
98
100 #if defined(_WIN32)
101 long unsigned int mode = 1;
102 if (ioctlsocket(descriptor, FIONBIO, &mode) != 0) {
103 std::string msg = "Could not set socket non blocked: ";
104 msg+= to_string(WSAGetLastError());
105 throw NetworkSocketException(msg);
106 }
107 #else
108 // get the server socket file descriptor control settings
109 int fdc = fcntl(descriptor, F_GETFL, 0);
110 if (fdc == -1) {
111 std::string msg = "Could not get socket file descriptor settings: ";
112 msg+= strerror(errno);
113 throw NetworkSocketException(msg);
114 }
115
116 // make the socket non blocked
117 if (fcntl(descriptor, F_SETFL, fdc | O_NONBLOCK) == -1) {
118 std::string msg = "Could not set socket non blocked: ";
119 msg+= strerror(errno);
120 throw NetworkSocketException(msg);
121 }
122 #endif
123}
124
126 #if defined(_WIN32)
127 ::closesocket(descriptor);
128 #else
130 #endif
131 descriptor = -1;
132}
133
135 return ip.find(':') != std::string::npos?IpVersion::IPV6:IpVersion::IPV4;
136}
137
138#if defined(__MINGW32__) && !defined(__MINGW64__)
139 // see: https://stackoverflow.com/questions/15370033/how-to-use-inet-pton-with-the-mingw-compiler
140 #define NS_INADDRSZ 4
141 #define NS_IN6ADDRSZ 16
142 #define NS_INT16SZ 2
143
144 // author: Paul Vixie, 1996.
145 int inet_pton4(const char* src, void* dst) {
146 uint8_t tmp[NS_INADDRSZ], *tp;
147
148 int saw_digit = 0;
149 int octets = 0;
150 *(tp = tmp) = 0;
151
152 int ch;
153 while ((ch = *src++) != '\0') {
154 if (ch >= '0' && ch <= '9') {
155 uint32_t n = *tp * 10 + (ch - '0');
156
157 if (saw_digit && *tp == 0)
158 return 0;
159
160 if (n > 255)
161 return 0;
162
163 *tp = n;
164 if (!saw_digit) {
165 if (++octets > 4)
166 return 0;
167 saw_digit = 1;
168 }
169 } else if (ch == '.' && saw_digit) {
170 if (octets == 4)
171 return 0;
172 *++tp = 0;
173 saw_digit = 0;
174 } else
175 return 0;
176 }
177 if (octets < 4)
178 return 0;
179
180 memcpy(dst, tmp, NS_INADDRSZ);
181
182 return 1;
183 }
184
185 // author: Paul Vixie, 1996.
186 int inet_pton6(int af, const char* src, void* dst) {
187 static const char xdigits[] = "0123456789abcdef";
188 uint8_t tmp[NS_IN6ADDRSZ];
189
190 uint8_t *tp = (uint8_t*) memset(tmp, '\0', NS_IN6ADDRSZ);
191 uint8_t *endp = tp + NS_IN6ADDRSZ;
192 uint8_t *colonp = NULL;
193
194 /* Leading :: requires some special handling. */
195 if (*src == ':') {
196 if (*++src != ':')
197 return 0;
198 }
199
200 const char *curtok = src;
201 int saw_xdigit = 0;
202 uint32_t val = 0;
203 int ch;
204 while ((ch = tolower(*src++)) != '\0') {
205 const char *pch = strchr(xdigits, ch);
206 if (pch != NULL) {
207 val <<= 4;
208 val |= (pch - xdigits);
209 if (val > 0xffff)
210 return 0;
211 saw_xdigit = 1;
212 continue;
213 }
214 if (ch == ':') {
215 curtok = src;
216 if (!saw_xdigit) {
217 if (colonp)
218 return 0;
219 colonp = tp;
220 continue;
221 } else if (*src == '\0') {
222 return 0;
223 }
224 if (tp + NS_INT16SZ > endp)
225 return 0;
226 *tp++ = (uint8_t) (val >> 8) & 0xff;
227 *tp++ = (uint8_t) val & 0xff;
228 saw_xdigit = 0;
229 val = 0;
230 continue;
231 }
232 if (ch == '.' && ((tp + NS_INADDRSZ) <= endp)
233 && inet_pton4(curtok, (char*) tp) > 0) {
234 tp += NS_INADDRSZ;
235 saw_xdigit = 0;
236 break; /* '\0' was seen by inet_pton4(). */
237 }
238 return 0;
239 }
240 if (saw_xdigit) {
241 if (tp + NS_INT16SZ > endp)
242 return 0;
243 *tp++ = (uint8_t) (val >> 8) & 0xff;
244 *tp++ = (uint8_t) val & 0xff;
245 }
246 if (colonp != NULL) {
247 /*
248 * Since some memmove()'s erroneously fail to handle
249 * overlapping regions, we'll do the shift by hand.
250 */
251 const int n = tp - colonp;
252
253 if (tp == endp)
254 return 0;
255
256 for (int i = 1; i <= n; i++) {
257 endp[-i] = colonp[n - i];
258 colonp[n - i] = 0;
259 }
260 tp = endp;
261 }
262 if (tp != endp)
263 return 0;
264
265 memcpy(dst, tmp, NS_IN6ADDRSZ);
266
267 return 1;
268 }
269
270 /*
271 * $OpenBSD: strlcpy.c,v 1.11 2006/05/05 15:27:38 millert Exp $
272 * Copyright (c) 1998 Todd C. Miller <Todd.Miller@courtesan.com>
273 *
274 * Permission to use, copy, modify, and distribute this software for any
275 * purpose with or without fee is hereby granted, provided that the above
276 * copyright notice and this permission notice appear in all copies.
277 *
278 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
279 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
280 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
281 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
282 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
283 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
284 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
285 */
286 size_t strlcpy(char* __restrict dst, const char* __restrict src, size_t siz)
287 {
288 char *d = dst;
289 const char *s = src;
290 size_t n = siz;
291
292 /* Copy as many bytes as will fit */
293 if (n != 0) {
294 while (--n != 0) {
295 if ((*d++ = *s++) == '\0')
296 break;
297 }
298 }
299
300 /* Not enough room in dst, add NUL and traverse rest of src */
301 if (n == 0) {
302 if (siz != 0)
303 *d = '\0'; /* NUL-terminate dst */
304 while (*s++)
305 ;
306 }
307
308 //
309 return (s - src - 1); /* count does not include NUL */
310 }
311
312 // author: Paul Vixie, 1996.
313 char* inet_ntop4(const void* src, char* dst, size_t size) {
314 static const char fmt[128] = "%u.%u.%u.%u";
315 char tmp[sizeof "255.255.255.255"];
316 int l;
317
318 // l = snprintf(tmp, sizeof(tmp), fmt, src[0], src[1], src[2], src[3]); // ****
319 std::sprintf(tmp, fmt, ((uint8_t*)src)[0], ((uint8_t*)src)[1], ((uint8_t*)src)[2], ((uint8_t*)src)[3]); // **** vc++ does not have snprintf
320 if (l <= 0 || (socklen_t) l >= size) {
321 return (NULL);
322 }
323 strlcpy(dst, tmp, size);
324 return (dst);
325 }
326
327 // author: Paul Vixie, 1996.
328 char* inet_ntop6(int af, const void* src, char* dst, size_t size)
329 {
330 /*
331 * Note that int32_t and int16_t need only be "at least" large enough
332 * to contain a value of the specified size. On some systems, like
333 * Crays, there is no such thing as an integer variable with 16 bits.
334 * Keep this in mind if you think this function should have been coded
335 * to use pointer overlays. All the world's not a VAX.
336 */
337 char tmp[sizeof "ffff:ffff:ffff:ffff:ffff:ffff:255.255.255.255"], *tp;
338 struct {
339 int base, len;
340 } best, cur;
341 u_int words[NS_IN6ADDRSZ / NS_INT16SZ];
342 int i;
343
344 /*
345 * Preprocess:
346 * Copy the input (bytewise) array into a wordwise array.
347 * Find the longest run of 0x00's in src[] for :: shorthanding.
348 */
349 memset(words, '\0', sizeof words);
350 for (i = 0; i < NS_IN6ADDRSZ; i++)
351 words[i / 2] |= (((uint8_t*)src)[i] << ((1 - (i % 2)) << 3));
352 best.base = -1;
353 best.len = 0;
354 cur.base = -1;
355 cur.len = 0;
356 for (i = 0; i < (NS_IN6ADDRSZ / NS_INT16SZ); i++) {
357 if (words[i] == 0) {
358 if (cur.base == -1)
359 cur.base = i, cur.len = 1;
360 else
361 cur.len++;
362 } else {
363 if (cur.base != -1) {
364 if (best.base == -1 || cur.len > best.len)
365 best = cur;
366 cur.base = -1;
367 }
368 }
369 }
370 if (cur.base != -1) {
371 if (best.base == -1 || cur.len > best.len)
372 best = cur;
373 }
374 if (best.base != -1 && best.len < 2)
375 best.base = -1;
376
377 /*
378 * Format the result.
379 */
380 tp = tmp;
381 for (i = 0; i < (NS_IN6ADDRSZ / NS_INT16SZ); i++) {
382 /* Are we inside the best run of 0x00's? */
383 if (best.base != -1 && i >= best.base && i < (best.base + best.len)) {
384 if (i == best.base)
385 *tp++ = ':';
386 continue;
387 }
388 /* Are we following an initial run of 0x00s or any real hex? */
389 if (i != 0)
390 *tp++ = ':';
391 /* Is this address an encapsulated IPv4? */
392 if (i == 6 && best.base == 0
393 && (best.len == 6 || (best.len == 7 && words[7] != 0x0001)
394 || (best.len == 5 && words[5] == 0xffff))) {
395 if (!inet_ntop4(src + 12, tp, sizeof tmp - (tp - tmp)))
396 return (NULL);
397 tp += strlen(tp);
398 break;
399 }
400 tp += std::sprintf(tp, "%x", words[i]); // ****
401 }
402 /* Was it a trailing run of 0x00's? */
403 if (best.base != -1
404 && (best.base + best.len) == (NS_IN6ADDRSZ / NS_INT16SZ))
405 *tp++ = ':';
406 *tp++ = '\0';
407
408 /*
409 * Check for overflow, copy, and we're done.
410 */
411 if ((socklen_t) (tp - tmp) > size) {
412 return (NULL);
413 }
414 strcpy(dst, tmp);
415 return (dst);
416 }
417
418#endif
Base class of network sockets.
Definition: NetworkSocket.h:17
void bind(const string &ip, const unsigned int port)
Binds a socket to local ip and port.
void close()
Closes the socket.
void shutdown()
shuts socket down for reading and writing
static IpVersion determineIpVersion(const string &ip)
Determine IP version.
const string & getAddress()
returns the end points ip address
void setNonBlocked()
sets the socket non blocked
const unsigned int getPort()
returns the end points port
virtual ~NetworkSocket()
public destructor
NetworkSocket()
Protected constructor.