Path: blob/master/drivers/windows/net_socket_winsock.cpp
9903 views
/**************************************************************************/1/* net_socket_winsock.cpp */2/**************************************************************************/3/* This file is part of: */4/* GODOT ENGINE */5/* https://godotengine.org */6/**************************************************************************/7/* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */8/* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */9/* */10/* Permission is hereby granted, free of charge, to any person obtaining */11/* a copy of this software and associated documentation files (the */12/* "Software"), to deal in the Software without restriction, including */13/* without limitation the rights to use, copy, modify, merge, publish, */14/* distribute, sublicense, and/or sell copies of the Software, and to */15/* permit persons to whom the Software is furnished to do so, subject to */16/* the following conditions: */17/* */18/* The above copyright notice and this permission notice shall be */19/* included in all copies or substantial portions of the Software. */20/* */21/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */22/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */23/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */24/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */25/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */26/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */27/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */28/**************************************************************************/2930#ifdef WINDOWS_ENABLED3132#include "net_socket_winsock.h"3334#include <winsock2.h>35#include <ws2tcpip.h>3637#include <mswsock.h>38// Workaround missing flag in MinGW39#if defined(__MINGW32__) && !defined(SIO_UDP_NETRESET)40#define SIO_UDP_NETRESET _WSAIOW(IOC_VENDOR, 15)41#endif4243size_t NetSocketWinSock::_set_addr_storage(struct sockaddr_storage *p_addr, const IPAddress &p_ip, uint16_t p_port, IP::Type p_ip_type) {44memset(p_addr, 0, sizeof(struct sockaddr_storage));45if (p_ip_type == IP::TYPE_IPV6 || p_ip_type == IP::TYPE_ANY) { // IPv6 socket.4647// IPv6 only socket with IPv4 address.48ERR_FAIL_COND_V(!p_ip.is_wildcard() && p_ip_type == IP::TYPE_IPV6 && p_ip.is_ipv4(), 0);4950struct sockaddr_in6 *addr6 = (struct sockaddr_in6 *)p_addr;51addr6->sin6_family = AF_INET6;52addr6->sin6_port = htons(p_port);53if (p_ip.is_valid()) {54memcpy(&addr6->sin6_addr.s6_addr, p_ip.get_ipv6(), 16);55} else {56addr6->sin6_addr = in6addr_any;57}58return sizeof(sockaddr_in6);59} else { // IPv4 socket.6061// IPv4 socket with IPv6 address.62ERR_FAIL_COND_V(!p_ip.is_wildcard() && !p_ip.is_ipv4(), 0);6364struct sockaddr_in *addr4 = (struct sockaddr_in *)p_addr;65addr4->sin_family = AF_INET;66addr4->sin_port = htons(p_port); // Short, network byte order.6768if (p_ip.is_valid()) {69memcpy(&addr4->sin_addr.s_addr, p_ip.get_ipv4(), 4);70} else {71addr4->sin_addr.s_addr = INADDR_ANY;72}7374return sizeof(sockaddr_in);75}76}7778void NetSocketWinSock::_set_ip_port(struct sockaddr_storage *p_addr, IPAddress *r_ip, uint16_t *r_port) {79if (p_addr->ss_family == AF_INET) {80struct sockaddr_in *addr4 = (struct sockaddr_in *)p_addr;81if (r_ip) {82r_ip->set_ipv4((uint8_t *)&(addr4->sin_addr.s_addr));83}84if (r_port) {85*r_port = ntohs(addr4->sin_port);86}87} else if (p_addr->ss_family == AF_INET6) {88struct sockaddr_in6 *addr6 = (struct sockaddr_in6 *)p_addr;89if (r_ip) {90r_ip->set_ipv6(addr6->sin6_addr.s6_addr);91}92if (r_port) {93*r_port = ntohs(addr6->sin6_port);94}95}96}9798NetSocket *NetSocketWinSock::_create_func() {99return memnew(NetSocketWinSock);100}101102void NetSocketWinSock::make_default() {103ERR_FAIL_COND(_create != nullptr);104105WSADATA data;106WSAStartup(MAKEWORD(2, 2), &data);107_create = _create_func;108}109110void NetSocketWinSock::cleanup() {111ERR_FAIL_COND(_create == nullptr);112113WSACleanup();114_create = nullptr;115}116117NetSocketWinSock::NetSocketWinSock() {118}119120NetSocketWinSock::~NetSocketWinSock() {121close();122}123124NetSocketWinSock::NetError NetSocketWinSock::_get_socket_error() const {125int err = WSAGetLastError();126if (err == WSAEISCONN) {127return ERR_NET_IS_CONNECTED;128}129if (err == WSAEINPROGRESS || err == WSAEALREADY) {130return ERR_NET_IN_PROGRESS;131}132if (err == WSAEWOULDBLOCK) {133return ERR_NET_WOULD_BLOCK;134}135if (err == WSAEADDRINUSE || err == WSAEADDRNOTAVAIL) {136return ERR_NET_ADDRESS_INVALID_OR_UNAVAILABLE;137}138if (err == WSAEACCES) {139return ERR_NET_UNAUTHORIZED;140}141if (err == WSAEMSGSIZE || err == WSAENOBUFS) {142return ERR_NET_BUFFER_TOO_SMALL;143}144print_verbose("Socket error: " + itos(err) + ".");145return ERR_NET_OTHER;146}147148bool NetSocketWinSock::_can_use_ip(const IPAddress &p_ip, const bool p_for_bind) const {149if (p_for_bind && !(p_ip.is_valid() || p_ip.is_wildcard())) {150return false;151} else if (!p_for_bind && !p_ip.is_valid()) {152return false;153}154// Check if socket support this IP type.155IP::Type type = p_ip.is_ipv4() ? IP::TYPE_IPV4 : IP::TYPE_IPV6;156return !(_ip_type != IP::TYPE_ANY && !p_ip.is_wildcard() && _ip_type != type);157}158159_FORCE_INLINE_ Error NetSocketWinSock::_change_multicast_group(IPAddress p_ip, String p_if_name, bool p_add) {160ERR_FAIL_COND_V(!is_open(), ERR_UNCONFIGURED);161ERR_FAIL_COND_V(!_can_use_ip(p_ip, false), ERR_INVALID_PARAMETER);162163// Need to force level and af_family to IP(v4) when using dual stacking and provided multicast group is IPv4.164IP::Type type = _ip_type == IP::TYPE_ANY && p_ip.is_ipv4() ? IP::TYPE_IPV4 : _ip_type;165// This needs to be the proper level for the multicast group, no matter if the socket is dual stacking.166int level = type == IP::TYPE_IPV4 ? IPPROTO_IP : IPPROTO_IPV6;167int ret = -1;168169IPAddress if_ip;170uint32_t if_v6id = 0;171HashMap<String, IP::Interface_Info> if_info;172IP::get_singleton()->get_local_interfaces(&if_info);173for (KeyValue<String, IP::Interface_Info> &E : if_info) {174IP::Interface_Info &c = E.value;175if (c.name != p_if_name) {176continue;177}178179if_v6id = (uint32_t)c.index.to_int();180if (type == IP::TYPE_IPV6) {181break; // IPv6 uses index.182}183184for (const IPAddress &F : c.ip_addresses) {185if (!F.is_ipv4()) {186continue; // Wrong IP type.187}188if_ip = F;189break;190}191break;192}193194if (level == IPPROTO_IP) {195ERR_FAIL_COND_V(!if_ip.is_valid(), ERR_INVALID_PARAMETER);196struct ip_mreq greq;197int sock_opt = p_add ? IP_ADD_MEMBERSHIP : IP_DROP_MEMBERSHIP;198memcpy(&greq.imr_multiaddr, p_ip.get_ipv4(), 4);199memcpy(&greq.imr_interface, if_ip.get_ipv4(), 4);200ret = setsockopt(_sock, level, sock_opt, (const char *)&greq, sizeof(greq));201} else {202struct ipv6_mreq greq;203int sock_opt = p_add ? IPV6_ADD_MEMBERSHIP : IPV6_DROP_MEMBERSHIP;204memcpy(&greq.ipv6mr_multiaddr, p_ip.get_ipv6(), 16);205greq.ipv6mr_interface = if_v6id;206ret = setsockopt(_sock, level, sock_opt, (const char *)&greq, sizeof(greq));207}208ERR_FAIL_COND_V(ret != 0, FAILED);209210return OK;211}212213void NetSocketWinSock::_set_socket(SOCKET p_sock, IP::Type p_ip_type, bool p_is_stream) {214_sock = p_sock;215_ip_type = p_ip_type;216_is_stream = p_is_stream;217}218219Error NetSocketWinSock::open(Type p_sock_type, IP::Type &ip_type) {220ERR_FAIL_COND_V(is_open(), ERR_ALREADY_IN_USE);221ERR_FAIL_COND_V(ip_type > IP::TYPE_ANY || ip_type < IP::TYPE_NONE, ERR_INVALID_PARAMETER);222223int family = ip_type == IP::TYPE_IPV4 ? AF_INET : AF_INET6;224int protocol = p_sock_type == TYPE_TCP ? IPPROTO_TCP : IPPROTO_UDP;225int type = p_sock_type == TYPE_TCP ? SOCK_STREAM : SOCK_DGRAM;226_sock = socket(family, type, protocol);227228if (_sock == INVALID_SOCKET && ip_type == IP::TYPE_ANY) {229// Careful here, changing the referenced parameter so the caller knows that we are using an IPv4 socket230// in place of a dual stack one, and further calls to _set_sock_addr will work as expected.231ip_type = IP::TYPE_IPV4;232family = AF_INET;233_sock = socket(family, type, protocol);234}235236ERR_FAIL_COND_V(_sock == INVALID_SOCKET, FAILED);237_ip_type = ip_type;238239if (family == AF_INET6) {240// Select IPv4 over IPv6 mapping.241set_ipv6_only_enabled(ip_type != IP::TYPE_ANY);242}243244if (protocol == IPPROTO_UDP) {245// Make sure to disable broadcasting for UDP sockets.246// Depending on the OS, this option might or might not be enabled by default. Let's normalize it.247set_broadcasting_enabled(false);248}249250_is_stream = p_sock_type == TYPE_TCP;251252if (!_is_stream) {253// Disable windows feature/bug reporting WSAECONNRESET/WSAENETRESET when254// recv/recvfrom and an ICMP reply was received from a previous send/sendto.255unsigned long disable = 0;256if (ioctlsocket(_sock, SIO_UDP_CONNRESET, &disable) == SOCKET_ERROR) {257print_verbose("Unable to turn off UDP WSAECONNRESET behavior on Windows.");258}259if (ioctlsocket(_sock, SIO_UDP_NETRESET, &disable) == SOCKET_ERROR) {260// This feature seems not to be supported on wine.261print_verbose("Unable to turn off UDP WSAENETRESET behavior on Windows.");262}263}264return OK;265}266267void NetSocketWinSock::close() {268if (_sock != INVALID_SOCKET) {269closesocket(_sock);270}271272_sock = INVALID_SOCKET;273_ip_type = IP::TYPE_NONE;274_is_stream = false;275}276277Error NetSocketWinSock::bind(IPAddress p_addr, uint16_t p_port) {278ERR_FAIL_COND_V(!is_open(), ERR_UNCONFIGURED);279ERR_FAIL_COND_V(!_can_use_ip(p_addr, true), ERR_INVALID_PARAMETER);280281sockaddr_storage addr;282size_t addr_size = _set_addr_storage(&addr, p_addr, p_port, _ip_type);283284if (::bind(_sock, (struct sockaddr *)&addr, addr_size) != 0) {285NetError err = _get_socket_error();286print_verbose("Failed to bind socket. Error: " + itos(err) + ".");287close();288return ERR_UNAVAILABLE;289}290291return OK;292}293294Error NetSocketWinSock::listen(int p_max_pending) {295ERR_FAIL_COND_V(!is_open(), ERR_UNCONFIGURED);296297if (::listen(_sock, p_max_pending) != 0) {298_get_socket_error();299print_verbose("Failed to listen from socket.");300close();301return FAILED;302}303304return OK;305}306307Error NetSocketWinSock::connect_to_host(IPAddress p_host, uint16_t p_port) {308ERR_FAIL_COND_V(!is_open(), ERR_UNCONFIGURED);309ERR_FAIL_COND_V(!_can_use_ip(p_host, false), ERR_INVALID_PARAMETER);310311struct sockaddr_storage addr;312size_t addr_size = _set_addr_storage(&addr, p_host, p_port, _ip_type);313314if (::WSAConnect(_sock, (struct sockaddr *)&addr, addr_size, nullptr, nullptr, nullptr, nullptr) != 0) {315NetError err = _get_socket_error();316317switch (err) {318// We are already connected.319case ERR_NET_IS_CONNECTED:320return OK;321// Still waiting to connect, try again in a while.322case ERR_NET_WOULD_BLOCK:323case ERR_NET_IN_PROGRESS:324return ERR_BUSY;325default:326print_verbose("Connection to remote host failed.");327close();328return FAILED;329}330}331332return OK;333}334335Error NetSocketWinSock::poll(PollType p_type, int p_timeout) const {336ERR_FAIL_COND_V(!is_open(), ERR_UNCONFIGURED);337338bool ready = false;339fd_set rd, wr, ex;340fd_set *rdp = nullptr;341fd_set *wrp = nullptr;342FD_ZERO(&rd);343FD_ZERO(&wr);344FD_ZERO(&ex);345FD_SET(_sock, &ex);346struct timeval timeout = { p_timeout / 1000, (p_timeout % 1000) * 1000 };347// For blocking operation, pass nullptr timeout pointer to select.348struct timeval *tp = nullptr;349if (p_timeout >= 0) {350// If timeout is non-negative, we want to specify the timeout instead.351tp = &timeout;352}353354switch (p_type) {355case POLL_TYPE_IN:356FD_SET(_sock, &rd);357rdp = &rd;358break;359case POLL_TYPE_OUT:360FD_SET(_sock, &wr);361wrp = ≀362break;363case POLL_TYPE_IN_OUT:364FD_SET(_sock, &rd);365FD_SET(_sock, &wr);366rdp = &rd;367wrp = ≀368}369// WSAPoll is broken: https://daniel.haxx.se/blog/2012/10/10/wsapoll-is-broken/.370int ret = select(1, rdp, wrp, &ex, tp);371372if (ret == SOCKET_ERROR) {373return FAILED;374}375376if (ret == 0) {377return ERR_BUSY;378}379380if (FD_ISSET(_sock, &ex)) {381_get_socket_error();382print_verbose("Exception when polling socket.");383return FAILED;384}385386if (rdp && FD_ISSET(_sock, rdp)) {387ready = true;388}389if (wrp && FD_ISSET(_sock, wrp)) {390ready = true;391}392393return ready ? OK : ERR_BUSY;394}395396Error NetSocketWinSock::recv(uint8_t *p_buffer, int p_len, int &r_read) {397ERR_FAIL_COND_V(!is_open(), ERR_UNCONFIGURED);398399r_read = ::recv(_sock, (char *)p_buffer, p_len, 0);400401if (r_read < 0) {402NetError err = _get_socket_error();403if (err == ERR_NET_WOULD_BLOCK) {404return ERR_BUSY;405}406407if (err == ERR_NET_BUFFER_TOO_SMALL) {408return ERR_OUT_OF_MEMORY;409}410411return FAILED;412}413414return OK;415}416417Error NetSocketWinSock::recvfrom(uint8_t *p_buffer, int p_len, int &r_read, IPAddress &r_ip, uint16_t &r_port, bool p_peek) {418ERR_FAIL_COND_V(!is_open(), ERR_UNCONFIGURED);419420struct sockaddr_storage from;421socklen_t len = sizeof(struct sockaddr_storage);422memset(&from, 0, len);423424r_read = ::recvfrom(_sock, (char *)p_buffer, p_len, p_peek ? MSG_PEEK : 0, (struct sockaddr *)&from, &len);425426if (r_read < 0) {427NetError err = _get_socket_error();428if (err == ERR_NET_WOULD_BLOCK) {429return ERR_BUSY;430}431432if (err == ERR_NET_BUFFER_TOO_SMALL) {433return ERR_OUT_OF_MEMORY;434}435436return FAILED;437}438439if (from.ss_family == AF_INET) {440struct sockaddr_in *sin_from = (struct sockaddr_in *)&from;441r_ip.set_ipv4((uint8_t *)&sin_from->sin_addr);442r_port = ntohs(sin_from->sin_port);443} else if (from.ss_family == AF_INET6) {444struct sockaddr_in6 *s6_from = (struct sockaddr_in6 *)&from;445r_ip.set_ipv6((uint8_t *)&s6_from->sin6_addr);446r_port = ntohs(s6_from->sin6_port);447} else {448// Unsupported socket family, should never happen.449ERR_FAIL_V(FAILED);450}451452return OK;453}454455Error NetSocketWinSock::send(const uint8_t *p_buffer, int p_len, int &r_sent) {456ERR_FAIL_COND_V(!is_open(), ERR_UNCONFIGURED);457458int flags = 0;459r_sent = ::send(_sock, (const char *)p_buffer, p_len, flags);460461if (r_sent < 0) {462NetError err = _get_socket_error();463if (err == ERR_NET_WOULD_BLOCK) {464return ERR_BUSY;465}466if (err == ERR_NET_BUFFER_TOO_SMALL) {467return ERR_OUT_OF_MEMORY;468}469470return FAILED;471}472473return OK;474}475476Error NetSocketWinSock::sendto(const uint8_t *p_buffer, int p_len, int &r_sent, IPAddress p_ip, uint16_t p_port) {477ERR_FAIL_COND_V(!is_open(), ERR_UNCONFIGURED);478479struct sockaddr_storage addr;480size_t addr_size = _set_addr_storage(&addr, p_ip, p_port, _ip_type);481r_sent = ::sendto(_sock, (const char *)p_buffer, p_len, 0, (struct sockaddr *)&addr, addr_size);482483if (r_sent < 0) {484NetError err = _get_socket_error();485if (err == ERR_NET_WOULD_BLOCK) {486return ERR_BUSY;487}488if (err == ERR_NET_BUFFER_TOO_SMALL) {489return ERR_OUT_OF_MEMORY;490}491492return FAILED;493}494495return OK;496}497498Error NetSocketWinSock::set_broadcasting_enabled(bool p_enabled) {499ERR_FAIL_COND_V(!is_open(), ERR_UNCONFIGURED);500// IPv6 has no broadcast support.501if (_ip_type == IP::TYPE_IPV6) {502return ERR_UNAVAILABLE;503}504505int par = p_enabled ? 1 : 0;506if (setsockopt(_sock, SOL_SOCKET, SO_BROADCAST, (const char *)&par, sizeof(int)) != 0) {507WARN_PRINT("Unable to change broadcast setting.");508return FAILED;509}510return OK;511}512513void NetSocketWinSock::set_blocking_enabled(bool p_enabled) {514ERR_FAIL_COND(!is_open());515516int ret = 0;517unsigned long par = p_enabled ? 0 : 1;518ret = ioctlsocket(_sock, FIONBIO, &par);519if (ret != 0) {520WARN_PRINT("Unable to change non-block mode.");521}522}523524void NetSocketWinSock::set_ipv6_only_enabled(bool p_enabled) {525ERR_FAIL_COND(!is_open());526// This option is only available in IPv6 sockets.527ERR_FAIL_COND(_ip_type == IP::TYPE_IPV4);528529int par = p_enabled ? 1 : 0;530if (setsockopt(_sock, IPPROTO_IPV6, IPV6_V6ONLY, (const char *)&par, sizeof(int)) != 0) {531WARN_PRINT("Unable to change IPv4 address mapping over IPv6 option.");532}533}534535void NetSocketWinSock::set_tcp_no_delay_enabled(bool p_enabled) {536ERR_FAIL_COND(!is_open());537ERR_FAIL_COND(!_is_stream); // Not TCP.538539int par = p_enabled ? 1 : 0;540if (setsockopt(_sock, IPPROTO_TCP, TCP_NODELAY, (const char *)&par, sizeof(int)) < 0) {541WARN_PRINT("Unable to set TCP no delay option.");542}543}544545void NetSocketWinSock::set_reuse_address_enabled(bool p_enabled) {546ERR_FAIL_COND(!is_open());547548// On Windows, enabling SO_REUSEADDR actually would also enable reuse port, very bad on TCP. Denying...549// Windows does not have this option, SO_REUSEADDR in this magical world means SO_REUSEPORT550}551552bool NetSocketWinSock::is_open() const {553return _sock != INVALID_SOCKET;554}555556int NetSocketWinSock::get_available_bytes() const {557ERR_FAIL_COND_V(!is_open(), -1);558559unsigned long len;560int ret = ioctlsocket(_sock, FIONREAD, &len);561if (ret == -1) {562_get_socket_error();563print_verbose("Error when checking available bytes on socket.");564return -1;565}566return len;567}568569Error NetSocketWinSock::get_socket_address(IPAddress *r_ip, uint16_t *r_port) const {570ERR_FAIL_COND_V(!is_open(), FAILED);571572struct sockaddr_storage saddr;573socklen_t len = sizeof(saddr);574if (getsockname(_sock, (struct sockaddr *)&saddr, &len) != 0) {575_get_socket_error();576print_verbose("Error when reading local socket address.");577return FAILED;578}579_set_ip_port(&saddr, r_ip, r_port);580return OK;581}582583Ref<NetSocket> NetSocketWinSock::accept(IPAddress &r_ip, uint16_t &r_port) {584Ref<NetSocket> out;585ERR_FAIL_COND_V(!is_open(), out);586587struct sockaddr_storage their_addr;588socklen_t size = sizeof(their_addr);589SOCKET fd = ::accept(_sock, (struct sockaddr *)&their_addr, &size);590if (fd == INVALID_SOCKET) {591_get_socket_error();592print_verbose("Error when accepting socket connection.");593return out;594}595596_set_ip_port(&their_addr, &r_ip, &r_port);597598NetSocketWinSock *ns = memnew(NetSocketWinSock);599ns->_set_socket(fd, _ip_type, _is_stream);600ns->set_blocking_enabled(false);601return Ref<NetSocket>(ns);602}603604Error NetSocketWinSock::join_multicast_group(const IPAddress &p_multi_address, const String &p_if_name) {605return _change_multicast_group(p_multi_address, p_if_name, true);606}607608Error NetSocketWinSock::leave_multicast_group(const IPAddress &p_multi_address, const String &p_if_name) {609return _change_multicast_group(p_multi_address, p_if_name, false);610}611612#endif // WINDOWS_ENABLED613614615