9 #if ((XTD_OS_WINDOWS | XTD_OS_MINGW) & XTD_OS)
12 static_assert(_WIN32_WINNT >= 0x600,
"unsupported target Windows version");
16 #if (XTD_OS_MINGW & XTD_OS)
20 #if ((XTD_OS_LINUX | XTD_OS_CYGWIN | XTD_OS_MSYS) & XTD_OS)
21 #include <sys/socket.h>
22 #include <sys/ioctl.h>
23 #include <netinet/in.h>
24 #include <netinet/udp.h>
25 #include <netinet/tcp.h>
26 #include <arpa/inet.h>
31 #include <type_traits>
39 #if (XTD_COMPILER_MSVC & XTD_COMPILER)
40 #pragma comment(lib, "ws2_32")
49 #if ((XTD_OS_LINUX | XTD_OS_CYGWIN | XTD_OS_MSYS) & XTD_OS)
51 #define closesocket close
52 #define ioctlsocket ioctl
53 #elif (XTD_OS_WINDOWS & XTD_OS)
54 using POLLFD = pollfd;
55 #elif (XTD_OS_MINGW & XTD_OS)
56 using POLLFD = WSAPOLLFD;
69 template <
typename _ReturnT,
typename _ExpressionT>
70 inline static _ReturnT _throw_if(
const xtd::source_location& source, _ReturnT ret, _ExpressionT exp,
const char* expstr){
81 template <
typename _Ty,
int level,
int optname>
class socket_option{
83 using value_type = _Ty;
84 static value_type
get(SOCKET s){
87 std::remove_pointer<typename get_parameter<4, decltype(getsockopt)>::type>::type iSize =
sizeof(value_type);
91 static void set(SOCKET s, value_type newval){
92 socket::exception::throw_if(setsockopt(s, level, optname, reinterpret_cast<char*>(&newval),
sizeof(newval)), [](
int i){
return (i<0); });
95 template <
int level,
int optname>
class socket_option<std::string, level, optname>{
97 using value_type = std::string;
98 static value_type
get(SOCKET s){
104 static void set(SOCKET s, value_type newval){
105 socket::exception::throw_if(setsockopt(s, level, optname, reinterpret_cast<char*>(&newval),
sizeof(newval)), [](
int i){
return (i<0); });
124 sin_family = AF_INET;
125 sin_addr.s_addr = inet_addr(sIP);
126 sin_port = htons(iPort);
132 if (&src !=
this) memcpy(
this, &src,
sizeof(
ipv4address));
142 TODO(
"Implement proper ipv6 address");
160 NOTE(
"add more socket protocols as needed defined in netinet/in.h")
163 TODO("Refactor out socket::serializer into something more universal")
167 #if ((XTD_OS_WINDOWS | XTD_OS_MINGW) & XTD_OS)
172 class winsock_initializer
sealed{
175 static winsock_initializer &
get(){
176 static winsock_initializer oInit;
179 winsock_initializer(
const winsock_initializer&) =
delete;
180 winsock_initializer& operator=(
const winsock_initializer&) =
delete;
182 winsock_initializer(){
185 if (oData.wVersion != 0x202)
throw crt_exception(here(),
"Invalid winsock version");
188 ~winsock_initializer(){ WSACleanup(); }
194 template<
typename _AddressT, socket_type, socket_protocol,
template<
class>
class ... _Policies>
class socket_base;
197 template<
typename _AddressT, socket_type _ST, socket_protocol _PR,
template<
class>
class _HeadT,
template<
class>
class ..._TailT>
198 class socket_base<_AddressT, _ST, _PR, _HeadT, _TailT...> :
public _HeadT<socket_base<_AddressT, _ST, _PR, _TailT...> >{
200 template<
typename ... _ArgsT>
explicit socket_base(_ArgsT &&...oArgs) : _HeadT<socket_base<_AddressT, _ST, _PR, _TailT...> >(std::forward<_ArgsT>(oArgs)...){}
207 template<
typename _AddressT, socket_type _SocketT, socket_protocol _Protocol>
208 class socket_base<_AddressT, _SocketT, _Protocol>{
230 #if ((XTD_OS_WINDOWS | XTD_OS_MINGW) & XTD_OS)
231 winsock_initializer::get();
233 _Socket =
xtd::crt_exception::throw_if(::socket(address_type::address_family, (
int)type, (
int)protocol), [](SOCKET s){
return static_cast<SOCKET
>(-1) == s; });
243 socket_base(
const socket_base&) =
delete;
247 socket_base& operator=(
const socket_base&) =
delete;
255 std::swap(_Socket, src._Socket);
263 operator SOCKET()
const{
return _Socket; }
270 template <
typename _Ty>
void write(
const _Ty& data){
278 template <
typename _Ty>
void read(_Ty& data){
286 template <
typename _Ty> _Ty
read(){
299 ::closesocket(_Socket);
308 uint32_t val = (blocking ? 0 : 1);
325 template <
typename _SuperT>
339 oPoll.events = POLLIN | POLLOUT;
340 oPoll.fd = _SuperT::_Socket;
345 if (oPoll.revents & POLLERR){
348 if (oPoll.revents & POLLHUP){
351 if (oPoll.revents & POLLIN){
354 if (oPoll.revents & POLLOUT){
359 template<
typename ... _ArgTs>
362 #if ((XTD_OS_WINDOWS | XTD_OS_MINGW) & XTD_OS)
363 int poll(POLLFD *ufds,
unsigned int nfds,
int timeout){ return ::WSAPoll(ufds, nfds, timeout); }
369 template <
typename _SuperT>
374 template<
typename ... _ArgTs>
378 void bind(
const typename _SuperT::address_type& addr){
379 exception::throw_if(::
bind(_SuperT::_Socket, reinterpret_cast<const sockaddr*>(&addr),
sizeof(
typename _SuperT::address_type)), [](
int i){
return i < 0; });
385 template <
typename _SuperT>
390 template<
typename ... _ArgTs>
394 void connect(
const typename _SuperT::address_type& addr){
395 exception::throw_if(::
connect(_SuperT::_Socket, reinterpret_cast<const sockaddr*>(&addr),
sizeof(
typename _SuperT::address_type)), [](
int i){
return i < 0; });
401 template <
typename _SuperT>
406 template<
typename ... _ArgTs>
415 template <
typename _ReturnT>
422 template <
typename _SuperT>
427 template<
typename ... _ArgTs>
431 bool keep_alive()
const{
return (_::socket_option<int, SOL_SOCKET, SO_KEEPALIVE>::get(_SuperT::_Socket) ?
true :
false); }
433 void keep_alive(
bool newval){ _::socket_option<int, SOL_SOCKET, SO_KEEPALIVE>::set(_SuperT::_Socket, newval); }
434 TODO(
"Add more SOL_SOCKET options");
438 template <
typename _SuperT>
443 template<
typename ... _ArgTs>
445 #if ((XTD_OS_MINGW | XTD_OS_WINDOWS) & XTD_OS)
446 bool dont_fragment()
const{
return (_::socket_option<int, IPPROTO_IP, IP_DONTFRAGMENT>::get(_SuperT::_Socket) ?
true :
false); }
449 void dont_fragment(
bool newval){ _::socket_option<int, IPPROTO_IP, IP_DONTFRAGMENT>::set(_SuperT::_Socket, newval); }
451 TODO(
"Add more IPPROTO_IP options");
455 template <
typename _SuperT>
460 template<
typename ... _ArgTs>
464 bool no_delay()
const{
return (_::socket_option<int, IPPROTO_TCP, TCP_NODELAY>::get(_SuperT::_Socket) ?
true :
false); }
466 void no_delay(
bool newval){ _::socket_option<int, IPPROTO_TCP, TCP_NODELAY>::set(_SuperT::_Socket, newval); }
467 TODO(
"Add more IPPROTO_TCP options");
471 template <
typename _SuperT>
476 template<
typename ... _ArgTs>
479 #if ((XTD_OS_MINGW | XTD_OS_WINDOWS) & XTD_OS)
480 bool no_checksum()
const{
return (_::socket_option<int, IPPROTO_UDP, UDP_NOCHECKSUM>::get(_SuperT::_Socket) ?
true :
false); }
483 void no_checksum(
bool newval){ _::socket_option<int, IPPROTO_UDP, UDP_NOCHECKSUM>::set(_SuperT::_Socket, newval); }
485 TODO(
"Add more IPPROTO_UDP options");
490 template <
typename _SuperT>
494 template<
typename ... _ArgTs>
513 FD_SET((SOCKET)*
this, &fdRead);
514 FD_SET((SOCKET)*
this, &fdWrite);
515 FD_SET((SOCKET)*
this, &fdErr);
516 tv.tv_sec = WaitMS / 1000;
518 tv.tv_usec = WaitMS / 1000;
519 #if (XTD_OS_MINGW & XTD_OS)
527 if (FD_ISSET((SOCKET)*
this, &fdErr)){
530 if (FD_ISSET((SOCKET)*
this, &fdRead)){
533 if (FD_ISSET((SOCKET)*
this, &fdWrite)){
541 TODO(
"Get rid of these")
542 template <typename _Ty>
546 template <
typename _SocketT>
547 static void write(_SocketT& oSocket,
const _Ty& src){
548 static_assert(std::is_pod<_Ty>::value,
"no acceptable specialization for type");
549 exception::throw_if(::send(oSocket, reinterpret_cast<const char*>(&src),
sizeof(_Ty), 0), [](
int i){
return i <= 0; });
552 template <
typename _SocketT>
553 static void read(_SocketT& oSocket, _Ty& src){
554 static_assert(std::is_pod<_Ty>::value,
"no acceptable specialization for type");
555 exception::throw_if(::recv(oSocket, reinterpret_cast<char*>(&src),
sizeof(_Ty), 0), [](
int i){
return i <= 0; });
560 template <
typename _Ty>
561 class NON_POD_Vector_Serializer{
564 template <
typename _SocketT>
565 static void write(_SocketT& oSocket,
const std::vector<_Ty>& src){
566 serializer<typename std::vector<_Ty>::size_type>::write(oSocket, src.size());
567 for (
const auto & oItem : src){
568 serializer<_Ty>::write(oSocket, oItem);
572 template <
typename _SocketT>
573 static void read(_SocketT& oSocket, std::vector<_Ty>& src){
574 typename std::vector<_Ty>::size_type count;
575 serializer<typename std::vector<_Ty>::size_type>::read(oSocket, count);
576 for (; count; --count){
578 serializer<_Ty>::read(oSocket, newval);
579 src.push_back(newval);
585 template <
typename _Ty>
586 class POD_Vector_Serializer{
589 template <
typename _SocketT >
590 static void write(_SocketT& oSocket,
const std::vector<_Ty>& src){
591 serializer<typename std::vector<_Ty>::size_type>::write(oSocket, src.size());
592 send(oSocket, reinterpret_cast<const char*>(&src[0]), (
int)(
sizeof(_Ty) * src.size()), 0);
595 template <
typename _SocketT>
596 static void read(_SocketT& oSocket, std::vector<_Ty>& src){
597 typename std::vector<_Ty>::size_type count;
598 serializer<typename std::vector<_Ty>::size_type>::read(oSocket, count);
600 recv(oSocket, reinterpret_cast<char*>(&src[0]), (
int)(
sizeof(_Ty) * count), 0);
606 template <
typename _Ty>
607 class serializer<std::vector<_Ty>> :
public std::conditional<std::is_pod<_Ty>::value, POD_Vector_Serializer<_Ty>, NON_POD_Vector_Serializer<_Ty>>::type{};
609 using ipv4_tcp_stream = socket_base<ipv4address, socket_type::stream, socket_protocol::tcp, socket_options, ip_options, tcp_options, connectable_socket, bindable_socket, listening_socket, selectable_socket>;
612 using ipv4_udp_socket = socket_base<ipv4address, socket_type::datagram, socket_protocol::udp, socket_options, ip_options, udp_options>;
ipv6 Internet Protocol v6
bindable_socket(_ArgTs &&...oArgs)
ctor
exception(const exception &ex)
constructors
void close()
Closes the open socket.
Serializes data on a socket.
exception(exception &&ex)
constructors
bool select(int WaitMS)
begin the select to wait for an event or timeout
callback< void()> disconnect_event
callback event fires when socket becomes disconnected
tcp Transmission Control Protocol
IP based socket properties.
IPv6 address wrapper around sockaddr_in6.
callback< void()> read_event
callback event fires when data is ready to read data
_Ty read()
reads data from the connected socket
std::shared_ptr< socket_base > shared_ptr
typedefs
exception(const source_location &loc, const xtd::string &swhat)
constructors
callback< void()> error_event
callback event fires when an error occurs
socket_base< ipv4address, socket_type::stream, socket_protocol::tcp, socket_options, ip_options, tcp_options, connectable_socket, bindable_socket, listening_socket, selectable_socket > ipv4_tcp_stream
General purpose IPV4 client and server socket type.
void listen(int Backlog=SOMAXCONN)
begins listening on the socket
selectable_socket(_ArgTs &&...oArgs)
ctor
polling_socket(_ArgTs &&...oArgs)
ctor
void connect(const typename _SuperT::address_type &addr)
initiates connection to a socket
std::unique_ptr< socket_base > unique_ptr
typedefs
xtd::callback< void()> onWrite
callback event fired when socket is ready to write
_ReturnT accept()
accepts an incoming connection request
Server side listening behavior.
socket_type
socket communication styles
Server side binding behavior.
void bind(const typename _SuperT::address_type &addr)
binds the socket to an address and port
_AddressT address_type
typedefs
connectable_socket(_ArgTs &&...oArgs)
ctor
socket_base(socket_base &&src)
constructors
static const int address_family
ipv6 address family
TCP based socket properties.
ipv4address(const char *sIP, uint16_t iPort)
constructor
static const int address_family
ipv4 address familt
void read(_Ty &data)
reads data from the connected socket
udp User Datagram Protocol
socket_base()
constructors
bool keep_alive() const
gets the SO_KEEPALIVE property
xtd::callback< void()> onError
callback event fired when a socket error occurs
Async IO select behavior.
specializations of std::basic_string for advanced and common string handling
host, target and build configurations and settings Various components are purpose built for specific ...
Client side connecting behavior.
#define throw_if(_test, _expression)
Simplifies use of exception::_throw_if.
void write(const _Ty &data)
writes data to the connected socket
ip_options(_ArgTs &&...oArgs)
ctor
datagram unreliable connectionless broadcast
generic and special purpose exceptions
icmp Internet Control Message Protocol
xtd::callback< void()> onRead
callback event fired when data is ready to be read
Single producer - multiple subscriber callback.
socket_base(SOCKET newval)
constructors
socket_base & operator=(socket_base &&src)
move assignment
socket_protocol
IP based protocols.
tcp_options(_ArgTs &&...oArgs)
ctor
SOCKET _Socket
OS/CRT inner SOCKET that is being managed by this wrapper.
Represents an socket error.
stream reliable FIFO stream on a remote socket
void poll(int Timeout)
begins polling the socket for events for a period of Timeout
listening_socket(_ArgTs &&...oArgs)
ctor
udp_options(_ArgTs &&...oArgs)
ctor
bool no_delay() const
gets the TCP_NODELAY property
void no_delay(bool newval)
sets the TCP_NODELAY property
IPv4 address wrapper around sockaddr_in.
void set_blocking(bool blocking)
sets the blocking mode of the socket
bool is_valid() const
test if the socket is valid
callback< void()> write_event
callback event fires when socket is ready to write data
socket_options(_ArgTs &&...oArgs)
ctor
socket_base< ipv4address, socket_type::datagram, socket_protocol::udp, socket_options, ip_options, udp_options > ipv4_udp_socket
General purpose UDP socket type.
void keep_alive(bool newval)
sets the SO_KEEPALIVE property
Contains information about the location of source code Used in error reporting and logging...
c++ wrapper around legacy errno based errors from the CRT