3978 lines
		
	
	
		
			108 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
		
		
			
		
	
	
			3978 lines
		
	
	
		
			108 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
|  | // | ||
|  | // detail/impl/socket_ops.ipp | ||
|  | // ~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||
|  | // | ||
|  | // Copyright (c) 2003-2023 Christopher M. Kohlhoff (chris at kohlhoff dot com) | ||
|  | // | ||
|  | // Distributed under the Boost Software License, Version 1.0. (See accompanying | ||
|  | // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) | ||
|  | // | ||
|  | 
 | ||
|  | #ifndef ASIO_DETAIL_SOCKET_OPS_IPP | ||
|  | #define ASIO_DETAIL_SOCKET_OPS_IPP | ||
|  | 
 | ||
|  | #if defined(_MSC_VER) && (_MSC_VER >= 1200) | ||
|  | # pragma once | ||
|  | #endif // defined(_MSC_VER) && (_MSC_VER >= 1200) | ||
|  | 
 | ||
|  | #include "asio/detail/config.hpp" | ||
|  | 
 | ||
|  | #include <cctype> | ||
|  | #include <cstdio> | ||
|  | #include <cstdlib> | ||
|  | #include <cstring> | ||
|  | #include <cerrno> | ||
|  | #include <new> | ||
|  | #include "asio/detail/assert.hpp" | ||
|  | #include "asio/detail/socket_ops.hpp" | ||
|  | #include "asio/error.hpp" | ||
|  | 
 | ||
|  | #if defined(ASIO_WINDOWS_RUNTIME) | ||
|  | # include <codecvt> | ||
|  | # include <locale> | ||
|  | # include <string> | ||
|  | #endif // defined(ASIO_WINDOWS_RUNTIME) | ||
|  | 
 | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) \ | ||
|  |   || defined(__MACH__) && defined(__APPLE__) | ||
|  | # if defined(ASIO_HAS_PTHREADS) | ||
|  | #  include <pthread.h> | ||
|  | # endif // defined(ASIO_HAS_PTHREADS) | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |        // || defined(__MACH__) && defined(__APPLE__) | ||
|  | 
 | ||
|  | #include "asio/detail/push_options.hpp" | ||
|  | 
 | ||
