/*************************************************************************/ /* tcp_server_posix.cpp */ /*************************************************************************/ /* This file is part of: */ /* GODOT ENGINE */ /* http://www.godotengine.org */ /*************************************************************************/ /* Copyright (c) 2007-2016 Juan Linietsky, Ariel Manzur. */ /* */ /* Permission is hereby granted, free of charge, to any person obtaining */ /* a copy of this software and associated documentation files (the */ /* "Software"), to deal in the Software without restriction, including */ /* without limitation the rights to use, copy, modify, merge, publish, */ /* distribute, sublicense, and/or sell copies of the Software, and to */ /* permit persons to whom the Software is furnished to do so, subject to */ /* the following conditions: */ /* */ /* The above copyright notice and this permission notice shall be */ /* included in all copies or substantial portions of the Software. */ /* */ /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */ /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */ /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/ /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */ /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */ /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */ /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ /*************************************************************************/ #include "tcp_server_posix.h" #include "stream_peer_tcp_posix.h" #ifdef UNIX_ENABLED #include #include #include #include #include #include #include #include #ifndef NO_FCNTL #ifdef __HAIKU__ #include #else #include #endif #else #include #endif #ifdef JAVASCRIPT_ENABLED #include #endif #include #include #include TCP_Server* TCPServerPosix::_create() { return memnew(TCPServerPosix); }; void TCPServerPosix::make_default() { TCP_Server::_create = TCPServerPosix::_create; }; Error TCPServerPosix::listen(uint16_t p_port, IP_Address::AddrType p_type, const List *p_accepted_hosts) { int sockfd; int family = p_type == IP_Address::TYPE_IPV6 ? AF_INET6 : AF_INET; sockfd = socket(family, SOCK_STREAM, 0); ERR_FAIL_COND_V(sockfd == -1, FAILED); #ifndef NO_FCNTL fcntl(sockfd, F_SETFL, O_NONBLOCK); #else int bval = 1; ioctl(sockfd, FIONBIO, &bval); #endif int reuse=1; if(setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, (char *)&reuse, sizeof(reuse)) < 0) { WARN_PRINT("REUSEADDR failed!") } sockaddr_storage addr = {0}; if (p_type == IP_Address::TYPE_IPV4) { struct sockaddr_in* addr4 = (struct sockaddr_in*)&addr; addr4->sin_family = AF_INET; addr4->sin_port = htons(p_port); addr4->sin_addr.s_addr = INADDR_ANY; } else { struct sockaddr_in6* addr6 = (struct sockaddr_in6*)&addr; addr6->sin6_family = AF_INET6; addr6->sin6_port = htons(p_port); addr6->sin6_addr = in6addr_any; }; // automatically fill with my IP TODO: use p_accepted_hosts if (bind(sockfd, (struct sockaddr *)&addr, sizeof addr) != -1) { if (::listen(sockfd, 1) == -1) { close(sockfd); ERR_FAIL_V(FAILED); }; } else { return ERR_ALREADY_IN_USE; }; if (listen_sockfd != -1) { stop(); }; listen_sockfd = sockfd; return OK; }; bool TCPServerPosix::is_connection_available() const { if (listen_sockfd == -1) { return false; }; struct pollfd pfd; pfd.fd = listen_sockfd; pfd.events = POLLIN; pfd.revents = 0; int ret = poll(&pfd, 1, 0); ERR_FAIL_COND_V(ret < 0, FAILED); if (ret && (pfd.revents & POLLIN)) { return true; }; return false; }; Ref TCPServerPosix::take_connection() { if (!is_connection_available()) { return Ref(); }; struct sockaddr_storage their_addr; socklen_t size = sizeof(their_addr); int fd = accept(listen_sockfd, (struct sockaddr *)&their_addr, &size); ERR_FAIL_COND_V(fd == -1, Ref()); #ifndef NO_FCNTL fcntl(fd, F_SETFL, O_NONBLOCK); #else int bval = 1; ioctl(fd, FIONBIO, &bval); #endif Ref conn = memnew(StreamPeerTCPPosix); IP_Address ip; if (their_addr.ss_family == AF_INET) { ip.type = IP_Address::TYPE_IPV4; struct sockaddr_in* addr4 = (struct sockaddr_in*)&their_addr; ip.field32[0] = (uint32_t)addr4->sin_addr.s_addr; conn->set_socket(fd, ip, ntohs(addr4->sin_port)); } else if (their_addr.ss_family == AF_INET6) { ip.type = IP_Address::TYPE_IPV6; struct sockaddr_in6* addr6 = (struct sockaddr_in6*)&their_addr; copymem(&addr6->sin6_addr.s6_addr, ip.field8, 16); conn->set_socket(fd, ip, ntohs(addr6->sin6_port)); }; return conn; }; void TCPServerPosix::stop() { if (listen_sockfd != -1) { int ret = close(listen_sockfd); ERR_FAIL_COND(ret!=0); }; listen_sockfd = -1; }; TCPServerPosix::TCPServerPosix() { listen_sockfd = -1; }; TCPServerPosix::~TCPServerPosix() { stop(); }; #endif