axmol/thirdparty/yasio/xxsocket.cpp

1003 lines
30 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-2023 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_CPP
#define YASIO__XXSOCKET_CPP
#include <assert.h>
#ifdef _DEBUG
# include <stdio.h>
#endif
#if !defined(YASIO_HEADER_ONLY)
# include "yasio/xxsocket.hpp"
#endif
#include "yasio/detail/utils.hpp"
#if !defined(_WIN32)
# include "yasio/detail/ifaddrs.hpp"
#endif
// For apple bsd socket implemention
#if !defined(TCP_KEEPIDLE)
# define TCP_KEEPIDLE TCP_KEEPALIVE
#endif
#if defined(_MSC_VER)
# pragma warning(push)
# pragma warning(disable : 4996)
#endif
#if defined(_WIN32)
static LPFN_ACCEPTEX __accept_ex = nullptr;
static LPFN_GETACCEPTEXSOCKADDRS __get_accept_ex_sockaddrs = nullptr;
static LPFN_CONNECTEX __connect_ex = nullptr;
#endif
#if !YASIO__HAS_NTOP
namespace yasio
{
YASIO__NS_INLINE
namespace inet
{
YASIO__NS_INLINE
namespace ip
{
namespace compat
{
# include "yasio/detail/inet_compat.inl"
} // namespace compat
} // namespace ip
} // namespace inet
} // namespace yasio
#endif
namespace yasio
{
YASIO__NS_INLINE
namespace inet
{
int xxsocket::xpconnect(const char* hostname, u_short port, u_short local_port)
{
auto flags = getipsv();
int error = -1;
xxsocket::resolve_i(
[&](const endpoint& ep) {
switch (ep.af())
{
case AF_INET:
if (flags & ipsv_ipv4)
{
error = pconnect(ep, local_port);
}
else if (flags & ipsv_ipv6)
{
xxsocket::resolve_i([&](const endpoint& ep6) { return 0 == (error = pconnect(ep6, local_port)); }, hostname, port, AF_INET6, AI_V4MAPPED);
}
break;
case AF_INET6:
if (flags & ipsv_ipv6)
error = pconnect(ep, local_port);
break;
}
return error == 0;
},
hostname, port, AF_UNSPEC, AI_ALL);
return error;
}
int xxsocket::xpconnect_n(const char* hostname, u_short port, const std::chrono::microseconds& wtimeout, u_short local_port)
{
auto flags = getipsv();
int error = -1;
xxsocket::resolve_i(
[&](const endpoint& ep) {
switch (ep.af())
{
case AF_INET:
if (flags & ipsv_ipv4)
error = pconnect_n(ep, wtimeout, local_port);
else if (flags & ipsv_ipv6)
{
xxsocket::resolve_i([&](const endpoint& ep6) { return 0 == (error = pconnect_n(ep6, wtimeout, local_port)); }, hostname, port, AF_INET6,
AI_V4MAPPED);
}
break;
case AF_INET6:
if (flags & ipsv_ipv6)
error = pconnect_n(ep, wtimeout, local_port);
break;
}
return error == 0;
},
hostname, port, AF_UNSPEC, AI_ALL);
return error;
}
int xxsocket::pconnect(const char* hostname, u_short port, u_short local_port)
{
int error = -1;
xxsocket::resolve_i([&](const endpoint& ep) { return 0 == (error = pconnect(ep, local_port)); }, hostname, port);
return error;
}
int xxsocket::pconnect_n(const char* hostname, u_short port, const std::chrono::microseconds& wtimeout, u_short local_port)
{
int error = -1;
xxsocket::resolve_i([&](const endpoint& ep) { return 0 == (error = pconnect_n(ep, wtimeout, local_port)); }, hostname, port);
return error;
}
int xxsocket::pconnect_n(const char* hostname, u_short port, u_short local_port)
{
int error = -1;
xxsocket::resolve_i(
[&](const endpoint& ep) {
(error = pconnect_n(ep, local_port));
return true;
},
hostname, port);
return error;
}
int xxsocket::pconnect(const endpoint& ep, u_short local_port)
{
if (this->reopen(ep.af()))
{
if (local_port != 0)
this->bind(YASIO_ADDR_ANY(ep.af()), local_port);
return this->connect(ep);
}
return -1;
}
int xxsocket::pconnect_n(const endpoint& ep, const std::chrono::microseconds& wtimeout, u_short local_port)
{
if (this->reopen(ep.af()))
{
if (local_port != 0)
this->bind(YASIO_ADDR_ANY(ep.af()), local_port);
return this->connect_n(ep, wtimeout);
}
return -1;
}
int xxsocket::pconnect_n(const endpoint& ep, u_short local_port)
{
if (this->reopen(ep.af()))
{
if (local_port != 0)
this->bind(YASIO_ADDR_ANY(ep.af()), local_port);
return xxsocket::connect_n(this->fd, ep);
}
return -1;
}
int xxsocket::pserve(const char* addr, u_short port) { return this->pserve(endpoint{addr, port}); }
int xxsocket::pserve(const endpoint& ep)
{
if (!this->reopen(ep.af()))
return -1;
set_optval(SOL_SOCKET, SO_REUSEADDR, 1);
int n = this->bind(ep);
if (n != 0)
return n;
return this->listen();
}
bool xxsocket::popen(int af, int type, int protocol)
{
#if defined(_WIN32)
bool ok = this->open_ex(af, type, protocol);
#else
bool ok = this->open(af, type, protocol);
#endif
if (ok)
set_nonblocking(true);
return ok;
}
int xxsocket::resolve(std::vector<endpoint>& endpoints, const char* hostname, unsigned short port, int socktype)
{
return resolve_i(
[&](const endpoint& ep) {
endpoints.push_back(ep);
return false;
},
hostname, port, AF_UNSPEC, AI_ALL, socktype);
}
int xxsocket::resolve_v4(std::vector<endpoint>& endpoints, const char* hostname, unsigned short port, int socktype)
{
return resolve_i(
[&](const endpoint& ep) {
endpoints.push_back(ep);
return false;
},
hostname, port, AF_INET, 0, socktype);
}
int xxsocket::resolve_v6(std::vector<endpoint>& endpoints, const char* hostname, unsigned short port, int socktype)
{
return resolve_i(
[&](const endpoint& ep) {
endpoints.push_back(ep);
return false;
},
hostname, port, AF_INET6, 0, socktype);
}
int xxsocket::resolve_v4to6(std::vector<endpoint>& endpoints, const char* hostname, unsigned short port, int socktype)
{
return xxsocket::resolve_i(
[&](const endpoint& ep) {
endpoints.push_back(ep);
return false;
},
hostname, port, AF_INET6, AI_V4MAPPED, socktype);
}
int xxsocket::resolve_tov6(std::vector<endpoint>& endpoints, const char* hostname, unsigned short port, int socktype)
{
return resolve_i(
[&](const endpoint& ep) {
endpoints.push_back(ep);
return false;
},
hostname, port, AF_INET6, AI_ALL | AI_V4MAPPED, socktype);
}
int xxsocket::getipsv(void)
{
int flags = 0;
xxsocket::traverse_local_address([&](const ip::endpoint& ep) -> bool {
switch (ep.af())
{
case AF_INET:
flags |= ipsv_ipv4;
break;
case AF_INET6:
flags |= ipsv_ipv6;
break;
}
return (flags == ipsv_dual_stack);
});
YASIO_LOG("xxsocket::getipsv: flags=%d", flags);
return flags;
}
void xxsocket::traverse_local_address(std::function<bool(const ip::endpoint&)> handler)
{
/* Only windows support use getaddrinfo to get local ip address(not loopback or linklocal),
Because nullptr same as "localhost": always return loopback address and at unix/linux the
gethostname always return "localhost"
*/
#if defined(_WIN32)
char hostname[256] = {0};
::gethostname(hostname, sizeof(hostname));
# if defined(_DEBUG)
YASIO_LOG("xxsocket::traverse_local_address: localhost=%s", hostname);
# endif
// ipv4 & ipv6
addrinfo hint, *ailist = nullptr;
::memset(&hint, 0x0, sizeof(hint));
endpoint ep;
int iret = getaddrinfo(hostname, nullptr, &hint, &ailist);
(void)iret;
if (ailist != nullptr)
{
for (auto aip = ailist; aip != nullptr; aip = aip->ai_next)
{
if (ep.as_is(aip))
{
YASIO_LOGV("xxsocket::traverse_local_address: ip=%s", ep.ip().c_str());
if (ep.is_global())
{
if (handler(ep))
break;
}
}
}
freeaddrinfo(ailist);
}
else
YASIO_LOGV("xxsocket::traverse_local_address fail with %s", xxsocket::gai_strerror(iret));
#else // unix like systems with <ifaddrs.h>
struct ifaddrs *ifaddr, *ifa;
/*
The value of ifa->ifa_name:
Android:
wifi: "w"
cellular: "r"
iOS:
wifi: "en0"
cellular: "pdp_ip0"
*/
if (yasio::getifaddrs(&ifaddr) == -1)
{
YASIO_LOG("xxsocket::traverse_local_address: getifaddrs fail!");
return;
}
endpoint ep;
/* Walk through linked list*/
for (ifa = ifaddr; ifa != nullptr; ifa = ifa->ifa_next)
{
if (ifa->ifa_addr == nullptr)
continue;
if (ep.as_is(ifa->ifa_addr))
{
YASIO_LOGV("xxsocket::traverse_local_address: ip=%s", ep.ip().c_str());
if (ep.is_global())
{
if (handler(ep))
break;
}
}
}
yasio::freeifaddrs(ifaddr);
#endif
}
xxsocket::xxsocket(void) : fd(invalid_socket) {}
xxsocket::xxsocket(socket_native_type h) : fd(h) {}
xxsocket::xxsocket(xxsocket&& right) YASIO__NOEXCEPT : fd(invalid_socket) { swap(right); }
xxsocket::xxsocket(int af, int type, int protocol) : fd(invalid_socket) { open(af, type, protocol); }
xxsocket::~xxsocket(void) { close(); }
xxsocket& xxsocket::operator=(socket_native_type handle) YASIO__NOEXCEPT
{
if (!this->is_open())
this->fd = handle;
return *this;
}
xxsocket& xxsocket::operator=(xxsocket&& right) YASIO__NOEXCEPT { return swap(right); }
xxsocket& xxsocket::swap(xxsocket& rhs)
{
std::swap(this->fd, rhs.fd);
return *this;
}
bool xxsocket::open(int af, int type, int protocol)
{
if (invalid_socket == this->fd)
this->fd = ::socket(af, type, protocol);
return is_open();
}
bool xxsocket::reopen(int af, int type, int protocol)
{
this->close();
return this->open(af, type, protocol);
}
#if defined(_WIN32)
bool xxsocket::open_ex(int af, int type, int protocol)
{
if (invalid_socket == this->fd)
{
this->fd = ::WSASocket(af, type, protocol, nullptr, 0, WSA_FLAG_OVERLAPPED);
DWORD dwBytes = 0;
if (nullptr == __accept_ex)
{
GUID guidAcceptEx = WSAID_ACCEPTEX;
(void)WSAIoctl(this->fd, SIO_GET_EXTENSION_FUNCTION_POINTER, &guidAcceptEx, sizeof(guidAcceptEx), &__accept_ex, sizeof(__accept_ex), &dwBytes, nullptr,
nullptr);
}
if (nullptr == __connect_ex)
{
GUID guidConnectEx = WSAID_CONNECTEX;
(void)WSAIoctl(this->fd, SIO_GET_EXTENSION_FUNCTION_POINTER, &guidConnectEx, sizeof(guidConnectEx), &__connect_ex, sizeof(__connect_ex), &dwBytes,
nullptr, nullptr);
}
if (nullptr == __get_accept_ex_sockaddrs)
{
GUID guidGetAcceptExSockaddrs = WSAID_GETACCEPTEXSOCKADDRS;
(void)WSAIoctl(this->fd, SIO_GET_EXTENSION_FUNCTION_POINTER, &guidGetAcceptExSockaddrs, sizeof(guidGetAcceptExSockaddrs), &__get_accept_ex_sockaddrs,
sizeof(__get_accept_ex_sockaddrs), &dwBytes, nullptr, nullptr);
}
}
return is_open();
}
bool xxsocket::accept_ex(SOCKET sockfd_listened, SOCKET sockfd_prepared, PVOID lpOutputBuffer, DWORD dwReceiveDataLength, DWORD dwLocalAddressLength,
DWORD dwRemoteAddressLength, LPDWORD lpdwBytesReceived, LPOVERLAPPED lpOverlapped)
{
return !!__accept_ex(sockfd_listened, sockfd_prepared, lpOutputBuffer, dwReceiveDataLength, dwLocalAddressLength, dwRemoteAddressLength, lpdwBytesReceived,
lpOverlapped);
}
bool xxsocket::connect_ex(SOCKET s, const struct sockaddr* name, int namelen, PVOID lpSendBuffer, DWORD dwSendDataLength, LPDWORD lpdwBytesSent,
LPOVERLAPPED lpOverlapped)
{
return !!__connect_ex(s, name, namelen, lpSendBuffer, dwSendDataLength, lpdwBytesSent, lpOverlapped);
}
void xxsocket::translate_sockaddrs(PVOID lpOutputBuffer, DWORD dwReceiveDataLength, DWORD dwLocalAddressLength, DWORD dwRemoteAddressLength,
sockaddr** LocalSockaddr, LPINT LocalSockaddrLength, sockaddr** RemoteSockaddr, LPINT RemoteSockaddrLength)
{
__get_accept_ex_sockaddrs(lpOutputBuffer, dwReceiveDataLength, dwLocalAddressLength, dwRemoteAddressLength, LocalSockaddr, LocalSockaddrLength,
RemoteSockaddr, RemoteSockaddrLength);
}
#endif
bool xxsocket::is_open(void) const { return this->fd != invalid_socket; }
socket_native_type xxsocket::native_handle(void) const { return this->fd; }
socket_native_type xxsocket::release_handle(void)
{
socket_native_type result = this->fd;
this->fd = invalid_socket;
return result;
}
int xxsocket::set_nonblocking(bool nonblocking) const { return set_nonblocking(this->fd, nonblocking); }
int xxsocket::set_nonblocking(socket_native_type s, bool nonblocking)
{
#if defined(_WIN32)
u_long argp = nonblocking;
return ::ioctlsocket(s, FIONBIO, &argp);
#else
int flags = ::fcntl(s, F_GETFL, 0);
return ::fcntl(s, F_SETFL, nonblocking ? (flags | O_NONBLOCK) : (flags & ~O_NONBLOCK));
#endif
}
int xxsocket::test_nonblocking() const { return xxsocket::test_nonblocking(this->fd); }
int xxsocket::test_nonblocking(socket_native_type s)
{
#if defined(_WIN32)
int r = 0;
unsigned char b[1];
r = xxsocket::recv(s, b, 0, 0);
if (r == 0)
return 1;
else if (r == -1 && GetLastError() == WSAEWOULDBLOCK)
return 0;
return -1; /* In case it is a connection socket (TCP) and it is not in connected state you will get here 10060 */
#else
int flags = ::fcntl(s, F_GETFL, 0);
return flags & O_NONBLOCK;
#endif
}
int xxsocket::bind(const char* addr, unsigned short port) const { return this->bind(endpoint(addr, port)); }
int xxsocket::bind(const endpoint& ep) const { return ::bind(this->fd, &ep, ep.len()); }
int xxsocket::bind_any(bool ipv6) const { return this->bind(endpoint(!ipv6 ? "0.0.0.0" : "::", 0)); }
int xxsocket::listen(int backlog) const { return ::listen(this->fd, backlog); }
xxsocket xxsocket::accept() const { return ::accept(this->fd, nullptr, nullptr); }
int xxsocket::accept_n(socket_native_type& new_sock) const
{
for (;;)
{
// Accept the waiting connection.
new_sock = ::accept(this->fd, nullptr, nullptr);
// Check if operation succeeded.
if (new_sock != invalid_socket)
{
xxsocket::set_nonblocking(new_sock, true);
return 0;
}
auto error = get_last_errno();
// Retry operation if interrupted by signal.
if (error == EINTR)
continue;
/* Operation failed.
** The error maybe EWOULDBLOCK, EAGAIN, ECONNABORTED, EPROTO,
** Simply Fall through to retry operation.
*/
return error;
}
}
int xxsocket::connect(const char* addr, u_short port) { return connect(endpoint(addr, port)); }
int xxsocket::connect(const endpoint& ep) { return xxsocket::connect(fd, ep); }
int xxsocket::connect(socket_native_type s, const char* addr, u_short port)
{
endpoint peer(addr, port);
return xxsocket::connect(s, peer);
}
int xxsocket::connect(socket_native_type s, const endpoint& ep) { return ::connect(s, &ep, ep.len()); }
int xxsocket::connect_n(const char* addr, u_short port, const std::chrono::microseconds& wtimeout) { return connect_n(ip::endpoint(addr, port), wtimeout); }
int xxsocket::connect_n(const endpoint& ep, const std::chrono::microseconds& wtimeout) { return this->connect_n(this->fd, ep, wtimeout); }
int xxsocket::connect_n(socket_native_type s, const endpoint& ep, const std::chrono::microseconds& wtimeout)
{
fd_set rset, wset;
int ret, error = 0;
set_nonblocking(s, true);
if ((ret = xxsocket::connect(s, ep)) < 0)
{
error = xxsocket::get_last_errno();
if (error != EINPROGRESS && error != EWOULDBLOCK)
return -1;
}
/* Do whatever we want while the connect is taking place. */
if (ret == 0)
goto done; /* connect completed immediately */
if (xxsocket::select(s, &rset, &wset, nullptr, wtimeout) <= 0)
error = xxsocket::get_last_errno();
else if ((FD_ISSET(s, &rset) || FD_ISSET(s, &wset)))
{ /* Everythings are ok */
socklen_t len = sizeof(error);
if (::getsockopt(s, SOL_SOCKET, SO_ERROR, (char*)&error, &len) < 0)
return (-1); /* Solaris pending error */
}
done:
if (error != 0)
{
::closesocket(s); /* just in case */
return (-1);
}
/* Since v3.31.2, we don't restore file status flags for unify behavior for all platforms */
// pitfall: because on win32, there is no way to test whether the s is non-blocking
// so, can't restore properly
return (0);
}
int xxsocket::connect_n(const endpoint& ep) { return xxsocket::connect_n(this->fd, ep); }
int xxsocket::connect_n(socket_native_type s, const endpoint& ep)
{
set_nonblocking(s, true);
return xxsocket::connect(s, ep);
}
int xxsocket::disconnect() const { return xxsocket::disconnect(this->fd); }
int xxsocket::disconnect(socket_native_type s)
{
#if defined(_WIN32)
sockaddr_storage addr_unspec{0};
return ::connect(s, (sockaddr*)&addr_unspec, sizeof(addr_unspec));
#else
# if defined(__MVS__)
sockaddr_storage addr_unspec{0};
# else
sockaddr addr_unspec{0};
addr_unspec.sa_family = AF_UNSPEC;
# endif
int ret, error;
for (;;)
{
ret = ::connect(s, &addr_unspec, sizeof(addr_unspec));
if (ret == 0)
return 0;
if ((error = xxsocket::get_last_errno()) == EINTR)
continue;
# if YASIO__OS_BSD_LIKE
/*
* From kernel source code of FreeBSD,NetBSD,OpenBSD,etc.
* The udp socket will be success disconnected by kernel function: `sodisconnect(upic_socket.c)`, then in the kernel, will continue try to
* connect with new sockaddr, but will failed with follow errno:
* a. EINVAL: addrlen mismatch
* b. EAFNOSUPPORT: family mismatch
* So, we just simply ignore them for the disconnect behavior.
*/
return (error == EAFNOSUPPORT || error == EINVAL) ? 0 : -1;
# else
return ret;
# endif
}
#endif
}
int xxsocket::send_n(const void* buf, int len, const std::chrono::microseconds& wtimeout, int flags)
{
return this->send_n(this->fd, buf, len, wtimeout, flags);
}
int xxsocket::send_n(socket_native_type s, const void* buf, int len, std::chrono::microseconds wtimeout, int flags)
{
int bytes_transferred = 0;
int n;
int error = 0;
xxsocket::set_nonblocking(s, true);
for (; bytes_transferred < len;)
{
// Try to transfer as much of the remaining data as possible.
// Since the socket is in non-blocking mode, this call will not
// block.
n = xxsocket::send(s, (const char*)buf + bytes_transferred, len - bytes_transferred, flags);
if (n > 0)
{
bytes_transferred += n;
continue;
}
// Check for possible blocking.
error = xxsocket::get_last_errno();
if (n == -1 && xxsocket::not_send_error(error))
{
// Wait upto <timeout> for the blocking to subside.
auto start = yasio::highp_clock();
int const rtn = handle_write_ready(s, wtimeout);
wtimeout -= std::chrono::microseconds(yasio::highp_clock() - start);
// Did select() succeed?
if (rtn != -1 && wtimeout.count() > 0)
{
// Blocking subsided in <timeout> period. Continue
// data transfer.
continue;
}
}
// Wait in select() timed out or other data transfer or
// select() failures.
break;
}
return bytes_transferred;
}
int xxsocket::recv_n(void* buf, int len, const std::chrono::microseconds& wtimeout, int flags) const
{
return this->recv_n(this->fd, buf, len, wtimeout, flags);
}
int xxsocket::recv_n(socket_native_type s, void* buf, int len, std::chrono::microseconds wtimeout, int flags)
{
int bytes_transferred = 0;
int n;
int error = 0;
xxsocket::set_nonblocking(s, true);
for (; bytes_transferred < len;)
{
// Try to transfer as much of the remaining data as possible.
// Since the socket is in non-blocking mode, this call will not
// block.
n = xxsocket::recv(s, static_cast<char*>(buf) + bytes_transferred, len - bytes_transferred, flags);
if (n > 0)
{
bytes_transferred += n;
continue;
}
// Check for possible blocking.
error = xxsocket::get_last_errno();
if (n == -1 && xxsocket::not_recv_error(error))
{
// Wait upto <timeout> for the blocking to subside.
auto start = yasio::highp_clock();
int const rtn = handle_read_ready(s, wtimeout);
wtimeout -= std::chrono::microseconds(yasio::highp_clock() - start);
// Did select() succeed?
if (rtn != -1 && wtimeout.count() > 0)
{
// Blocking subsided in <timeout> period. Continue
// data transfer.
continue;
}
}
// Wait in select() timed out or other data transfer or
// select() failures.
break;
}
return bytes_transferred;
}
int xxsocket::send(const void* buf, int len, int flags) const { return static_cast<int>(::send(this->fd, (const char*)buf, len, flags)); }
int xxsocket::send(socket_native_type s, const void* buf, int len, int flags) { return static_cast<int>(::send(s, (const char*)buf, len, flags)); }
int xxsocket::recv(void* buf, int len, int flags) const { return static_cast<int>(this->recv(this->fd, buf, len, flags)); }
int xxsocket::recv(socket_native_type s, void* buf, int len, int flags) { return static_cast<int>(::recv(s, (char*)buf, len, flags)); }
int xxsocket::sendto(const void* buf, int len, const endpoint& to, int flags) const
{
return static_cast<int>(::sendto(this->fd, (const char*)buf, len, flags, &to, to.len()));
}
int xxsocket::recvfrom(void* buf, int len, endpoint& from, int flags) const
{
socklen_t addrlen{sizeof(from)};
int n = static_cast<int>(::recvfrom(this->fd, (char*)buf, len, flags, &from, &addrlen));
from.len(addrlen);
return n;
}
int xxsocket::handle_write_ready(const std::chrono::microseconds& wtimeout) const { return handle_write_ready(this->fd, wtimeout); }
int xxsocket::handle_write_ready(socket_native_type s, const std::chrono::microseconds& wtimeout)
{
fd_set writefds;
return xxsocket::select(s, nullptr, &writefds, nullptr, wtimeout);
}
int xxsocket::handle_read_ready(const std::chrono::microseconds& wtimeout) const { return handle_read_ready(this->fd, wtimeout); }
int xxsocket::handle_read_ready(socket_native_type s, const std::chrono::microseconds& wtimeout)
{
fd_set readfds;
return xxsocket::select(s, &readfds, nullptr, nullptr, wtimeout);
}
int xxsocket::select(socket_native_type s, fd_set* readfds, fd_set* writefds, fd_set* exceptfds, std::chrono::microseconds wtimeout)
{
int n = 0;
for (;;)
{
reregister_descriptor(s, readfds);
reregister_descriptor(s, writefds);
reregister_descriptor(s, exceptfds);
timeval waitd_tv = {static_cast<decltype(timeval::tv_sec)>(wtimeout.count() / std::micro::den),
static_cast<decltype(timeval::tv_usec)>(wtimeout.count() % std::micro::den)};
long long start = highp_clock();
n = ::select(static_cast<int>(s + 1), readfds, writefds, exceptfds, &waitd_tv);
wtimeout -= std::chrono::microseconds(highp_clock() - start);
if (n < 0 && xxsocket::get_last_errno() == EINTR)
{
if (wtimeout.count() > 0)
continue;
n = 0;
}
if (n == 0)
xxsocket::set_last_errno(ETIMEDOUT);
break;
}
return n;
}
void xxsocket::reregister_descriptor(socket_native_type s, fd_set* fds)
{
if (fds)
{
FD_ZERO(fds);
FD_SET(s, fds);
}
}
endpoint xxsocket::local_endpoint(void) const { return local_endpoint(this->fd); }
endpoint xxsocket::local_endpoint(socket_native_type fd)
{
endpoint ep;
socklen_t socklen = sizeof(ep);
getsockname(fd, &ep, &socklen);
ep.len(socklen);
return ep;
}
endpoint xxsocket::peer_endpoint(void) const { return peer_endpoint(this->fd); }
endpoint xxsocket::peer_endpoint(socket_native_type fd)
{
endpoint ep;
socklen_t socklen = sizeof(ep);
getpeername(fd, &ep, &socklen);
ep.len(socklen);
return ep;
}
int xxsocket::set_keepalive(int flag, int idle, int interval, int probes) { return set_keepalive(this->fd, flag, idle, interval, probes); }
int xxsocket::set_keepalive(socket_native_type s, int flag, int idle, int interval, int probes)
{
#if defined(_WIN32)
tcp_keepalive buffer_in;
buffer_in.onoff = flag;
buffer_in.keepalivetime = idle * 1000;
buffer_in.keepaliveinterval = interval * 1000;
return WSAIoctl(s, SIO_KEEPALIVE_VALS, &buffer_in, sizeof(buffer_in), nullptr, 0, (DWORD*)&probes, nullptr, nullptr);
#else
if (set_optval(s, SOL_SOCKET, SO_KEEPALIVE, flag) != 0)
return -1;
if (set_optval(s, IPPROTO_TCP, TCP_KEEPIDLE, idle) != 0)
return -1;
if (set_optval(s, IPPROTO_TCP, TCP_KEEPINTVL, interval) != 0)
return -1;
return set_optval(s, IPPROTO_TCP, TCP_KEEPCNT, probes);
#endif
}
void xxsocket::reuse_address(bool reuse)
{
int optval = reuse ? 1 : 0;
// All operating systems have 'SO_REUSEADDR'
this->set_optval(SOL_SOCKET, SO_REUSEADDR, optval);
#if defined(SO_REUSEPORT) // macos,ios,linux,android
this->set_optval(SOL_SOCKET, SO_REUSEPORT, optval);
#endif
}
void xxsocket::exclusive_address(bool exclusive)
{
#if defined(SO_EXCLUSIVEADDRUSE)
this->set_optval(SOL_SOCKET, SO_EXCLUSIVEADDRUSE, exclusive ? 1 : 0);
#elif defined(SO_EXCLBIND)
this->set_optval(SOL_SOCKET, SO_EXCLBIND, exclusive ? 1 : 0);
#endif
}
xxsocket::operator socket_native_type(void) const { return this->fd; }
int xxsocket::shutdown(int how) const { return ::shutdown(this->fd, how); }
void xxsocket::close(int shut_how)
{
if (is_open())
{
#if !defined(__EMSCRIPTEN__)
if (shut_how >= 0)
::shutdown(this->fd, shut_how);
#endif
::closesocket(this->fd);
this->fd = invalid_socket;
}
}
unsigned int xxsocket::tcp_rtt() const { return xxsocket::tcp_rtt(this->fd); }
unsigned int xxsocket::tcp_rtt(socket_native_type s)
{
#if defined(_WIN32)
# if defined(NTDDI_WIN10_RS2) && NTDDI_VERSION >= NTDDI_WIN10_RS2
TCP_INFO_v0 info;
DWORD tcpi_ver = 0, bytes_transferred = 0;
int status = WSAIoctl(s, SIO_TCP_INFO,
(LPVOID)&tcpi_ver, // lpvInBuffer pointer to a DWORD, version of tcp info
(DWORD)sizeof(tcpi_ver), // size, in bytes, of the input buffer
(LPVOID)&info, // pointer to a TCP_INFO_v0 structure
(DWORD)sizeof(info), // size of the output buffer
(LPDWORD)&bytes_transferred, // number of bytes returned
(LPWSAOVERLAPPED) nullptr, // OVERLAPPED structure
(LPWSAOVERLAPPED_COMPLETION_ROUTINE) nullptr);
/*
info.RttUs: The current estimated round-trip time for the connection, in microseconds.
info.MinRttUs: The minimum sampled round trip time, in microseconds.
*/
if (status == 0)
return info.RttUs;
# endif
#elif defined(__linux__) || defined(__FreeBSD__) || defined(__NetBSD__)
struct tcp_info info;
if (0 == xxsocket::get_optval(s, IPPROTO_TCP, TCP_INFO, info))
return info.tcpi_rtt;
#elif defined(__APPLE__)
struct tcp_connection_info info;
/*
info.tcpi_srtt: average RTT in ms
info.tcpi_rttcur: most recent RTT in ms
*/
if (0 == xxsocket::get_optval(s, IPPROTO_TCP, TCP_CONNECTION_INFO, info))
return info.tcpi_srtt * std::milli::den;
#endif
return 0;
}
void xxsocket::init_ws32_lib(void) {}
int xxsocket::get_last_errno(void)
{
#if defined(_WIN32)
return ::WSAGetLastError();
#else
return errno;
#endif
}
void xxsocket::set_last_errno(int error)
{
#if defined(_WIN32)
::WSASetLastError(error);
#else
errno = error;
#endif
}
bool xxsocket::not_send_error(int error) { return (error == EWOULDBLOCK || error == EAGAIN || error == EINTR || error == ENOBUFS); }
bool xxsocket::not_recv_error(int error) { return (error == EWOULDBLOCK || error == EAGAIN || error == EINTR); }
const char* xxsocket::strerror(int error)
{
#if defined(_WIN32)
static char error_msg[256];
return xxsocket::strerror_r(error, error_msg, sizeof(error_msg));
#else
return ::strerror(error);
#endif
}
const char* xxsocket::strerror_r(int error, char* buf, size_t buflen)
{
#if defined(_WIN32)
ZeroMemory(buf, buflen);
::FormatMessageA(FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS | FORMAT_MESSAGE_MAX_WIDTH_MASK /* remove line-end charactors \r\n */, NULL,
error, MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US), // english language
buf, static_cast<DWORD>(buflen), nullptr);
return buf;
#else
// XSI-compliant return int not const char*, refer to: https://linux.die.net/man/3/strerror_r
auto YASIO__UNUSED ret = ::strerror_r(error, buf, buflen);
return buf;
#endif
}
const char* xxsocket::gai_strerror(int error)
{
#if defined(_WIN32)
return xxsocket::strerror(error);
#else
return ::gai_strerror(error);
#endif
}
} // namespace inet
} // namespace yasio
// initialize win32 socket library
#ifdef _WIN32
namespace
{
struct ws2_32_gc {
ws2_32_gc(void)
{
WSADATA dat = {0};
(void)WSAStartup(0x0202, &dat);
}
~ws2_32_gc(void) { WSACleanup(); }
};
ws2_32_gc __ws32_lib_gc;
} // namespace
#endif
#if defined(_MSC_VER)
# pragma warning(pop)
#endif
#endif