/* Server.cpp - Server class for Raspberry Pi Copyright (c) 2016 Hristo Gochkov All right reserved. This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2.1 of the License, or (at your option) any later version. This library 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 Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this library; if not, write to the Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ #include "WiFiServer.h" #include #include #undef write #undef close int WiFiServer::setTimeout(uint32_t seconds){ struct timeval tv; tv.tv_sec = seconds; tv.tv_usec = 0; if(setsockopt(sockfd, SOL_SOCKET, SO_RCVTIMEO, (char *)&tv, sizeof(struct timeval)) < 0) return -1; return setsockopt(sockfd, SOL_SOCKET, SO_SNDTIMEO, (char *)&tv, sizeof(struct timeval)); } size_t WiFiServer::write(const uint8_t *data, size_t len){ return 0; } void WiFiServer::stopAll(){} WiFiClient WiFiServer::available(){ if(!_listening) return WiFiClient(); int client_sock; if (_accepted_sockfd >= 0) { client_sock = _accepted_sockfd; _accepted_sockfd = -1; } else { struct sockaddr_in _client; int cs = sizeof(struct sockaddr_in); #ifdef ESP_IDF_VERSION_MAJOR client_sock = lwip_accept(sockfd, (struct sockaddr *)&_client, (socklen_t*)&cs); #else client_sock = lwip_accept_r(sockfd, (struct sockaddr *)&_client, (socklen_t*)&cs); #endif } if(client_sock >= 0){ int val = 1; if(setsockopt(client_sock, SOL_SOCKET, SO_KEEPALIVE, (char*)&val, sizeof(int)) == ESP_OK) { val = _noDelay; if(setsockopt(client_sock, IPPROTO_TCP, TCP_NODELAY, (char*)&val, sizeof(int)) == ESP_OK) return WiFiClient(client_sock); } } return WiFiClient(); } void WiFiServer::begin(uint16_t port){ begin(port, 1); } void WiFiServer::begin(uint16_t port, int enable){ if(_listening) return; if(port){ _port = port; } struct sockaddr_in server; sockfd = socket(AF_INET , SOCK_STREAM, 0); if (sockfd < 0) return; setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, &enable, sizeof(int)); server.sin_family = AF_INET; server.sin_addr.s_addr = _addr; server.sin_port = htons(_port); if(bind(sockfd, (struct sockaddr *)&server, sizeof(server)) < 0) return; if(listen(sockfd , _max_clients) < 0) return; fcntl(sockfd, F_SETFL, O_NONBLOCK); _listening = true; _noDelay = false; _accepted_sockfd = -1; } void WiFiServer::setNoDelay(bool nodelay) { _noDelay = nodelay; } bool WiFiServer::getNoDelay() { return _noDelay; } bool WiFiServer::hasClient() { if (_accepted_sockfd >= 0) { return true; } struct sockaddr_in _client; int cs = sizeof(struct sockaddr_in); #ifdef ESP_IDF_VERSION_MAJOR _accepted_sockfd = lwip_accept(sockfd, (struct sockaddr *)&_client, (socklen_t*)&cs); #else _accepted_sockfd = lwip_accept_r(sockfd, (struct sockaddr *)&_client, (socklen_t*)&cs); #endif if (_accepted_sockfd >= 0) { return true; } return false; } void WiFiServer::end(){ #ifdef ESP_IDF_VERSION_MAJOR lwip_close(sockfd); #else lwip_close_r(sockfd); #endif sockfd = -1; _listening = false; } void WiFiServer::close(){ end(); } void WiFiServer::stop(){ end(); }