mirror of
https://github.com/esp8266/Arduino.git
synced 2025-04-19 23:22:16 +03:00
177 lines
4.8 KiB
C++
177 lines
4.8 KiB
C++
|
|
/*
|
|
Arduino emulation - socket part of ClientContext
|
|
Copyright (c) 2018 david gauchard. All rights reserved.
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a
|
|
copy of this software and associated documentation files (the "Software"),
|
|
to deal with 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:
|
|
|
|
- Redistributions of source code must retain the above copyright notice,
|
|
this list of conditions and the following disclaimers.
|
|
|
|
- Redistributions in binary form must reproduce the above copyright notice,
|
|
this list of conditions and the following disclaimers in the
|
|
documentation and/or other materials provided with the distribution.
|
|
|
|
- The names of its contributors may not be used to endorse or promote
|
|
products derived from this Software without specific prior written
|
|
permission.
|
|
|
|
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 CONTRIBUTORS 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 WITH THE SOFTWARE.
|
|
*/
|
|
// separated from lwIP to avoid type conflicts
|
|
|
|
#include <unistd.h>
|
|
#include <sys/socket.h>
|
|
#include <netinet/tcp.h>
|
|
#include <arpa/inet.h>
|
|
#include <poll.h>
|
|
#include <fcntl.h>
|
|
#include <errno.h>
|
|
|
|
int mockSockSetup (int sock)
|
|
{
|
|
if (fcntl(sock, F_SETFL, O_NONBLOCK) == -1)
|
|
{
|
|
fprintf(stderr, MOCK "socket fcntl(O_NONBLOCK): %s\n", strerror(errno));
|
|
close(sock);
|
|
return -1;
|
|
}
|
|
|
|
#ifndef MSG_NOSIGNAL
|
|
int i = 1;
|
|
if (setsockopt(sock, SOL_SOCKET, SO_NOSIGPIPE, &i, sizeof i) == -1)
|
|
{
|
|
fprintf(stderr, MOCK "sockopt( SO_NOSIGPIPE)(macOS): %s\n", strerror(errno));
|
|
return -1;
|
|
}
|
|
#endif
|
|
|
|
return sock;
|
|
}
|
|
|
|
int mockConnect (uint32_t ipv4, int& sock, int port)
|
|
{
|
|
struct sockaddr_in server;
|
|
if ((sock = ::socket(AF_INET, SOCK_STREAM, 0)) == -1)
|
|
{
|
|
perror(MOCK "ClientContext:connect: ::socket()");
|
|
return 0;
|
|
}
|
|
server.sin_family = AF_INET;
|
|
server.sin_port = htons(port);
|
|
memcpy(&server.sin_addr, &ipv4, 4);
|
|
if (::connect(sock, (struct sockaddr*)&server, sizeof(server)) == -1)
|
|
{
|
|
perror(MOCK "ClientContext::connect: ::connect()");
|
|
return 0;
|
|
}
|
|
|
|
return mockSockSetup(sock) == -1? 0: 1;
|
|
}
|
|
|
|
ssize_t mockFillInBuf (int sock, char* ccinbuf, size_t& ccinbufsize)
|
|
{
|
|
size_t maxread = CCBUFSIZE - ccinbufsize;
|
|
ssize_t ret = ::read(sock, ccinbuf + ccinbufsize, maxread);
|
|
if (ret == -1)
|
|
{
|
|
if (errno != EAGAIN)
|
|
{
|
|
fprintf(stderr, MOCK "ClientContext::(read/peek fd=%i): filling buffer for %zd bytes: %s\n", sock, maxread, strerror(errno));
|
|
return -1;
|
|
}
|
|
ret = 0;
|
|
}
|
|
ccinbufsize += ret;
|
|
return ret;
|
|
}
|
|
|
|
ssize_t mockPeekBytes (int sock, char* dst, size_t usersize, int timeout_ms, char* ccinbuf, size_t& ccinbufsize)
|
|
{
|
|
if (usersize > CCBUFSIZE)
|
|
fprintf(stderr, MOCK "CCBUFSIZE(%d) should be increased by %zd bytes (-> %zd)\n", CCBUFSIZE, usersize - CCBUFSIZE, usersize);
|
|
|
|
struct pollfd p;
|
|
size_t retsize = 0;
|
|
do
|
|
{
|
|
if (usersize <= ccinbufsize)
|
|
{
|
|
// data already buffered
|
|
retsize = usersize;
|
|
break;
|
|
}
|
|
|
|
// check incoming data data
|
|
if (mockFillInBuf(sock, ccinbuf, ccinbufsize) < 0)
|
|
return -1;
|
|
if (usersize <= ccinbufsize)
|
|
{
|
|
// data just received
|
|
retsize = usersize;
|
|
break;
|
|
}
|
|
|
|
// wait for more data until timeout
|
|
p.fd = sock;
|
|
p.events = POLLIN;
|
|
} while (poll(&p, 1, timeout_ms) == 1);
|
|
|
|
memcpy(dst, ccinbuf, retsize);
|
|
return retsize;
|
|
}
|
|
|
|
ssize_t mockRead (int sock, char* dst, size_t size, int timeout_ms, char* ccinbuf, size_t& ccinbufsize)
|
|
{
|
|
ssize_t copied = mockPeekBytes(sock, dst, size, timeout_ms, ccinbuf, ccinbufsize);
|
|
if (copied < 0)
|
|
return -1;
|
|
// swallow (XXX use a circular buffer)
|
|
memmove(ccinbuf, ccinbuf + copied, ccinbufsize - copied);
|
|
ccinbufsize -= copied;
|
|
return copied;
|
|
}
|
|
|
|
ssize_t mockWrite (int sock, const uint8_t* data, size_t size, int timeout_ms)
|
|
{
|
|
struct pollfd p;
|
|
p.fd = sock;
|
|
p.events = POLLOUT;
|
|
int ret = poll(&p, 1, timeout_ms);
|
|
if (ret == -1)
|
|
{
|
|
fprintf(stderr, MOCK "ClientContext::write: poll(%d): %s\n", sock, strerror(errno));
|
|
return 0;
|
|
}
|
|
if (ret)
|
|
{
|
|
#ifndef MSG_NOSIGNAL
|
|
ret = ::write(sock, data, size);
|
|
#else
|
|
ret = ::send(sock, data, size, MSG_NOSIGNAL);
|
|
#endif
|
|
if (ret == -1)
|
|
{
|
|
fprintf(stderr, MOCK "ClientContext::read: write(%d): %s\n", sock, strerror(errno));
|
|
return -1;
|
|
}
|
|
if (ret != (int)size)
|
|
{
|
|
fprintf(stderr, MOCK "ClientContext::write: short write (%d < %zd) (TODO)\n", ret, size);
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
}
|
|
return ret;
|
|
}
|