/* * This file is part of OpenTTD. * OpenTTD is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, version 2. * OpenTTD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. * See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with OpenTTD. If not, see . */ /** * @file os_abstraction.h Network stuff has many things that needs to be * included and/or implemented by default. * All those things are in this file. */ #ifndef NETWORK_CORE_OS_ABSTRACTION_H #define NETWORK_CORE_OS_ABSTRACTION_H /** * Abstraction of a network error where all implementation details of the * error codes are encapsulated in this class and the abstraction layer. */ class NetworkError { private: int error; ///< The underlying error number from errno or WSAGetLastError. mutable std::string message; ///< The string representation of the error (set on first call to #AsString). public: NetworkError(int error); bool HasError() const; bool WouldBlock() const; bool IsConnectionReset() const; bool IsConnectInProgress() const; const std::string &AsString() const; static NetworkError GetLast(); }; /* Include standard stuff per OS */ /* Windows stuff */ #if defined(_WIN32) #include #include #include #include /* Windows has some different names for some types */ typedef unsigned long in_addr_t; /* Handle cross-compilation with --build=*-*-cygwin --host=*-*-mingw32 */ #if defined(__MINGW32__) && !defined(AI_ADDRCONFIG) # define AI_ADDRCONFIG 0x00000400 #endif #if !(defined(__MINGW32__) || defined(__CYGWIN__)) /* Windows has some different names for some types */ typedef SSIZE_T ssize_t; typedef int socklen_t; # define IPPROTO_IPV6 41 #endif /* !(__MINGW32__ && __CYGWIN__) */ #endif /* _WIN32 */ /* UNIX stuff */ #if defined(UNIX) && !defined(__OS2__) # if defined(OPENBSD) || defined(__NetBSD__) # define AI_ADDRCONFIG 0 # endif # define SOCKET int # define INVALID_SOCKET -1 # define closesocket close /* Need this for FIONREAD on solaris */ # define BSD_COMP /* Includes needed for UNIX-like systems */ # include # include # include # include # include # include # include /* According to glibc/NEWS, appeared in glibc-2.3. */ # if !defined(__sgi__) && !defined(SUNOS) && !defined(__INNOTEK_LIBC__) \ && !(defined(__GLIBC__) && (__GLIBC__ <= 2) && (__GLIBC_MINOR__ <= 2)) && !defined(__dietlibc__) && !defined(HPUX) /* If for any reason ifaddrs.h does not exist on your system, comment out * the following two lines and an alternative way will be used to fetch * the list of IPs from the system. */ # include # define HAVE_GETIFADDRS # endif # if !defined(INADDR_NONE) # define INADDR_NONE 0xffffffff # endif # if defined(__GLIBC__) && (__GLIBC__ <= 2) && (__GLIBC_MINOR__ <= 1) typedef uint32_t in_addr_t; # endif # include # include # include # if defined(__EMSCRIPTEN__) /* Emscripten doesn't support AI_ADDRCONFIG and errors out on it. */ # undef AI_ADDRCONFIG # define AI_ADDRCONFIG 0 /* Emscripten says it supports FD_SETSIZE fds, but it really only supports 64. * https://github.com/emscripten-core/emscripten/issues/1711 */ # undef FD_SETSIZE # define FD_SETSIZE 64 # endif /* Haiku says it supports FD_SETSIZE fds, but it really only supports 512. */ # if defined(__HAIKU__) # undef FD_SETSIZE # define FD_SETSIZE 512 # endif #endif /* UNIX */ /* OS/2 stuff */ #if defined(__OS2__) # define SOCKET int # define INVALID_SOCKET -1 # define closesocket close /* Includes needed for OS/2 systems */ # include # include # include # include # include # include # include # include # include # include # include # include # define INADDR_NONE 0xffffffff # include "../../3rdparty/os2/getaddrinfo.h" # include "../../3rdparty/os2/getnameinfo.h" #define IPV6_V6ONLY 27 /* * IPv6 address */ struct in6_addr { union { uint8_t __u6_addr8[16]; uint16_t __u6_addr16[8]; uint32_t __u6_addr32[4]; } __u6_addr; /* 128-bit IP6 address */ }; #define s6_addr __u6_addr.__u6_addr8 struct sockaddr_in6 { uint8_t sin6_len; /* length of this struct */ sa_family_t sin6_family; /* AF_INET6 */ in_port_t sin6_port; /* Transport layer port # */ uint32_t sin6_flowinfo; /* IP6 flow information */ struct in6_addr sin6_addr; /* IP6 address */ uint32_t sin6_scope_id; /* scope zone index */ }; typedef int socklen_t; #if !defined(__INNOTEK_LIBC__) typedef unsigned long in_addr_t; #endif /* __INNOTEK_LIBC__ */ #endif /* OS/2 */ #ifdef __EMSCRIPTEN__ /** * Emscripten doesn't set 'addrlen' for accept(), getsockname(), getpeername() * and recvfrom(), which confuses other functions and causes them to crash. * This function needs to be called after these four functions to make sure * 'addrlen' is patched up. * * https://github.com/emscripten-core/emscripten/issues/12996 * * @param address The address returned by those four functions. * @return The correct value for addrlen. */ static inline socklen_t FixAddrLenForEmscripten(struct sockaddr_storage &address) { switch (address.ss_family) { case AF_INET6: return sizeof(struct sockaddr_in6); case AF_INET: return sizeof(struct sockaddr_in); default: NOT_REACHED(); } } #endif bool SetNonBlocking(SOCKET d); bool SetNoDelay(SOCKET d); bool SetReusePort(SOCKET d); NetworkError GetSocketError(SOCKET d); /* Make sure these structures have the size we expect them to be */ static_assert(sizeof(in_addr) == 4); ///< IPv4 addresses should be 4 bytes. static_assert(sizeof(in6_addr) == 16); ///< IPv6 addresses should be 16 bytes. #endif /* NETWORK_CORE_OS_ABSTRACTION_H */