TDME2 1.9.121
TCPSocket.cpp
Go to the documentation of this file.
2
3#include <string>
4
5#include <tdme/tdme.h>
10
11#include <errno.h>
12#include <fcntl.h>
13#include <string.h>
14
15#if defined(_WIN32)
16 #include <winsock2.h>
17 #include <ws2tcpip.h>
18 #define socklen_t int
19 #define BUF_CAST(buf) ((char*)buf)
20 #if defined(_MSC_VER)
21 #define ssize_t int
22 #endif
23#else
24 #include <arpa/inet.h>
25 #include <netinet/tcp.h>
26 #include <netinet/in.h>
27 #include <sys/socket.h>
28 #define BUF_CAST(buf) ((void*)buf)
29#endif
30
32
33using std::to_string;
34
39
40/**
41 * @brief public destructor
42 */
43TCPSocket::~TCPSocket() {
44}
45
46size_t TCPSocket::read(void* buf, const size_t bytes) {
47 ssize_t bytesRead = ::recv(descriptor, BUF_CAST(buf), bytes, 0);
48 if (bytesRead == -1) {
49 std::string msg = "error while reading from socket: ";
50 msg+= strerror(errno);
51 throw NetworkIOException(msg);
52 } else
53 if (bytesRead == 0) {
54 throw NetworkSocketClosedException("end of stream");
55 }
56 //
57 return (size_t)bytesRead;
58}
59
60size_t TCPSocket::write(void* buf, const size_t bytes) {
61 #if defined(__APPLE__) || defined(_WIN32)
62 ssize_t bytesWritten = ::send(descriptor, BUF_CAST(buf), bytes, 0);
63 #else
64 ssize_t bytesWritten = ::send(descriptor, BUF_CAST(buf), bytes, MSG_NOSIGNAL);
65 #endif
66 if (bytesWritten == -1) {
67 if (errno == ECONNRESET || errno == EPIPE) {
68 std::string msg = "end of stream: ";
69 msg+= strerror(errno);
71 } else {
72 std::string msg = "error while writing to socket: ";
73 msg+= strerror(errno);
74 throw NetworkIOException(msg);
75 }
76 }
77 //
78 return (size_t)bytesWritten;
79}
80
81void TCPSocket::create(TCPSocket& socket, IpVersion ipVersion) {
82 socket.ipVersion = ipVersion;
83 socket.descriptor = ::socket(ipVersion == IPV6?PF_INET6:PF_INET, SOCK_STREAM, IPPROTO_TCP);
84 if (socket.descriptor == -1) {
85 std::string msg = "Could not create socket: ";
86 msg+= strerror(errno);
87 throw NetworkSocketException(msg);
88 }
89 #if defined(__APPLE__)
90 int flag = 1;
91 if (setsockopt(socket.descriptor, SOL_SOCKET, SO_NOSIGPIPE, (void*)&flag, sizeof(flag)) == -1) {
92 std::string msg = "Could not set no sig pipe on socket: ";
93 msg+= strerror(errno);
94 throw NetworkSocketException(msg);
95 }
96 #endif
97}
98
99void TCPSocket::connect(const std::string& ip, const unsigned int port) {
100 // determine IP version
102
103 // socket address in setup
104 socklen_t sinLen = 0;
105 void* sin;
106 sockaddr_in sinIPV4;
107 sockaddr_in6 sinIPV6;
108 switch(ipVersion) {
109 case IPV4:
110 {
111 sinLen = sizeof(sinIPV4);
112 memset(&sinIPV4, 0, sinLen);
113 sinIPV4.sin_family = AF_INET;
114 sinIPV4.sin_port = htons(port);
115 sinIPV4.sin_addr.s_addr = inet_addr(ip.c_str());
116 sin = &sinIPV4;
117 }
118 break;
119 case IPV6:
120 {
121 sinLen = sizeof(sinIPV6);
122 memset(&sinIPV6, 0, sinLen);
123 sinIPV6.sin6_family = AF_INET6;
124 sinIPV6.sin6_port = htons(port);
125 inet_pton(AF_INET6, ip.c_str(), &sinIPV6.sin6_addr);
126 sin = &sinIPV6;
127 }
128 break;
129 }
130
131 // bind
132 if (::connect(descriptor, (const struct sockaddr*)sin, sinLen) == -1) {
133 std::string msg = "Could not connect socket: ";
134 #if defined(_WIN32)
135 msg+= to_string(WSAGetLastError());
136 #else
137 msg+= strerror(errno);
138 #endif
139 throw NetworkSocketException(msg);
140 }
141
142 // set address
143 this->ip = ip;
144 this->port = port;
145}
146void TCPSocket::createServerSocket(TCPSocket& socket, const std::string& ip, const unsigned int port, const int backlog) {
147 // create socket
149
150 try {
151 // set non blocked
152 socket.setNonBlocked();
153
154 // bind
155 socket.bind(ip, port);
156
157 // make socket listen, backlog is 10% of max CCU
158 if (listen(socket.descriptor, backlog) == -1) {
159 std::string msg = "Could not set socket to listen: ";
160 msg+= strerror(errno);
161 throw NetworkSocketException(msg);
162 }
163 } catch (NetworkSocketException &exception) {
164 socket.close();
165 throw;
166 }
167}
168
170 int flag = 1;
171 #if defined(_WIN32)
172 if (setsockopt(descriptor, IPPROTO_TCP, TCP_NODELAY, (const char*)&flag, sizeof(flag)) == -1) {
173 #else
174 if (setsockopt(descriptor, IPPROTO_TCP, TCP_NODELAY, (void*)&flag, sizeof(flag)) == -1) {
175 #endif
176 std::string msg = "Could not set tcp no delay: ";
177 msg+= strerror(errno);
178 throw NetworkSocketException(msg);
179 }
180}
181
183 struct sockaddr_in _sin;
184 socklen_t _sinSize = sizeof(_sin);
185
186 // accept socket
187 int _descriptor = ::accept(descriptor, (struct sockaddr *)&_sin, &_sinSize);
188 if (_descriptor == -1) {
189 // no more connections are present
190 if (errno == EAGAIN ||
191 errno == EWOULDBLOCK) {
192 return false;
193 }
194 std::string msg = "Could not accept socket: ";
195 msg+= strerror(errno);
196 throw NetworkSocketException(msg);
197 }
198
199 // create client socket, return it
200 _socket.descriptor = _descriptor;
201 _socket.ip = inet_ntoa(_sin.sin_addr);
202 _socket.port = ntohs(_sin.sin_port);
203 _socket.setNonBlocked();
204 _socket.setTCPNoDelay();
205
206 // success
207 return true;
208}
#define BUF_CAST(buf)
Definition: TCPSocket.cpp:28
Base exception class for network IO exceptions.
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.
static IpVersion determineIpVersion(const string &ip)
Determine IP version.
void setNonBlocked()
sets the socket non blocked
Class representing a TCP socket.
Definition: TCPSocket.h:15
size_t read(void *buf, const size_t bytes)
Reads up to "bytes" bytes from socket.
Definition: TCPSocket.cpp:46
size_t write(void *buf, const size_t bytes)
Writes up to "bytes" bytes to socket.
Definition: TCPSocket.cpp:60
bool accept(TCPSocket &_socket)
Accepts a socket from a server socket.
Definition: TCPSocket.cpp:182
static void create(TCPSocket &socket, IpVersion ipVersion)
Creates a TCP socket.
Definition: TCPSocket.cpp:81
static void createServerSocket(TCPSocket &socket, const std::string &ip, const unsigned int port, const int backlog)
Creates a TCP server socket.
Definition: TCPSocket.cpp:146
void setTCPNoDelay()
Disables nagle's algorithm.
Definition: TCPSocket.cpp:169
void connect(const string &ip, const unsigned int port)
Connects a socket to given IP and port.
Definition: TCPSocket.cpp:99