| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498 |
- /*
- Copyright (c) 2013 Daniele Bartolini, Michele Rossi
- Copyright (c) 2012 Daniele Bartolini, Simone Boscaratto
- Permission is hereby granted, free of charge, to any person
- obtaining a copy of this software and associated documentation
- files (the "Software"), to deal in the Software without
- restriction, including without limitation the rights to use,
- copy, modify, merge, publish, distribute, sublicense, and/or sell
- copies of the Software, and to permit persons to whom the
- Software is furnished to do so, subject to the following
- conditions:
- The above copyright notice and this permission notice shall be
- included in all copies or substantial portions of the Software.
- THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
- EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
- OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
- NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
- HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
- WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
- FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
- OTHER DEALINGS IN THE SOFTWARE.
- */
- #pragma once
- #include "config.h"
- #include "types.h"
- #include "assert.h"
- #include "macros.h"
- #include "net_address.h"
- #if CROWN_PLATFORM_POSIX
- #include <sys/socket.h>
- #include <sys/types.h>
- #include <netinet/in.h>
- #include <fcntl.h>
- #include <unistd.h>
- #include <errno.h>
- #elif CROWN_PLATFORM_WINDOWS
- //Undefined in WinHeaders.h, but winsock2 ecc need it.
- #ifndef NEAR
- #define NEAR
- #endif
- #ifndef FAR
- #define FAR
- #endif
- #include <winsock2.h>
- #include "win_headers.h"
- #pragma comment(lib, "Ws2_32.lib")
- #endif
- namespace crown
- {
- struct ConnectResult
- {
- enum { NO_ERROR, BAD_SOCKET, REFUSED, TIMEOUT, UNKNOWN } error;
- };
- struct ReadResult
- {
- enum { NO_ERROR, BAD_SOCKET, REMOTE_CLOSED, TIMEOUT, UNKNOWN } error;
- size_t bytes_read;
- };
- struct WriteResult
- {
- enum { NO_ERROR, BAD_SOCKET, REMOTE_CLOSED, TIMEOUT, UNKNOWN } error;
- size_t bytes_wrote;
- };
- struct AcceptResult
- {
- enum { NO_ERROR, BAD_SOCKET, NO_CONNECTION, UNKNOWN } error;
- };
- /// TCP socket
- ///
- /// @ingroup Network
- struct TCPSocket
- {
- //-----------------------------------------------------------------------------
- TCPSocket()
- #if CROWN_PLATFORM_POSIX
- : m_socket(0)
- #elif CROWN_PLATFORM_WINDOWS
- : m_socket(INVALID_SOCKET)
- #endif
- {
- }
- //-----------------------------------------------------------------------------
- void open()
- {
- #if CROWN_PLATFORM_POSIX
- m_socket = ::socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
- CE_ASSERT(m_socket >= 0, "socket: errno = %d", errno);
- #elif CROWN_PLATFORM_WINDOWS
- m_socket = ::socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
- CE_ASSERT(m_socket >= 0, "socket: WSAGetLastError = %d", WSAGetLastError());
- #endif
- }
- //-----------------------------------------------------------------------------
- ConnectResult connect(const NetAddress& ip, uint16_t port)
- {
- close();
- open();
- ConnectResult cr;
- cr.error = ConnectResult::NO_ERROR;
- #if CROWN_PLATFORM_POSIX
- sockaddr_in addr_in;
- addr_in.sin_family = AF_INET;
- addr_in.sin_addr.s_addr = htonl(ip.address());
- addr_in.sin_port = htons(port);
- int err = ::connect(m_socket, (const sockaddr*)&addr_in, sizeof(sockaddr_in));
- if (err == 0)
- return cr;
- if (errno == ECONNREFUSED)
- cr.error = ConnectResult::REFUSED;
- else if (errno == ETIMEDOUT)
- cr.error = ConnectResult::TIMEOUT;
- else
- cr.error = ConnectResult::UNKNOWN;
- return cr;
- #elif CROWN_PLATFORM_WINDOWS
- sockaddr_in addr_in;
- addr_in.sin_family = AF_INET;
- addr_in.sin_addr.s_addr = ::htonl(ip.address());
- addr_in.sin_port = ::htons(port);
- int err = ::connect(m_socket, (const sockaddr*)&addr_in, sizeof(sockaddr_in));
- if (err == 0)
- return cr;
- int wsaerr = WSAGetLastError();
- if (wsaerr == WSAECONNREFUSED)
- cr.error = ConnectResult::REFUSED;
- else if (wsaerr == WSAETIMEDOUT)
- cr.error = ConnectResult::TIMEOUT;
- else
- cr.error = ConnectResult::UNKNOWN;
- return cr;
- #endif
- }
- //-----------------------------------------------------------------------------
- bool bind(uint16_t port)
- {
- close();
- open();
- set_reuse_address(true);
- #if CROWN_PLATFORM_POSIX
- sockaddr_in address;
- address.sin_family = AF_INET;
- address.sin_addr.s_addr = htonl(INADDR_ANY);
- address.sin_port = htons(port);
- int err = ::bind(m_socket, (const sockaddr*) &address, sizeof(sockaddr_in));
- CE_ASSERT(err == 0, "bind: errno = %d", errno);
- CE_UNUSED(err);
- return true;
- #elif CROWN_PLATFORM_WINDOWS
- sockaddr_in address;
- address.sin_family = AF_INET;
- address.sin_addr.s_addr = htonl(INADDR_ANY);
- address.sin_port = htons(port);
- int err = ::bind(m_socket, (const sockaddr*) &address, sizeof(sockaddr_in));
- CE_ASSERT(err == 0, "bind: WSAGetLastError = %d", WSAGetLastError());
- CE_UNUSED(err);
- return true;
- #endif
- }
- //-----------------------------------------------------------------------------
- void listen(uint32_t max)
- {
- #if CROWN_PLATFORM_POSIX
- int err = ::listen(m_socket, max);
- CE_ASSERT(err == 0, "listen: errno = %d", errno);
- CE_UNUSED(err);
- #elif CROWN_PLATFORM_WINDOWS
- int err = ::listen(m_socket, max);
- CE_ASSERT(err == 0, "listen: WSAGetLastError = %d", WSAGetLastError());
- CE_UNUSED(err);
- #endif
- }
- AcceptResult accept_internal(TCPSocket& c)
- {
- AcceptResult ar;
- ar.error = AcceptResult::NO_ERROR;
- #if CROWN_PLATFORM_POSIX
- int err = ::accept(m_socket, NULL, NULL);
- if (err >= 0)
- c.m_socket = err;
- else if (err == -1 && errno == EBADF)
- ar.error = AcceptResult::BAD_SOCKET;
- else if (err == -1 && (errno == EAGAIN || errno == EWOULDBLOCK))
- ar.error = AcceptResult::NO_CONNECTION;
- else
- ar.error = AcceptResult::UNKNOWN;
- return ar;
- #elif CROWN_PLATFORM_WINDOWS
- int err = ::accept(m_socket, NULL, NULL);
- if (err != INVALID_SOCKET)
- {
- c.m_socket = err;
- return ar;
- }
- int wsaerr = WSAGetLastError();
- if (wsaerr == WSAEWOULDBLOCK)
- ar.error = AcceptResult::NO_CONNECTION;
- else
- ar.error = AcceptResult::UNKNOWN;
- return ar;
- #endif
- }
- //-----------------------------------------------------------------------------
- AcceptResult accept_nonblock(TCPSocket& c)
- {
- set_blocking(false);
- return accept_internal(c);
- }
- //-----------------------------------------------------------------------------
- AcceptResult accept(TCPSocket& c)
- {
- set_blocking(true);
- return accept_internal(c);
- }
- //-----------------------------------------------------------------------------
- void close()
- {
- #if CROWN_PLATFORM_POSIX
- if (m_socket != 0)
- {
- ::close(m_socket);
- m_socket = 0;
- }
- #elif CROWN_PLATFORM_WINDOWS
- if (m_socket != INVALID_SOCKET)
- {
- ::closesocket(m_socket);
- m_socket = INVALID_SOCKET;
- }
- #endif
- }
- ReadResult read_internal(void* data, size_t size)
- {
- ReadResult rr;
- rr.error = ReadResult::NO_ERROR;
- rr.bytes_read = 0;
- #if CROWN_PLATFORM_POSIX
- char* buf = (char*) data;
- size_t to_read = size;
- while (to_read > 0)
- {
- ssize_t read_bytes = ::recv(m_socket, buf, to_read, 0);
- if (read_bytes == -1 && (errno == EAGAIN || errno == EWOULDBLOCK))
- {
- return rr;
- }
- else if (read_bytes == -1 && errno == ETIMEDOUT)
- {
- rr.error = ReadResult::TIMEOUT;
- return rr;
- }
- else if (read_bytes == 0)
- {
- rr.error = ReadResult::REMOTE_CLOSED;
- return rr;
- }
- buf += read_bytes;
- to_read -= read_bytes;
- rr.bytes_read += read_bytes;
- }
- return rr;
- #elif CROWN_PLATFORM_WINDOWS
- char* buf = (char*) data;
- size_t to_read = size;
- while (to_read > 0)
- {
- int read_bytes = ::recv(m_socket, buf, to_read, 0);
- if (read_bytes == SOCKET_ERROR && WSAGetLastError() == WSAEWOULDBLOCK)
- return rr;
- else if (read_bytes == SOCKET_ERROR && WSAGetLastError() == WSAETIMEDOUT)
- {
- rr.error = ReadResult::TIMEOUT;
- return rr;
- }
- else if (read_bytes == 0)
- {
- rr.error = ReadResult::REMOTE_CLOSED;
- return rr;
- }
- buf += read_bytes;
- to_read -= read_bytes;
- rr.bytes_read += read_bytes;
- }
- return rr;
- #endif
- }
- //-----------------------------------------------------------------------------
- ReadResult read_nonblock(void* data, size_t size)
- {
- set_blocking(false);
- return read_internal(data, size);
- }
- //-----------------------------------------------------------------------------
- ReadResult read(void* data, size_t size)
- {
- set_blocking(true);
- return read_internal(data, size);
- }
- WriteResult write_internal(const void* data, size_t size)
- {
- WriteResult wr;
- wr.error = WriteResult::NO_ERROR;
- wr.bytes_wrote = 0;
- #if CROWN_PLATFORM_POSIX
- const char* buf = (const char*) data;
- size_t to_send = size;
- while (to_send > 0)
- {
- ssize_t bytes_wrote = ::send(m_socket, (const char*) buf, to_send, 0);
- if (bytes_wrote == -1 && (errno == EAGAIN || errno == EWOULDBLOCK))
- return wr;
- else if (bytes_wrote == -1 && errno == ETIMEDOUT)
- {
- wr.error = WriteResult::TIMEOUT;
- return wr;
- }
- else if (bytes_wrote == 0)
- {
- wr.error = WriteResult::REMOTE_CLOSED;
- return wr;
- }
- else
- {
- wr.error = WriteResult::UNKNOWN;
- return wr;
- }
- buf += bytes_wrote;
- to_send -= bytes_wrote;
- wr.bytes_wrote += bytes_wrote;
- }
- return wr;
- #elif CROWN_PLATFORM_WINDOWS
- const char* buf = (const char*) data;
- size_t to_send = size;
- while (to_send > 0)
- {
- int bytes_wrote = ::send(m_socket, (const char*) buf, to_send, 0);
- if (bytes_wrote == SOCKET_ERROR && WSAGetLastError() == WSAEWOULDBLOCK)
- {
- return wr;
- }
- else if (bytes_wrote == SOCKET_ERROR && WSAGetLastError() == WSAETIMEDOUT)
- {
- wr.error = WriteResult::TIMEOUT;
- return wr;
- }
- else if (bytes_wrote == 0)
- {
- wr.error = WriteResult::REMOTE_CLOSED;
- return wr;
- }
- else
- {
- wr.error = WriteResult::UNKNOWN;
- return wr;
- }
- buf += bytes_wrote;
- to_send -= bytes_wrote;
- wr.bytes_wrote += bytes_wrote;
- }
- return wr;
- #endif
- }
- //-----------------------------------------------------------------------------
- WriteResult write_nonblock(const void* data, size_t size)
- {
- set_blocking(false);
- return write_internal(data, size);
- }
- //-----------------------------------------------------------------------------
- WriteResult write(const void* data, size_t size)
- {
- set_blocking(true);
- return write_internal(data, size);
- }
- //-----------------------------------------------------------------------------
- void set_blocking(bool blocking)
- {
- #if CROWN_PLATFORM_POSIX
- int flags = fcntl(m_socket, F_GETFL, 0);
- fcntl(m_socket, F_SETFL, blocking ? (flags & ~O_NONBLOCK) : O_NONBLOCK);
- #elif CROWN_PLATFORM_WINDOWS
- //Warning! http://www.sockets.com/winsock.htm#IoctlSocket
- u_long non_blocking = blocking ? 0 : 1;
- ioctlsocket(m_socket, FIONBIO, &non_blocking);
- #endif
- }
- //-----------------------------------------------------------------------------
- void set_reuse_address(bool reuse)
- {
- #if CROWN_PLATFORM_POSIX
- int optval = (int) reuse;
- int err = setsockopt(m_socket, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(optval));
- CE_ASSERT(err == 0, "setsockopt: errno = %d", errno);
- #elif CROWN_PLATFORM_WINDOWS
- int optval = (int) reuse;
- int err = setsockopt(m_socket, SOL_SOCKET, SO_REUSEADDR, (const char*) &optval, sizeof(optval));
- CE_ASSERT(err == 0, "setsockopt: WSAGetLastError = %d", WSAGetLastError());
- #endif
- }
- //-----------------------------------------------------------------------------
- void set_timeout(uint32_t seconds)
- {
- #if CROWN_PLATFORM_POSIX
- struct timeval timeout;
- timeout.tv_sec = seconds;
- timeout.tv_usec = 0;
- int err;
- err = setsockopt (m_socket, SOL_SOCKET, SO_RCVTIMEO, (char*) &timeout, sizeof(timeout));
- CE_ASSERT(err == 0, "setsockopt: errno: %d", errno);
- err = setsockopt (m_socket, SOL_SOCKET, SO_SNDTIMEO, (char*) &timeout, sizeof(timeout));
- CE_ASSERT(err == 0, "setsockopt: errno: %d", errno);
- #elif CROWN_PLATFORM_WINDOWS
- struct timeval timeout;
- timeout.tv_sec = seconds;
- timeout.tv_usec = 0;
- int err;
- err = setsockopt (m_socket, SOL_SOCKET, SO_RCVTIMEO, (char*) &timeout, sizeof(timeout));
- CE_ASSERT(err == 0, "setsockopt: WSAGetLastError: %d", WSAGetLastError());
- err = setsockopt (m_socket, SOL_SOCKET, SO_SNDTIMEO, (char*) &timeout, sizeof(timeout));
- CE_ASSERT(err == 0, "setsockopt: WSAGetLastError: %d", WSAGetLastError());
- #endif
- }
- private:
- #if CROWN_PLATFORM_POSIX
- int m_socket;
- #elif CROWN_PLATFORM_WINDOWS
- SOCKET m_socket;
- #endif
- };
- } // namespace crown
|