axmol/external/yasio/xxsocket.hpp

1211 lines
39 KiB
C++

//////////////////////////////////////////////////////////////////////////////////////////
// A multi-platform support c++11 library with focus on asynchronous socket I/O for any
// client application.
//
//////////////////////////////////////////////////////////////////////////////////////////
/*
The MIT License (MIT)
Copyright (c) 2012-2021 HALX99
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.
*/
#ifndef YASIO__XXSOCKET_HPP
#define YASIO__XXSOCKET_HPP
#include <stddef.h>
#include <errno.h>
#include <string.h>
#include <stdio.h>
#include <stdint.h>
#include <sstream>
#include <vector>
#include <chrono>
#include <functional>
#include "yasio/detail/config.hpp"
#include "yasio/detail/logging.hpp"
#if defined(_MSC_VER)
# pragma warning(push)
# pragma warning(disable : 4996)
#endif
#ifdef _WIN32
# if !defined(WIN32_LEAN_AND_MEAN)
# define WIN32_LEAN_AND_MEAN
# endif
# include <WinSock2.h>
# include <Windows.h>
# if defined(_WIN32) && !defined(_WINSTORE)
# include <Mswsock.h>
# include <Mstcpip.h>
# endif
# include <Ws2tcpip.h>
# if defined(YASIO_NT_COMPAT_GAI)
# include <Wspiapi.h>
# endif
# if YASIO__HAS_UDS
# include <afunix.h>
# endif
typedef SOCKET socket_native_type;
typedef int socklen_t;
# define poll WSAPoll
# pragma comment(lib, "ws2_32.lib")
# undef gai_strerror
#else
# include <unistd.h>
# include <signal.h>
# include <sys/ioctl.h>
# include <netdb.h>
# include <sys/types.h>
# include <sys/poll.h>
# if defined(__linux__)
# include <sys/epoll.h>
# endif
# include <sys/select.h>
# include <sys/socket.h>
# include <sys/un.h>
# include <netinet/in.h>
# include <netinet/tcp.h>
# include <net/if.h>
# include <arpa/inet.h>
# if !defined(SD_RECEIVE)
# define SD_RECEIVE SHUT_RD
# endif
# if !defined(SD_SEND)
# define SD_SEND SHUT_WR
# endif
# if !defined(SD_BOTH)
# define SD_BOTH SHUT_RDWR
# endif
# if !defined(closesocket)
# define closesocket close
# endif
# if !defined(ioctlsocket)
# define ioctlsocket ioctl
# endif
# if defined(__linux__)
# define SO_NOSIGPIPE MSG_NOSIGNAL
# endif
typedef int socket_native_type;
# undef socket
#endif
#define SD_NONE -1
#include <fcntl.h> // common platform header
// redefine socket error code for posix api
#ifdef _WIN32
# undef EWOULDBLOCK
# undef EINPROGRESS
# undef EALREADY
# undef ENOTSOCK
# undef EDESTADDRREQ
# undef EMSGSIZE
# undef EPROTOTYPE
# undef ENOPROTOOPT
# undef EPROTONOSUPPORT
# undef ESOCKTNOSUPPORT
# undef EOPNOTSUPP
# undef EPFNOSUPPORT
# undef EAFNOSUPPORT
# undef EADDRINUSE
# undef EADDRNOTAVAIL
# undef ENETDOWN
# undef ENETUNREACH
# undef ENETRESET
# undef ECONNABORTED
# undef ECONNRESET
# undef ENOBUFS
# undef EISCONN
# undef ENOTCONN
# undef ESHUTDOWN
# undef ETOOMANYREFS
# undef ETIMEDOUT
# undef ECONNREFUSED
# undef ELOOP
# undef ENAMETOOLONG
# undef EHOSTDOWN
# undef EHOSTUNREACH
# undef ENOTEMPTY
# undef EPROCLIM
# undef EUSERS
# undef EDQUOT
# undef ESTALE
# undef EREMOTE
# undef EBADF
# undef EFAULT
# undef EAGAIN
# define EWOULDBLOCK WSAEWOULDBLOCK
# define EINPROGRESS WSAEINPROGRESS
# define EALREADY WSAEALREADY
# define ENOTSOCK WSAENOTSOCK
# define EDESTADDRREQ WSAEDESTADDRREQ
# define EMSGSIZE WSAEMSGSIZE
# define EPROTOTYPE WSAEPROTOTYPE
# define ENOPROTOOPT WSAENOPROTOOPT
# define EPROTONOSUPPORT WSAEPROTONOSUPPORT
# define ESOCKTNOSUPPORT WSAESOCKTNOSUPPORT
# define EOPNOTSUPP WSAEOPNOTSUPP
# define EPFNOSUPPORT WSAEPFNOSUPPORT
# define EAFNOSUPPORT WSAEAFNOSUPPORT
# define EADDRINUSE WSAEADDRINUSE
# define EADDRNOTAVAIL WSAEADDRNOTAVAIL
# define ENETDOWN WSAENETDOWN
# define ENETUNREACH WSAENETUNREACH
# define ENETRESET WSAENETRESET
# define ECONNABORTED WSAECONNABORTED
# define ECONNRESET WSAECONNRESET
# define ENOBUFS WSAENOBUFS
# define EISCONN WSAEISCONN
# define ENOTCONN WSAENOTCONN
# define ESHUTDOWN WSAESHUTDOWN
# define ETOOMANYREFS WSAETOOMANYREFS
# define ETIMEDOUT WSAETIMEDOUT
# define ECONNREFUSED WSAECONNREFUSED
# define ELOOP WSAELOOP
# define ENAMETOOLONG WSAENAMETOOLONG
# define EHOSTDOWN WSAEHOSTDOWN
# define EHOSTUNREACH WSAEHOSTUNREACH
# define ENOTEMPTY WSAENOTEMPTY
# define EPROCLIM WSAEPROCLIM
# define EUSERS WSAEUSERS
# define EDQUOT WSAEDQUOT
# define ESTALE WSAESTALE
# define EREMOTE WSAEREMOTE
# define EBADF WSAEBADF
# define EFAULT WSAEFAULT
# define EAGAIN WSATRY_AGAIN
#endif
#if !defined(MAXNS)
# define MAXNS 3
#endif
#define IN_MAX_ADDRSTRLEN INET6_ADDRSTRLEN
#if !defined(_WS2IPDEF_)
inline bool IN4_IS_ADDR_LOOPBACK(const in_addr* a)
{
return ((a->s_addr & 0xff) == 0x7f); // 127/8
}
inline bool IN4_IS_ADDR_LINKLOCAL(const in_addr* a)
{
return ((a->s_addr & 0xffff) == 0xfea9); // 169.254/16
}
inline bool IN6_IS_ADDR_GLOBAL(const in6_addr* a)
{
//
// Check the format prefix and exclude addresses
// whose high 4 bits are all zero or all one.
// This is a cheap way of excluding v4-compatible,
// v4-mapped, loopback, multicast, link-local, site-local.
//
unsigned int High = (a->s6_addr[0] & 0xf0);
return ((High != 0) && (High != 0xf0));
}
#endif
#define YASIO_ADDR_ANY(af) (af == AF_INET ? "0.0.0.0" : "::")
namespace yasio
{
namespace inet
{
// #define _make_dotted_decimal(b1,b2,b3,b4) ( ( ((uint32_t)(b4) << 24) & 0xff000000 ) | (
// ((uint32_t)(b3) << 16) & 0x00ff0000 ) | ( ((uint32_t)(b2) << 8) & 0x0000ff00 ) | ( (uint32_t)(b1)
// & 0x000000ff ) )
static const socket_native_type invalid_socket = (socket_native_type)-1;
YASIO__NS_INLINE
namespace ip
{
#pragma pack(push, 1)
// ip packet
struct ip_header {
// header size; 5+
unsigned char header_length : 4;
// IP version: 0100/0x04(IPv4), 0110/0x05(IPv6)
unsigned char version : 4;
// type of service:
union {
unsigned char value;
struct {
unsigned char priority : 3;
unsigned char D : 1; // delay: 0(normal), 1(as little as possible)
unsigned char T : 1; // throughput: 0(normal), 1(as big as possible)
unsigned char R : 1; // reliability: 0(normal), 1(as big as possible)
unsigned char C : 1; // transmission cost: 0(normal), 1(as little as possible)
unsigned char reserved : 1; // always be zero
} detail;
} TOS;
// total size, header + data; MAX length is: 65535
unsigned short total_length;
// identifier: all split small packet set as the same value.
unsigned short identifier;
// flags and frag
unsigned short flags : 3;
unsigned short frag : 13;
// time of living, decreased by route, if zero, this packet will by dropped
// avoid foward looply.
unsigned char TTL;
// protocol
// 1: ICMP
// 2: IGMP
// 6: TCP
// 0x11/17: UDP
// 0x58/88: IGRP
// 0x59/89: OSPF
unsigned char protocol; // TCP / UDP / Other
// check header of IP-PACKET 's correctness.
unsigned short checksum;
typedef union {
unsigned int value;
struct {
unsigned int B1 : 8, B2 : 8, B3 : 8, B4 : 8;
} detail;
} dotted_decimal_t;
// source ip address
dotted_decimal_t src_ip;
// destination ip address
dotted_decimal_t dst_ip;
};
struct psd_header {
unsigned long src_addr;
unsigned long dst_addr;
char mbz;
char protocol;
unsigned short tcp_length;
};
struct tcp_header {
unsigned short src_port; // lgtm [cpp/class-many-fields]
unsigned short dst_port;
unsigned int seqno;
unsigned int ackno;
unsigned char header_length : 4;
unsigned char reserved : 4;
unsigned char flg_fin : 1, flg_syn : 1, flg_rst : 1, flg_psh : 1, flg_ack : 1, flg_urg : 1, flg_reserved : 2;
unsigned short win_length;
unsigned short checksum;
unsigned short urp;
};
struct udp_header {
unsigned short src_port;
unsigned short dst_port;
unsigned short length;
unsigned short checksum;
};
struct icmp_header {
unsigned char type; // 8bit type
unsigned char code; // 8bit code
unsigned short checksum; // 16bit check sum
unsigned short id; // identifier: usually use process id
unsigned short seqno; // message sequence NO.
};
struct eth_header {
unsigned dst_eth[6];
unsigned src_eth[6];
unsigned eth_type;
};
struct arp_header {
unsigned short arp_hw; // format of hardware address
unsigned short arp_pro; // format of protocol address
unsigned char arp_hlen; // length of hardware address
unsigned char arp_plen; // length of protocol address
unsigned short arp_op; // arp operation
unsigned char arp_oha[6]; // sender hardware address
unsigned long arp_opa; // sender protocol address
unsigned char arp_tha; // target hardware address
unsigned long arp_tpa; // target protocol address;
};
struct arp_packet {
eth_header ethhdr;
arp_header arphdr;
};
#pragma pack(pop)
namespace compat
{
#if YASIO__HAS_NTOP
using ::inet_ntop;
using ::inet_pton;
#else
YASIO__DECL const char* inet_ntop(int af, const void* src, char* dst, socklen_t);
YASIO__DECL int inet_pton(int af, const char* src, void* dst);
#endif
} // namespace compat
inline bool is_global_in4_addr(const in_addr* addr) { return !IN4_IS_ADDR_LOOPBACK(addr) && !IN4_IS_ADDR_LINKLOCAL(addr); };
inline bool is_global_in6_addr(const in6_addr* addr) { return !!IN6_IS_ADDR_GLOBAL(addr); };
struct endpoint {
public:
endpoint() { this->zeroset(); }
endpoint(const endpoint& rhs) { this->as_is(rhs); }
explicit endpoint(const addrinfo* info) { as_is(info); }
explicit endpoint(const sockaddr* info) { as_is(info); }
explicit endpoint(const char* addr, unsigned short port = 0) { as_in(addr, port); }
explicit endpoint(uint32_t addr, unsigned short port = 0) { as_in(addr, port); }
endpoint(int family, const void* addr, unsigned short port = 0) { as_in(family, addr, port); }
explicit operator bool() const { return this->af() != AF_UNSPEC; }
endpoint& operator=(const endpoint& rhs) { return this->as_is(rhs); }
endpoint& as_is(const endpoint& rhs)
{
this->zeroset();
memcpy(this, &rhs, sizeof(rhs));
return *this;
}
endpoint& as_is(const addrinfo* info) { return this->as_is_raw(info->ai_addr, info->ai_addrlen); }
endpoint& as_is(const sockaddr* addr)
{
this->zeroset();
switch (addr->sa_family)
{
case AF_INET:
::memcpy(&in4_, addr, sizeof(sockaddr_in));
this->len(sizeof(sockaddr_in));
break;
case AF_INET6:
::memcpy(&in6_, addr, sizeof(sockaddr_in6));
this->len(sizeof(sockaddr_in6));
break;
#if defined(YASIO_ENABLE_UDS) && YASIO__HAS_UDS
case AF_UNIX:
as_un(((sockaddr_un*)addr)->sun_path);
break;
#endif
}
return *this;
}
endpoint& as_in(int family, const void* addr_in, u_short port)
{
this->zeroset();
this->af(family);
this->port(port);
switch (family)
{
case AF_INET:
::memcpy(&in4_.sin_addr, addr_in, sizeof(in_addr));
this->len(sizeof(sockaddr_in));
break;
case AF_INET6:
::memcpy(&in6_.sin6_addr, addr_in, sizeof(in6_addr));
this->len(sizeof(sockaddr_in6));
break;
}
return *this;
}
endpoint& as_in(const char* addr, unsigned short port)
{
this->zeroset();
/*
* Windows XP no inet_pton or inet_ntop
*/
if (strchr(addr, ':') == nullptr)
{ // ipv4
if (compat::inet_pton(AF_INET, addr, &this->in4_.sin_addr) == 1)
{
this->in4_.sin_family = AF_INET;
this->in4_.sin_port = htons(port);
this->len(sizeof(sockaddr_in));
}
}
else
{ // ipv6
if (compat::inet_pton(AF_INET6, addr, &this->in6_.sin6_addr) == 1)
{
this->in6_.sin6_family = AF_INET6;
this->in6_.sin6_port = htons(port);
this->len(sizeof(sockaddr_in6));
}
}
return *this;
}
endpoint& as_in(uint32_t addr, u_short port)
{
this->zeroset();
this->af(AF_INET);
this->addr_v4(addr);
this->port(port);
this->len(sizeof(sockaddr_in));
return *this;
}
#if defined(YASIO_ENABLE_UDS) && YASIO__HAS_UDS
endpoint& as_un(const char* name)
{
int n = snprintf(un_.sun_path, sizeof(un_.sun_path) - 1, "%s", name);
if (n > 0)
{
un_.sun_family = AF_UNIX;
this->len(offsetof(struct sockaddr_un, sun_path) + n + 1);
}
else
{
un_.sun_family = AF_UNSPEC;
this->len(0);
}
return *this;
}
#endif
endpoint& as_is_raw(const void* ai_addr, size_t ai_addrlen)
{
this->zeroset();
::memcpy(this, ai_addr, ai_addrlen);
this->len(ai_addrlen);
return *this;
}
void zeroset() { ::memset(this, 0x0, sizeof(*this)); }
void af(int v) { sa_.sa_family = v; }
int af() const { return sa_.sa_family; }
void ip(const char* addr)
{
/*
* Windows XP no inet_pton or inet_ntop
*/
if (strchr(addr, ':') == nullptr)
{ // ipv4
this->in4_.sin_family = AF_INET;
compat::inet_pton(AF_INET, addr, &this->in4_.sin_addr);
}
else
{ // ipv6
this->in6_.sin6_family = AF_INET6;
compat::inet_pton(AF_INET6, addr, &this->in6_.sin6_addr);
}
}
std::string ip() const
{
std::string ipstring(IN_MAX_ADDRSTRLEN - 1, '\0');
auto str = inaddr_to_string(
&ipstring.front(), [](const in_addr*) { return true; }, [](const in6_addr*) { return true; });
ipstring.resize(str ? strlen(str) : 0);
return ipstring;
}
// to_string with port, can simply add prefix "http::" or "https://" for url
std::string to_string() const
{
std::string addr(IN_MAX_ADDRSTRLEN + sizeof("65535") + 2, '[');
size_t n = 0;
switch (sa_.sa_family)
{
case AF_INET:
n = strlen(compat::inet_ntop(AF_INET, &in4_.sin_addr, &addr.front(), static_cast<socklen_t>(addr.length())));
n += sprintf(&addr.front() + n, ":%u", this->port());
break;
case AF_INET6:
n = strlen(compat::inet_ntop(AF_INET6, &in6_.sin6_addr, &addr.front() + 1, static_cast<socklen_t>(addr.length() - 1)));
n += sprintf(&addr.front() + n, "]:%u", this->port());
break;
#if defined(YASIO_ENABLE_UDS) && YASIO__HAS_UDS
case AF_UNIX:
n = this->len();
addr.assign(un_.sun_path, n);
break;
#endif
}
addr.resize(n);
return addr;
}
unsigned short port() const { return ntohs(in4_.sin_port); }
void port(unsigned short value) { in4_.sin_port = htons(value); }
void addr_v4(uint32_t addr) { in4_.sin_addr.s_addr = htonl(addr); }
uint32_t addr_v4() const { return ntohl(in4_.sin_addr.s_addr); }
/*
%N: s_net 127
%H: s_host 0
%L: s_lh 0
%M: s_impno 1
%l: low byte of port
%h: high byte of port
*/
std::string format_v4(const char* foramt)
{
static const char* const _SIN_FORMATS[] = {"%N", "%H", "%L", "%M", "%l", "%h"};
unsigned char addr_bytes[sizeof(in4_.sin_addr.s_addr) + sizeof(u_short)];
memcpy(addr_bytes, &in4_.sin_addr.s_addr, sizeof(in4_.sin_addr.s_addr));
memcpy(addr_bytes + sizeof(in4_.sin_addr.s_addr), &in4_.sin_port, sizeof(in4_.sin_port));
char snum[sizeof("255")] = {0};
const size_t _N0 = sizeof("%N") - 1;
std::string s = foramt;
for (size_t idx = 0; idx < YASIO_ARRAYSIZE(_SIN_FORMATS); ++idx)
{
auto fmt = _SIN_FORMATS[idx];
auto offst = s.find(fmt);
if (offst != std::string::npos)
{
sprintf(snum, "%u", addr_bytes[idx]);
s.replace(offst, _N0, snum);
}
}
return s;
}
// in_addr(ip) to string with pred
template <typename _Pred4, typename _Pred6> const char* inaddr_to_string(char* str /*[IN_MAX_ADDRSTRLEN]*/, _Pred4&& pred4, _Pred6&& pred6) const
{
switch (af())
{
case AF_INET:
if (pred4(&in4_.sin_addr))
return compat::inet_ntop(AF_INET, &in4_.sin_addr, str, INET_ADDRSTRLEN);
break;
case AF_INET6:
if (pred6(&in6_.sin6_addr))
return compat::inet_ntop(AF_INET6, &in6_.sin6_addr, str, INET6_ADDRSTRLEN);
break;
}
return nullptr;
}
// in_addr(ip) to csv without loopback or linklocal address
void inaddr_to_csv_nl(std::string& csv)
{
char str[INET6_ADDRSTRLEN] = {0};
if (inaddr_to_string(str, is_global_in4_addr, is_global_in6_addr))
{
csv += str;
csv += ',';
}
}
// the in_addr(from sockaddr) to csv string helper function without loopback or linklocal
// address
static void inaddr_to_csv_nl(const sockaddr* addr, std::string& csv) { endpoint(addr).inaddr_to_csv_nl(csv); }
// the in_addr/in6_addr to csv string helper function without loopback or linklocal address
// the inaddr should be union of in_addr,in6_addr or ensure it's memory enough when
// family=AF_INET6
static void inaddr_to_csv_nl(int family, const void* inaddr, std::string& csv) { endpoint(family, inaddr).inaddr_to_csv_nl(csv); }
void len(size_t n)
{
#if !YASIO__HAS_SA_LEN
len_ = static_cast<uint8_t>(n);
#else
sa_.sa_len = static_cast<uint8_t>(n);
#endif
}
socklen_t len() const
{
#if !YASIO__HAS_SA_LEN
return len_;
#else
return sa_.sa_len;
#endif
}
union {
sockaddr sa_;
sockaddr_in in4_;
sockaddr_in6 in6_;
#if defined(YASIO_ENABLE_UDS) && YASIO__HAS_UDS
sockaddr_un un_;
#endif
};
#if !YASIO__HAS_SA_LEN
uint8_t len_;
#endif
};
// supported internet protocol flags
enum : u_short
{
ipsv_unavailable = 0,
ipsv_ipv4 = 1,
ipsv_ipv6 = 2,
ipsv_dual_stack = ipsv_ipv4 | ipsv_ipv6,
};
} // namespace ip
#if !YASIO__HAS_NS_INLINE
using namespace yasio::inet::ip;
#endif
/*
** CLASS xxsocket: a posix socket wrapper
*/
class YASIO_API xxsocket {
public: /// portable connect APIs
// easy to connect a server ipv4 or ipv6 with local ip protocol version detect
// for support ipv6 ONLY network.
YASIO__DECL int xpconnect(const char* hostname, u_short port, u_short local_port = 0);
YASIO__DECL int xpconnect_n(const char* hostname, u_short port, const std::chrono::microseconds& wtimeout, u_short local_port = 0);
// easy to connect a server ipv4 or ipv6.
YASIO__DECL int pconnect(const char* hostname, u_short port, u_short local_port = 0);
YASIO__DECL int pconnect_n(const char* hostname, u_short port, const std::chrono::microseconds& wtimeout, u_short local_port = 0);
YASIO__DECL int pconnect_n(const char* hostname, u_short port, u_short local_port = 0);
// easy to connect a server ipv4 or ipv6.
YASIO__DECL int pconnect(const endpoint& ep, u_short local_port = 0);
YASIO__DECL int pconnect_n(const endpoint& ep, const std::chrono::microseconds& wtimeout, u_short local_port = 0);
YASIO__DECL int pconnect_n(const endpoint& ep, u_short local_port = 0);
// easy to create a tcp ipv4 or ipv6 server socket.
YASIO__DECL int pserve(const char* addr, u_short port);
YASIO__DECL int pserve(const endpoint& ep);
public:
// Construct a empty socket object
YASIO__DECL xxsocket();
// Construct with a exist socket handle
YASIO__DECL xxsocket(socket_native_type handle);
// Disable copy constructor
YASIO__DECL xxsocket(const xxsocket&) = delete;
// Construct with a exist socket, it will replace the source
YASIO__DECL xxsocket(xxsocket&&);
YASIO__DECL xxsocket& operator=(socket_native_type handle);
// Disable copy assign operator
YASIO__DECL xxsocket& operator=(const xxsocket&) = delete;
// Construct with a exist socket, it will replace the source
YASIO__DECL xxsocket& operator=(xxsocket&&);
// See also as function: open
YASIO__DECL xxsocket(int af, int type, int protocol);
YASIO__DECL ~xxsocket();
// swap with other when this fd is closed.
YASIO__DECL xxsocket& swap(xxsocket& who);
/* @brief: Open new socket
** @params:
** af : Usually is [AF_INET]
** type : [SOCK_STREAM-->TCP] and [SOCK_DGRAM-->UDP]
** protocol: Usually is [0]
** @returns: false: check reason by errno
*/
YASIO__DECL bool open(int af = AF_INET, int type = SOCK_STREAM, int protocol = 0);
YASIO__DECL bool reopen(int af = AF_INET, int type = SOCK_STREAM, int protocol = 0);
#ifdef _WIN32
YASIO__DECL bool open_ex(int af = AF_INET, int type = SOCK_STREAM, int protocol = 0);
YASIO__DECL static bool accept_ex(SOCKET sockfd_listened, SOCKET sockfd_prepared, PVOID lpOutputBuffer, DWORD dwReceiveDataLength, DWORD dwLocalAddressLength,
DWORD dwRemoteAddressLength, LPDWORD lpdwBytesReceived, LPOVERLAPPED lpOverlapped);
YASIO__DECL static bool connect_ex(SOCKET s, const struct sockaddr* name, int namelen, PVOID lpSendBuffer, DWORD dwSendDataLength, LPDWORD lpdwBytesSent,
LPOVERLAPPED lpOverlapped);
YASIO__DECL static void translate_sockaddrs(PVOID lpOutputBuffer, DWORD dwReceiveDataLength, DWORD dwLocalAddressLength, DWORD dwRemoteAddressLength,
sockaddr** LocalSockaddr, LPINT LocalSockaddrLength, sockaddr** RemoteSockaddr, LPINT RemoteSockaddrLength);
#endif
/** Is this socket opened **/
YASIO__DECL bool is_open() const;
/** Gets the socket fd value **/
YASIO__DECL socket_native_type native_handle() const;
/** Release ownership of the underlying native socket handle **/
YASIO__DECL socket_native_type release_handle();
/* @brief: Set this socket io mode to nonblocking
** @params:
**
** @returns: [0] succeed, otherwise, a value of SOCKET_ERROR is returned.
*/
YASIO__DECL int set_nonblocking(bool nonblocking) const;
YASIO__DECL static int set_nonblocking(socket_native_type s, bool nonblocking);
/* @brief: Test whether the socket has nonblocking flag
** @params:
**
** @returns: [1] yes. [0] no
** @pitfall: for wsock2, will return [-1] when it's a unconnected SOCK_STREAM
*/
YASIO__DECL int test_nonblocking() const;
YASIO__DECL static int test_nonblocking(socket_native_type s);
/* @brief: Associates a local address with this socket
** @params:
** addr: four point address, if set "0.0.0.0" ipv4, "::" ipv6, the socket will listen at
*any.
** port: @$$#s
** @returns:
** If no error occurs, bind returns [0]. Otherwise, it returns SOCKET_ERROR
*/
YASIO__DECL int bind(const char* addr, unsigned short port) const;
YASIO__DECL int bind(const endpoint&) const;
YASIO__DECL int bind_any(bool ipv6 = false) const;
/* @brief: Places this socket in a state in which it is listening for an incoming connection
** @params:
** Ommit
**
** @returns:
** If no error occurs, bind returns [0]. Otherwise, it returns SOCKET_ERROR
*/
YASIO__DECL int listen(int backlog = SOMAXCONN) const;
/* @brief: Permits an incoming connection attempt on this socket
** @returns:
** If no error occurs, accept returns a new socket on which
** the actual connection is made.
** Otherwise, a value of [invalid_socket] is returned
*/
YASIO__DECL xxsocket accept() const;
/* @brief: Permits an incoming connection attempt on this socket
** @params:
** @returns:
** If no error occurs, return 0, and the new_sock will be the actual connection is made.
** Otherwise, a EWOULDBLOCK,EAGAIN or other value is returned
*/
YASIO__DECL int accept_n(socket_native_type& new_sock) const;
/* @brief: Establishes a connection to a specified this socket
** @params:
** addr: Usually is a IPV4 address
** port: Server Listenning Port
**
** @returns:
** If no error occurs, returns [0].
** Otherwise, it returns SOCKET_ERROR
*/
YASIO__DECL int connect(const char* addr, u_short port);
YASIO__DECL int connect(const endpoint& ep);
YASIO__DECL static int connect(socket_native_type s, const char* addr, u_short port);
YASIO__DECL static int connect(socket_native_type s, const endpoint& ep);
/* @brief: Establishes a connection to a specified this socket with nonblocking
** @params:
** timeout: connection timeout, millseconds
**
** @returns: [0].succeed, [-1].failed
** @remark: Because on win32, there is no way to test whether the socket is non-blocking,
** so, after this function called, the socket will be always set to blocking mode.
*/
YASIO__DECL int connect_n(const char* addr, u_short port, const std::chrono::microseconds& wtimeout);
YASIO__DECL int connect_n(const endpoint& ep, const std::chrono::microseconds& wtimeout);
YASIO__DECL static int connect_n(socket_native_type s, const endpoint& ep, const std::chrono::microseconds& wtimeout);
/* @brief: Establishes a connection to a specified this socket with nonblocking
** @params:
**
** @returns: [0].succeed, [-1].failed
** @remark: this function will return immediately, for tcp, you should detect whether the
** handshake complete by handle_write_ready.
*/
YASIO__DECL int connect_n(const endpoint& ep);
YASIO__DECL static int connect_n(socket_native_type s, const endpoint& ep);
/* @brief: Disconnect a connectionless socket (such as SOCK_DGRAM)
**
*/
YASIO__DECL int disconnect() const;
YASIO__DECL static int disconnect(socket_native_type s);
/* @brief: nonblock send
** @params: omit
**
** @returns:
** If no error occurs, retval == len,
** Oterwise, If retval < len && not_recv_error(get_last_errno()), should close socket.
*/
YASIO__DECL int send_n(const void* buf, int len, const std::chrono::microseconds& wtimeout, int flags = 0);
YASIO__DECL static int send_n(socket_native_type s, const void* buf, int len, std::chrono::microseconds wtimeout, int flags = 0);
/* @brief: nonblock recv
** @params:
** The timeout is in microseconds
** @returns:
** If no error occurs, retval == len,
** Oterwise, If retval < len && not_recv_error(get_last_errno()), should close socket.
*/
YASIO__DECL int recv_n(void* buf, int len, const std::chrono::microseconds& wtimeout, int flags = 0) const;
YASIO__DECL static int recv_n(socket_native_type s, void* buf, int len, std::chrono::microseconds wtimeout, int flags = 0);
/* @brief: Sends data on this connected socket
** @params: omit
**
** @returns:
** If no error occurs, send returns the total number of bytes sent,
** which can be less than the number requested to be sent in the len parameter.
** Otherwise, a value of SOCKET_ERROR is returned.
*/
YASIO__DECL int send(const void* buf, int len, int flags = 0) const;
YASIO__DECL static int send(socket_native_type fd, const void* buf, int len, int flags = 0);
/* @brief: Receives data from this connected socket or a bound connectionless socket.
** @params: omit
**
** @returns:
** If no error occurs, recv returns the number of bytes received and
** the buffer pointed to by the buf parameter will contain this data received.
** If the connection has been gracefully closed, the return value is [0].
*/
YASIO__DECL int recv(void* buf, int len, int flags = 0) const;
YASIO__DECL static int recv(socket_native_type s, void* buf, int len, int flags);
/* @brief: Sends data on this connected socket
** @params: omit
**
** @returns:
** If no error occurs, send returns the total number of bytes sent,
** which can be less than the number requested to be sent in the len parameter.
** Otherwise, a value of SOCKET_ERROR is returned.
*/
YASIO__DECL int sendto(const void* buf, int len, const endpoint& to, int flags = 0) const;
/* @brief: Receives a datagram and stores the source address
** @params: omit
**
** @returns:
** If no error occurs, recv returns the number of bytes received and
** the buffer pointed to by the buf parameter will contain this data received.
** If the connection has been gracefully closed, the return value is [0].
*/
YASIO__DECL int recvfrom(void* buf, int len, endpoint& peer, int flags = 0) const;
YASIO__DECL int handle_write_ready(const std::chrono::microseconds& wtimeout) const;
YASIO__DECL static int handle_write_ready(socket_native_type s, const std::chrono::microseconds& wtimeout);
YASIO__DECL int handle_read_ready(const std::chrono::microseconds& wtimeout) const;
YASIO__DECL static int handle_read_ready(socket_native_type s, const std::chrono::microseconds& wtimeout);
/* @brief: Get local address info
** @params : None
**
** @returns:
*/
YASIO__DECL endpoint local_endpoint() const;
YASIO__DECL static endpoint local_endpoint(socket_native_type);
/* @brief: Get peer address info
** @params : None
**
** @returns:
* @remark: if this a listening socket fd, will return "0.0.0.0:0"
*/
YASIO__DECL endpoint peer_endpoint() const;
YASIO__DECL static endpoint peer_endpoint(socket_native_type);
/* @brief: Configure TCP keepalive
** @params : flag: 1.on, 0.off
** idle: time(secs) to send keepalive when no data interaction
** interval: keepalive send interval(secs)
** probes: count to try when no response
**
** @returns: [0].successfully
** [<0].one or more errors occured
*/
YASIO__DECL int set_keepalive(int flag = 1, int idle = 7200, int interval = 75, int probes = 10);
YASIO__DECL static int set_keepalive(socket_native_type s, int flag, int idle, int interval, int probes);
YASIO__DECL void reuse_address(bool reuse);
YASIO__DECL void exclusive_address(bool exclusive);
/* @brief: Sets the socket option
** @params :
** level: The level at which the option is defined (for example, SOL_SOCKET).
** optname: The socket option for which the value is to be set (for example, SO_BROADCAST).
** The optname parameter must be a socket option defined within the specified level,
** or behavior is undefined.
** optval: The option value.
** @examples:
** set_optval(SOL_SOCKET, SO_SNDBUF, 4096);
** set_optval(SOL_SOCKET, SO_RCVBUF, 4096);
**
** @remark: for more detail, please see:
** windows: https://docs.microsoft.com/en-us/windows/win32/api/winsock/nf-winsock-setsockopt
** linux: https://linux.die.net/man/3/setsockopt
**
** @returns: If no error occurs, set_optval returns zero. Otherwise, a value of SOCKET_ERROR is
** returned
*/
template <typename _Ty> inline int set_optval(int level, int optname, const _Ty& optval) { return set_optval(this->fd, level, optname, optval); }
template <typename _Ty> inline static int set_optval(socket_native_type sockfd, int level, int optname, const _Ty& optval)
{
return set_optval(sockfd, level, optname, &optval, static_cast<socklen_t>(sizeof(_Ty)));
}
int set_optval(int level, int optname, const void* optval, socklen_t optlen) { return set_optval(this->fd, level, optname, optval, optlen); }
static int set_optval(socket_native_type sockfd, int level, int optname, const void* optval, socklen_t optlen)
{
return ::setsockopt(sockfd, level, optname, static_cast<const char*>(optval), optlen);
}
/* @brief: Retrieves a socket option
** @params :
** level: The level at which the option is defined. Example: SOL_SOCKET.
** optname: The socket option for which the value is to be retrieved.
** Example: SO_ACCEPTCONN. The optname value must be a socket option defined within the
** specified level, or behavior is undefined.
** optval: A variable to the buffer in which the value for the requested option is to be
** returned.
**
** @returns: If no error occurs, get_optval returns zero. Otherwise, a value of SOCKET_ERROR is
*returned
*/
template <typename _Ty> inline _Ty get_optval(int level, int optname) const
{
_Ty optval = {};
get_optval(this->fd, level, optname, optval);
return optval;
}
template <typename _Ty> inline int get_optval(int level, int optname, _Ty& optval) const { return get_optval(this->fd, level, optname, optval); }
template <typename _Ty> inline static int get_optval(socket_native_type sockfd, int level, int optname, _Ty& optval)
{
socklen_t optlen = static_cast<socklen_t>(sizeof(_Ty));
return get_optval(sockfd, level, optname, &optval, &optlen);
}
static int get_optval(socket_native_type sockfd, int level, int optname, void* optval, socklen_t* optlen)
{
return ::getsockopt(sockfd, level, optname, static_cast<char*>(optval), optlen);
}
/* @brief: control the socket
** @params :
** see MSDN or man page
** @returns: If no error occurs, ioctl returns zero. Otherwise, a value of SOCKET_ERROR is
** returned
**
**
**
*/
template <typename _Ty> inline int ioctl(long cmd, const _Ty& value) const { return xxsocket::ioctl(this->fd, cmd, value); }
template <typename _Ty> inline static int ioctl(socket_native_type s, long cmd, const _Ty& value)
{
u_long argp = static_cast<u_long>(value);
return ::ioctlsocket(s, cmd, &argp);
}
/* @brief: wrapper system select, hide signal EINTR
** @params:
** s: the socket fd, it's different with system
** see MSDN or man page
** @returns: If no error occurs, returns >= 0. Otherwise, a value of -1 is
** returned
*/
int select(fd_set* readfds, fd_set* writefds, fd_set* exceptfds, const std::chrono::microseconds& wtimeout) const
{
return xxsocket::select(this->fd, readfds, writefds, exceptfds, wtimeout);
}
YASIO__DECL static int select(socket_native_type s, fd_set* readfds, fd_set* writefds, fd_set* exceptfds, std::chrono::microseconds wtimeout);
/* @brief: Disables sends or receives on this socket
** @params:
** how: [SD_SEND] or [SD_RECEIVE] or [SD_BOTH]
**
** @returns: [0] succeed, otherwise, a value of SOCKET_ERROR is returned.
*/
YASIO__DECL int shutdown(int how = SD_BOTH) const;
/* @brief: close sends
** @params:
** shut_how: [SD_SEND] or [SD_RECEIVE] or [SD_BOTH] or [SD_NONE]
**
** @returns: [0] succeed, otherwise, a value of SOCKET_ERROR is returned.
*/
YASIO__DECL void close(int shut_how = SD_BOTH);
/* @brief: Retrive tcp socket rtt in microseconds
** @params:
** non
**
** @returns: [0] succeed, otherwise, a value of SOCKET_ERROR is returned.
*/
YASIO__DECL unsigned int tcp_rtt() const;
YASIO__DECL static unsigned int tcp_rtt(socket_native_type s);
YASIO__DECL operator socket_native_type() const;
/// <summary>
/// this function just for windows platform
/// </summary>
YASIO__DECL static void init_ws32_lib();
YASIO__DECL static int get_last_errno();
YASIO__DECL static void set_last_errno(int error);
YASIO__DECL static bool not_send_error(int error);
YASIO__DECL static bool not_recv_error(int error);
YASIO__DECL static const char* strerror(int error);
YASIO__DECL static const char* gai_strerror(int error);
/// <summary>
/// Resolve all as ipv4 or ipv6 endpoints
/// </summary>
YASIO__DECL static int resolve(std::vector<endpoint>& endpoints, const char* hostname, unsigned short port = 0, int socktype = SOCK_STREAM);
/// <summary>
/// Resolve as ipv4 address only.
/// </summary>
YASIO__DECL static int resolve_v4(std::vector<endpoint>& endpoints, const char* hostname, unsigned short port = 0, int socktype = SOCK_STREAM);
/// <summary>
/// Resolve as ipv6 address only.
/// </summary>
YASIO__DECL static int resolve_v6(std::vector<endpoint>& endpoints, const char* hostname, unsigned short port = 0, int socktype = SOCK_STREAM);
/// <summary>
/// Resolve as ipv4 address only and convert to V4MAPPED format.
/// </summary>
YASIO__DECL static int resolve_v4to6(std::vector<endpoint>& endpoints, const char* hostname, unsigned short port = 0, int socktype = SOCK_STREAM);
/// <summary>
/// Force resolve all addres to ipv6 endpoints, IP4 with AI_V4MAPPED
/// </summary>
YASIO__DECL static int resolve_tov6(std::vector<endpoint>& endpoints, const char* hostname, unsigned short port = 0, int socktype = SOCK_STREAM);
/// <summary>
/// Resolve as ipv4 or ipv6 endpoints with callback
/// </summary>
template <typename _Fty>
inline static int resolve_i(const _Fty& callback, const char* hostname, unsigned short port = 0, int af = 0, int flags = 0, int socktype = SOCK_STREAM)
{
addrinfo hint;
memset(&hint, 0x0, sizeof(hint));
hint.ai_flags = flags;
hint.ai_family = af;
hint.ai_socktype = socktype;
addrinfo* answerlist = nullptr;
char buffer[sizeof "65535"] = {'\0'};
const char* service = nullptr;
if (port > 0)
{
sprintf(buffer, "%u", port); // It's enough for unsigned short, so use sprintf ok.
service = buffer;
}
int error = getaddrinfo(hostname, service, &hint, &answerlist);
if (nullptr == answerlist)
return error;
for (auto ai = answerlist; ai != nullptr; ai = ai->ai_next)
{
if (ai->ai_family == AF_INET6 || ai->ai_family == AF_INET)
{
if (callback(endpoint(ai)))
break;
}
}
freeaddrinfo(answerlist);
return error;
}
/*
** @brief:: Gets supported internet protocol versions of localhost
** @returns:
** ipsv_unavailable: The network unavailable.
** ipsv_ipv4: Support ipv4 only.
** ipsv_ipv6: Support ipv6 only.
** ipsv_dual_stack:
** Support ipv4 or ipv6, but for multi network adapters device, you should always
** use ipv4 preferred, such as smart phone with wifi & cellular network. The smart phone's os
** will choose wifi when it is available to avoid consume user's cash, when the cellular
*support
** ipv6/ipv4 but the wifi only support ipv4, then use ipv6 will cause network issue.
** For more detail, see: https://github.com/halx99/yasio/issues/130
*/
YASIO__DECL static int getipsv();
/*
** @brief: Traverse local device network adapter address with valid ip
** @params:
** handler: prototype is [](const ip::endpoint& ep)->bool
*/
YASIO__DECL static void traverse_local_address(std::function<bool(const ip::endpoint&)> handler);
protected:
YASIO__DECL static void reregister_descriptor(socket_native_type s, fd_set* fds);
private:
socket_native_type fd;
}; // namespace inet
} // namespace inet
namespace net = inet;
} // namespace yasio
namespace std
{ // VS2013 the operator must be at namespace std
inline bool operator<(const yasio::inet::ip::endpoint& lhs, const yasio::inet::ip::endpoint& rhs)
{ // apply operator < to operands
if (lhs.af() == AF_INET)
return (static_cast<uint64_t>(lhs.in4_.sin_addr.s_addr) + lhs.in4_.sin_port) < (static_cast<uint64_t>(rhs.in4_.sin_addr.s_addr) + rhs.in4_.sin_port);
return ::memcmp(&lhs, &rhs, sizeof(rhs)) < 0;
}
inline bool operator==(const yasio::inet::ip::endpoint& lhs, const yasio::inet::ip::endpoint& rhs)
{ // apply operator == to operands
return !(lhs < rhs) && !(rhs < lhs);
}
} // namespace std
#if defined(_MSC_VER)
# pragma warning(pop)
#endif
#if defined(YASIO_HEADER_ONLY)
# include "yasio/xxsocket.cpp" // lgtm [cpp/include-non-header]
#endif
#endif