|  | namespace asio { | ||
|  | namespace detail { | ||
|  | namespace socket_ops { | ||
|  | 
 | ||
|  | #if !defined(ASIO_WINDOWS_RUNTIME) | ||
|  | 
 | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | struct msghdr { int msg_namelen; }; | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | 
 | ||
|  | #if defined(__hpux) | ||
|  | // HP-UX doesn't declare these functions extern "C", so they are declared again | ||
|  | // here to avoid linker errors about undefined symbols. | ||
|  | extern "C" char* if_indextoname(unsigned int, char*); | ||
|  | extern "C" unsigned int if_nametoindex(const char*); | ||
|  | #endif // defined(__hpux) | ||
|  | 
 | ||
|  | #endif // !defined(ASIO_WINDOWS_RUNTIME) | ||
|  | 
 | ||
|  | inline void clear_last_error() | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   WSASetLastError(0); | ||
|  | #else | ||
|  |   errno = 0; | ||
|  | #endif | ||
|  | } | ||
|  | 
 | ||
|  | #if !defined(ASIO_WINDOWS_RUNTIME) | ||
|  | 
 | ||
|  | inline void get_last_error( | ||
|  |     asio::error_code& ec, bool is_error_condition) | ||
|  | { | ||
|  |   if (!is_error_condition) | ||
|  |   { | ||
|  |     asio::error::clear(ec); | ||
|  |   } | ||
|  |   else | ||
|  |   { | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |     ec = asio::error_code(WSAGetLastError(), | ||
|  |         asio::error::get_system_category()); | ||
|  | #else | ||
|  |     ec = asio::error_code(errno, | ||
|  |         asio::error::get_system_category()); | ||
|  | #endif | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | template <typename SockLenType> | ||
|  | inline socket_type call_accept(SockLenType msghdr::*, | ||
|  |     socket_type s, void* addr, std::size_t* addrlen) | ||
|  | { | ||
|  |   SockLenType tmp_addrlen = addrlen ? (SockLenType)*addrlen : 0; | ||
|  |   socket_type result = ::accept(s, | ||
|  |       static_cast<socket_addr_type*>(addr), | ||
|  |       addrlen ? &tmp_addrlen : 0); | ||
|  |   if (addrlen) | ||
|  |     *addrlen = (std::size_t)tmp_addrlen; | ||
|  |   return result; | ||
|  | } | ||
|  | 
 | ||
|  | socket_type accept(socket_type s, void* addr, | ||
|  |     std::size_t* addrlen, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return invalid_socket; | ||
|  |   } | ||
|  | 
 | ||
|  |   socket_type new_s = call_accept(&msghdr::msg_namelen, s, addr, addrlen); | ||
|  |   get_last_error(ec, new_s == invalid_socket); | ||
|  |   if (new_s == invalid_socket) | ||
|  |     return new_s; | ||
|  | 
 | ||
|  | #if defined(__MACH__) && defined(__APPLE__) || defined(__FreeBSD__) | ||
|  |   int optval = 1; | ||
|  |   int result = ::setsockopt(new_s, SOL_SOCKET, | ||
|  |       SO_NOSIGPIPE, &optval, sizeof(optval)); | ||
|  |   get_last_error(ec, result != 0); | ||
|  |   if (result != 0) | ||
|  |   { | ||
|  |     ::close(new_s); | ||
|  |     return invalid_socket; | ||
|  |   } | ||
|  | #endif | ||
|  | 
 | ||
|  |   asio::error::clear(ec); | ||
|  |   return new_s; | ||
|  | } | ||
|  | 
 | ||
|  | socket_type sync_accept(socket_type s, state_type state, | ||
|  |     void* addr, std::size_t* addrlen, asio::error_code& ec) | ||
|  | { | ||
|  |   // Accept a socket. | ||
|  |   for (;;) | ||
|  |   { | ||
|  |     // Try to complete the operation without blocking. | ||
|  |     socket_type new_socket = socket_ops::accept(s, addr, addrlen, ec); | ||
|  | 
 | ||
|  |     // Check if operation succeeded. | ||
|  |     if (new_socket != invalid_socket) | ||
|  |       return new_socket; | ||
|  | 
 | ||
|  |     // Operation failed. | ||
|  |     if (ec == asio::error::would_block | ||
|  |         || ec == asio::error::try_again) | ||
|  |     { | ||
|  |       if (state & user_set_non_blocking) | ||
|  |         return invalid_socket; | ||
|  |       // Fall through to retry operation. | ||
|  |     } | ||
|  |     else if (ec == asio::error::connection_aborted) | ||
|  |     { | ||
|  |       if (state & enable_connection_aborted) | ||
|  |         return invalid_socket; | ||
|  |       // Fall through to retry operation. | ||
|  |     } | ||
|  | #if defined(EPROTO) | ||
|  |     else if (ec.value() == EPROTO) | ||
|  |     { | ||
|  |       if (state & enable_connection_aborted) | ||
|  |         return invalid_socket; | ||
|  |       // Fall through to retry operation. | ||
|  |     } | ||
|  | #endif // defined(EPROTO) | ||
|  |     else | ||
|  |       return invalid_socket; | ||
|  | 
 | ||
|  |     // Wait for socket to become ready. | ||
|  |     if (socket_ops::poll_read(s, 0, -1, ec) < 0) | ||
|  |       return invalid_socket; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | #if defined(ASIO_HAS_IOCP) | ||
|  | 
 | ||
|  | void complete_iocp_accept(socket_type s, void* output_buffer, | ||
|  |     DWORD address_length, void* addr, std::size_t* addrlen, | ||
|  |     socket_type new_socket, asio::error_code& ec) | ||
|  | { | ||
|  |   // Map non-portable errors to their portable counterparts. | ||
|  |   if (ec.value() == ERROR_NETNAME_DELETED) | ||
|  |     ec = asio::error::connection_aborted; | ||
|  | 
 | ||
|  |   if (!ec) | ||
|  |   { | ||
|  |     // Get the address of the peer. | ||
|  |     if (addr && addrlen) | ||
|  |     { | ||
|  |       LPSOCKADDR local_addr = 0; | ||
|  |       int local_addr_length = 0; | ||
|  |       LPSOCKADDR remote_addr = 0; | ||
|  |       int remote_addr_length = 0; | ||
|  |       GetAcceptExSockaddrs(output_buffer, 0, address_length, | ||
|  |           address_length, &local_addr, &local_addr_length, | ||
|  |           &remote_addr, &remote_addr_length); | ||
|  |       if (static_cast<std::size_t>(remote_addr_length) > *addrlen) | ||
|  |       { | ||
|  |         ec = asio::error::invalid_argument; | ||
|  |       } | ||
|  |       else | ||
|  |       { | ||
|  |         using namespace std; // For memcpy. | ||
|  |         memcpy(addr, remote_addr, remote_addr_length); | ||
|  |         *addrlen = static_cast<std::size_t>(remote_addr_length); | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     // Need to set the SO_UPDATE_ACCEPT_CONTEXT option so that getsockname | ||
|  |     // and getpeername will work on the accepted socket. | ||
|  |     SOCKET update_ctx_param = s; | ||
|  |     socket_ops::state_type state = 0; | ||
|  |     socket_ops::setsockopt(new_socket, state, | ||
|  |           SOL_SOCKET, SO_UPDATE_ACCEPT_CONTEXT, | ||
|  |           &update_ctx_param, sizeof(SOCKET), ec); | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | #else // defined(ASIO_HAS_IOCP) | ||
|  | 
 | ||
|  | bool non_blocking_accept(socket_type s, | ||
|  |     state_type state, void* addr, std::size_t* addrlen, | ||
|  |     asio::error_code& ec, socket_type& new_socket) | ||
|  | { | ||
|  |   for (;;) | ||
|  |   { | ||
|  |     // Accept the waiting connection. | ||
|  |     new_socket = socket_ops::accept(s, addr, addrlen, ec); | ||
|  | 
 | ||
|  |     // Check if operation succeeded. | ||
|  |     if (new_socket != invalid_socket) | ||
|  |       return true; | ||
|  | 
 | ||
|  |     // Retry operation if interrupted by signal. | ||
|  |     if (ec == asio::error::interrupted) | ||
|  |       continue; | ||
|  | 
 | ||
|  |     // Operation failed. | ||
|  |     if (ec == asio::error::would_block | ||
|  |         || ec == asio::error::try_again) | ||
|  |     { | ||
|  |       // Fall through to retry operation. | ||
|  |     } | ||
|  |     else if (ec == asio::error::connection_aborted) | ||
|  |     { | ||
|  |       if (state & enable_connection_aborted) | ||
|  |         return true; | ||
|  |       // Fall through to retry operation. | ||
|  |     } | ||
|  | #if defined(EPROTO) | ||
|  |     else if (ec.value() == EPROTO) | ||
|  |     { | ||
|  |       if (state & enable_connection_aborted) | ||
|  |         return true; | ||
|  |       // Fall through to retry operation. | ||
|  |     } | ||
|  | #endif // defined(EPROTO) | ||
|  |     else | ||
|  |       return true; | ||
|  | 
 | ||
|  |     return false; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | #endif // defined(ASIO_HAS_IOCP) | ||
|  | 
 | ||
|  | template <typename SockLenType> | ||
|  | inline int call_bind(SockLenType msghdr::*, | ||
|  |     socket_type s, const void* addr, std::size_t addrlen) | ||
|  | { | ||
|  |   return ::bind(s, static_cast<const socket_addr_type*>(addr), | ||
|  |       (SockLenType)addrlen); | ||
|  | } | ||
|  | 
 | ||
|  | int bind(socket_type s, const void* addr, | ||
|  |     std::size_t addrlen, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return socket_error_retval; | ||
|  |   } | ||
|  | 
 | ||
|  |   int result = call_bind(&msghdr::msg_namelen, s, addr, addrlen); | ||
|  |   get_last_error(ec, result != 0); | ||
|  |   return result; | ||
|  | } | ||
|  | 
 | ||
|  | int close(socket_type s, state_type& state, | ||
|  |     bool destruction, asio::error_code& ec) | ||
|  | { | ||
|  |   int result = 0; | ||
|  |   if (s != invalid_socket) | ||
|  |   { | ||
|  |     // We don't want the destructor to block, so set the socket to linger in | ||
|  |     // the background. If the user doesn't like this behaviour then they need | ||
|  |     // to explicitly close the socket. | ||
|  |     if (destruction && (state & user_set_linger)) | ||
|  |     { | ||
|  |       ::linger opt; | ||
|  |       opt.l_onoff = 0; | ||
|  |       opt.l_linger = 0; | ||
|  |       asio::error_code ignored_ec; | ||
|  |       socket_ops::setsockopt(s, state, SOL_SOCKET, | ||
|  |           SO_LINGER, &opt, sizeof(opt), ignored_ec); | ||
|  |     } | ||
|  | 
 | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |     result = ::closesocket(s); | ||
|  | #else // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |     result = ::close(s); | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |     get_last_error(ec, result != 0); | ||
|  | 
 | ||
|  |     if (result != 0 | ||
|  |         && (ec == asio::error::would_block | ||
|  |           || ec == asio::error::try_again)) | ||
|  |     { | ||
|  |       // According to UNIX Network Programming Vol. 1, it is possible for | ||
|  |       // close() to fail with EWOULDBLOCK under certain circumstances. What | ||
|  |       // isn't clear is the state of the descriptor after this error. The one | ||
|  |       // current OS where this behaviour is seen, Windows, says that the socket | ||
|  |       // remains open. Therefore we'll put the descriptor back into blocking | ||
|  |       // mode and have another attempt at closing it. | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |       ioctl_arg_type arg = 0; | ||
|  |       ::ioctlsocket(s, FIONBIO, &arg); | ||
|  | #else // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | # if defined(__SYMBIAN32__) || defined(__EMSCRIPTEN__) | ||
|  |       int flags = ::fcntl(s, F_GETFL, 0); | ||
|  |       if (flags >= 0) | ||
|  |         ::fcntl(s, F_SETFL, flags & ~O_NONBLOCK); | ||
|  | # else // defined(__SYMBIAN32__) || defined(__EMSCRIPTEN__) | ||
|  |       ioctl_arg_type arg = 0; | ||
|  |       ::ioctl(s, FIONBIO, &arg); | ||
|  | # endif // defined(__SYMBIAN32__) || defined(__EMSCRIPTEN__) | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |       state &= ~non_blocking; | ||
|  | 
 | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |       result = ::closesocket(s); | ||
|  | #else // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |       result = ::close(s); | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |       get_last_error(ec, result != 0); | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   return result; | ||
|  | } | ||
|  | 
 | ||
|  | bool set_user_non_blocking(socket_type s, | ||
|  |     state_type& state, bool value, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return false; | ||
|  |   } | ||
|  | 
 | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   ioctl_arg_type arg = (value ? 1 : 0); | ||
|  |   int result = ::ioctlsocket(s, FIONBIO, &arg); | ||
|  |   get_last_error(ec, result < 0); | ||
|  | #elif defined(__SYMBIAN32__) || defined(__EMSCRIPTEN__) | ||
|  |   int result = ::fcntl(s, F_GETFL, 0); | ||
|  |   get_last_error(ec, result < 0); | ||
|  |   if (result >= 0) | ||
|  |   { | ||
|  |     int flag = (value ? (result | O_NONBLOCK) : (result & ~O_NONBLOCK)); | ||
|  |     result = ::fcntl(s, F_SETFL, flag); | ||
|  |     get_last_error(ec, result < 0); | ||
|  |   } | ||
|  | #else | ||
|  |   ioctl_arg_type arg = (value ? 1 : 0); | ||
|  |   int result = ::ioctl(s, FIONBIO, &arg); | ||
|  |   get_last_error(ec, result < 0); | ||
|  | #endif | ||
|  | 
 | ||
|  |   if (result >= 0) | ||
|  |   { | ||
|  |     if (value) | ||
|  |       state |= user_set_non_blocking; | ||
|  |     else | ||
|  |     { | ||
|  |       // Clearing the user-set non-blocking mode always overrides any | ||
|  |       // internally-set non-blocking flag. Any subsequent asynchronous | ||
|  |       // operations will need to re-enable non-blocking I/O. | ||
|  |       state &= ~(user_set_non_blocking | internal_non_blocking); | ||
|  |     } | ||
|  |     return true; | ||
|  |   } | ||
|  | 
 | ||
|  |   return false; | ||
|  | } | ||
|  | 
 | ||
|  | bool set_internal_non_blocking(socket_type s, | ||
|  |     state_type& state, bool value, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return false; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!value && (state & user_set_non_blocking)) | ||
|  |   { | ||
|  |     // It does not make sense to clear the internal non-blocking flag if the | ||
|  |     // user still wants non-blocking behaviour. Return an error and let the | ||
|  |     // caller figure out whether to update the user-set non-blocking flag. | ||
|  |     ec = asio::error::invalid_argument; | ||
|  |     return false; | ||
|  |   } | ||
|  | 
 | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   ioctl_arg_type arg = (value ? 1 : 0); | ||
|  |   int result = ::ioctlsocket(s, FIONBIO, &arg); | ||
|  |   get_last_error(ec, result < 0); | ||
|  | #elif defined(__SYMBIAN32__) || defined(__EMSCRIPTEN__) | ||
|  |   int result = ::fcntl(s, F_GETFL, 0); | ||
|  |   get_last_error(ec, result < 0); | ||
|  |   if (result >= 0) | ||
|  |   { | ||
|  |     int flag = (value ? (result | O_NONBLOCK) : (result & ~O_NONBLOCK)); | ||
|  |     result = ::fcntl(s, F_SETFL, flag); | ||
|  |     get_last_error(ec, result < 0); | ||
|  |   } | ||
|  | #else | ||
|  |   ioctl_arg_type arg = (value ? 1 : 0); | ||
|  |   int result = ::ioctl(s, FIONBIO, &arg); | ||
|  |   get_last_error(ec, result < 0); | ||
|  | #endif | ||
|  | 
 | ||
|  |   if (result >= 0) | ||
|  |   { | ||
|  |     if (value) | ||
|  |       state |= internal_non_blocking; | ||
|  |     else | ||
|  |       state &= ~internal_non_blocking; | ||
|  |     return true; | ||
|  |   } | ||
|  | 
 | ||
|  |   return false; | ||
|  | } | ||
|  | 
 | ||
|  | int shutdown(socket_type s, int what, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return socket_error_retval; | ||
|  |   } | ||
|  | 
 | ||
|  |   int result = ::shutdown(s, what); | ||
|  |   get_last_error(ec, result != 0); | ||
|  |   return result; | ||
|  | } | ||
|  | 
 | ||
|  | template <typename SockLenType> | ||
|  | inline int call_connect(SockLenType msghdr::*, | ||
|  |     socket_type s, const void* addr, std::size_t addrlen) | ||
|  | { | ||
|  |   return ::connect(s, static_cast<const socket_addr_type*>(addr), | ||
|  |       (SockLenType)addrlen); | ||
|  | } | ||
|  | 
 | ||
|  | int connect(socket_type s, const void* addr, | ||
|  |     std::size_t addrlen, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return socket_error_retval; | ||
|  |   } | ||
|  | 
 | ||
|  |   int result = call_connect(&msghdr::msg_namelen, s, addr, addrlen); | ||
|  |   get_last_error(ec, result != 0); | ||
|  | #if defined(__linux__) | ||
|  |   if (result != 0 && ec == asio::error::try_again) | ||
|  |   { | ||
|  |     if (static_cast<const socket_addr_type*>(addr)->sa_family == AF_UNIX) | ||
|  |       ec = asio::error::in_progress; | ||
|  |     else | ||
|  |       ec = asio::error::no_buffer_space; | ||
|  |   } | ||
|  | #endif // defined(__linux__) | ||
|  |   return result; | ||
|  | } | ||
|  | 
 | ||
|  | void sync_connect(socket_type s, const void* addr, | ||
|  |     std::size_t addrlen, asio::error_code& ec) | ||
|  | { | ||
|  |   // Perform the connect operation. | ||
|  |   socket_ops::connect(s, addr, addrlen, ec); | ||
|  |   if (ec != asio::error::in_progress | ||
|  |       && ec != asio::error::would_block) | ||
|  |   { | ||
|  |     // The connect operation finished immediately. | ||
|  |     return; | ||
|  |   } | ||
|  | 
 | ||
|  |   // Wait for socket to become ready. | ||
|  |   if (socket_ops::poll_connect(s, -1, ec) < 0) | ||
|  |     return; | ||
|  | 
 | ||
|  |   // Get the error code from the connect operation. | ||
|  |   int connect_error = 0; | ||
|  |   size_t connect_error_len = sizeof(connect_error); | ||
|  |   if (socket_ops::getsockopt(s, 0, SOL_SOCKET, SO_ERROR, | ||
|  |         &connect_error, &connect_error_len, ec) == socket_error_retval) | ||
|  |     return; | ||
|  | 
 | ||
|  |   // Return the result of the connect operation. | ||
|  |   ec = asio::error_code(connect_error, | ||
|  |       asio::error::get_system_category()); | ||
|  | } | ||
|  | 
 | ||
|  | #if defined(ASIO_HAS_IOCP) | ||
|  | 
 | ||
|  | void complete_iocp_connect(socket_type s, asio::error_code& ec) | ||
|  | { | ||
|  |   // Map non-portable errors to their portable counterparts. | ||
|  |   switch (ec.value()) | ||
|  |   { | ||
|  |   case ERROR_CONNECTION_REFUSED: | ||
|  |     ec = asio::error::connection_refused; | ||
|  |     break; | ||
|  |   case ERROR_NETWORK_UNREACHABLE: | ||
|  |     ec = asio::error::network_unreachable; | ||
|  |     break; | ||
|  |   case ERROR_HOST_UNREACHABLE: | ||
|  |     ec = asio::error::host_unreachable; | ||
|  |     break; | ||
|  |   case ERROR_SEM_TIMEOUT: | ||
|  |     ec = asio::error::timed_out; | ||
|  |     break; | ||
|  |   default: | ||
|  |     break; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (!ec) | ||
|  |   { | ||
|  |     // Need to set the SO_UPDATE_CONNECT_CONTEXT option so that getsockname | ||
|  |     // and getpeername will work on the connected socket. | ||
|  |     socket_ops::state_type state = 0; | ||
|  |     const int so_update_connect_context = 0x7010; | ||
|  |     socket_ops::setsockopt(s, state, SOL_SOCKET, | ||
|  |         so_update_connect_context, 0, 0, ec); | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | #endif // defined(ASIO_HAS_IOCP) | ||
|  | 
 | ||
|  | bool non_blocking_connect(socket_type s, asio::error_code& ec) | ||
|  | { | ||
|  |   // Check if the connect operation has finished. This is required since we may | ||
|  |   // get spurious readiness notifications from the reactor. | ||
|  | #if defined(ASIO_WINDOWS) \ | ||
|  |   || defined(__CYGWIN__) \ | ||
|  |   || defined(__SYMBIAN32__) | ||
|  |   fd_set write_fds; | ||
|  |   FD_ZERO(&write_fds); | ||
|  |   FD_SET(s, &write_fds); | ||
|  |   fd_set except_fds; | ||
|  |   FD_ZERO(&except_fds); | ||
|  |   FD_SET(s, &except_fds); | ||
|  |   timeval zero_timeout; | ||
|  |   zero_timeout.tv_sec = 0; | ||
|  |   zero_timeout.tv_usec = 0; | ||
|  |   int ready = ::select(s + 1, 0, &write_fds, &except_fds, &zero_timeout); | ||
|  | #else // defined(ASIO_WINDOWS) | ||
|  |       // || defined(__CYGWIN__) | ||
|  |       // || defined(__SYMBIAN32__) | ||
|  |   pollfd fds; | ||
|  |   fds.fd = s; | ||
|  |   fds.events = POLLOUT; | ||
|  |   fds.revents = 0; | ||
|  |   int ready = ::poll(&fds, 1, 0); | ||
|  | #endif // defined(ASIO_WINDOWS) | ||
|  |        // || defined(__CYGWIN__) | ||
|  |        // || defined(__SYMBIAN32__) | ||
|  |   if (ready == 0) | ||
|  |   { | ||
|  |     // The asynchronous connect operation is still in progress. | ||
|  |     return false; | ||
|  |   } | ||
|  | 
 | ||
|  |   // Get the error code from the connect operation. | ||
|  |   int connect_error = 0; | ||
|  |   size_t connect_error_len = sizeof(connect_error); | ||
|  |   if (socket_ops::getsockopt(s, 0, SOL_SOCKET, SO_ERROR, | ||
|  |         &connect_error, &connect_error_len, ec) == 0) | ||
|  |   { | ||
|  |     if (connect_error) | ||
|  |     { | ||
|  |       ec = asio::error_code(connect_error, | ||
|  |           asio::error::get_system_category()); | ||
|  |     } | ||
|  |     else | ||
|  |       asio::error::clear(ec); | ||
|  |   } | ||
|  | 
 | ||
|  |   return true; | ||
|  | } | ||
|  | 
 | ||
|  | int socketpair(int af, int type, int protocol, | ||
|  |     socket_type sv[2], asio::error_code& ec) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   (void)(af); | ||
|  |   (void)(type); | ||
|  |   (void)(protocol); | ||
|  |   (void)(sv); | ||
|  |   ec = asio::error::operation_not_supported; | ||
|  |   return socket_error_retval; | ||
|  | #else | ||
|  |   int result = ::socketpair(af, type, protocol, sv); | ||
|  |   get_last_error(ec, result != 0); | ||
|  |   return result; | ||
|  | #endif | ||
|  | } | ||
|  | 
 | ||
|  | bool sockatmark(socket_type s, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return false; | ||
|  |   } | ||
|  | 
 | ||
|  | #if defined(SIOCATMARK) | ||
|  |   ioctl_arg_type value = 0; | ||
|  | # if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   int result = ::ioctlsocket(s, SIOCATMARK, &value); | ||
|  | # else // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   int result = ::ioctl(s, SIOCATMARK, &value); | ||
|  | # endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   get_last_error(ec, result < 0); | ||
|  | # if defined(ENOTTY) | ||
|  |   if (ec.value() == ENOTTY) | ||
|  |     ec = asio::error::not_socket; | ||
|  | # endif // defined(ENOTTY) | ||
|  | #else // defined(SIOCATMARK) | ||
|  |   int value = ::sockatmark(s); | ||
|  |   get_last_error(ec, value < 0); | ||
|  | #endif // defined(SIOCATMARK) | ||
|  | 
 | ||
|  |   return ec ? false : value != 0; | ||
|  | } | ||
|  | 
 | ||
|  | size_t available(socket_type s, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   ioctl_arg_type value = 0; | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   int result = ::ioctlsocket(s, FIONREAD, &value); | ||
|  | #else // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   int result = ::ioctl(s, FIONREAD, &value); | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   get_last_error(ec, result < 0); | ||
|  | #if defined(ENOTTY) | ||
|  |   if (ec.value() == ENOTTY) | ||
|  |     ec = asio::error::not_socket; | ||
|  | #endif // defined(ENOTTY) | ||
|  | 
 | ||
|  |   return ec ? static_cast<size_t>(0) : static_cast<size_t>(value); | ||
|  | } | ||
|  | 
 | ||
|  | int listen(socket_type s, int backlog, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return socket_error_retval; | ||
|  |   } | ||
|  | 
 | ||
|  |   int result = ::listen(s, backlog); | ||
|  |   get_last_error(ec, result != 0); | ||
|  |   return result; | ||
|  | } | ||
|  | 
 | ||
|  | inline void init_buf_iov_base(void*& base, void* addr) | ||
|  | { | ||
|  |   base = addr; | ||
|  | } | ||
|  | 
 | ||
|  | template <typename T> | ||
|  | inline void init_buf_iov_base(T& base, void* addr) | ||
|  | { | ||
|  |   base = static_cast<T>(addr); | ||
|  | } | ||
|  | 
 | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | typedef WSABUF buf; | ||
|  | #else // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | typedef iovec buf; | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | 
 | ||
|  | void init_buf(buf& b, void* data, size_t size) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   b.buf = static_cast<char*>(data); | ||
|  |   b.len = static_cast<u_long>(size); | ||
|  | #else // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   init_buf_iov_base(b.iov_base, data); | ||
|  |   b.iov_len = size; | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | } | ||
|  | 
 | ||
|  | void init_buf(buf& b, const void* data, size_t size) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   b.buf = static_cast<char*>(const_cast<void*>(data)); | ||
|  |   b.len = static_cast<u_long>(size); | ||
|  | #else // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   init_buf_iov_base(b.iov_base, const_cast<void*>(data)); | ||
|  |   b.iov_len = size; | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | } | ||
|  | 
 | ||
|  | inline void init_msghdr_msg_name(void*& name, void* addr) | ||
|  | { | ||
|  |   name = static_cast<socket_addr_type*>(addr); | ||
|  | } | ||
|  | 
 | ||
|  | inline void init_msghdr_msg_name(void*& name, const socket_addr_type* addr) | ||
|  | { | ||
|  |   name = const_cast<socket_addr_type*>(addr); | ||
|  | } | ||
|  | 
 | ||
|  | template <typename T> | ||
|  | inline void init_msghdr_msg_name(T& name, void* addr) | ||
|  | { | ||
|  |   name = static_cast<T>(addr); | ||
|  | } | ||
|  | 
 | ||
|  | template <typename T> | ||
|  | inline void init_msghdr_msg_name(T& name, const void* addr) | ||
|  | { | ||
|  |   name = static_cast<T>(const_cast<void*>(addr)); | ||
|  | } | ||
|  | 
 | ||
|  | signed_size_type recv(socket_type s, buf* bufs, size_t count, | ||
|  |     int flags, asio::error_code& ec) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   // Receive some data. | ||
|  |   DWORD recv_buf_count = static_cast<DWORD>(count); | ||
|  |   DWORD bytes_transferred = 0; | ||
|  |   DWORD recv_flags = flags; | ||
|  |   int result = ::WSARecv(s, bufs, recv_buf_count, | ||
|  |       &bytes_transferred, &recv_flags, 0, 0); | ||
|  |   get_last_error(ec, true); | ||
|  |   if (ec.value() == ERROR_NETNAME_DELETED) | ||
|  |     ec = asio::error::connection_reset; | ||
|  |   else if (ec.value() == ERROR_PORT_UNREACHABLE) | ||
|  |     ec = asio::error::connection_refused; | ||
|  |   else if (ec.value() == WSAEMSGSIZE || ec.value() == ERROR_MORE_DATA) | ||
|  |     result = 0; | ||
|  |   if (result != 0) | ||
|  |     return socket_error_retval; | ||
|  |   asio::error::clear(ec); | ||
|  |   return bytes_transferred; | ||
|  | #else // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   msghdr msg = msghdr(); | ||
|  |   msg.msg_iov = bufs; | ||
|  |   msg.msg_iovlen = static_cast<int>(count); | ||
|  |   signed_size_type result = ::recvmsg(s, &msg, flags); | ||
|  |   get_last_error(ec, result < 0); | ||
|  |   return result; | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | } | ||
|  | 
 | ||
|  | signed_size_type recv1(socket_type s, void* data, size_t size, | ||
|  |     int flags, asio::error_code& ec) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   // Receive some data. | ||
|  |   WSABUF buf; | ||
|  |   buf.buf = const_cast<char*>(static_cast<const char*>(data)); | ||
|  |   buf.len = static_cast<ULONG>(size); | ||
|  |   DWORD bytes_transferred = 0; | ||
|  |   DWORD recv_flags = flags; | ||
|  |   int result = ::WSARecv(s, &buf, 1, | ||
|  |       &bytes_transferred, &recv_flags, 0, 0); | ||
|  |   get_last_error(ec, true); | ||
|  |   if (ec.value() == ERROR_NETNAME_DELETED) | ||
|  |     ec = asio::error::connection_reset; | ||
|  |   else if (ec.value() == ERROR_PORT_UNREACHABLE) | ||
|  |     ec = asio::error::connection_refused; | ||
|  |   else if (ec.value() == WSAEMSGSIZE || ec.value() == ERROR_MORE_DATA) | ||
|  |     result = 0; | ||
|  |   if (result != 0) | ||
|  |     return socket_error_retval; | ||
|  |   asio::error::clear(ec); | ||
|  |   return bytes_transferred; | ||
|  | #else // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   signed_size_type result = ::recv(s, static_cast<char*>(data), size, flags); | ||
|  |   get_last_error(ec, result < 0); | ||
|  |   return result; | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | } | ||
|  | 
 | ||
|  | size_t sync_recv(socket_type s, state_type state, buf* bufs, | ||
|  |     size_t count, int flags, bool all_empty, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   // A request to read 0 bytes on a stream is a no-op. | ||
|  |   if (all_empty && (state & stream_oriented)) | ||
|  |   { | ||
|  |     asio::error::clear(ec); | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   // Read some data. | ||
|  |   for (;;) | ||
|  |   { | ||
|  |     // Try to complete the operation without blocking. | ||
|  |     signed_size_type bytes = socket_ops::recv(s, bufs, count, flags, ec); | ||
|  | 
 | ||
|  |     // Check for EOF. | ||
|  |     if ((state & stream_oriented) && bytes == 0) | ||
|  |     { | ||
|  |       ec = asio::error::eof; | ||
|  |       return 0; | ||
|  |     } | ||
|  | 
 | ||
|  |     // Check if operation succeeded. | ||
|  |     if (bytes >= 0) | ||
|  |       return bytes; | ||
|  | 
 | ||
|  |     // Operation failed. | ||
|  |     if ((state & user_set_non_blocking) | ||
|  |         || (ec != asio::error::would_block | ||
|  |           && ec != asio::error::try_again)) | ||
|  |       return 0; | ||
|  | 
 | ||
|  |     // Wait for socket to become ready. | ||
|  |     if (socket_ops::poll_read(s, 0, -1, ec) < 0) | ||
|  |       return 0; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | size_t sync_recv1(socket_type s, state_type state, void* data, | ||
|  |     size_t size, int flags, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   // A request to read 0 bytes on a stream is a no-op. | ||
|  |   if (size == 0 && (state & stream_oriented)) | ||
|  |   { | ||
|  |     asio::error::clear(ec); | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   // Read some data. | ||
|  |   for (;;) | ||
|  |   { | ||
|  |     // Try to complete the operation without blocking. | ||
|  |     signed_size_type bytes = socket_ops::recv1(s, data, size, flags, ec); | ||
|  | 
 | ||
|  |     // Check for EOF. | ||
|  |     if ((state & stream_oriented) && bytes == 0) | ||
|  |     { | ||
|  |       ec = asio::error::eof; | ||
|  |       return 0; | ||
|  |     } | ||
|  | 
 | ||
|  |     // Check if operation succeeded. | ||
|  |     if (bytes >= 0) | ||
|  |       return bytes; | ||
|  | 
 | ||
|  |     // Operation failed. | ||
|  |     if ((state & user_set_non_blocking) | ||
|  |         || (ec != asio::error::would_block | ||
|  |           && ec != asio::error::try_again)) | ||
|  |       return 0; | ||
|  | 
 | ||
|  |     // Wait for socket to become ready. | ||
|  |     if (socket_ops::poll_read(s, 0, -1, ec) < 0) | ||
|  |       return 0; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | #if defined(ASIO_HAS_IOCP) | ||
|  | 
 | ||
|  | void complete_iocp_recv(state_type state, | ||
|  |     const weak_cancel_token_type& cancel_token, bool all_empty, | ||
|  |     asio::error_code& ec, size_t bytes_transferred) | ||
|  | { | ||
|  |   // Map non-portable errors to their portable counterparts. | ||
|  |   if (ec.value() == ERROR_NETNAME_DELETED) | ||
|  |   { | ||
|  |     if (cancel_token.expired()) | ||
|  |       ec = asio::error::operation_aborted; | ||
|  |     else | ||
|  |       ec = asio::error::connection_reset; | ||
|  |   } | ||
|  |   else if (ec.value() == ERROR_PORT_UNREACHABLE) | ||
|  |   { | ||
|  |     ec = asio::error::connection_refused; | ||
|  |   } | ||
|  |   else if (ec.value() == WSAEMSGSIZE || ec.value() == ERROR_MORE_DATA) | ||
|  |   { | ||
|  |     asio::error::clear(ec); | ||
|  |   } | ||
|  | 
 | ||
|  |   // Check for connection closed. | ||
|  |   else if (!ec && bytes_transferred == 0 | ||
|  |       && (state & stream_oriented) != 0 | ||
|  |       && !all_empty) | ||
|  |   { | ||
|  |     ec = asio::error::eof; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | #else // defined(ASIO_HAS_IOCP) | ||
|  | 
 | ||
|  | bool non_blocking_recv(socket_type s, | ||
|  |     buf* bufs, size_t count, int flags, bool is_stream, | ||
|  |     asio::error_code& ec, size_t& bytes_transferred) | ||
|  | { | ||
|  |   for (;;) | ||
|  |   { | ||
|  |     // Read some data. | ||
|  |     signed_size_type bytes = socket_ops::recv(s, bufs, count, flags, ec); | ||
|  | 
 | ||
|  |     // Check for end of stream. | ||
|  |     if (is_stream && bytes == 0) | ||
|  |     { | ||
|  |       ec = asio::error::eof; | ||
|  |       return true; | ||
|  |     } | ||
|  | 
 | ||
|  |     // Check if operation succeeded. | ||
|  |     if (bytes >= 0) | ||
|  |     { | ||
|  |       bytes_transferred = bytes; | ||
|  |       return true; | ||
|  |     } | ||
|  | 
 | ||
|  |     // Retry operation if interrupted by signal. | ||
|  |     if (ec == asio::error::interrupted) | ||
|  |       continue; | ||
|  | 
 | ||
|  |     // Check if we need to run the operation again. | ||
|  |     if (ec == asio::error::would_block | ||
|  |         || ec == asio::error::try_again) | ||
|  |       return false; | ||
|  | 
 | ||
|  |     // Operation failed. | ||
|  |     bytes_transferred = 0; | ||
|  |     return true; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | bool non_blocking_recv1(socket_type s, | ||
|  |     void* data, size_t size, int flags, bool is_stream, | ||
|  |     asio::error_code& ec, size_t& bytes_transferred) | ||
|  | { | ||
|  |   for (;;) | ||
|  |   { | ||
|  |     // Read some data. | ||
|  |     signed_size_type bytes = socket_ops::recv1(s, data, size, flags, ec); | ||
|  | 
 | ||
|  |     // Check for end of stream. | ||
|  |     if (is_stream && bytes == 0) | ||
|  |     { | ||
|  |       ec = asio::error::eof; | ||
|  |       return true; | ||
|  |     } | ||
|  | 
 | ||
|  |     // Check if operation succeeded. | ||
|  |     if (bytes >= 0) | ||
|  |     { | ||
|  |       bytes_transferred = bytes; | ||
|  |       return true; | ||
|  |     } | ||
|  | 
 | ||
|  |     // Retry operation if interrupted by signal. | ||
|  |     if (ec == asio::error::interrupted) | ||
|  |       continue; | ||
|  | 
 | ||
|  |     // Check if we need to run the operation again. | ||
|  |     if (ec == asio::error::would_block | ||
|  |         || ec == asio::error::try_again) | ||
|  |       return false; | ||
|  | 
 | ||
|  |     // Operation failed. | ||
|  |     bytes_transferred = 0; | ||
|  |     return true; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | #endif // defined(ASIO_HAS_IOCP) | ||
|  | 
 | ||
|  | signed_size_type recvfrom(socket_type s, buf* bufs, size_t count, | ||
|  |     int flags, void* addr, std::size_t* addrlen, asio::error_code& ec) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   // Receive some data. | ||
|  |   DWORD recv_buf_count = static_cast<DWORD>(count); | ||
|  |   DWORD bytes_transferred = 0; | ||
|  |   DWORD recv_flags = flags; | ||
|  |   int tmp_addrlen = (int)*addrlen; | ||
|  |   int result = ::WSARecvFrom(s, bufs, recv_buf_count, &bytes_transferred, | ||
|  |       &recv_flags, static_cast<socket_addr_type*>(addr), &tmp_addrlen, 0, 0); | ||
|  |   get_last_error(ec, true); | ||
|  |   *addrlen = (std::size_t)tmp_addrlen; | ||
|  |   if (ec.value() == ERROR_NETNAME_DELETED) | ||
|  |     ec = asio::error::connection_reset; | ||
|  |   else if (ec.value() == ERROR_PORT_UNREACHABLE) | ||
|  |     ec = asio::error::connection_refused; | ||
|  |   else if (ec.value() == WSAEMSGSIZE || ec.value() == ERROR_MORE_DATA) | ||
|  |     result = 0; | ||
|  |   if (result != 0) | ||
|  |     return socket_error_retval; | ||
|  |   asio::error::clear(ec); | ||
|  |   return bytes_transferred; | ||
|  | #else // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   msghdr msg = msghdr(); | ||
|  |   init_msghdr_msg_name(msg.msg_name, addr); | ||
|  |   msg.msg_namelen = static_cast<int>(*addrlen); | ||
|  |   msg.msg_iov = bufs; | ||
|  |   msg.msg_iovlen = static_cast<int>(count); | ||
|  |   signed_size_type result = ::recvmsg(s, &msg, flags); | ||
|  |   get_last_error(ec, result < 0); | ||
|  |   *addrlen = msg.msg_namelen; | ||
|  |   return result; | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | } | ||
|  | 
 | ||
|  | template <typename SockLenType> | ||
|  | inline signed_size_type call_recvfrom(SockLenType msghdr::*, socket_type s, | ||
|  |     void* data, size_t size, int flags, void* addr, std::size_t* addrlen) | ||
|  | { | ||
|  |   SockLenType tmp_addrlen = addrlen ? (SockLenType)*addrlen : 0; | ||
|  |   signed_size_type result = ::recvfrom(s, static_cast<char*>(data), size, | ||
|  |       flags, static_cast<socket_addr_type*>(addr), addrlen ? &tmp_addrlen : 0); | ||
|  |   if (addrlen) | ||
|  |     *addrlen = (std::size_t)tmp_addrlen; | ||
|  |   return result; | ||
|  | } | ||
|  | 
 | ||
|  | signed_size_type recvfrom1(socket_type s, void* data, size_t size, | ||
|  |     int flags, void* addr, std::size_t* addrlen, asio::error_code& ec) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   // Receive some data. | ||
|  |   WSABUF buf; | ||
|  |   buf.buf = static_cast<char*>(data); | ||
|  |   buf.len = static_cast<ULONG>(size); | ||
|  |   DWORD bytes_transferred = 0; | ||
|  |   DWORD recv_flags = flags; | ||
|  |   int tmp_addrlen = (int)*addrlen; | ||
|  |   int result = ::WSARecvFrom(s, &buf, 1, &bytes_transferred, &recv_flags, | ||
|  |       static_cast<socket_addr_type*>(addr), &tmp_addrlen, 0, 0); | ||
|  |   get_last_error(ec, true); | ||
|  |   *addrlen = (std::size_t)tmp_addrlen; | ||
|  |   if (ec.value() == ERROR_NETNAME_DELETED) | ||
|  |     ec = asio::error::connection_reset; | ||
|  |   else if (ec.value() == ERROR_PORT_UNREACHABLE) | ||
|  |     ec = asio::error::connection_refused; | ||
|  |   else if (ec.value() == WSAEMSGSIZE || ec.value() == ERROR_MORE_DATA) | ||
|  |     result = 0; | ||
|  |   if (result != 0) | ||
|  |     return socket_error_retval; | ||
|  |   asio::error::clear(ec); | ||
|  |   return bytes_transferred; | ||
|  | #else // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   signed_size_type result = call_recvfrom(&msghdr::msg_namelen, | ||
|  |       s, data, size, flags, addr, addrlen); | ||
|  |   get_last_error(ec, result < 0); | ||
|  |   return result; | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | } | ||
|  | 
 | ||
|  | size_t sync_recvfrom(socket_type s, state_type state, buf* bufs, size_t count, | ||
|  |     int flags, void* addr, std::size_t* addrlen, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   // Read some data. | ||
|  |   for (;;) | ||
|  |   { | ||
|  |     // Try to complete the operation without blocking. | ||
|  |     signed_size_type bytes = socket_ops::recvfrom( | ||
|  |         s, bufs, count, flags, addr, addrlen, ec); | ||
|  | 
 | ||
|  |     // Check if operation succeeded. | ||
|  |     if (bytes >= 0) | ||
|  |       return bytes; | ||
|  | 
 | ||
|  |     // Operation failed. | ||
|  |     if ((state & user_set_non_blocking) | ||
|  |         || (ec != asio::error::would_block | ||
|  |           && ec != asio::error::try_again)) | ||
|  |       return 0; | ||
|  | 
 | ||
|  |     // Wait for socket to become ready. | ||
|  |     if (socket_ops::poll_read(s, 0, -1, ec) < 0) | ||
|  |       return 0; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | size_t sync_recvfrom1(socket_type s, state_type state, void* data, size_t size, | ||
|  |     int flags, void* addr, std::size_t* addrlen, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   // Read some data. | ||
|  |   for (;;) | ||
|  |   { | ||
|  |     // Try to complete the operation without blocking. | ||
|  |     signed_size_type bytes = socket_ops::recvfrom1( | ||
|  |         s, data, size, flags, addr, addrlen, ec); | ||
|  | 
 | ||
|  |     // Check if operation succeeded. | ||
|  |     if (bytes >= 0) | ||
|  |       return bytes; | ||
|  | 
 | ||
|  |     // Operation failed. | ||
|  |     if ((state & user_set_non_blocking) | ||
|  |         || (ec != asio::error::would_block | ||
|  |           && ec != asio::error::try_again)) | ||
|  |       return 0; | ||
|  | 
 | ||
|  |     // Wait for socket to become ready. | ||
|  |     if (socket_ops::poll_read(s, 0, -1, ec) < 0) | ||
|  |       return 0; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | #if defined(ASIO_HAS_IOCP) | ||
|  | 
 | ||
|  | void complete_iocp_recvfrom( | ||
|  |     const weak_cancel_token_type& cancel_token, | ||
|  |     asio::error_code& ec) | ||
|  | { | ||
|  |   // Map non-portable errors to their portable counterparts. | ||
|  |   if (ec.value() == ERROR_NETNAME_DELETED) | ||
|  |   { | ||
|  |     if (cancel_token.expired()) | ||
|  |       ec = asio::error::operation_aborted; | ||
|  |     else | ||
|  |       ec = asio::error::connection_reset; | ||
|  |   } | ||
|  |   else if (ec.value() == ERROR_PORT_UNREACHABLE) | ||
|  |   { | ||
|  |     ec = asio::error::connection_refused; | ||
|  |   } | ||
|  |   else if (ec.value() == WSAEMSGSIZE || ec.value() == ERROR_MORE_DATA) | ||
|  |   { | ||
|  |     asio::error::clear(ec); | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | #else // defined(ASIO_HAS_IOCP) | ||
|  | 
 | ||
|  | bool non_blocking_recvfrom(socket_type s, buf* bufs, | ||
|  |     size_t count, int flags, void* addr, std::size_t* addrlen, | ||
|  |     asio::error_code& ec, size_t& bytes_transferred) | ||
|  | { | ||
|  |   for (;;) | ||
|  |   { | ||
|  |     // Read some data. | ||
|  |     signed_size_type bytes = socket_ops::recvfrom( | ||
|  |         s, bufs, count, flags, addr, addrlen, ec); | ||
|  | 
 | ||
|  |     // Check if operation succeeded. | ||
|  |     if (bytes >= 0) | ||
|  |     { | ||
|  |       bytes_transferred = bytes; | ||
|  |       return true; | ||
|  |     } | ||
|  | 
 | ||
|  |     // Retry operation if interrupted by signal. | ||
|  |     if (ec == asio::error::interrupted) | ||
|  |       continue; | ||
|  | 
 | ||
|  |     // Check if we need to run the operation again. | ||
|  |     if (ec == asio::error::would_block | ||
|  |         || ec == asio::error::try_again) | ||
|  |       return false; | ||
|  | 
 | ||
|  |     // Operation failed. | ||
|  |     bytes_transferred = 0; | ||
|  |     return true; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | bool non_blocking_recvfrom1(socket_type s, void* data, | ||
|  |     size_t size, int flags, void* addr, std::size_t* addrlen, | ||
|  |     asio::error_code& ec, size_t& bytes_transferred) | ||
|  | { | ||
|  |   for (;;) | ||
|  |   { | ||
|  |     // Read some data. | ||
|  |     signed_size_type bytes = socket_ops::recvfrom1( | ||
|  |         s, data, size, flags, addr, addrlen, ec); | ||
|  | 
 | ||
|  |     // Check if operation succeeded. | ||
|  |     if (bytes >= 0) | ||
|  |     { | ||
|  |       bytes_transferred = bytes; | ||
|  |       return true; | ||
|  |     } | ||
|  | 
 | ||
|  |     // Retry operation if interrupted by signal. | ||
|  |     if (ec == asio::error::interrupted) | ||
|  |       continue; | ||
|  | 
 | ||
|  |     // Check if we need to run the operation again. | ||
|  |     if (ec == asio::error::would_block | ||
|  |         || ec == asio::error::try_again) | ||
|  |       return false; | ||
|  | 
 | ||
|  |     // Operation failed. | ||
|  |     bytes_transferred = 0; | ||
|  |     return true; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | #endif // defined(ASIO_HAS_IOCP) | ||
|  | 
 | ||
|  | signed_size_type recvmsg(socket_type s, buf* bufs, size_t count, | ||
|  |     int in_flags, int& out_flags, asio::error_code& ec) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   out_flags = 0; | ||
|  |   return socket_ops::recv(s, bufs, count, in_flags, ec); | ||
|  | #else // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   msghdr msg = msghdr(); | ||
|  |   msg.msg_iov = bufs; | ||
|  |   msg.msg_iovlen = static_cast<int>(count); | ||
|  |   signed_size_type result = ::recvmsg(s, &msg, in_flags); | ||
|  |   get_last_error(ec, result < 0); | ||
|  |   if (result >= 0) | ||
|  |     out_flags = msg.msg_flags; | ||
|  |   else | ||
|  |     out_flags = 0; | ||
|  |   return result; | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | } | ||
|  | 
 | ||
|  | size_t sync_recvmsg(socket_type s, state_type state, | ||
|  |     buf* bufs, size_t count, int in_flags, int& out_flags, | ||
|  |     asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   // Read some data. | ||
|  |   for (;;) | ||
|  |   { | ||
|  |     // Try to complete the operation without blocking. | ||
|  |     signed_size_type bytes = socket_ops::recvmsg( | ||
|  |         s, bufs, count, in_flags, out_flags, ec); | ||
|  | 
 | ||
|  |     // Check if operation succeeded. | ||
|  |     if (bytes >= 0) | ||
|  |       return bytes; | ||
|  | 
 | ||
|  |     // Operation failed. | ||
|  |     if ((state & user_set_non_blocking) | ||
|  |         || (ec != asio::error::would_block | ||
|  |           && ec != asio::error::try_again)) | ||
|  |       return 0; | ||
|  | 
 | ||
|  |     // Wait for socket to become ready. | ||
|  |     if (socket_ops::poll_read(s, 0, -1, ec) < 0) | ||
|  |       return 0; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | #if defined(ASIO_HAS_IOCP) | ||
|  | 
 | ||
|  | void complete_iocp_recvmsg( | ||
|  |     const weak_cancel_token_type& cancel_token, | ||
|  |     asio::error_code& ec) | ||
|  | { | ||
|  |   // Map non-portable errors to their portable counterparts. | ||
|  |   if (ec.value() == ERROR_NETNAME_DELETED) | ||
|  |   { | ||
|  |     if (cancel_token.expired()) | ||
|  |       ec = asio::error::operation_aborted; | ||
|  |     else | ||
|  |       ec = asio::error::connection_reset; | ||
|  |   } | ||
|  |   else if (ec.value() == ERROR_PORT_UNREACHABLE) | ||
|  |   { | ||
|  |     ec = asio::error::connection_refused; | ||
|  |   } | ||
|  |   else if (ec.value() == WSAEMSGSIZE || ec.value() == ERROR_MORE_DATA) | ||
|  |   { | ||
|  |     asio::error::clear(ec); | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | #else // defined(ASIO_HAS_IOCP) | ||
|  | 
 | ||
|  | bool non_blocking_recvmsg(socket_type s, | ||
|  |     buf* bufs, size_t count, int in_flags, int& out_flags, | ||
|  |     asio::error_code& ec, size_t& bytes_transferred) | ||
|  | { | ||
|  |   for (;;) | ||
|  |   { | ||
|  |     // Read some data. | ||
|  |     signed_size_type bytes = socket_ops::recvmsg( | ||
|  |         s, bufs, count, in_flags, out_flags, ec); | ||
|  | 
 | ||
|  |     // Check if operation succeeded. | ||
|  |     if (bytes >= 0) | ||
|  |     { | ||
|  |       bytes_transferred = bytes; | ||
|  |       return true; | ||
|  |     } | ||
|  | 
 | ||
|  |     // Retry operation if interrupted by signal. | ||
|  |     if (ec == asio::error::interrupted) | ||
|  |       continue; | ||
|  | 
 | ||
|  |     // Check if we need to run the operation again. | ||
|  |     if (ec == asio::error::would_block | ||
|  |         || ec == asio::error::try_again) | ||
|  |       return false; | ||
|  | 
 | ||
|  |     // Operation failed. | ||
|  |     bytes_transferred = 0; | ||
|  |     return true; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | #endif // defined(ASIO_HAS_IOCP) | ||
|  | 
 | ||
|  | signed_size_type send(socket_type s, const buf* bufs, size_t count, | ||
|  |     int flags, asio::error_code& ec) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   // Send the data. | ||
|  |   DWORD send_buf_count = static_cast<DWORD>(count); | ||
|  |   DWORD bytes_transferred = 0; | ||
|  |   DWORD send_flags = flags; | ||
|  |   int result = ::WSASend(s, const_cast<buf*>(bufs), | ||
|  |         send_buf_count, &bytes_transferred, send_flags, 0, 0); | ||
|  |   get_last_error(ec, true); | ||
|  |   if (ec.value() == ERROR_NETNAME_DELETED) | ||
|  |     ec = asio::error::connection_reset; | ||
|  |   else if (ec.value() == ERROR_PORT_UNREACHABLE) | ||
|  |     ec = asio::error::connection_refused; | ||
|  |   if (result != 0) | ||
|  |     return socket_error_retval; | ||
|  |   asio::error::clear(ec); | ||
|  |   return bytes_transferred; | ||
|  | #else // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   msghdr msg = msghdr(); | ||
|  |   msg.msg_iov = const_cast<buf*>(bufs); | ||
|  |   msg.msg_iovlen = static_cast<int>(count); | ||
|  | #if defined(ASIO_HAS_MSG_NOSIGNAL) | ||
|  |   flags |= MSG_NOSIGNAL; | ||
|  | #endif // defined(ASIO_HAS_MSG_NOSIGNAL) | ||
|  |   signed_size_type result = ::sendmsg(s, &msg, flags); | ||
|  |   get_last_error(ec, result < 0); | ||
|  |   return result; | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | } | ||
|  | 
 | ||
|  | signed_size_type send1(socket_type s, const void* data, size_t size, | ||
|  |     int flags, asio::error_code& ec) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   // Send the data. | ||
|  |   WSABUF buf; | ||
|  |   buf.buf = const_cast<char*>(static_cast<const char*>(data)); | ||
|  |   buf.len = static_cast<ULONG>(size); | ||
|  |   DWORD bytes_transferred = 0; | ||
|  |   DWORD send_flags = flags; | ||
|  |   int result = ::WSASend(s, &buf, 1, | ||
|  |         &bytes_transferred, send_flags, 0, 0); | ||
|  |   get_last_error(ec, true); | ||
|  |   if (ec.value() == ERROR_NETNAME_DELETED) | ||
|  |     ec = asio::error::connection_reset; | ||
|  |   else if (ec.value() == ERROR_PORT_UNREACHABLE) | ||
|  |     ec = asio::error::connection_refused; | ||
|  |   if (result != 0) | ||
|  |     return socket_error_retval; | ||
|  |   asio::error::clear(ec); | ||
|  |   return bytes_transferred; | ||
|  | #else // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | #if defined(ASIO_HAS_MSG_NOSIGNAL) | ||
|  |   flags |= MSG_NOSIGNAL; | ||
|  | #endif // defined(ASIO_HAS_MSG_NOSIGNAL) | ||
|  |   signed_size_type result = ::send(s, | ||
|  |       static_cast<const char*>(data), size, flags); | ||
|  |   get_last_error(ec, result < 0); | ||
|  |   return result; | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | } | ||
|  | 
 | ||
|  | size_t sync_send(socket_type s, state_type state, const buf* bufs, | ||
|  |     size_t count, int flags, bool all_empty, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   // A request to write 0 bytes to a stream is a no-op. | ||
|  |   if (all_empty && (state & stream_oriented)) | ||
|  |   { | ||
|  |     asio::error::clear(ec); | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   // Read some data. | ||
|  |   for (;;) | ||
|  |   { | ||
|  |     // Try to complete the operation without blocking. | ||
|  |     signed_size_type bytes = socket_ops::send(s, bufs, count, flags, ec); | ||
|  | 
 | ||
|  |     // Check if operation succeeded. | ||
|  |     if (bytes >= 0) | ||
|  |       return bytes; | ||
|  | 
 | ||
|  |     // Operation failed. | ||
|  |     if ((state & user_set_non_blocking) | ||
|  |         || (ec != asio::error::would_block | ||
|  |           && ec != asio::error::try_again)) | ||
|  |       return 0; | ||
|  | 
 | ||
|  |     // Wait for socket to become ready. | ||
|  |     if (socket_ops::poll_write(s, 0, -1, ec) < 0) | ||
|  |       return 0; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | size_t sync_send1(socket_type s, state_type state, const void* data, | ||
|  |     size_t size, int flags, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   // A request to write 0 bytes to a stream is a no-op. | ||
|  |   if (size == 0 && (state & stream_oriented)) | ||
|  |   { | ||
|  |     asio::error::clear(ec); | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   // Read some data. | ||
|  |   for (;;) | ||
|  |   { | ||
|  |     // Try to complete the operation without blocking. | ||
|  |     signed_size_type bytes = socket_ops::send1(s, data, size, flags, ec); | ||
|  | 
 | ||
|  |     // Check if operation succeeded. | ||
|  |     if (bytes >= 0) | ||
|  |       return bytes; | ||
|  | 
 | ||
|  |     // Operation failed. | ||
|  |     if ((state & user_set_non_blocking) | ||
|  |         || (ec != asio::error::would_block | ||
|  |           && ec != asio::error::try_again)) | ||
|  |       return 0; | ||
|  | 
 | ||
|  |     // Wait for socket to become ready. | ||
|  |     if (socket_ops::poll_write(s, 0, -1, ec) < 0) | ||
|  |       return 0; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | #if defined(ASIO_HAS_IOCP) | ||
|  | 
 | ||
|  | void complete_iocp_send( | ||
|  |     const weak_cancel_token_type& cancel_token, | ||
|  |     asio::error_code& ec) | ||
|  | { | ||
|  |   // Map non-portable errors to their portable counterparts. | ||
|  |   if (ec.value() == ERROR_NETNAME_DELETED) | ||
|  |   { | ||
|  |     if (cancel_token.expired()) | ||
|  |       ec = asio::error::operation_aborted; | ||
|  |     else | ||
|  |       ec = asio::error::connection_reset; | ||
|  |   } | ||
|  |   else if (ec.value() == ERROR_PORT_UNREACHABLE) | ||
|  |   { | ||
|  |     ec = asio::error::connection_refused; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | #else // defined(ASIO_HAS_IOCP) | ||
|  | 
 | ||
|  | bool non_blocking_send(socket_type s, | ||
|  |     const buf* bufs, size_t count, int flags, | ||
|  |     asio::error_code& ec, size_t& bytes_transferred) | ||
|  | { | ||
|  |   for (;;) | ||
|  |   { | ||
|  |     // Write some data. | ||
|  |     signed_size_type bytes = socket_ops::send(s, bufs, count, flags, ec); | ||
|  | 
 | ||
|  |     // Check if operation succeeded. | ||
|  |     if (bytes >= 0) | ||
|  |     { | ||
|  |       bytes_transferred = bytes; | ||
|  |       return true; | ||
|  |     } | ||
|  | 
 | ||
|  |     // Retry operation if interrupted by signal. | ||
|  |     if (ec == asio::error::interrupted) | ||
|  |       continue; | ||
|  | 
 | ||
|  |     // Check if we need to run the operation again. | ||
|  |     if (ec == asio::error::would_block | ||
|  |         || ec == asio::error::try_again) | ||
|  |       return false; | ||
|  | 
 | ||
|  |     // Operation failed. | ||
|  |     bytes_transferred = 0; | ||
|  |     return true; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | bool non_blocking_send1(socket_type s, | ||
|  |     const void* data, size_t size, int flags, | ||
|  |     asio::error_code& ec, size_t& bytes_transferred) | ||
|  | { | ||
|  |   for (;;) | ||
|  |   { | ||
|  |     // Write some data. | ||
|  |     signed_size_type bytes = socket_ops::send1(s, data, size, flags, ec); | ||
|  | 
 | ||
|  |     // Check if operation succeeded. | ||
|  |     if (bytes >= 0) | ||
|  |     { | ||
|  |       bytes_transferred = bytes; | ||
|  |       return true; | ||
|  |     } | ||
|  | 
 | ||
|  |     // Retry operation if interrupted by signal. | ||
|  |     if (ec == asio::error::interrupted) | ||
|  |       continue; | ||
|  | 
 | ||
|  |     // Check if we need to run the operation again. | ||
|  |     if (ec == asio::error::would_block | ||
|  |         || ec == asio::error::try_again) | ||
|  |       return false; | ||
|  | 
 | ||
|  |     // Operation failed. | ||
|  |     bytes_transferred = 0; | ||
|  |     return true; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | #endif // defined(ASIO_HAS_IOCP) | ||
|  | 
 | ||
|  | signed_size_type sendto(socket_type s, const buf* bufs, | ||
|  |     size_t count, int flags, const void* addr, | ||
|  |     std::size_t addrlen, asio::error_code& ec) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   // Send the data. | ||
|  |   DWORD send_buf_count = static_cast<DWORD>(count); | ||
|  |   DWORD bytes_transferred = 0; | ||
|  |   int result = ::WSASendTo(s, const_cast<buf*>(bufs), | ||
|  |         send_buf_count, &bytes_transferred, flags, | ||
|  |         static_cast<const socket_addr_type*>(addr), | ||
|  |         static_cast<int>(addrlen), 0, 0); | ||
|  |   get_last_error(ec, true); | ||
|  |   if (ec.value() == ERROR_NETNAME_DELETED) | ||
|  |     ec = asio::error::connection_reset; | ||
|  |   else if (ec.value() == ERROR_PORT_UNREACHABLE) | ||
|  |     ec = asio::error::connection_refused; | ||
|  |   if (result != 0) | ||
|  |     return socket_error_retval; | ||
|  |   asio::error::clear(ec); | ||
|  |   return bytes_transferred; | ||
|  | #else // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   msghdr msg = msghdr(); | ||
|  |   init_msghdr_msg_name(msg.msg_name, addr); | ||
|  |   msg.msg_namelen = static_cast<int>(addrlen); | ||
|  |   msg.msg_iov = const_cast<buf*>(bufs); | ||
|  |   msg.msg_iovlen = static_cast<int>(count); | ||
|  | #if defined(ASIO_HAS_MSG_NOSIGNAL) | ||
|  |   flags |= MSG_NOSIGNAL; | ||
|  | #endif // defined(ASIO_HAS_MSG_NOSIGNAL) | ||
|  |   signed_size_type result = ::sendmsg(s, &msg, flags); | ||
|  |   get_last_error(ec, result < 0); | ||
|  |   return result; | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | } | ||
|  | 
 | ||
|  | template <typename SockLenType> | ||
|  | inline signed_size_type call_sendto(SockLenType msghdr::*, | ||
|  |     socket_type s, const void* data, size_t size, int flags, | ||
|  |     const void* addr, std::size_t addrlen) | ||
|  | { | ||
|  |   return ::sendto(s, static_cast<char*>(const_cast<void*>(data)), size, flags, | ||
|  |       static_cast<const socket_addr_type*>(addr), (SockLenType)addrlen); | ||
|  | } | ||
|  | 
 | ||
|  | signed_size_type sendto1(socket_type s, const void* data, | ||
|  |     size_t size, int flags, const void* addr, | ||
|  |     std::size_t addrlen, asio::error_code& ec) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   // Send the data. | ||
|  |   WSABUF buf; | ||
|  |   buf.buf = const_cast<char*>(static_cast<const char*>(data)); | ||
|  |   buf.len = static_cast<ULONG>(size); | ||
|  |   DWORD bytes_transferred = 0; | ||
|  |   int result = ::WSASendTo(s, &buf, 1, &bytes_transferred, flags, | ||
|  |       static_cast<const socket_addr_type*>(addr), | ||
|  |       static_cast<int>(addrlen), 0, 0); | ||
|  |   get_last_error(ec, true); | ||
|  |   if (ec.value() == ERROR_NETNAME_DELETED) | ||
|  |     ec = asio::error::connection_reset; | ||
|  |   else if (ec.value() == ERROR_PORT_UNREACHABLE) | ||
|  |     ec = asio::error::connection_refused; | ||
|  |   if (result != 0) | ||
|  |     return socket_error_retval; | ||
|  |   asio::error::clear(ec); | ||
|  |   return bytes_transferred; | ||
|  | #else // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | #if defined(ASIO_HAS_MSG_NOSIGNAL) | ||
|  |   flags |= MSG_NOSIGNAL; | ||
|  | #endif // defined(ASIO_HAS_MSG_NOSIGNAL) | ||
|  |   signed_size_type result = call_sendto(&msghdr::msg_namelen, | ||
|  |       s, data, size, flags, addr, addrlen); | ||
|  |   get_last_error(ec, result < 0); | ||
|  |   return result; | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | } | ||
|  | 
 | ||
|  | size_t sync_sendto(socket_type s, state_type state, | ||
|  |     const buf* bufs, size_t count, int flags, const void* addr, | ||
|  |     std::size_t addrlen, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   // Write some data. | ||
|  |   for (;;) | ||
|  |   { | ||
|  |     // Try to complete the operation without blocking. | ||
|  |     signed_size_type bytes = socket_ops::sendto( | ||
|  |         s, bufs, count, flags, addr, addrlen, ec); | ||
|  | 
 | ||
|  |     // Check if operation succeeded. | ||
|  |     if (bytes >= 0) | ||
|  |       return bytes; | ||
|  | 
 | ||
|  |     // Operation failed. | ||
|  |     if ((state & user_set_non_blocking) | ||
|  |         || (ec != asio::error::would_block | ||
|  |           && ec != asio::error::try_again)) | ||
|  |       return 0; | ||
|  | 
 | ||
|  |     // Wait for socket to become ready. | ||
|  |     if (socket_ops::poll_write(s, 0, -1, ec) < 0) | ||
|  |       return 0; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | size_t sync_sendto1(socket_type s, state_type state, | ||
|  |     const void* data, size_t size, int flags, const void* addr, | ||
|  |     std::size_t addrlen, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   // Write some data. | ||
|  |   for (;;) | ||
|  |   { | ||
|  |     // Try to complete the operation without blocking. | ||
|  |     signed_size_type bytes = socket_ops::sendto1( | ||
|  |         s, data, size, flags, addr, addrlen, ec); | ||
|  | 
 | ||
|  |     // Check if operation succeeded. | ||
|  |     if (bytes >= 0) | ||
|  |       return bytes; | ||
|  | 
 | ||
|  |     // Operation failed. | ||
|  |     if ((state & user_set_non_blocking) | ||
|  |         || (ec != asio::error::would_block | ||
|  |           && ec != asio::error::try_again)) | ||
|  |       return 0; | ||
|  | 
 | ||
|  |     // Wait for socket to become ready. | ||
|  |     if (socket_ops::poll_write(s, 0, -1, ec) < 0) | ||
|  |       return 0; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | #if !defined(ASIO_HAS_IOCP) | ||
|  | 
 | ||
|  | bool non_blocking_sendto(socket_type s, | ||
|  |     const buf* bufs, size_t count, int flags, | ||
|  |     const void* addr, std::size_t addrlen, | ||
|  |     asio::error_code& ec, size_t& bytes_transferred) | ||
|  | { | ||
|  |   for (;;) | ||
|  |   { | ||
|  |     // Write some data. | ||
|  |     signed_size_type bytes = socket_ops::sendto( | ||
|  |         s, bufs, count, flags, addr, addrlen, ec); | ||
|  | 
 | ||
|  |     // Check if operation succeeded. | ||
|  |     if (bytes >= 0) | ||
|  |     { | ||
|  |       bytes_transferred = bytes; | ||
|  |       return true; | ||
|  |     } | ||
|  | 
 | ||
|  |     // Retry operation if interrupted by signal. | ||
|  |     if (ec == asio::error::interrupted) | ||
|  |       continue; | ||
|  | 
 | ||
|  |     // Check if we need to run the operation again. | ||
|  |     if (ec == asio::error::would_block | ||
|  |         || ec == asio::error::try_again) | ||
|  |       return false; | ||
|  | 
 | ||
|  |     // Operation failed. | ||
|  |     bytes_transferred = 0; | ||
|  |     return true; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | bool non_blocking_sendto1(socket_type s, | ||
|  |     const void* data, size_t size, int flags, | ||
|  |     const void* addr, std::size_t addrlen, | ||
|  |     asio::error_code& ec, size_t& bytes_transferred) | ||
|  | { | ||
|  |   for (;;) | ||
|  |   { | ||
|  |     // Write some data. | ||
|  |     signed_size_type bytes = socket_ops::sendto1( | ||
|  |         s, data, size, flags, addr, addrlen, ec); | ||
|  | 
 | ||
|  |     // Check if operation succeeded. | ||
|  |     if (bytes >= 0) | ||
|  |     { | ||
|  |       bytes_transferred = bytes; | ||
|  |       return true; | ||
|  |     } | ||
|  | 
 | ||
|  |     // Retry operation if interrupted by signal. | ||
|  |     if (ec == asio::error::interrupted) | ||
|  |       continue; | ||
|  | 
 | ||
|  |     // Check if we need to run the operation again. | ||
|  |     if (ec == asio::error::would_block | ||
|  |         || ec == asio::error::try_again) | ||
|  |       return false; | ||
|  | 
 | ||
|  |     // Operation failed. | ||
|  |     bytes_transferred = 0; | ||
|  |     return true; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | #endif // !defined(ASIO_HAS_IOCP) | ||
|  | 
 | ||
|  | socket_type socket(int af, int type, int protocol, | ||
|  |     asio::error_code& ec) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   socket_type s = ::WSASocketW(af, type, protocol, 0, 0, WSA_FLAG_OVERLAPPED); | ||
|  |   get_last_error(ec, s == invalid_socket); | ||
|  |   if (s == invalid_socket) | ||
|  |     return s; | ||
|  | 
 | ||
|  |   if (af == ASIO_OS_DEF(AF_INET6)) | ||
|  |   { | ||
|  |     // Try to enable the POSIX default behaviour of having IPV6_V6ONLY set to | ||
|  |     // false. This will only succeed on Windows Vista and later versions of | ||
|  |     // Windows, where a dual-stack IPv4/v6 implementation is available. | ||
|  |     DWORD optval = 0; | ||
|  |     ::setsockopt(s, IPPROTO_IPV6, IPV6_V6ONLY, | ||
|  |         reinterpret_cast<const char*>(&optval), sizeof(optval)); | ||
|  |   } | ||
|  | 
 | ||
|  |   return s; | ||
|  | #elif defined(__MACH__) && defined(__APPLE__) || defined(__FreeBSD__) | ||
|  |   socket_type s = ::socket(af, type, protocol); | ||
|  |   get_last_error(ec, s == invalid_socket); | ||
|  |   if (s == invalid_socket) | ||
|  |     return s; | ||
|  | 
 | ||
|  |   int optval = 1; | ||
|  |   int result = ::setsockopt(s, SOL_SOCKET, | ||
|  |       SO_NOSIGPIPE, &optval, sizeof(optval)); | ||
|  |   get_last_error(ec, result != 0); | ||
|  |   if (result != 0) | ||
|  |   { | ||
|  |     ::close(s); | ||
|  |     return invalid_socket; | ||
|  |   } | ||
|  | 
 | ||
|  |   return s; | ||
|  | #else | ||
|  |   int s = ::socket(af, type, protocol); | ||
|  |   get_last_error(ec, s < 0); | ||
|  |   return s; | ||
|  | #endif | ||
|  | } | ||
|  | 
 | ||
|  | template <typename SockLenType> | ||
|  | inline int call_setsockopt(SockLenType msghdr::*, | ||
|  |     socket_type s, int level, int optname, | ||
|  |     const void* optval, std::size_t optlen) | ||
|  | { | ||
|  |   return ::setsockopt(s, level, optname, | ||
|  |       (const char*)optval, (SockLenType)optlen); | ||
|  | } | ||
|  | 
 | ||
|  | int setsockopt(socket_type s, state_type& state, int level, int optname, | ||
|  |     const void* optval, std::size_t optlen, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return socket_error_retval; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (level == custom_socket_option_level && optname == always_fail_option) | ||
|  |   { | ||
|  |     ec = asio::error::invalid_argument; | ||
|  |     return socket_error_retval; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (level == custom_socket_option_level | ||
|  |       && optname == enable_connection_aborted_option) | ||
|  |   { | ||
|  |     if (optlen != sizeof(int)) | ||
|  |     { | ||
|  |       ec = asio::error::invalid_argument; | ||
|  |       return socket_error_retval; | ||
|  |     } | ||
|  | 
 | ||
|  |     if (*static_cast<const int*>(optval)) | ||
|  |       state |= enable_connection_aborted; | ||
|  |     else | ||
|  |       state &= ~enable_connection_aborted; | ||
|  |     asio::error::clear(ec); | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (level == SOL_SOCKET && optname == SO_LINGER) | ||
|  |     state |= user_set_linger; | ||
|  | 
 | ||
|  | #if defined(__BORLANDC__) | ||
|  |   // Mysteriously, using the getsockopt and setsockopt functions directly with | ||
|  |   // Borland C++ results in incorrect values being set and read. The bug can be | ||
|  |   // worked around by using function addresses resolved with GetProcAddress. | ||
|  |   if (HMODULE winsock_module = ::GetModuleHandleA("ws2_32")) | ||
|  |   { | ||
|  |     typedef int (WSAAPI *sso_t)(SOCKET, int, int, const char*, int); | ||
|  |     if (sso_t sso = (sso_t)::GetProcAddress(winsock_module, "setsockopt")) | ||
|  |     { | ||
|  |       int result = sso(s, level, optname, | ||
|  |             reinterpret_cast<const char*>(optval), | ||
|  |             static_cast<int>(optlen)); | ||
|  |       get_last_error(ec, result != 0); | ||
|  |       return result; | ||
|  |     } | ||
|  |   } | ||
|  |   ec = asio::error::fault; | ||
|  |   return socket_error_retval; | ||
|  | #else // defined(__BORLANDC__) | ||
|  |   int result = call_setsockopt(&msghdr::msg_namelen, | ||
|  |         s, level, optname, optval, optlen); | ||
|  |   get_last_error(ec, result != 0); | ||
|  |   if (result == 0) | ||
|  |   { | ||
|  | #if defined(__MACH__) && defined(__APPLE__) \ | ||
|  |   || defined(__NetBSD__) || defined(__FreeBSD__) \ | ||
|  |   || defined(__OpenBSD__) || defined(__QNX__) | ||
|  |     // To implement portable behaviour for SO_REUSEADDR with UDP sockets we | ||
|  |     // need to also set SO_REUSEPORT on BSD-based platforms. | ||
|  |     if ((state & datagram_oriented) | ||
|  |         && level == SOL_SOCKET && optname == SO_REUSEADDR) | ||
|  |     { | ||
|  |       call_setsockopt(&msghdr::msg_namelen, s, | ||
|  |           SOL_SOCKET, SO_REUSEPORT, optval, optlen); | ||
|  |     } | ||
|  | #endif | ||
|  |   } | ||
|  | 
 | ||
|  |   return result; | ||
|  | #endif // defined(__BORLANDC__) | ||
|  | } | ||
|  | 
 | ||
|  | template <typename SockLenType> | ||
|  | inline int call_getsockopt(SockLenType msghdr::*, | ||
|  |     socket_type s, int level, int optname, | ||
|  |     void* optval, std::size_t* optlen) | ||
|  | { | ||
|  |   SockLenType tmp_optlen = (SockLenType)*optlen; | ||
|  |   int result = ::getsockopt(s, level, optname, (char*)optval, &tmp_optlen); | ||
|  |   *optlen = (std::size_t)tmp_optlen; | ||
|  |   return result; | ||
|  | } | ||
|  | 
 | ||
|  | int getsockopt(socket_type s, state_type state, int level, int optname, | ||
|  |     void* optval, size_t* optlen, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return socket_error_retval; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (level == custom_socket_option_level && optname == always_fail_option) | ||
|  |   { | ||
|  |     ec = asio::error::invalid_argument; | ||
|  |     return socket_error_retval; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (level == custom_socket_option_level | ||
|  |       && optname == enable_connection_aborted_option) | ||
|  |   { | ||
|  |     if (*optlen != sizeof(int)) | ||
|  |     { | ||
|  |       ec = asio::error::invalid_argument; | ||
|  |       return socket_error_retval; | ||
|  |     } | ||
|  | 
 | ||
|  |     *static_cast<int*>(optval) = (state & enable_connection_aborted) ? 1 : 0; | ||
|  |     asio::error::clear(ec); | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  | #if defined(__BORLANDC__) | ||
|  |   // Mysteriously, using the getsockopt and setsockopt functions directly with | ||
|  |   // Borland C++ results in incorrect values being set and read. The bug can be | ||
|  |   // worked around by using function addresses resolved with GetProcAddress. | ||
|  |   if (HMODULE winsock_module = ::GetModuleHandleA("ws2_32")) | ||
|  |   { | ||
|  |     typedef int (WSAAPI *gso_t)(SOCKET, int, int, char*, int*); | ||
|  |     if (gso_t gso = (gso_t)::GetProcAddress(winsock_module, "getsockopt")) | ||
|  |     { | ||
|  |       int tmp_optlen = static_cast<int>(*optlen); | ||
|  |       int result = gso(s, level, optname, | ||
|  |             reinterpret_cast<char*>(optval), &tmp_optlen); | ||
|  |       get_last_error(ec, result != 0); | ||
|  |       *optlen = static_cast<size_t>(tmp_optlen); | ||
|  |       if (result != 0 && level == IPPROTO_IPV6 && optname == IPV6_V6ONLY | ||
|  |           && ec.value() == WSAENOPROTOOPT && *optlen == sizeof(DWORD)) | ||
|  |       { | ||
|  |         // Dual-stack IPv4/v6 sockets, and the IPV6_V6ONLY socket option, are | ||
|  |         // only supported on Windows Vista and later. To simplify program logic | ||
|  |         // we will fake success of getting this option and specify that the | ||
|  |         // value is non-zero (i.e. true). This corresponds to the behavior of | ||
|  |         // IPv6 sockets on Windows platforms pre-Vista. | ||
|  |         *static_cast<DWORD*>(optval) = 1; | ||
|  |         asio::error::clear(ec); | ||
|  |       } | ||
|  |       return result; | ||
|  |     } | ||
|  |   } | ||
|  |   ec = asio::error::fault; | ||
|  |   return socket_error_retval; | ||
|  | #elif defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   int result = call_getsockopt(&msghdr::msg_namelen, | ||
|  |         s, level, optname, optval, optlen); | ||
|  |   get_last_error(ec, result != 0); | ||
|  |   if (result != 0 && level == IPPROTO_IPV6 && optname == IPV6_V6ONLY | ||
|  |       && ec.value() == WSAENOPROTOOPT && *optlen == sizeof(DWORD)) | ||
|  |   { | ||
|  |     // Dual-stack IPv4/v6 sockets, and the IPV6_V6ONLY socket option, are only | ||
|  |     // supported on Windows Vista and later. To simplify program logic we will | ||
|  |     // fake success of getting this option and specify that the value is | ||
|  |     // non-zero (i.e. true). This corresponds to the behavior of IPv6 sockets | ||
|  |     // on Windows platforms pre-Vista. | ||
|  |     *static_cast<DWORD*>(optval) = 1; | ||
|  |     asio::error::clear(ec); | ||
|  |   } | ||
|  |   return result; | ||
|  | #else // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   int result = call_getsockopt(&msghdr::msg_namelen, | ||
|  |         s, level, optname, optval, optlen); | ||
|  |   get_last_error(ec, result != 0); | ||
|  | #if defined(__linux__) | ||
|  |   if (result == 0 && level == SOL_SOCKET && *optlen == sizeof(int) | ||
|  |       && (optname == SO_SNDBUF || optname == SO_RCVBUF)) | ||
|  |   { | ||
|  |     // On Linux, setting SO_SNDBUF or SO_RCVBUF to N actually causes the kernel | ||
|  |     // to set the buffer size to N*2. Linux puts additional stuff into the | ||
|  |     // buffers so that only about half is actually available to the application. | ||
|  |     // The retrieved value is divided by 2 here to make it appear as though the | ||
|  |     // correct value has been set. | ||
|  |     *static_cast<int*>(optval) /= 2; | ||
|  |   } | ||
|  | #endif // defined(__linux__) | ||
|  |   return result; | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | } | ||
|  | 
 | ||
|  | template <typename SockLenType> | ||
|  | inline int call_getpeername(SockLenType msghdr::*, | ||
|  |     socket_type s, void* addr, std::size_t* addrlen) | ||
|  | { | ||
|  |   SockLenType tmp_addrlen = (SockLenType)*addrlen; | ||
|  |   int result = ::getpeername(s, | ||
|  |       static_cast<socket_addr_type*>(addr), &tmp_addrlen); | ||
|  |   *addrlen = (std::size_t)tmp_addrlen; | ||
|  |   return result; | ||
|  | } | ||
|  | 
 | ||
|  | int getpeername(socket_type s, void* addr, std::size_t* addrlen, | ||
|  |     bool cached, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return socket_error_retval; | ||
|  |   } | ||
|  | 
 | ||
|  | #if defined(ASIO_WINDOWS) && !defined(ASIO_WINDOWS_APP) \ | ||
|  |   || defined(__CYGWIN__) | ||
|  |   if (cached) | ||
|  |   { | ||
|  |     // Check if socket is still connected. | ||
|  |     DWORD connect_time = 0; | ||
|  |     size_t connect_time_len = sizeof(connect_time); | ||
|  |     if (socket_ops::getsockopt(s, 0, SOL_SOCKET, SO_CONNECT_TIME, | ||
|  |           &connect_time, &connect_time_len, ec) == socket_error_retval) | ||
|  |     { | ||
|  |       return socket_error_retval; | ||
|  |     } | ||
|  |     if (connect_time == 0xFFFFFFFF) | ||
|  |     { | ||
|  |       ec = asio::error::not_connected; | ||
|  |       return socket_error_retval; | ||
|  |     } | ||
|  | 
 | ||
|  |     // The cached value is still valid. | ||
|  |     asio::error::clear(ec); | ||
|  |     return 0; | ||
|  |   } | ||
|  | #else // defined(ASIO_WINDOWS) && !defined(ASIO_WINDOWS_APP) | ||
|  |       // || defined(__CYGWIN__) | ||
|  |   (void)cached; | ||
|  | #endif // defined(ASIO_WINDOWS) && !defined(ASIO_WINDOWS_APP) | ||
|  |        // || defined(__CYGWIN__) | ||
|  | 
 | ||
|  |   int result = call_getpeername(&msghdr::msg_namelen, s, addr, addrlen); | ||
|  |   get_last_error(ec, result != 0); | ||
|  |   return result; | ||
|  | } | ||
|  | 
 | ||
|  | template <typename SockLenType> | ||
|  | inline int call_getsockname(SockLenType msghdr::*, | ||
|  |     socket_type s, void* addr, std::size_t* addrlen) | ||
|  | { | ||
|  |   SockLenType tmp_addrlen = (SockLenType)*addrlen; | ||
|  |   int result = ::getsockname(s, | ||
|  |       static_cast<socket_addr_type*>(addr), &tmp_addrlen); | ||
|  |   *addrlen = (std::size_t)tmp_addrlen; | ||
|  |   return result; | ||
|  | } | ||
|  | 
 | ||
|  | int getsockname(socket_type s, void* addr, | ||
|  |     std::size_t* addrlen, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return socket_error_retval; | ||
|  |   } | ||
|  | 
 | ||
|  |   int result = call_getsockname(&msghdr::msg_namelen, s, addr, addrlen); | ||
|  |   get_last_error(ec, result != 0); | ||
|  |   return result; | ||
|  | } | ||
|  | 
 | ||
|  | int ioctl(socket_type s, state_type& state, int cmd, | ||
|  |     ioctl_arg_type* arg, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return socket_error_retval; | ||
|  |   } | ||
|  | 
 | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   int result = ::ioctlsocket(s, cmd, arg); | ||
|  | #elif defined(__MACH__) && defined(__APPLE__) \ | ||
|  |   || defined(__NetBSD__) || defined(__FreeBSD__) || defined(__OpenBSD__) | ||
|  |   int result = ::ioctl(s, static_cast<unsigned int>(cmd), arg); | ||
|  | #else | ||
|  |   int result = ::ioctl(s, cmd, arg); | ||
|  | #endif | ||
|  |   get_last_error(ec, result < 0); | ||
|  |   if (result >= 0) | ||
|  |   { | ||
|  |     // When updating the non-blocking mode we always perform the ioctl syscall, | ||
|  |     // even if the flags would otherwise indicate that the socket is already in | ||
|  |     // the correct state. This ensures that the underlying socket is put into | ||
|  |     // the state that has been requested by the user. If the ioctl syscall was | ||
|  |     // successful then we need to update the flags to match. | ||
|  |     if (cmd == static_cast<int>(FIONBIO)) | ||
|  |     { | ||
|  |       if (*arg) | ||
|  |       { | ||
|  |         state |= user_set_non_blocking; | ||
|  |       } | ||
|  |       else | ||
|  |       { | ||
|  |         // Clearing the non-blocking mode always overrides any internally-set | ||
|  |         // non-blocking flag. Any subsequent asynchronous operations will need | ||
|  |         // to re-enable non-blocking I/O. | ||
|  |         state &= ~(user_set_non_blocking | internal_non_blocking); | ||
|  |       } | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   return result; | ||
|  | } | ||
|  | 
 | ||
|  | int select(int nfds, fd_set* readfds, fd_set* writefds, | ||
|  |     fd_set* exceptfds, timeval* timeout, asio::error_code& ec) | ||
|  | { | ||
|  | #if defined(__EMSCRIPTEN__) | ||
|  |   exceptfds = 0; | ||
|  | #endif // defined(__EMSCRIPTEN__) | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   if (!readfds && !writefds && !exceptfds && timeout) | ||
|  |   { | ||
|  |     DWORD milliseconds = timeout->tv_sec * 1000 + timeout->tv_usec / 1000; | ||
|  |     if (milliseconds == 0) | ||
|  |       milliseconds = 1; // Force context switch. | ||
|  |     ::Sleep(milliseconds); | ||
|  |     asio::error::clear(ec); | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   // The select() call allows timeout values measured in microseconds, but the | ||
|  |   // system clock (as wrapped by boost::posix_time::microsec_clock) typically | ||
|  |   // has a resolution of 10 milliseconds. This can lead to a spinning select | ||
|  |   // reactor, meaning increased CPU usage, when waiting for the earliest | ||
|  |   // scheduled timeout if it's less than 10 milliseconds away. To avoid a tight | ||
|  |   // spin we'll use a minimum timeout of 1 millisecond. | ||
|  |   if (timeout && timeout->tv_sec == 0 | ||
|  |       && timeout->tv_usec > 0 && timeout->tv_usec < 1000) | ||
|  |     timeout->tv_usec = 1000; | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | 
 | ||
|  | #if defined(__hpux) && defined(__SELECT) | ||
|  |   timespec ts; | ||
|  |   ts.tv_sec = timeout ? timeout->tv_sec : 0; | ||
|  |   ts.tv_nsec = timeout ? timeout->tv_usec * 1000 : 0; | ||
|  |   int result = ::pselect(nfds, readfds, | ||
|  |         writefds, exceptfds, timeout ? &ts : 0, 0); | ||
|  | #else | ||
|  |   int result = ::select(nfds, readfds, writefds, exceptfds, timeout); | ||
|  | #endif | ||
|  |   get_last_error(ec, result < 0); | ||
|  |   return result; | ||
|  | } | ||
|  | 
 | ||
|  | int poll_read(socket_type s, state_type state, | ||
|  |     int msec, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return socket_error_retval; | ||
|  |   } | ||
|  | 
 | ||
|  | #if defined(ASIO_WINDOWS) \ | ||
|  |   || defined(__CYGWIN__) \ | ||
|  |   || defined(__SYMBIAN32__) | ||
|  |   fd_set fds; | ||
|  |   FD_ZERO(&fds); | ||
|  |   FD_SET(s, &fds); | ||
|  |   timeval timeout_obj; | ||
|  |   timeval* timeout; | ||
|  |   if (state & user_set_non_blocking) | ||
|  |   { | ||
|  |     timeout_obj.tv_sec = 0; | ||
|  |     timeout_obj.tv_usec = 0; | ||
|  |     timeout = &timeout_obj; | ||
|  |   } | ||
|  |   else if (msec >= 0) | ||
|  |   { | ||
|  |     timeout_obj.tv_sec = msec / 1000; | ||
|  |     timeout_obj.tv_usec = (msec % 1000) * 1000; | ||
|  |     timeout = &timeout_obj; | ||
|  |   } | ||
|  |   else | ||
|  |     timeout = 0; | ||
|  |   int result = ::select(s + 1, &fds, 0, 0, timeout); | ||
|  |   get_last_error(ec, result < 0); | ||
|  | #else // defined(ASIO_WINDOWS) | ||
|  |       // || defined(__CYGWIN__) | ||
|  |       // || defined(__SYMBIAN32__) | ||
|  |   pollfd fds; | ||
|  |   fds.fd = s; | ||
|  |   fds.events = POLLIN; | ||
|  |   fds.revents = 0; | ||
|  |   int timeout = (state & user_set_non_blocking) ? 0 : msec; | ||
|  |   int result = ::poll(&fds, 1, timeout); | ||
|  |   get_last_error(ec, result < 0); | ||
|  | #endif // defined(ASIO_WINDOWS) | ||
|  |        // || defined(__CYGWIN__) | ||
|  |        // || defined(__SYMBIAN32__) | ||
|  |   if (result == 0) | ||
|  |     if (state & user_set_non_blocking) | ||
|  |       ec = asio::error::would_block; | ||
|  |   return result; | ||
|  | } | ||
|  | 
 | ||
|  | int poll_write(socket_type s, state_type state, | ||
|  |     int msec, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return socket_error_retval; | ||
|  |   } | ||
|  | 
 | ||
|  | #if defined(ASIO_WINDOWS) \ | ||
|  |   || defined(__CYGWIN__) \ | ||
|  |   || defined(__SYMBIAN32__) | ||
|  |   fd_set fds; | ||
|  |   FD_ZERO(&fds); | ||
|  |   FD_SET(s, &fds); | ||
|  |   timeval timeout_obj; | ||
|  |   timeval* timeout; | ||
|  |   if (state & user_set_non_blocking) | ||
|  |   { | ||
|  |     timeout_obj.tv_sec = 0; | ||
|  |     timeout_obj.tv_usec = 0; | ||
|  |     timeout = &timeout_obj; | ||
|  |   } | ||
|  |   else if (msec >= 0) | ||
|  |   { | ||
|  |     timeout_obj.tv_sec = msec / 1000; | ||
|  |     timeout_obj.tv_usec = (msec % 1000) * 1000; | ||
|  |     timeout = &timeout_obj; | ||
|  |   } | ||
|  |   else | ||
|  |     timeout = 0; | ||
|  |   int result = ::select(s + 1, 0, &fds, 0, timeout); | ||
|  |   get_last_error(ec, result < 0); | ||
|  | #else // defined(ASIO_WINDOWS) | ||
|  |       // || defined(__CYGWIN__) | ||
|  |       // || defined(__SYMBIAN32__) | ||
|  |   pollfd fds; | ||
|  |   fds.fd = s; | ||
|  |   fds.events = POLLOUT; | ||
|  |   fds.revents = 0; | ||
|  |   int timeout = (state & user_set_non_blocking) ? 0 : msec; | ||
|  |   int result = ::poll(&fds, 1, timeout); | ||
|  |   get_last_error(ec, result < 0); | ||
|  | #endif // defined(ASIO_WINDOWS) | ||
|  |        // || defined(__CYGWIN__) | ||
|  |        // || defined(__SYMBIAN32__) | ||
|  |   if (result == 0) | ||
|  |     if (state & user_set_non_blocking) | ||
|  |       ec = asio::error::would_block; | ||
|  |   return result; | ||
|  | } | ||
|  | 
 | ||
|  | int poll_error(socket_type s, state_type state, | ||
|  |     int msec, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return socket_error_retval; | ||
|  |   } | ||
|  | 
 | ||
|  | #if defined(ASIO_WINDOWS) \ | ||
|  |   || defined(__CYGWIN__) \ | ||
|  |   || defined(__SYMBIAN32__) | ||
|  |   fd_set fds; | ||
|  |   FD_ZERO(&fds); | ||
|  |   FD_SET(s, &fds); | ||
|  |   timeval timeout_obj; | ||
|  |   timeval* timeout; | ||
|  |   if (state & user_set_non_blocking) | ||
|  |   { | ||
|  |     timeout_obj.tv_sec = 0; | ||
|  |     timeout_obj.tv_usec = 0; | ||
|  |     timeout = &timeout_obj; | ||
|  |   } | ||
|  |   else if (msec >= 0) | ||
|  |   { | ||
|  |     timeout_obj.tv_sec = msec / 1000; | ||
|  |     timeout_obj.tv_usec = (msec % 1000) * 1000; | ||
|  |     timeout = &timeout_obj; | ||
|  |   } | ||
|  |   else | ||
|  |     timeout = 0; | ||
|  |   int result = ::select(s + 1, 0, 0, &fds, timeout); | ||
|  |   get_last_error(ec, result < 0); | ||
|  | #else // defined(ASIO_WINDOWS) | ||
|  |       // || defined(__CYGWIN__) | ||
|  |       // || defined(__SYMBIAN32__) | ||
|  |   pollfd fds; | ||
|  |   fds.fd = s; | ||
|  |   fds.events = POLLPRI | POLLERR | POLLHUP; | ||
|  |   fds.revents = 0; | ||
|  |   int timeout = (state & user_set_non_blocking) ? 0 : msec; | ||
|  |   int result = ::poll(&fds, 1, timeout); | ||
|  |   get_last_error(ec, result < 0); | ||
|  | #endif // defined(ASIO_WINDOWS) | ||
|  |        // || defined(__CYGWIN__) | ||
|  |        // || defined(__SYMBIAN32__) | ||
|  |   if (result == 0) | ||
|  |     if (state & user_set_non_blocking) | ||
|  |       ec = asio::error::would_block; | ||
|  |   return result; | ||
|  | } | ||
|  | 
 | ||
|  | int poll_connect(socket_type s, int msec, asio::error_code& ec) | ||
|  | { | ||
|  |   if (s == invalid_socket) | ||
|  |   { | ||
|  |     ec = asio::error::bad_descriptor; | ||
|  |     return socket_error_retval; | ||
|  |   } | ||
|  | 
 | ||
|  | #if defined(ASIO_WINDOWS) \ | ||
|  |   || defined(__CYGWIN__) \ | ||
|  |   || defined(__SYMBIAN32__) | ||
|  |   fd_set write_fds; | ||
|  |   FD_ZERO(&write_fds); | ||
|  |   FD_SET(s, &write_fds); | ||
|  |   fd_set except_fds; | ||
|  |   FD_ZERO(&except_fds); | ||
|  |   FD_SET(s, &except_fds); | ||
|  |   timeval timeout_obj; | ||
|  |   timeval* timeout; | ||
|  |   if (msec >= 0) | ||
|  |   { | ||
|  |     timeout_obj.tv_sec = msec / 1000; | ||
|  |     timeout_obj.tv_usec = (msec % 1000) * 1000; | ||
|  |     timeout = &timeout_obj; | ||
|  |   } | ||
|  |   else | ||
|  |     timeout = 0; | ||
|  |   int result = ::select(s + 1, 0, &write_fds, &except_fds, timeout); | ||
|  |   get_last_error(ec, result < 0); | ||
|  |   return result; | ||
|  | #else // defined(ASIO_WINDOWS) | ||
|  |       // || defined(__CYGWIN__) | ||
|  |       // || defined(__SYMBIAN32__) | ||
|  |   pollfd fds; | ||
|  |   fds.fd = s; | ||
|  |   fds.events = POLLOUT; | ||
|  |   fds.revents = 0; | ||
|  |   int result = ::poll(&fds, 1, msec); | ||
|  |   get_last_error(ec, result < 0); | ||
|  |   return result; | ||
|  | #endif // defined(ASIO_WINDOWS) | ||
|  |        // || defined(__CYGWIN__) | ||
|  |        // || defined(__SYMBIAN32__) | ||
|  | } | ||
|  | 
 | ||
|  | #endif // !defined(ASIO_WINDOWS_RUNTIME) | ||
|  | 
 | ||
|  | const char* inet_ntop(int af, const void* src, char* dest, size_t length, | ||
|  |     unsigned long scope_id, asio::error_code& ec) | ||
|  | { | ||
|  |   clear_last_error(); | ||
|  | #if defined(ASIO_WINDOWS_RUNTIME) | ||
|  |   using namespace std; // For sprintf. | ||
|  |   const unsigned char* bytes = static_cast<const unsigned char*>(src); | ||
|  |   if (af == ASIO_OS_DEF(AF_INET)) | ||
|  |   { | ||
|  |     sprintf_s(dest, length, "%u.%u.%u.%u", | ||
|  |         bytes[0], bytes[1], bytes[2], bytes[3]); | ||
|  |     return dest; | ||
|  |   } | ||
|  |   else if (af == ASIO_OS_DEF(AF_INET6)) | ||
|  |   { | ||
|  |     size_t n = 0, b = 0, z = 0; | ||
|  |     while (n < length && b < 16) | ||
|  |     { | ||
|  |       if (bytes[b] == 0 && bytes[b + 1] == 0 && z == 0) | ||
|  |       { | ||
|  |         do b += 2; while (b < 16 && bytes[b] == 0 && bytes[b + 1] == 0); | ||
|  |         n += sprintf_s(dest + n, length - n, ":%s", b < 16 ? "" : ":"), ++z; | ||
|  |       } | ||
|  |       else | ||
|  |       { | ||
|  |         n += sprintf_s(dest + n, length - n, "%s%x", b ? ":" : "", | ||
|  |             (static_cast<u_long_type>(bytes[b]) << 8) | bytes[b + 1]); | ||
|  |         b += 2; | ||
|  |       } | ||
|  |     } | ||
|  |     if (scope_id) | ||
|  |       n += sprintf_s(dest + n, length - n, "%%%lu", scope_id); | ||
|  |     return dest; | ||
|  |   } | ||
|  |   else | ||
|  |   { | ||
|  |     ec = asio::error::address_family_not_supported; | ||
|  |     return 0; | ||
|  |   } | ||
|  | #elif defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   using namespace std; // For memcpy. | ||
|  | 
 | ||
|  |   if (af != ASIO_OS_DEF(AF_INET) && af != ASIO_OS_DEF(AF_INET6)) | ||
|  |   { | ||
|  |     ec = asio::error::address_family_not_supported; | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   union | ||
|  |   { | ||
|  |     socket_addr_type base; | ||
|  |     sockaddr_storage_type storage; | ||
|  |     sockaddr_in4_type v4; | ||
|  |     sockaddr_in6_type v6; | ||
|  |   } address; | ||
|  |   DWORD address_length; | ||
|  |   if (af == ASIO_OS_DEF(AF_INET)) | ||
|  |   { | ||
|  |     address_length = sizeof(sockaddr_in4_type); | ||
|  |     address.v4.sin_family = ASIO_OS_DEF(AF_INET); | ||
|  |     address.v4.sin_port = 0; | ||
|  |     memcpy(&address.v4.sin_addr, src, sizeof(in4_addr_type)); | ||
|  |   } | ||
|  |   else // AF_INET6 | ||
|  |   { | ||
|  |     address_length = sizeof(sockaddr_in6_type); | ||
|  |     address.v6.sin6_family = ASIO_OS_DEF(AF_INET6); | ||
|  |     address.v6.sin6_port = 0; | ||
|  |     address.v6.sin6_flowinfo = 0; | ||
|  |     address.v6.sin6_scope_id = scope_id; | ||
|  |     memcpy(&address.v6.sin6_addr, src, sizeof(in6_addr_type)); | ||
|  |   } | ||
|  | 
 | ||
|  |   DWORD string_length = static_cast<DWORD>(length); | ||
|  | #if defined(BOOST_NO_ANSI_APIS) || (defined(_MSC_VER) && (_MSC_VER >= 1800)) | ||
|  |   LPWSTR string_buffer = (LPWSTR)_alloca(length * sizeof(WCHAR)); | ||
|  |   int result = ::WSAAddressToStringW(&address.base, | ||
|  |         address_length, 0, string_buffer, &string_length); | ||
|  |   get_last_error(ec, true); | ||
|  |   ::WideCharToMultiByte(CP_ACP, 0, string_buffer, -1, | ||
|  |       dest, static_cast<int>(length), 0, 0); | ||
|  | #else | ||
|  |   int result = ::WSAAddressToStringA(&address.base, | ||
|  |       address_length, 0, dest, &string_length); | ||
|  |   get_last_error(ec, true); | ||
|  | #endif | ||
|  | 
 | ||
|  |   // Windows may set error code on success. | ||
|  |   if (result != socket_error_retval) | ||
|  |     asio::error::clear(ec); | ||
|  | 
 | ||
|  |   // Windows may not set an error code on failure. | ||
|  |   else if (result == socket_error_retval && !ec) | ||
|  |     ec = asio::error::invalid_argument; | ||
|  | 
 | ||
|  |   return result == socket_error_retval ? 0 : dest; | ||
|  | #else // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   const char* result = ::inet_ntop(af, src, dest, static_cast<int>(length)); | ||
|  |   get_last_error(ec, true); | ||
|  |   if (result == 0 && !ec) | ||
|  |     ec = asio::error::invalid_argument; | ||
|  |   if (result != 0 && af == ASIO_OS_DEF(AF_INET6) && scope_id != 0) | ||
|  |   { | ||
|  |     using namespace std; // For strcat and sprintf. | ||
|  |     char if_name[(IF_NAMESIZE > 21 ? IF_NAMESIZE : 21) + 1] = "%"; | ||
|  |     const in6_addr_type* ipv6_address = static_cast<const in6_addr_type*>(src); | ||
|  |     bool is_link_local = ((ipv6_address->s6_addr[0] == 0xfe) | ||
|  |         && ((ipv6_address->s6_addr[1] & 0xc0) == 0x80)); | ||
|  |     bool is_multicast_link_local = ((ipv6_address->s6_addr[0] == 0xff) | ||
|  |         && ((ipv6_address->s6_addr[1] & 0x0f) == 0x02)); | ||
|  |     if ((!is_link_local && !is_multicast_link_local) | ||
|  |         || if_indextoname(static_cast<unsigned>(scope_id), if_name + 1) == 0) | ||
|  | #if defined(ASIO_HAS_SNPRINTF) | ||
|  |       snprintf(if_name + 1, sizeof(if_name) - 1, "%lu", scope_id); | ||
|  | #else // defined(ASIO_HAS_SNPRINTF) | ||
|  |       sprintf(if_name + 1, "%lu", scope_id); | ||
|  | #endif // defined(ASIO_HAS_SNPRINTF) | ||
|  |     strcat(dest, if_name); | ||
|  |   } | ||
|  |   return result; | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | } | ||
|  | 
 | ||
|  | int inet_pton(int af, const char* src, void* dest, | ||
|  |     unsigned long* scope_id, asio::error_code& ec) | ||
|  | { | ||
|  |   clear_last_error(); | ||
|  | #if defined(ASIO_WINDOWS_RUNTIME) | ||
|  |   using namespace std; // For sscanf. | ||
|  |   unsigned char* bytes = static_cast<unsigned char*>(dest); | ||
|  |   if (af == ASIO_OS_DEF(AF_INET)) | ||
|  |   { | ||
|  |     unsigned int b0, b1, b2, b3; | ||
|  |     if (sscanf_s(src, "%u.%u.%u.%u", &b0, &b1, &b2, &b3) != 4) | ||
|  |     { | ||
|  |       ec = asio::error::invalid_argument; | ||
|  |       return -1; | ||
|  |     } | ||
|  |     if (b0 > 255 || b1 > 255 || b2 > 255 || b3 > 255) | ||
|  |     { | ||
|  |       ec = asio::error::invalid_argument; | ||
|  |       return -1; | ||
|  |     } | ||
|  |     bytes[0] = static_cast<unsigned char>(b0); | ||
|  |     bytes[1] = static_cast<unsigned char>(b1); | ||
|  |     bytes[2] = static_cast<unsigned char>(b2); | ||
|  |     bytes[3] = static_cast<unsigned char>(b3); | ||
|  |     asio::error::clear(ec); | ||
|  |     return 1; | ||
|  |   } | ||
|  |   else if (af == ASIO_OS_DEF(AF_INET6)) | ||
|  |   { | ||
|  |     unsigned char* bytes = static_cast<unsigned char*>(dest); | ||
|  |     std::memset(bytes, 0, 16); | ||
|  |     unsigned char back_bytes[16] = { 0 }; | ||
|  |     int num_front_bytes = 0, num_back_bytes = 0; | ||
|  |     const char* p = src; | ||
|  | 
 | ||
|  |     enum { fword, fcolon, bword, scope, done } state = fword; | ||
|  |     unsigned long current_word = 0; | ||
|  |     while (state != done) | ||
|  |     { | ||
|  |       if (current_word > 0xFFFF) | ||
|  |       { | ||
|  |         ec = asio::error::invalid_argument; | ||
|  |         return -1; | ||
|  |       } | ||
|  | 
 | ||
|  |       switch (state) | ||
|  |       { | ||
|  |       case fword: | ||
|  |         if (*p >= '0' && *p <= '9') | ||
|  |           current_word = current_word * 16 + *p++ - '0'; | ||
|  |         else if (*p >= 'a' && *p <= 'f') | ||
|  |           current_word = current_word * 16 + *p++ - 'a' + 10; | ||
|  |         else if (*p >= 'A' && *p <= 'F') | ||
|  |           current_word = current_word * 16 + *p++ - 'A' + 10; | ||
|  |         else | ||
|  |         { | ||
|  |           if (num_front_bytes == 16) | ||
|  |           { | ||
|  |             ec = asio::error::invalid_argument; | ||
|  |             return -1; | ||
|  |           } | ||
|  | 
 | ||
|  |           bytes[num_front_bytes++] = (current_word >> 8) & 0xFF; | ||
|  |           bytes[num_front_bytes++] = current_word & 0xFF; | ||
|  |           current_word = 0; | ||
|  | 
 | ||
|  |           if (*p == ':') | ||
|  |             state = fcolon, ++p; | ||
|  |           else if (*p == '%') | ||
|  |             state = scope, ++p; | ||
|  |           else if (*p == 0) | ||
|  |             state = done; | ||
|  |           else | ||
|  |           { | ||
|  |             ec = asio::error::invalid_argument; | ||
|  |             return -1; | ||
|  |           } | ||
|  |         } | ||
|  |         break; | ||
|  | 
 | ||
|  |       case fcolon: | ||
|  |         if (*p == ':') | ||
|  |           state = bword, ++p; | ||
|  |         else | ||
|  |           state = fword; | ||
|  |         break; | ||
|  | 
 | ||
|  |       case bword: | ||
|  |         if (*p >= '0' && *p <= '9') | ||
|  |           current_word = current_word * 16 + *p++ - '0'; | ||
|  |         else if (*p >= 'a' && *p <= 'f') | ||
|  |           current_word = current_word * 16 + *p++ - 'a' + 10; | ||
|  |         else if (*p >= 'A' && *p <= 'F') | ||
|  |           current_word = current_word * 16 + *p++ - 'A' + 10; | ||
|  |         else | ||
|  |         { | ||
|  |           if (num_front_bytes + num_back_bytes == 16) | ||
|  |           { | ||
|  |             ec = asio::error::invalid_argument; | ||
|  |             return -1; | ||
|  |           } | ||
|  | 
 | ||
|  |           back_bytes[num_back_bytes++] = (current_word >> 8) & 0xFF; | ||
|  |           back_bytes[num_back_bytes++] = current_word & 0xFF; | ||
|  |           current_word = 0; | ||
|  | 
 | ||
|  |           if (*p == ':') | ||
|  |             state = bword, ++p; | ||
|  |           else if (*p == '%') | ||
|  |             state = scope, ++p; | ||
|  |           else if (*p == 0) | ||
|  |             state = done; | ||
|  |           else | ||
|  |           { | ||
|  |             ec = asio::error::invalid_argument; | ||
|  |             return -1; | ||
|  |           } | ||
|  |         } | ||
|  |         break; | ||
|  | 
 | ||
|  |       case scope: | ||
|  |         if (*p >= '0' && *p <= '9') | ||
|  |           current_word = current_word * 10 + *p++ - '0'; | ||
|  |         else if (*p == 0) | ||
|  |           *scope_id = current_word, state = done; | ||
|  |         else | ||
|  |         { | ||
|  |           ec = asio::error::invalid_argument; | ||
|  |           return -1; | ||
|  |         } | ||
|  |         break; | ||
|  | 
 | ||
|  |       default: | ||
|  |         break; | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     for (int i = 0; i < num_back_bytes; ++i) | ||
|  |       bytes[16 - num_back_bytes + i] = back_bytes[i]; | ||
|  | 
 | ||
|  |     asio::error::clear(ec); | ||
|  |     return 1; | ||
|  |   } | ||
|  |   else | ||
|  |   { | ||
|  |     ec = asio::error::address_family_not_supported; | ||
|  |     return -1; | ||
|  |   } | ||
|  | #elif defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   using namespace std; // For memcpy and strcmp. | ||
|  | 
 | ||
|  |   if (af != ASIO_OS_DEF(AF_INET) && af != ASIO_OS_DEF(AF_INET6)) | ||
|  |   { | ||
|  |     ec = asio::error::address_family_not_supported; | ||
|  |     return -1; | ||
|  |   } | ||
|  | 
 | ||
|  |   union | ||
|  |   { | ||
|  |     socket_addr_type base; | ||
|  |     sockaddr_storage_type storage; | ||
|  |     sockaddr_in4_type v4; | ||
|  |     sockaddr_in6_type v6; | ||
|  |   } address; | ||
|  |   int address_length = sizeof(sockaddr_storage_type); | ||
|  | #if defined(BOOST_NO_ANSI_APIS) || (defined(_MSC_VER) && (_MSC_VER >= 1800)) | ||
|  |   int num_wide_chars = static_cast<int>(strlen(src)) + 1; | ||
|  |   LPWSTR wide_buffer = (LPWSTR)_alloca(num_wide_chars * sizeof(WCHAR)); | ||
|  |   ::MultiByteToWideChar(CP_ACP, 0, src, -1, wide_buffer, num_wide_chars); | ||
|  |   int result = ::WSAStringToAddressW(wide_buffer, | ||
|  |       af, 0, &address.base, &address_length); | ||
|  |   get_last_error(ec, true); | ||
|  | #else | ||
|  |   int result = ::WSAStringToAddressA(const_cast<char*>(src), | ||
|  |       af, 0, &address.base, &address_length); | ||
|  |   get_last_error(ec, true); | ||
|  | #endif | ||
|  | 
 | ||
|  |   if (af == ASIO_OS_DEF(AF_INET)) | ||
|  |   { | ||
|  |     if (result != socket_error_retval) | ||
|  |     { | ||
|  |       memcpy(dest, &address.v4.sin_addr, sizeof(in4_addr_type)); | ||
|  |       asio::error::clear(ec); | ||
|  |     } | ||
|  |     else if (strcmp(src, "255.255.255.255") == 0) | ||
|  |     { | ||
|  |       static_cast<in4_addr_type*>(dest)->s_addr = INADDR_NONE; | ||
|  |       asio::error::clear(ec); | ||
|  |     } | ||
|  |   } | ||
|  |   else // AF_INET6 | ||
|  |   { | ||
|  |     if (result != socket_error_retval) | ||
|  |     { | ||
|  |       memcpy(dest, &address.v6.sin6_addr, sizeof(in6_addr_type)); | ||
|  |       if (scope_id) | ||
|  |         *scope_id = address.v6.sin6_scope_id; | ||
|  |       asio::error::clear(ec); | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   // Windows may not set an error code on failure. | ||
|  |   if (result == socket_error_retval && !ec) | ||
|  |     ec = asio::error::invalid_argument; | ||
|  | 
 | ||
|  |   if (result != socket_error_retval) | ||
|  |     asio::error::clear(ec); | ||
|  | 
 | ||
|  |   return result == socket_error_retval ? -1 : 1; | ||
|  | #else // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   using namespace std; // For strchr, memcpy and atoi. | ||
|  | 
 | ||
|  |   // On some platforms, inet_pton fails if an address string contains a scope | ||
|  |   // id. Detect and remove the scope id before passing the string to inet_pton. | ||
|  |   const bool is_v6 = (af == ASIO_OS_DEF(AF_INET6)); | ||
|  |   const char* if_name = is_v6 ? strchr(src, '%') : 0; | ||
|  |   char src_buf[max_addr_v6_str_len + 1]; | ||
|  |   const char* src_ptr = src; | ||
|  |   if (if_name != 0) | ||
|  |   { | ||
|  |     if (if_name - src > max_addr_v6_str_len) | ||
|  |     { | ||
|  |       ec = asio::error::invalid_argument; | ||
|  |       return 0; | ||
|  |     } | ||
|  |     memcpy(src_buf, src, if_name - src); | ||
|  |     src_buf[if_name - src] = 0; | ||
|  |     src_ptr = src_buf; | ||
|  |   } | ||
|  | 
 | ||
|  |   int result = ::inet_pton(af, src_ptr, dest); | ||
|  |   get_last_error(ec, true); | ||
|  |   if (result <= 0 && !ec) | ||
|  |     ec = asio::error::invalid_argument; | ||
|  |   if (result > 0 && is_v6 && scope_id) | ||
|  |   { | ||
|  |     using namespace std; // For strchr and atoi. | ||
|  |     *scope_id = 0; | ||
|  |     if (if_name != 0) | ||
|  |     { | ||
|  |       in6_addr_type* ipv6_address = static_cast<in6_addr_type*>(dest); | ||
|  |       bool is_link_local = ((ipv6_address->s6_addr[0] == 0xfe) | ||
|  |           && ((ipv6_address->s6_addr[1] & 0xc0) == 0x80)); | ||
|  |       bool is_multicast_link_local = ((ipv6_address->s6_addr[0] == 0xff) | ||
|  |           && ((ipv6_address->s6_addr[1] & 0x0f) == 0x02)); | ||
|  |       if (is_link_local || is_multicast_link_local) | ||
|  |         *scope_id = if_nametoindex(if_name + 1); | ||
|  |       if (*scope_id == 0) | ||
|  |         *scope_id = atoi(if_name + 1); | ||
|  |     } | ||
|  |   } | ||
|  |   return result; | ||
|  | #endif // defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | } | ||
|  | 
 | ||
|  | int gethostname(char* name, int namelen, asio::error_code& ec) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS_RUNTIME) | ||
|  |   try | ||
|  |   { | ||
|  |     using namespace Windows::Foundation::Collections; | ||
|  |     using namespace Windows::Networking; | ||
|  |     using namespace Windows::Networking::Connectivity; | ||
|  |     IVectorView<HostName^>^ hostnames = NetworkInformation::GetHostNames(); | ||
|  |     for (unsigned i = 0; i < hostnames->Size; ++i) | ||
|  |     { | ||
|  |       HostName^ hostname = hostnames->GetAt(i); | ||
|  |       if (hostname->Type == HostNameType::DomainName) | ||
|  |       { | ||
|  |         std::wstring_convert<std::codecvt_utf8<wchar_t>> converter; | ||
|  |         std::string raw_name = converter.to_bytes(hostname->RawName->Data()); | ||
|  |         if (namelen > 0 && raw_name.size() < static_cast<std::size_t>(namelen)) | ||
|  |         { | ||
|  |           strcpy_s(name, namelen, raw_name.c_str()); | ||
|  |           return 0; | ||
|  |         } | ||
|  |       } | ||
|  |     } | ||
|  |     return -1; | ||
|  |   } | ||
|  |   catch (Platform::Exception^ e) | ||
|  |   { | ||
|  |     ec = asio::error_code(e->HResult, | ||
|  |         asio::system_category()); | ||
|  |     return -1; | ||
|  |   } | ||
|  | #else // defined(ASIO_WINDOWS_RUNTIME) | ||
|  |   int result = ::gethostname(name, namelen); | ||
|  |   get_last_error(ec, result != 0); | ||
|  |   return result; | ||
|  | #endif // defined(ASIO_WINDOWS_RUNTIME) | ||
|  | } | ||
|  | 
 | ||
|  | #if !defined(ASIO_WINDOWS_RUNTIME) | ||
|  | 
 | ||
|  | #if !defined(ASIO_HAS_GETADDRINFO) | ||
|  | 
 | ||
|  | // The following functions are only needed for emulation of getaddrinfo and | ||
|  | // getnameinfo. | ||
|  | 
 | ||
|  | inline asio::error_code translate_netdb_error(int error) | ||
|  | { | ||
|  |   switch (error) | ||
|  |   { | ||
|  |   case 0: | ||
|  |     return asio::error_code(); | ||
|  |   case HOST_NOT_FOUND: | ||
|  |     return asio::error::host_not_found; | ||
|  |   case TRY_AGAIN: | ||
|  |     return asio::error::host_not_found_try_again; | ||
|  |   case NO_RECOVERY: | ||
|  |     return asio::error::no_recovery; | ||
|  |   case NO_DATA: | ||
|  |     return asio::error::no_data; | ||
|  |   default: | ||
|  |     ASIO_ASSERT(false); | ||
|  |     return asio::error::invalid_argument; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | inline hostent* gethostbyaddr(const char* addr, int length, int af, | ||
|  |     hostent* result, char* buffer, int buflength, asio::error_code& ec) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   (void)(buffer); | ||
|  |   (void)(buflength); | ||
|  |   hostent* retval = ::gethostbyaddr(addr, length, af); | ||
|  |   get_last_error(ec, !retval); | ||
|  |   if (!retval) | ||
|  |     return 0; | ||
|  |   *result = *retval; | ||
|  |   return retval; | ||
|  | #elif defined(__sun) || defined(__QNX__) | ||
|  |   int error = 0; | ||
|  |   hostent* retval = ::gethostbyaddr_r(addr, length, | ||
|  |       af, result, buffer, buflength, &error); | ||
|  |   get_last_error(ec, !retval); | ||
|  |   if (error) | ||
|  |     ec = translate_netdb_error(error); | ||
|  |   return retval; | ||
|  | #elif defined(__MACH__) && defined(__APPLE__) | ||
|  |   (void)(buffer); | ||
|  |   (void)(buflength); | ||
|  |   int error = 0; | ||
|  |   hostent* retval = ::getipnodebyaddr(addr, length, af, &error); | ||
|  |   get_last_error(ec, !retval); | ||
|  |   if (error) | ||
|  |     ec = translate_netdb_error(error); | ||
|  |   if (!retval) | ||
|  |     return 0; | ||
|  |   *result = *retval; | ||
|  |   return retval; | ||
|  | #else | ||
|  |   hostent* retval = 0; | ||
|  |   int error = 0; | ||
|  |   clear_last_error(); | ||
|  |   ::gethostbyaddr_r(addr, length, af, result, | ||
|  |       buffer, buflength, &retval, &error); | ||
|  |   get_last_error(ec, true); | ||
|  |   if (error) | ||
|  |     ec = translate_netdb_error(error); | ||
|  |   return retval; | ||
|  | #endif | ||
|  | } | ||
|  | 
 | ||
|  | inline hostent* gethostbyname(const char* name, int af, struct hostent* result, | ||
|  |     char* buffer, int buflength, int ai_flags, asio::error_code& ec) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |   (void)(buffer); | ||
|  |   (void)(buflength); | ||
|  |   (void)(ai_flags); | ||
|  |   if (af != ASIO_OS_DEF(AF_INET)) | ||
|  |   { | ||
|  |     ec = asio::error::address_family_not_supported; | ||
|  |     return 0; | ||
|  |   } | ||
|  |   hostent* retval = ::gethostbyname(name); | ||
|  |   get_last_error(ec, !retval); | ||
|  |   if (!retval) | ||
|  |     return 0; | ||
|  |   *result = *retval; | ||
|  |   return result; | ||
|  | #elif defined(__sun) || defined(__QNX__) | ||
|  |   (void)(ai_flags); | ||
|  |   if (af != ASIO_OS_DEF(AF_INET)) | ||
|  |   { | ||
|  |     ec = asio::error::address_family_not_supported; | ||
|  |     return 0; | ||
|  |   } | ||
|  |   int error = 0; | ||
|  |   hostent* retval = ::gethostbyname_r(name, result, buffer, buflength, &error); | ||
|  |   get_last_error(ec, !retval); | ||
|  |   if (error) | ||
|  |     ec = translate_netdb_error(error); | ||
|  |   return retval; | ||
|  | #elif defined(__MACH__) && defined(__APPLE__) | ||
|  |   (void)(buffer); | ||
|  |   (void)(buflength); | ||
|  |   int error = 0; | ||
|  |   hostent* retval = ::getipnodebyname(name, af, ai_flags, &error); | ||
|  |   get_last_error(ec, !retval); | ||
|  |   if (error) | ||
|  |     ec = translate_netdb_error(error); | ||
|  |   if (!retval) | ||
|  |     return 0; | ||
|  |   *result = *retval; | ||
|  |   return retval; | ||
|  | #else | ||
|  |   (void)(ai_flags); | ||
|  |   if (af != ASIO_OS_DEF(AF_INET)) | ||
|  |   { | ||
|  |     ec = asio::error::address_family_not_supported; | ||
|  |     return 0; | ||
|  |   } | ||
|  |   hostent* retval = 0; | ||
|  |   int error = 0; | ||
|  |   clear_last_error(); | ||
|  |   ::gethostbyname_r(name, result, buffer, buflength, &retval, &error); | ||
|  |   get_last_error(ec, true); | ||
|  |   if (error) | ||
|  |     ec = translate_netdb_error(error); | ||
|  |   return retval; | ||
|  | #endif | ||
|  | } | ||
|  | 
 | ||
|  | inline void freehostent(hostent* h) | ||
|  | { | ||
|  | #if defined(__MACH__) && defined(__APPLE__) | ||
|  |   if (h) | ||
|  |     ::freehostent(h); | ||
|  | #else | ||
|  |   (void)(h); | ||
|  | #endif | ||
|  | } | ||
|  | 
 | ||
|  | // Emulation of getaddrinfo based on implementation in: | ||
|  | // Stevens, W. R., UNIX Network Programming Vol. 1, 2nd Ed., Prentice-Hall 1998. | ||
|  | 
 | ||
|  | struct gai_search | ||
|  | { | ||
|  |   const char* host; | ||
|  |   int family; | ||
|  | }; | ||
|  | 
 | ||
|  | inline int gai_nsearch(const char* host, | ||
|  |     const addrinfo_type* hints, gai_search (&search)[2]) | ||
|  | { | ||
|  |   int search_count = 0; | ||
|  |   if (host == 0 || host[0] == '\0') | ||
|  |   { | ||
|  |     if (hints->ai_flags & AI_PASSIVE) | ||
|  |     { | ||
|  |       // No host and AI_PASSIVE implies wildcard bind. | ||
|  |       switch (hints->ai_family) | ||
|  |       { | ||
|  |       case ASIO_OS_DEF(AF_INET): | ||
|  |         search[search_count].host = "0.0.0.0"; | ||
|  |         search[search_count].family = ASIO_OS_DEF(AF_INET); | ||
|  |         ++search_count; | ||
|  |         break; | ||
|  |       case ASIO_OS_DEF(AF_INET6): | ||
|  |         search[search_count].host = "0::0"; | ||
|  |         search[search_count].family = ASIO_OS_DEF(AF_INET6); | ||
|  |         ++search_count; | ||
|  |         break; | ||
|  |       case ASIO_OS_DEF(AF_UNSPEC): | ||
|  |         search[search_count].host = "0::0"; | ||
|  |         search[search_count].family = ASIO_OS_DEF(AF_INET6); | ||
|  |         ++search_count; | ||
|  |         search[search_count].host = "0.0.0.0"; | ||
|  |         search[search_count].family = ASIO_OS_DEF(AF_INET); | ||
|  |         ++search_count; | ||
|  |         break; | ||
|  |       default: | ||
|  |         break; | ||
|  |       } | ||
|  |     } | ||
|  |     else | ||
|  |     { | ||
|  |       // No host and not AI_PASSIVE means connect to local host. | ||
|  |       switch (hints->ai_family) | ||
|  |       { | ||
|  |       case ASIO_OS_DEF(AF_INET): | ||
|  |         search[search_count].host = "localhost"; | ||
|  |         search[search_count].family = ASIO_OS_DEF(AF_INET); | ||
|  |         ++search_count; | ||
|  |         break; | ||
|  |       case ASIO_OS_DEF(AF_INET6): | ||
|  |         search[search_count].host = "localhost"; | ||
|  |         search[search_count].family = ASIO_OS_DEF(AF_INET6); | ||
|  |         ++search_count; | ||
|  |         break; | ||
|  |       case ASIO_OS_DEF(AF_UNSPEC): | ||
|  |         search[search_count].host = "localhost"; | ||
|  |         search[search_count].family = ASIO_OS_DEF(AF_INET6); | ||
|  |         ++search_count; | ||
|  |         search[search_count].host = "localhost"; | ||
|  |         search[search_count].family = ASIO_OS_DEF(AF_INET); | ||
|  |         ++search_count; | ||
|  |         break; | ||
|  |       default: | ||
|  |         break; | ||
|  |       } | ||
|  |     } | ||
|  |   } | ||
|  |   else | ||
|  |   { | ||
|  |     // Host is specified. | ||
|  |     switch (hints->ai_family) | ||
|  |     { | ||
|  |     case ASIO_OS_DEF(AF_INET): | ||
|  |       search[search_count].host = host; | ||
|  |       search[search_count].family = ASIO_OS_DEF(AF_INET); | ||
|  |       ++search_count; | ||
|  |       break; | ||
|  |     case ASIO_OS_DEF(AF_INET6): | ||
|  |       search[search_count].host = host; | ||
|  |       search[search_count].family = ASIO_OS_DEF(AF_INET6); | ||
|  |       ++search_count; | ||
|  |       break; | ||
|  |     case ASIO_OS_DEF(AF_UNSPEC): | ||
|  |       search[search_count].host = host; | ||
|  |       search[search_count].family = ASIO_OS_DEF(AF_INET6); | ||
|  |       ++search_count; | ||
|  |       search[search_count].host = host; | ||
|  |       search[search_count].family = ASIO_OS_DEF(AF_INET); | ||
|  |       ++search_count; | ||
|  |       break; | ||
|  |     default: | ||
|  |       break; | ||
|  |     } | ||
|  |   } | ||
|  |   return search_count; | ||
|  | } | ||
|  | 
 | ||
|  | template <typename T> | ||
|  | inline T* gai_alloc(std::size_t size = sizeof(T)) | ||
|  | { | ||
|  |   using namespace std; | ||
|  |   T* p = static_cast<T*>(::operator new(size, std::nothrow)); | ||
|  |   if (p) | ||
|  |     memset(p, 0, size); | ||
|  |   return p; | ||
|  | } | ||
|  | 
 | ||
|  | inline void gai_free(void* p) | ||
|  | { | ||
|  |   ::operator delete(p); | ||
|  | } | ||
|  | 
 | ||
|  | inline void gai_strcpy(char* target, const char* source, std::size_t max_size) | ||
|  | { | ||
|  |   using namespace std; | ||
|  | #if defined(ASIO_HAS_SECURE_RTL) | ||
|  |   strcpy_s(target, max_size, source); | ||
|  | #else // defined(ASIO_HAS_SECURE_RTL) | ||
|  |   *target = 0; | ||
|  |   if (max_size > 0) | ||
|  |     strncat(target, source, max_size - 1); | ||
|  | #endif // defined(ASIO_HAS_SECURE_RTL) | ||
|  | } | ||
|  | 
 | ||
|  | enum { gai_clone_flag = 1 << 30 }; | ||
|  | 
 | ||
|  | inline int gai_aistruct(addrinfo_type*** next, const addrinfo_type* hints, | ||
|  |     const void* addr, int family) | ||
|  | { | ||
|  |   using namespace std; | ||
|  | 
 | ||
|  |   addrinfo_type* ai = gai_alloc<addrinfo_type>(); | ||
|  |   if (ai == 0) | ||
|  |     return EAI_MEMORY; | ||
|  | 
 | ||
|  |   ai->ai_next = 0; | ||
|  |   **next = ai; | ||
|  |   *next = &ai->ai_next; | ||
|  | 
 | ||
|  |   ai->ai_canonname = 0; | ||
|  |   ai->ai_socktype = hints->ai_socktype; | ||
|  |   if (ai->ai_socktype == 0) | ||
|  |     ai->ai_flags |= gai_clone_flag; | ||
|  |   ai->ai_protocol = hints->ai_protocol; | ||
|  |   ai->ai_family = family; | ||
|  | 
 | ||
|  |   switch (ai->ai_family) | ||
|  |   { | ||
|  |   case ASIO_OS_DEF(AF_INET): | ||
|  |     { | ||
|  |       sockaddr_in4_type* sinptr = gai_alloc<sockaddr_in4_type>(); | ||
|  |       if (sinptr == 0) | ||
|  |         return EAI_MEMORY; | ||
|  |       sinptr->sin_family = ASIO_OS_DEF(AF_INET); | ||
|  |       memcpy(&sinptr->sin_addr, addr, sizeof(in4_addr_type)); | ||
|  |       ai->ai_addr = reinterpret_cast<sockaddr*>(sinptr); | ||
|  |       ai->ai_addrlen = sizeof(sockaddr_in4_type); | ||
|  |       break; | ||
|  |     } | ||
|  |   case ASIO_OS_DEF(AF_INET6): | ||
|  |     { | ||
|  |       sockaddr_in6_type* sin6ptr = gai_alloc<sockaddr_in6_type>(); | ||
|  |       if (sin6ptr == 0) | ||
|  |         return EAI_MEMORY; | ||
|  |       sin6ptr->sin6_family = ASIO_OS_DEF(AF_INET6); | ||
|  |       memcpy(&sin6ptr->sin6_addr, addr, sizeof(in6_addr_type)); | ||
|  |       ai->ai_addr = reinterpret_cast<sockaddr*>(sin6ptr); | ||
|  |       ai->ai_addrlen = sizeof(sockaddr_in6_type); | ||
|  |       break; | ||
|  |     } | ||
|  |   default: | ||
|  |     break; | ||
|  |   } | ||
|  | 
 | ||
|  |   return 0; | ||
|  | } | ||
|  | 
 | ||
|  | inline addrinfo_type* gai_clone(addrinfo_type* ai) | ||
|  | { | ||
|  |   using namespace std; | ||
|  | 
 | ||
|  |   addrinfo_type* new_ai = gai_alloc<addrinfo_type>(); | ||
|  |   if (new_ai == 0) | ||
|  |     return new_ai; | ||
|  | 
 | ||
|  |   new_ai->ai_next = ai->ai_next; | ||
|  |   ai->ai_next = new_ai; | ||
|  | 
 | ||
|  |   new_ai->ai_flags = 0; | ||
|  |   new_ai->ai_family = ai->ai_family; | ||
|  |   new_ai->ai_socktype = ai->ai_socktype; | ||
|  |   new_ai->ai_protocol = ai->ai_protocol; | ||
|  |   new_ai->ai_canonname = 0; | ||
|  |   new_ai->ai_addrlen = ai->ai_addrlen; | ||
|  |   new_ai->ai_addr = gai_alloc<sockaddr>(ai->ai_addrlen); | ||
|  |   memcpy(new_ai->ai_addr, ai->ai_addr, ai->ai_addrlen); | ||
|  | 
 | ||
|  |   return new_ai; | ||
|  | } | ||
|  | 
 | ||
|  | inline int gai_port(addrinfo_type* aihead, int port, int socktype) | ||
|  | { | ||
|  |   int num_found = 0; | ||
|  | 
 | ||
|  |   for (addrinfo_type* ai = aihead; ai; ai = ai->ai_next) | ||
|  |   { | ||
|  |     if (ai->ai_flags & gai_clone_flag) | ||
|  |     { | ||
|  |       if (ai->ai_socktype != 0) | ||
|  |       { | ||
|  |         ai = gai_clone(ai); | ||
|  |         if (ai == 0) | ||
|  |           return -1; | ||
|  |         // ai now points to newly cloned entry. | ||
|  |       } | ||
|  |     } | ||
|  |     else if (ai->ai_socktype != socktype) | ||
|  |     { | ||
|  |       // Ignore if mismatch on socket type. | ||
|  |       continue; | ||
|  |     } | ||
|  | 
 | ||
|  |     ai->ai_socktype = socktype; | ||
|  | 
 | ||
|  |     switch (ai->ai_family) | ||
|  |     { | ||
|  |     case ASIO_OS_DEF(AF_INET): | ||
|  |       { | ||
|  |         sockaddr_in4_type* sinptr = | ||
|  |           reinterpret_cast<sockaddr_in4_type*>(ai->ai_addr); | ||
|  |         sinptr->sin_port = port; | ||
|  |         ++num_found; | ||
|  |         break; | ||
|  |       } | ||
|  |     case ASIO_OS_DEF(AF_INET6): | ||
|  |       { | ||
|  |         sockaddr_in6_type* sin6ptr = | ||
|  |           reinterpret_cast<sockaddr_in6_type*>(ai->ai_addr); | ||
|  |         sin6ptr->sin6_port = port; | ||
|  |         ++num_found; | ||
|  |         break; | ||
|  |       } | ||
|  |     default: | ||
|  |       break; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   return num_found; | ||
|  | } | ||
|  | 
 | ||
|  | inline int gai_serv(addrinfo_type* aihead, | ||
|  |     const addrinfo_type* hints, const char* serv) | ||
|  | { | ||
|  |   using namespace std; | ||
|  | 
 | ||
|  |   int num_found = 0; | ||
|  | 
 | ||
|  |   if ( | ||
|  | #if defined(AI_NUMERICSERV) | ||
|  |       (hints->ai_flags & AI_NUMERICSERV) || | ||
|  | #endif | ||
|  |       isdigit(static_cast<unsigned char>(serv[0]))) | ||
|  |   { | ||
|  |     int port = htons(atoi(serv)); | ||
|  |     if (hints->ai_socktype) | ||
|  |     { | ||
|  |       // Caller specifies socket type. | ||
|  |       int rc = gai_port(aihead, port, hints->ai_socktype); | ||
|  |       if (rc < 0) | ||
|  |         return EAI_MEMORY; | ||
|  |       num_found += rc; | ||
|  |     } | ||
|  |     else | ||
|  |     { | ||
|  |       // Caller does not specify socket type. | ||
|  |       int rc = gai_port(aihead, port, SOCK_STREAM); | ||
|  |       if (rc < 0) | ||
|  |         return EAI_MEMORY; | ||
|  |       num_found += rc; | ||
|  |       rc = gai_port(aihead, port, SOCK_DGRAM); | ||
|  |       if (rc < 0) | ||
|  |         return EAI_MEMORY; | ||
|  |       num_found += rc; | ||
|  |     } | ||
|  |   } | ||
|  |   else | ||
|  |   { | ||
|  |     // Try service name with TCP first, then UDP. | ||
|  |     if (hints->ai_socktype == 0 || hints->ai_socktype == SOCK_STREAM) | ||
|  |     { | ||
|  |       servent* sptr = getservbyname(serv, "tcp"); | ||
|  |       if (sptr != 0) | ||
|  |       { | ||
|  |         int rc = gai_port(aihead, sptr->s_port, SOCK_STREAM); | ||
|  |         if (rc < 0) | ||
|  |           return EAI_MEMORY; | ||
|  |         num_found += rc; | ||
|  |       } | ||
|  |     } | ||
|  |     if (hints->ai_socktype == 0 || hints->ai_socktype == SOCK_DGRAM) | ||
|  |     { | ||
|  |       servent* sptr = getservbyname(serv, "udp"); | ||
|  |       if (sptr != 0) | ||
|  |       { | ||
|  |         int rc = gai_port(aihead, sptr->s_port, SOCK_DGRAM); | ||
|  |         if (rc < 0) | ||
|  |           return EAI_MEMORY; | ||
|  |         num_found += rc; | ||
|  |       } | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   if (num_found == 0) | ||
|  |   { | ||
|  |     if (hints->ai_socktype == 0) | ||
|  |     { | ||
|  |       // All calls to getservbyname() failed. | ||
|  |       return EAI_NONAME; | ||
|  |     } | ||
|  |     else | ||
|  |     { | ||
|  |       // Service not supported for socket type. | ||
|  |       return EAI_SERVICE; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   return 0; | ||
|  | } | ||
|  | 
 | ||
|  | inline int gai_echeck(const char* host, const char* service, | ||
|  |     int flags, int family, int socktype, int protocol) | ||
|  | { | ||
|  |   (void)(flags); | ||
|  |   (void)(protocol); | ||
|  | 
 | ||
|  |   // Host or service must be specified. | ||
|  |   if (host == 0 || host[0] == '\0') | ||
|  |     if (service == 0 || service[0] == '\0') | ||
|  |       return EAI_NONAME; | ||
|  | 
 | ||
|  |   // Check combination of family and socket type. | ||
|  |   switch (family) | ||
|  |   { | ||
|  |   case ASIO_OS_DEF(AF_UNSPEC): | ||
|  |     break; | ||
|  |   case ASIO_OS_DEF(AF_INET): | ||
|  |   case ASIO_OS_DEF(AF_INET6): | ||
|  |     if (service != 0 && service[0] != '\0') | ||
|  |       if (socktype != 0 && socktype != SOCK_STREAM && socktype != SOCK_DGRAM) | ||
|  |         return EAI_SOCKTYPE; | ||
|  |     break; | ||
|  |   default: | ||
|  |     return EAI_FAMILY; | ||
|  |   } | ||
|  | 
 | ||
|  |   return 0; | ||
|  | } | ||
|  | 
 | ||
|  | inline void freeaddrinfo_emulation(addrinfo_type* aihead) | ||
|  | { | ||
|  |   addrinfo_type* ai = aihead; | ||
|  |   while (ai) | ||
|  |   { | ||
|  |     gai_free(ai->ai_addr); | ||
|  |     gai_free(ai->ai_canonname); | ||
|  |     addrinfo_type* ainext = ai->ai_next; | ||
|  |     gai_free(ai); | ||
|  |     ai = ainext; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | inline int getaddrinfo_emulation(const char* host, const char* service, | ||
|  |     const addrinfo_type* hintsp, addrinfo_type** result) | ||
|  | { | ||
|  |   // Set up linked list of addrinfo structures. | ||
|  |   addrinfo_type* aihead = 0; | ||
|  |   addrinfo_type** ainext = &aihead; | ||
|  |   char* canon = 0; | ||
|  | 
 | ||
|  |   // Supply default hints if not specified by caller. | ||
|  |   addrinfo_type hints = addrinfo_type(); | ||
|  |   hints.ai_family = ASIO_OS_DEF(AF_UNSPEC); | ||
|  |   if (hintsp) | ||
|  |     hints = *hintsp; | ||
|  | 
 | ||
|  |   // If the resolution is not specifically for AF_INET6, remove the AI_V4MAPPED | ||
|  |   // and AI_ALL flags. | ||
|  | #if defined(AI_V4MAPPED) | ||
|  |   if (hints.ai_family != ASIO_OS_DEF(AF_INET6)) | ||
|  |     hints.ai_flags &= ~AI_V4MAPPED; | ||
|  | #endif | ||
|  | #if defined(AI_ALL) | ||
|  |   if (hints.ai_family != ASIO_OS_DEF(AF_INET6)) | ||
|  |     hints.ai_flags &= ~AI_ALL; | ||
|  | #endif | ||
|  | 
 | ||
|  |   // Basic error checking. | ||
|  |   int rc = gai_echeck(host, service, hints.ai_flags, hints.ai_family, | ||
|  |       hints.ai_socktype, hints.ai_protocol); | ||
|  |   if (rc != 0) | ||
|  |   { | ||
|  |     freeaddrinfo_emulation(aihead); | ||
|  |     return rc; | ||
|  |   } | ||
|  | 
 | ||
|  |   gai_search search[2]; | ||
|  |   int search_count = gai_nsearch(host, &hints, search); | ||
|  |   for (gai_search* sptr = search; sptr < search + search_count; ++sptr) | ||
|  |   { | ||
|  |     // Check for IPv4 dotted decimal string. | ||
|  |     in4_addr_type inaddr; | ||
|  |     asio::error_code ec; | ||
|  |     if (socket_ops::inet_pton(ASIO_OS_DEF(AF_INET), | ||
|  |           sptr->host, &inaddr, 0, ec) == 1) | ||
|  |     { | ||
|  |       if (hints.ai_family != ASIO_OS_DEF(AF_UNSPEC) | ||
|  |           && hints.ai_family != ASIO_OS_DEF(AF_INET)) | ||
|  |       { | ||
|  |         freeaddrinfo_emulation(aihead); | ||
|  |         gai_free(canon); | ||
|  |         return EAI_FAMILY; | ||
|  |       } | ||
|  |       if (sptr->family == ASIO_OS_DEF(AF_INET)) | ||
|  |       { | ||
|  |         rc = gai_aistruct(&ainext, &hints, &inaddr, ASIO_OS_DEF(AF_INET)); | ||
|  |         if (rc != 0) | ||
|  |         { | ||
|  |           freeaddrinfo_emulation(aihead); | ||
|  |           gai_free(canon); | ||
|  |           return rc; | ||
|  |         } | ||
|  |       } | ||
|  |       continue; | ||
|  |     } | ||
|  | 
 | ||
|  |     // Check for IPv6 hex string. | ||
|  |     in6_addr_type in6addr; | ||
|  |     if (socket_ops::inet_pton(ASIO_OS_DEF(AF_INET6), | ||
|  |           sptr->host, &in6addr, 0, ec) == 1) | ||
|  |     { | ||
|  |       if (hints.ai_family != ASIO_OS_DEF(AF_UNSPEC) | ||
|  |           && hints.ai_family != ASIO_OS_DEF(AF_INET6)) | ||
|  |       { | ||
|  |         freeaddrinfo_emulation(aihead); | ||
|  |         gai_free(canon); | ||
|  |         return EAI_FAMILY; | ||
|  |       } | ||
|  |       if (sptr->family == ASIO_OS_DEF(AF_INET6)) | ||
|  |       { | ||
|  |         rc = gai_aistruct(&ainext, &hints, &in6addr, | ||
|  |             ASIO_OS_DEF(AF_INET6)); | ||
|  |         if (rc != 0) | ||
|  |         { | ||
|  |           freeaddrinfo_emulation(aihead); | ||
|  |           gai_free(canon); | ||
|  |           return rc; | ||
|  |         } | ||
|  |       } | ||
|  |       continue; | ||
|  |     } | ||
|  | 
 | ||
|  |     // Look up hostname. | ||
|  |     hostent hent; | ||
|  |     char hbuf[8192] = ""; | ||
|  |     hostent* hptr = socket_ops::gethostbyname(sptr->host, | ||
|  |         sptr->family, &hent, hbuf, sizeof(hbuf), hints.ai_flags, ec); | ||
|  |     if (hptr == 0) | ||
|  |     { | ||
|  |       if (search_count == 2) | ||
|  |       { | ||
|  |         // Failure is OK if there are multiple searches. | ||
|  |         continue; | ||
|  |       } | ||
|  |       freeaddrinfo_emulation(aihead); | ||
|  |       gai_free(canon); | ||
|  |       if (ec == asio::error::host_not_found) | ||
|  |         return EAI_NONAME; | ||
|  |       if (ec == asio::error::host_not_found_try_again) | ||
|  |         return EAI_AGAIN; | ||
|  |       if (ec == asio::error::no_recovery) | ||
|  |         return EAI_FAIL; | ||
|  |       if (ec == asio::error::no_data) | ||
|  |         return EAI_NONAME; | ||
|  |       return EAI_NONAME; | ||
|  |     } | ||
|  | 
 | ||
|  |     // Check for address family mismatch if one was specified. | ||
|  |     if (hints.ai_family != ASIO_OS_DEF(AF_UNSPEC) | ||
|  |         && hints.ai_family != hptr->h_addrtype) | ||
|  |     { | ||
|  |       freeaddrinfo_emulation(aihead); | ||
|  |       gai_free(canon); | ||
|  |       socket_ops::freehostent(hptr); | ||
|  |       return EAI_FAMILY; | ||
|  |     } | ||
|  | 
 | ||
|  |     // Save canonical name first time. | ||
|  |     if (host != 0 && host[0] != '\0' && hptr->h_name && hptr->h_name[0] | ||
|  |         && (hints.ai_flags & AI_CANONNAME) && canon == 0) | ||
|  |     { | ||
|  |       std::size_t canon_len = strlen(hptr->h_name) + 1; | ||
|  |       canon = gai_alloc<char>(canon_len); | ||
|  |       if (canon == 0) | ||
|  |       { | ||
|  |         freeaddrinfo_emulation(aihead); | ||
|  |         socket_ops::freehostent(hptr); | ||
|  |         return EAI_MEMORY; | ||
|  |       } | ||
|  |       gai_strcpy(canon, hptr->h_name, canon_len); | ||
|  |     } | ||
|  | 
 | ||
|  |     // Create an addrinfo structure for each returned address. | ||
|  |     for (char** ap = hptr->h_addr_list; *ap; ++ap) | ||
|  |     { | ||
|  |       rc = gai_aistruct(&ainext, &hints, *ap, hptr->h_addrtype); | ||
|  |       if (rc != 0) | ||
|  |       { | ||
|  |         freeaddrinfo_emulation(aihead); | ||
|  |         gai_free(canon); | ||
|  |         socket_ops::freehostent(hptr); | ||
|  |         return EAI_FAMILY; | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     socket_ops::freehostent(hptr); | ||
|  |   } | ||
|  | 
 | ||
|  |   // Check if we found anything. | ||
|  |   if (aihead == 0) | ||
|  |   { | ||
|  |     gai_free(canon); | ||
|  |     return EAI_NONAME; | ||
|  |   } | ||
|  | 
 | ||
|  |   // Return canonical name in first entry. | ||
|  |   if (host != 0 && host[0] != '\0' && (hints.ai_flags & AI_CANONNAME)) | ||
|  |   { | ||
|  |     if (canon) | ||
|  |     { | ||
|  |       aihead->ai_canonname = canon; | ||
|  |       canon = 0; | ||
|  |     } | ||
|  |     else | ||
|  |     { | ||
|  |       std::size_t canonname_len = strlen(search[0].host) + 1; | ||
|  |       aihead->ai_canonname = gai_alloc<char>(canonname_len); | ||
|  |       if (aihead->ai_canonname == 0) | ||
|  |       { | ||
|  |         freeaddrinfo_emulation(aihead); | ||
|  |         return EAI_MEMORY; | ||
|  |       } | ||
|  |       gai_strcpy(aihead->ai_canonname, search[0].host, canonname_len); | ||
|  |     } | ||
|  |   } | ||
|  |   gai_free(canon); | ||
|  | 
 | ||
|  |   // Process the service name. | ||
|  |   if (service != 0 && service[0] != '\0') | ||
|  |   { | ||
|  |     rc = gai_serv(aihead, &hints, service); | ||
|  |     if (rc != 0) | ||
|  |     { | ||
|  |       freeaddrinfo_emulation(aihead); | ||
|  |       return rc; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   // Return result to caller. | ||
|  |   *result = aihead; | ||
|  |   return 0; | ||
|  | } | ||
|  | 
 | ||
|  | inline asio::error_code getnameinfo_emulation( | ||
|  |     const socket_addr_type* sa, std::size_t salen, char* host, | ||
|  |     std::size_t hostlen, char* serv, std::size_t servlen, int flags, | ||
|  |     asio::error_code& ec) | ||
|  | { | ||
|  |   using namespace std; | ||
|  | 
 | ||
|  |   const char* addr; | ||
|  |   size_t addr_len; | ||
|  |   unsigned short port; | ||
|  |   switch (sa->sa_family) | ||
|  |   { | ||
|  |   case ASIO_OS_DEF(AF_INET): | ||
|  |     if (salen != sizeof(sockaddr_in4_type)) | ||
|  |     { | ||
|  |       return ec = asio::error::invalid_argument; | ||
|  |     } | ||
|  |     addr = reinterpret_cast<const char*>( | ||
|  |         &reinterpret_cast<const sockaddr_in4_type*>(sa)->sin_addr); | ||
|  |     addr_len = sizeof(in4_addr_type); | ||
|  |     port = reinterpret_cast<const sockaddr_in4_type*>(sa)->sin_port; | ||
|  |     break; | ||
|  |   case ASIO_OS_DEF(AF_INET6): | ||
|  |     if (salen != sizeof(sockaddr_in6_type)) | ||
|  |     { | ||
|  |       return ec = asio::error::invalid_argument; | ||
|  |     } | ||
|  |     addr = reinterpret_cast<const char*>( | ||
|  |         &reinterpret_cast<const sockaddr_in6_type*>(sa)->sin6_addr); | ||
|  |     addr_len = sizeof(in6_addr_type); | ||
|  |     port = reinterpret_cast<const sockaddr_in6_type*>(sa)->sin6_port; | ||
|  |     break; | ||
|  |   default: | ||
|  |     return ec = asio::error::address_family_not_supported; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (host && hostlen > 0) | ||
|  |   { | ||
|  |     if (flags & NI_NUMERICHOST) | ||
|  |     { | ||
|  |       if (socket_ops::inet_ntop(sa->sa_family, addr, host, hostlen, 0, ec) == 0) | ||
|  |       { | ||
|  |         return ec; | ||
|  |       } | ||
|  |     } | ||
|  |     else | ||
|  |     { | ||
|  |       hostent hent; | ||
|  |       char hbuf[8192] = ""; | ||
|  |       hostent* hptr = socket_ops::gethostbyaddr(addr, | ||
|  |           static_cast<int>(addr_len), sa->sa_family, | ||
|  |           &hent, hbuf, sizeof(hbuf), ec); | ||
|  |       if (hptr && hptr->h_name && hptr->h_name[0] != '\0') | ||
|  |       { | ||
|  |         if (flags & NI_NOFQDN) | ||
|  |         { | ||
|  |           char* dot = strchr(hptr->h_name, '.'); | ||
|  |           if (dot) | ||
|  |           { | ||
|  |             *dot = 0; | ||
|  |           } | ||
|  |         } | ||
|  |         gai_strcpy(host, hptr->h_name, hostlen); | ||
|  |         socket_ops::freehostent(hptr); | ||
|  |       } | ||
|  |       else | ||
|  |       { | ||
|  |         socket_ops::freehostent(hptr); | ||
|  |         if (flags & NI_NAMEREQD) | ||
|  |         { | ||
|  |           return ec = asio::error::host_not_found; | ||
|  |         } | ||
|  |         if (socket_ops::inet_ntop(sa->sa_family, | ||
|  |               addr, host, hostlen, 0, ec) == 0) | ||
|  |         { | ||
|  |           return ec; | ||
|  |         } | ||
|  |       } | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   if (serv && servlen > 0) | ||
|  |   { | ||
|  |     if (flags & NI_NUMERICSERV) | ||
|  |     { | ||
|  |       if (servlen < 6) | ||
|  |       { | ||
|  |         return ec = asio::error::no_buffer_space; | ||
|  |       } | ||
|  | #if defined(ASIO_HAS_SNPRINTF) | ||
|  |       snprintf(serv, servlen, "%u", ntohs(port)); | ||
|  | #elif defined(ASIO_HAS_SECURE_RTL) | ||
|  |       sprintf_s(serv, servlen, "%u", ntohs(port)); | ||
|  | #else // defined(ASIO_HAS_SECURE_RTL) | ||
|  |       sprintf(serv, "%u", ntohs(port)); | ||
|  | #endif // defined(ASIO_HAS_SECURE_RTL) | ||
|  |     } | ||
|  |     else | ||
|  |     { | ||
|  | #if defined(ASIO_HAS_PTHREADS) | ||
|  |       static ::pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER; | ||
|  |       ::pthread_mutex_lock(&mutex); | ||
|  | #endif // defined(ASIO_HAS_PTHREADS) | ||
|  |       servent* sptr = ::getservbyport(port, (flags & NI_DGRAM) ? "udp" : 0); | ||
|  |       if (sptr && sptr->s_name && sptr->s_name[0] != '\0') | ||
|  |       { | ||
|  |         gai_strcpy(serv, sptr->s_name, servlen); | ||
|  |       } | ||
|  |       else | ||
|  |       { | ||
|  |         if (servlen < 6) | ||
|  |         { | ||
|  |           return ec = asio::error::no_buffer_space; | ||
|  |         } | ||
|  | #if defined(ASIO_HAS_SNPRINTF) | ||
|  |         snprintf(serv, servlen, "%u", ntohs(port)); | ||
|  | #elif defined(ASIO_HAS_SECURE_RTL) | ||
|  |         sprintf_s(serv, servlen, "%u", ntohs(port)); | ||
|  | #else // defined(ASIO_HAS_SECURE_RTL) | ||
|  |         sprintf(serv, "%u", ntohs(port)); | ||
|  | #endif // defined(ASIO_HAS_SECURE_RTL) | ||
|  |       } | ||
|  | #if defined(ASIO_HAS_PTHREADS) | ||
|  |       ::pthread_mutex_unlock(&mutex); | ||
|  | #endif // defined(ASIO_HAS_PTHREADS) | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   asio::error::clear(ec); | ||
|  |   return ec; | ||
|  | } | ||
|  | 
 | ||
|  | #endif // !defined(ASIO_HAS_GETADDRINFO) | ||
|  | 
 | ||
|  | inline asio::error_code translate_addrinfo_error(int error) | ||
|  | { | ||
|  |   switch (error) | ||
|  |   { | ||
|  |   case 0: | ||
|  |     return asio::error_code(); | ||
|  |   case EAI_AGAIN: | ||
|  |     return asio::error::host_not_found_try_again; | ||
|  |   case EAI_BADFLAGS: | ||
|  |     return asio::error::invalid_argument; | ||
|  |   case EAI_FAIL: | ||
|  |     return asio::error::no_recovery; | ||
|  |   case EAI_FAMILY: | ||
|  |     return asio::error::address_family_not_supported; | ||
|  |   case EAI_MEMORY: | ||
|  |     return asio::error::no_memory; | ||
|  |   case EAI_NONAME: | ||
|  | #if defined(EAI_ADDRFAMILY) | ||
|  |   case EAI_ADDRFAMILY: | ||
|  | #endif | ||
|  | #if defined(EAI_NODATA) && (EAI_NODATA != EAI_NONAME) | ||
|  |   case EAI_NODATA: | ||
|  | #endif | ||
|  |     return asio::error::host_not_found; | ||
|  |   case EAI_SERVICE: | ||
|  |     return asio::error::service_not_found; | ||
|  |   case EAI_SOCKTYPE: | ||
|  |     return asio::error::socket_type_not_supported; | ||
|  |   default: // Possibly the non-portable EAI_SYSTEM. | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  |     return asio::error_code( | ||
|  |         WSAGetLastError(), asio::error::get_system_category()); | ||
|  | #else | ||
|  |     return asio::error_code( | ||
|  |         errno, asio::error::get_system_category()); | ||
|  | #endif | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | asio::error_code getaddrinfo(const char* host, | ||
|  |     const char* service, const addrinfo_type& hints, | ||
|  |     addrinfo_type** result, asio::error_code& ec) | ||
|  | { | ||
|  |   host = (host && *host) ? host : 0; | ||
|  |   service = (service && *service) ? service : 0; | ||
|  |   clear_last_error(); | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | # if defined(ASIO_HAS_GETADDRINFO) | ||
|  |   // Building for Windows XP, Windows Server 2003, or later. | ||
|  |   int error = ::getaddrinfo(host, service, &hints, result); | ||
|  |   return ec = translate_addrinfo_error(error); | ||
|  | # else | ||
|  |   // Building for Windows 2000 or earlier. | ||
|  |   typedef int (WSAAPI *gai_t)(const char*, | ||
|  |       const char*, const addrinfo_type*, addrinfo_type**); | ||
|  |   if (HMODULE winsock_module = ::GetModuleHandleA("ws2_32")) | ||
|  |   { | ||
|  |     if (gai_t gai = (gai_t)::GetProcAddress(winsock_module, "getaddrinfo")) | ||
|  |     { | ||
|  |       int error = gai(host, service, &hints, result); | ||
|  |       return ec = translate_addrinfo_error(error); | ||
|  |     } | ||
|  |   } | ||
|  |   int error = getaddrinfo_emulation(host, service, &hints, result); | ||
|  |   return ec = translate_addrinfo_error(error); | ||
|  | # endif | ||
|  | #elif !defined(ASIO_HAS_GETADDRINFO) | ||
|  |   int error = getaddrinfo_emulation(host, service, &hints, result); | ||
|  |   return ec = translate_addrinfo_error(error); | ||
|  | #else | ||
|  |   int error = ::getaddrinfo(host, service, &hints, result); | ||
|  | #if defined(__MACH__) && defined(__APPLE__) | ||
|  |   using namespace std; // For isdigit and atoi. | ||
|  |   if (error == 0 && service && isdigit(static_cast<unsigned char>(service[0]))) | ||
|  |   { | ||
|  |     u_short_type port = host_to_network_short(atoi(service)); | ||
|  |     for (addrinfo_type* ai = *result; ai; ai = ai->ai_next) | ||
|  |     { | ||
|  |       switch (ai->ai_family) | ||
|  |       { | ||
|  |       case ASIO_OS_DEF(AF_INET): | ||
|  |         { | ||
|  |           sockaddr_in4_type* sinptr = | ||
|  |             reinterpret_cast<sockaddr_in4_type*>(ai->ai_addr); | ||
|  |           if (sinptr->sin_port == 0) | ||
|  |             sinptr->sin_port = port; | ||
|  |           break; | ||
|  |         } | ||
|  |       case ASIO_OS_DEF(AF_INET6): | ||
|  |         { | ||
|  |           sockaddr_in6_type* sin6ptr = | ||
|  |             reinterpret_cast<sockaddr_in6_type*>(ai->ai_addr); | ||
|  |           if (sin6ptr->sin6_port == 0) | ||
|  |             sin6ptr->sin6_port = port; | ||
|  |           break; | ||
|  |         } | ||
|  |       default: | ||
|  |         break; | ||
|  |       } | ||
|  |     } | ||
|  |   } | ||
|  | #endif | ||
|  |   return ec = translate_addrinfo_error(error); | ||
|  | #endif | ||
|  | } | ||
|  | 
 | ||
|  | asio::error_code background_getaddrinfo( | ||
|  |     const weak_cancel_token_type& cancel_token, const char* host, | ||
|  |     const char* service, const addrinfo_type& hints, | ||
|  |     addrinfo_type** result, asio::error_code& ec) | ||
|  | { | ||
|  |   if (cancel_token.expired()) | ||
|  |     ec = asio::error::operation_aborted; | ||
|  |   else | ||
|  |     socket_ops::getaddrinfo(host, service, hints, result, ec); | ||
|  |   return ec; | ||
|  | } | ||
|  | 
 | ||
|  | void freeaddrinfo(addrinfo_type* ai) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | # if defined(ASIO_HAS_GETADDRINFO) | ||
|  |   // Building for Windows XP, Windows Server 2003, or later. | ||
|  |   ::freeaddrinfo(ai); | ||
|  | # else | ||
|  |   // Building for Windows 2000 or earlier. | ||
|  |   typedef int (WSAAPI *fai_t)(addrinfo_type*); | ||
|  |   if (HMODULE winsock_module = ::GetModuleHandleA("ws2_32")) | ||
|  |   { | ||
|  |     if (fai_t fai = (fai_t)::GetProcAddress(winsock_module, "freeaddrinfo")) | ||
|  |     { | ||
|  |       fai(ai); | ||
|  |       return; | ||
|  |     } | ||
|  |   } | ||
|  |   freeaddrinfo_emulation(ai); | ||
|  | # endif | ||
|  | #elif !defined(ASIO_HAS_GETADDRINFO) | ||
|  |   freeaddrinfo_emulation(ai); | ||
|  | #else | ||
|  |   ::freeaddrinfo(ai); | ||
|  | #endif | ||
|  | } | ||
|  | 
 | ||
|  | asio::error_code getnameinfo(const void* addr, | ||
|  |     std::size_t addrlen, char* host, std::size_t hostlen, | ||
|  |     char* serv, std::size_t servlen, int flags, asio::error_code& ec) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS) || defined(__CYGWIN__) | ||
|  | # if defined(ASIO_HAS_GETADDRINFO) | ||
|  |   // Building for Windows XP, Windows Server 2003, or later. | ||
|  |   clear_last_error(); | ||
|  |   int error = ::getnameinfo(static_cast<const socket_addr_type*>(addr), | ||
|  |       static_cast<socklen_t>(addrlen), host, static_cast<DWORD>(hostlen), | ||
|  |       serv, static_cast<DWORD>(servlen), flags); | ||
|  |   return ec = translate_addrinfo_error(error); | ||
|  | # else | ||
|  |   // Building for Windows 2000 or earlier. | ||
|  |   typedef int (WSAAPI *gni_t)(const socket_addr_type*, | ||
|  |       int, char*, DWORD, char*, DWORD, int); | ||
|  |   if (HMODULE winsock_module = ::GetModuleHandleA("ws2_32")) | ||
|  |   { | ||
|  |     if (gni_t gni = (gni_t)::GetProcAddress(winsock_module, "getnameinfo")) | ||
|  |     { | ||
|  |       clear_last_error(); | ||
|  |       int error = gni(static_cast<const socket_addr_type*>(addr), | ||
|  |           static_cast<int>(addrlen), host, static_cast<DWORD>(hostlen), | ||
|  |           serv, static_cast<DWORD>(servlen), flags); | ||
|  |       return ec = translate_addrinfo_error(error); | ||
|  |     } | ||
|  |   } | ||
|  |   clear_last_error(); | ||
|  |   return getnameinfo_emulation(static_cast<const socket_addr_type*>(addr), | ||
|  |       addrlen, host, hostlen, serv, servlen, flags, ec); | ||
|  | # endif | ||
|  | #elif !defined(ASIO_HAS_GETADDRINFO) | ||
|  |   using namespace std; // For memcpy. | ||
|  |   sockaddr_storage_type tmp_addr; | ||
|  |   memcpy(&tmp_addr, addr, addrlen); | ||
|  |   addr = &tmp_addr; | ||
|  |   clear_last_error(); | ||
|  |   return getnameinfo_emulation(static_cast<const socket_addr_type*>(addr), | ||
|  |       addrlen, host, hostlen, serv, servlen, flags, ec); | ||
|  | #else | ||
|  |   clear_last_error(); | ||
|  |   int error = ::getnameinfo(static_cast<const socket_addr_type*>(addr), | ||
|  |       addrlen, host, hostlen, serv, servlen, flags); | ||
|  |   return ec = translate_addrinfo_error(error); | ||
|  | #endif | ||
|  | } | ||
|  | 
 | ||
|  | asio::error_code sync_getnameinfo(const void* addr, | ||
|  |     std::size_t addrlen, char* host, std::size_t hostlen, char* serv, | ||
|  |     std::size_t servlen, int sock_type, asio::error_code& ec) | ||
|  | { | ||
|  |   // First try resolving with the service name. If that fails try resolving | ||
|  |   // but allow the service to be returned as a number. | ||
|  |   int flags = (sock_type == SOCK_DGRAM) ? NI_DGRAM : 0; | ||
|  |   socket_ops::getnameinfo(addr, addrlen, host, | ||
|  |       hostlen, serv, servlen, flags, ec); | ||
|  |   if (ec) | ||
|  |   { | ||
|  |     socket_ops::getnameinfo(addr, addrlen, host, hostlen, | ||
|  |         serv, servlen, flags | NI_NUMERICSERV, ec); | ||
|  |   } | ||
|  | 
 | ||
|  |   return ec; | ||
|  | } | ||
|  | 
 | ||
|  | asio::error_code background_getnameinfo( | ||
|  |     const weak_cancel_token_type& cancel_token, | ||
|  |     const void* addr, std::size_t addrlen, | ||
|  |     char* host, std::size_t hostlen, char* serv, | ||
|  |     std::size_t servlen, int sock_type, asio::error_code& ec) | ||
|  | { | ||
|  |   if (cancel_token.expired()) | ||
|  |   { | ||
|  |     ec = asio::error::operation_aborted; | ||
|  |   } | ||
|  |   else | ||
|  |   { | ||
|  |     // First try resolving with the service name. If that fails try resolving | ||
|  |     // but allow the service to be returned as a number. | ||
|  |     int flags = (sock_type == SOCK_DGRAM) ? NI_DGRAM : 0; | ||
|  |     socket_ops::getnameinfo(addr, addrlen, host, | ||
|  |         hostlen, serv, servlen, flags, ec); | ||
|  |     if (ec) | ||
|  |     { | ||
|  |       socket_ops::getnameinfo(addr, addrlen, host, hostlen, | ||
|  |           serv, servlen, flags | NI_NUMERICSERV, ec); | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   return ec; | ||
|  | } | ||
|  | 
 | ||
|  | #endif // !defined(ASIO_WINDOWS_RUNTIME) | ||
|  | 
 | ||
|  | u_long_type network_to_host_long(u_long_type value) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS_RUNTIME) | ||
|  |   unsigned char* value_p = reinterpret_cast<unsigned char*>(&value); | ||
|  |   u_long_type result = (static_cast<u_long_type>(value_p[0]) << 24) | ||
|  |     | (static_cast<u_long_type>(value_p[1]) << 16) | ||
|  |     | (static_cast<u_long_type>(value_p[2]) << 8) | ||
|  |     | static_cast<u_long_type>(value_p[3]); | ||
|  |   return result; | ||
|  | #else // defined(ASIO_WINDOWS_RUNTIME) | ||
|  |   return ntohl(value); | ||
|  | #endif // defined(ASIO_WINDOWS_RUNTIME) | ||
|  | } | ||
|  | 
 | ||
|  | u_long_type host_to_network_long(u_long_type value) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS_RUNTIME) | ||
|  |   u_long_type result; | ||
|  |   unsigned char* result_p = reinterpret_cast<unsigned char*>(&result); | ||
|  |   result_p[0] = static_cast<unsigned char>((value >> 24) & 0xFF); | ||
|  |   result_p[1] = static_cast<unsigned char>((value >> 16) & 0xFF); | ||
|  |   result_p[2] = static_cast<unsigned char>((value >> 8) & 0xFF); | ||
|  |   result_p[3] = static_cast<unsigned char>(value & 0xFF); | ||
|  |   return result; | ||
|  | #else // defined(ASIO_WINDOWS_RUNTIME) | ||
|  |   return htonl(value); | ||
|  | #endif // defined(ASIO_WINDOWS_RUNTIME) | ||
|  | } | ||
|  | 
 | ||
|  | u_short_type network_to_host_short(u_short_type value) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS_RUNTIME) | ||
|  |   unsigned char* value_p = reinterpret_cast<unsigned char*>(&value); | ||
|  |   u_short_type result = (static_cast<u_short_type>(value_p[0]) << 8) | ||
|  |     | static_cast<u_short_type>(value_p[1]); | ||
|  |   return result; | ||
|  | #else // defined(ASIO_WINDOWS_RUNTIME) | ||
|  |   return ntohs(value); | ||
|  | #endif // defined(ASIO_WINDOWS_RUNTIME) | ||
|  | } | ||
|  | 
 | ||
|  | u_short_type host_to_network_short(u_short_type value) | ||
|  | { | ||
|  | #if defined(ASIO_WINDOWS_RUNTIME) | ||
|  |   u_short_type result; | ||
|  |   unsigned char* result_p = reinterpret_cast<unsigned char*>(&result); | ||
|  |   result_p[0] = static_cast<unsigned char>((value >> 8) & 0xFF); | ||
|  |   result_p[1] = static_cast<unsigned char>(value & 0xFF); | ||
|  |   return result; | ||
|  | #else // defined(ASIO_WINDOWS_RUNTIME) | ||
|  |   return htons(value); | ||
|  | #endif // defined(ASIO_WINDOWS_RUNTIME) | ||
|  | } | ||
|  | 
 | ||
|  | } // namespace socket_ops | ||
|  | } // namespace detail | ||
|  | } // namespace asio | ||
|  | 
 | ||
|  | #include "asio/detail/pop_options.hpp" | ||
|  | 
 | ||
|  | #endif // ASIO_DETAIL_SOCKET_OPS_IPP |