10 #define SHUT_RDWR SD_BOTH
15 #include <netinet/in.h>
16 #include <sys/socket.h>
17 #include <arpa/inet.h>
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());
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);
85 std::string msg =
"Could not bind socket: ";
87 msg+= to_string(WSAGetLastError());
89 msg+= strerror(errno);
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());
111 std::string msg =
"Could not get socket file descriptor settings: ";
112 msg+= strerror(errno);
117 if (fcntl(
descriptor, F_SETFL, fdc | O_NONBLOCK) == -1) {
118 std::string msg =
"Could not set socket non blocked: ";
119 msg+= strerror(errno);
135 return ip.find(
':') != std::string::npos?IpVersion::IPV6:IpVersion::IPV4;
138#if defined(__MINGW32__) && !defined(__MINGW64__)
140 #define NS_INADDRSZ 4
141 #define NS_IN6ADDRSZ 16
145 int inet_pton4(
const char* src,
void* dst) {
146 uint8_t tmp[NS_INADDRSZ], *tp;
153 while ((ch = *src++) !=
'\0') {
154 if (ch >=
'0' && ch <=
'9') {
155 uint32_t n = *tp * 10 + (ch -
'0');
157 if (saw_digit && *tp == 0)
169 }
else if (ch ==
'.' && saw_digit) {
180 memcpy(dst, tmp, NS_INADDRSZ);
186 int inet_pton6(
int af,
const char* src,
void* dst) {
187 static const char xdigits[] =
"0123456789abcdef";
188 uint8_t tmp[NS_IN6ADDRSZ];
190 uint8_t *tp = (uint8_t*) memset(tmp,
'\0', NS_IN6ADDRSZ);
191 uint8_t *endp = tp + NS_IN6ADDRSZ;
192 uint8_t *colonp = NULL;
200 const char *curtok = src;
204 while ((ch = tolower(*src++)) !=
'\0') {
205 const char *pch = strchr(xdigits, ch);
208 val |= (pch - xdigits);
221 }
else if (*src ==
'\0') {
224 if (tp + NS_INT16SZ > endp)
226 *tp++ = (uint8_t) (val >> 8) & 0xff;
227 *tp++ = (uint8_t) val & 0xff;
232 if (ch ==
'.' && ((tp + NS_INADDRSZ) <= endp)
233 && inet_pton4(curtok, (
char*) tp) > 0) {
241 if (tp + NS_INT16SZ > endp)
243 *tp++ = (uint8_t) (val >> 8) & 0xff;
244 *tp++ = (uint8_t) val & 0xff;
246 if (colonp != NULL) {
251 const int n = tp - colonp;
256 for (
int i = 1; i <= n; i++) {
257 endp[-i] = colonp[n - i];
265 memcpy(dst, tmp, NS_IN6ADDRSZ);
286 size_t strlcpy(
char* __restrict dst,
const char* __restrict src,
size_t siz)
295 if ((*d++ = *s++) ==
'\0')
309 return (s - src - 1);
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"];
319 std::sprintf(tmp, fmt, ((uint8_t*)src)[0], ((uint8_t*)src)[1], ((uint8_t*)src)[2], ((uint8_t*)src)[3]);
320 if (l <= 0 || (socklen_t) l >= size) {
323 strlcpy(dst, tmp, size);
328 char* inet_ntop6(
int af,
const void* src,
char* dst,
size_t size)
337 char tmp[
sizeof "ffff:ffff:ffff:ffff:ffff:ffff:255.255.255.255"], *tp;
341 u_int words[NS_IN6ADDRSZ / NS_INT16SZ];
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));
356 for (i = 0; i < (NS_IN6ADDRSZ / NS_INT16SZ); i++) {
359 cur.base = i, cur.len = 1;
363 if (cur.base != -1) {
364 if (best.base == -1 || cur.len > best.len)
370 if (cur.base != -1) {
371 if (best.base == -1 || cur.len > best.len)
374 if (best.base != -1 && best.len < 2)
381 for (i = 0; i < (NS_IN6ADDRSZ / NS_INT16SZ); i++) {
383 if (best.base != -1 && i >= best.base && i < (best.base + best.len)) {
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)))
400 tp += std::sprintf(tp,
"%x", words[i]);
404 && (best.base + best.len) == (NS_IN6ADDRSZ / NS_INT16SZ))
411 if ((socklen_t) (tp - tmp) > size) {
Base class of network sockets.
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.