mirror of
https://libwebsockets.org/repo/libwebsockets
synced 2024-11-22 00:57:52 +00:00
686 lines
16 KiB
C
686 lines
16 KiB
C
/*
|
|
* libwebsockets - small server side websockets and web server implementation
|
|
*
|
|
* Copyright (C) 2010 - 2023 Andy Green <andy@warmcat.com>
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
#if !defined(_GNU_SOURCE)
|
|
#define _GNU_SOURCE
|
|
#endif
|
|
#include "private-lib-core.h"
|
|
|
|
#if defined(LWS_HAVE_LINUX_IPV6_H)
|
|
#include <linux/ipv6.h>
|
|
#endif
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#if !defined(LWS_DETECTED_PLAT_IOS)
|
|
#include <net/route.h>
|
|
#endif
|
|
|
|
#include <net/if.h>
|
|
|
|
#include <pwd.h>
|
|
#include <grp.h>
|
|
|
|
#if defined(LWS_WITH_MBEDTLS)
|
|
#if defined(LWS_HAVE_MBEDTLS_NET_SOCKETS)
|
|
#include "mbedtls/net_sockets.h"
|
|
#else
|
|
#include "mbedtls/net.h"
|
|
#endif
|
|
#endif
|
|
|
|
#include <netinet/ip.h>
|
|
|
|
int
|
|
lws_send_pipe_choked(struct lws *wsi)
|
|
{
|
|
struct lws_pollfd fds;
|
|
struct lws *wsi_eff;
|
|
|
|
#if !defined(LWS_WITHOUT_EXTENSIONS)
|
|
if (wsi->ws && wsi->ws->tx_draining_ext)
|
|
return 1;
|
|
#endif
|
|
|
|
#if defined(LWS_WITH_HTTP2)
|
|
wsi_eff = lws_get_network_wsi(wsi);
|
|
#else
|
|
wsi_eff = wsi;
|
|
#endif
|
|
|
|
/* the fact we checked implies we avoided back-to-back writes */
|
|
wsi_eff->could_have_pending = 0;
|
|
|
|
/* treat the fact we got a truncated send pending as if we're choked */
|
|
if (lws_has_buffered_out(wsi_eff)
|
|
#if defined(LWS_WITH_HTTP_STREAM_COMPRESSION)
|
|
||wsi->http.comp_ctx.buflist_comp ||
|
|
wsi->http.comp_ctx.may_have_more
|
|
#endif
|
|
)
|
|
return 1;
|
|
|
|
fds.fd = wsi_eff->desc.sockfd;
|
|
fds.events = POLLOUT;
|
|
fds.revents = 0;
|
|
|
|
if (poll(&fds, 1, 0) != 1)
|
|
return 1;
|
|
|
|
if ((fds.revents & POLLOUT) == 0)
|
|
return 1;
|
|
|
|
/* okay to send another packet without blocking */
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
lws_plat_set_nonblocking(lws_sockfd_type fd)
|
|
{
|
|
return fcntl(fd, F_SETFL, O_NONBLOCK) < 0;
|
|
}
|
|
|
|
int
|
|
lws_plat_set_socket_options(struct lws_vhost *vhost, int fd, int unix_skt)
|
|
{
|
|
int optval = 1;
|
|
socklen_t optlen = sizeof(optval);
|
|
|
|
#if defined(__APPLE__) || \
|
|
defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || \
|
|
defined(__NetBSD__) || \
|
|
defined(__OpenBSD__) || \
|
|
defined(__HAIKU__)
|
|
struct protoent *tcp_proto;
|
|
#endif
|
|
|
|
(void)fcntl(fd, F_SETFD, FD_CLOEXEC);
|
|
|
|
if (!unix_skt && vhost->ka_time) {
|
|
/* enable keepalive on this socket */
|
|
optval = 1;
|
|
if (setsockopt(fd, SOL_SOCKET, SO_KEEPALIVE,
|
|
(const void *)&optval, optlen) < 0)
|
|
return 1;
|
|
|
|
#if defined(__APPLE__) || \
|
|
defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || \
|
|
defined(__NetBSD__) || \
|
|
defined(__CYGWIN__) || defined(__OpenBSD__) || defined (__sun) || \
|
|
defined(__HAIKU__)
|
|
|
|
/*
|
|
* didn't find a way to set these per-socket, need to
|
|
* tune kernel systemwide values
|
|
*/
|
|
#else
|
|
/* set the keepalive conditions we want on it too */
|
|
|
|
#if defined(LWS_HAVE_TCP_USER_TIMEOUT)
|
|
optval = 1000 * (vhost->ka_time +
|
|
(vhost->ka_interval * vhost->ka_probes));
|
|
if (setsockopt(fd, IPPROTO_TCP, TCP_USER_TIMEOUT,
|
|
(const void *)&optval, optlen) < 0)
|
|
return 1;
|
|
#endif
|
|
optval = vhost->ka_time;
|
|
if (setsockopt(fd, IPPROTO_TCP, TCP_KEEPIDLE,
|
|
(const void *)&optval, optlen) < 0)
|
|
return 1;
|
|
|
|
optval = vhost->ka_interval;
|
|
if (setsockopt(fd, IPPROTO_TCP, TCP_KEEPINTVL,
|
|
(const void *)&optval, optlen) < 0)
|
|
return 1;
|
|
|
|
optval = vhost->ka_probes;
|
|
if (setsockopt(fd, IPPROTO_TCP, TCP_KEEPCNT,
|
|
(const void *)&optval, optlen) < 0)
|
|
return 1;
|
|
#endif
|
|
}
|
|
|
|
#if defined(SO_BINDTODEVICE)
|
|
if (!unix_skt && vhost->bind_iface && vhost->iface) {
|
|
lwsl_info("binding listen skt to %s using SO_BINDTODEVICE\n", vhost->iface);
|
|
if (setsockopt(fd, SOL_SOCKET, SO_BINDTODEVICE, vhost->iface,
|
|
(socklen_t)strlen(vhost->iface)) < 0) {
|
|
lwsl_warn("Failed to bind to device %s\n", vhost->iface);
|
|
return 1;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/* Disable Nagle */
|
|
optval = 1;
|
|
#if defined (__sun) || defined(__QNX__) || defined(__NuttX__)
|
|
if (!unix_skt && setsockopt(fd, IPPROTO_TCP, TCP_NODELAY, (const void *)&optval, optlen) < 0)
|
|
return 1;
|
|
#elif !defined(__APPLE__) && \
|
|
!defined(__FreeBSD__) && !defined(__FreeBSD_kernel__) && \
|
|
!defined(__NetBSD__) && \
|
|
!defined(__OpenBSD__) && \
|
|
!defined(__HAIKU__)
|
|
if (!unix_skt && setsockopt(fd, SOL_TCP, TCP_NODELAY, (const void *)&optval, optlen) < 0)
|
|
return 1;
|
|
#else
|
|
tcp_proto = getprotobyname("TCP");
|
|
if (!unix_skt && setsockopt(fd, tcp_proto->p_proto, TCP_NODELAY, &optval, optlen) < 0)
|
|
return 1;
|
|
#endif
|
|
|
|
return lws_plat_set_nonblocking(fd);
|
|
}
|
|
|
|
#if !defined(__NuttX__)
|
|
static const int ip_opt_lws_flags[] = {
|
|
LCCSCF_IP_LOW_LATENCY, LCCSCF_IP_HIGH_THROUGHPUT
|
|
#if !defined(__OpenBSD__) && !defined(__sun) && !defined(__QNX__)
|
|
, LCCSCF_IP_HIGH_RELIABILITY
|
|
, LCCSCF_IP_LOW_COST
|
|
#endif
|
|
}, ip_opt_val[] = {
|
|
IPTOS_LOWDELAY, IPTOS_THROUGHPUT
|
|
#if !defined(__OpenBSD__) && !defined(__sun) && !defined(__QNX__)
|
|
, IPTOS_RELIABILITY
|
|
, IPTOS_MINCOST
|
|
#endif
|
|
};
|
|
#if !defined(LWS_WITH_NO_LOGS)
|
|
static const char *ip_opt_names[] = {
|
|
"LOWDELAY", "THROUGHPUT"
|
|
#if !defined(__OpenBSD__) && !defined(__sun) && !defined(__QNX__)
|
|
, "RELIABILITY"
|
|
, "MINCOST"
|
|
#endif
|
|
};
|
|
#endif
|
|
#endif
|
|
|
|
int
|
|
lws_plat_set_socket_options_ip(lws_sockfd_type fd, uint8_t pri, int lws_flags)
|
|
{
|
|
int optval = (int)pri, ret = 0, n;
|
|
socklen_t optlen = sizeof(optval);
|
|
#if (_LWS_ENABLED_LOGS & LLL_WARN)
|
|
int en;
|
|
#endif
|
|
|
|
#if 0
|
|
#if defined(TCP_FASTOPEN_CONNECT)
|
|
optval = 1;
|
|
if (setsockopt(fd, IPPROTO_TCP, TCP_FASTOPEN_CONNECT, (void *)&optval,
|
|
sizeof(optval)))
|
|
lwsl_warn("%s: FASTOPEN_CONNECT failed\n", __func__);
|
|
optval = (int)pri;
|
|
#endif
|
|
#endif
|
|
|
|
#if !defined(__APPLE__) && \
|
|
!defined(__FreeBSD__) && !defined(__FreeBSD_kernel__) && \
|
|
!defined(__NetBSD__) && \
|
|
!defined(__OpenBSD__) && \
|
|
!defined(__sun) && \
|
|
!defined(__HAIKU__) && \
|
|
!defined(__CYGWIN__) && \
|
|
!defined(__QNX__) && \
|
|
!defined(__NuttX__)
|
|
|
|
/* the BSDs don't have SO_PRIORITY */
|
|
|
|
if (pri) { /* 0 is the default already */
|
|
if (setsockopt(fd, SOL_SOCKET, SO_PRIORITY,
|
|
(const void *)&optval, optlen) < 0) {
|
|
#if (_LWS_ENABLED_LOGS & LLL_WARN)
|
|
en = errno;
|
|
lwsl_warn("%s: unable to set socket pri %d: errno %d\n",
|
|
__func__, (int)pri, en);
|
|
#endif
|
|
ret = 1;
|
|
} else
|
|
lwsl_notice("%s: set pri %u\n", __func__, pri);
|
|
}
|
|
#endif
|
|
|
|
if (lws_flags & LCCSCF_ALLOW_REUSE_ADDR) {
|
|
optval = 1;
|
|
if (setsockopt(fd, SOL_SOCKET, SO_REUSEADDR,
|
|
(const void *)&optval, optlen) < 0) {
|
|
#if (_LWS_ENABLED_LOGS & LLL_WARN)
|
|
en = errno;
|
|
lwsl_warn("%s: unable to reuse local addresses: errno %d\n",
|
|
__func__, en);
|
|
#endif
|
|
ret = 1;
|
|
} else
|
|
lwsl_notice("%s: set reuse addresses\n", __func__);
|
|
}
|
|
|
|
|
|
if (lws_flags & LCCSCF_IPV6_PREFER_PUBLIC_ADDR) {
|
|
#if defined(LWS_WITH_IPV6) && defined(IPV6_PREFER_SRC_PUBLIC)
|
|
optval = IPV6_PREFER_SRC_PUBLIC;
|
|
|
|
if (setsockopt(fd, IPPROTO_IPV6, IPV6_ADDR_PREFERENCES,
|
|
(const void *)&optval, optlen) < 0) {
|
|
#if (_LWS_ENABLED_LOGS & LLL_WARN)
|
|
en = errno;
|
|
lwsl_warn("%s: unable to set IPV6_PREFER_SRC_PUBLIC: errno %d\n",
|
|
__func__, en);
|
|
#endif
|
|
ret = 1;
|
|
} else
|
|
lwsl_notice("%s: set IPV6_PREFER_SRC_PUBLIC\n", __func__);
|
|
#else
|
|
lwsl_err("%s: IPV6_PREFER_SRC_PUBLIC UNIMPLEMENTED on this platform\n", __func__);
|
|
#endif
|
|
}
|
|
|
|
|
|
#if !defined(__NuttX__)
|
|
/* array size differs by platform */
|
|
for (n = 0; n < (int)LWS_ARRAY_SIZE(ip_opt_lws_flags); n++) {
|
|
if (!(lws_flags & ip_opt_lws_flags[n]))
|
|
continue;
|
|
|
|
optval = (int)ip_opt_val[n];
|
|
if (setsockopt(fd, IPPROTO_IP, IP_TOS, (const void *)&optval,
|
|
optlen) < 0) {
|
|
#if !defined(LWS_WITH_NO_LOGS)
|
|
en = errno;
|
|
lwsl_warn("%s: unable to set %s: errno %d\n", __func__,
|
|
ip_opt_names[n], en);
|
|
#endif
|
|
ret = 1;
|
|
} else
|
|
lwsl_notice("%s: set ip flag %s\n", __func__,
|
|
ip_opt_names[n]);
|
|
}
|
|
#endif
|
|
|
|
return ret;
|
|
}
|
|
|
|
/* cast a struct sockaddr_in6 * into addr for ipv6 */
|
|
|
|
enum {
|
|
IP_SCORE_NONE,
|
|
IP_SCORE_NONNATIVE,
|
|
IP_SCORE_IPV6_SCOPE_BASE,
|
|
/* ipv6 scopes */
|
|
IP_SCORE_GLOBAL_NATIVE = 18
|
|
};
|
|
|
|
int
|
|
lws_interface_to_sa(int ipv6, const char *ifname, struct sockaddr_in *addr,
|
|
size_t addrlen)
|
|
{
|
|
int rc = LWS_ITOSA_NOT_EXIST;
|
|
|
|
struct ifaddrs *ifr;
|
|
struct ifaddrs *ifc;
|
|
#if defined(LWS_WITH_IPV6)
|
|
struct sockaddr_in6 *addr6 = (struct sockaddr_in6 *)addr;
|
|
unsigned long sco = IP_SCORE_NONE;
|
|
unsigned long ts;
|
|
const uint8_t *p;
|
|
#endif
|
|
|
|
if (getifaddrs(&ifr)) {
|
|
lwsl_err("%s: unable to getifaddrs: errno %d\n", __func__, errno);
|
|
|
|
return LWS_ITOSA_USABLE;
|
|
}
|
|
for (ifc = ifr; ifc != NULL; ifc = ifc->ifa_next) {
|
|
if (!ifc->ifa_addr || !ifc->ifa_name)
|
|
continue;
|
|
|
|
lwsl_debug(" interface %s vs %s (fam %d) ipv6 %d\n",
|
|
ifc->ifa_name, ifname,
|
|
ifc->ifa_addr->sa_family, ipv6);
|
|
|
|
if (strcmp(ifc->ifa_name, ifname))
|
|
continue;
|
|
|
|
switch (ifc->ifa_addr->sa_family) {
|
|
#if defined(AF_PACKET)
|
|
case AF_PACKET:
|
|
/* interface exists but is not usable */
|
|
if (rc == LWS_ITOSA_NOT_EXIST)
|
|
rc = LWS_ITOSA_NOT_USABLE;
|
|
continue;
|
|
#endif
|
|
|
|
case AF_INET:
|
|
#if defined(LWS_WITH_IPV6)
|
|
if (ipv6) {
|
|
/* any existing solution is better than this */
|
|
if (sco != IP_SCORE_NONE)
|
|
break;
|
|
sco = IP_SCORE_NONNATIVE;
|
|
rc = LWS_ITOSA_USABLE;
|
|
/* map IPv4 to IPv6 */
|
|
memset((char *)&addr6->sin6_addr, 0,
|
|
sizeof(struct in6_addr));
|
|
addr6->sin6_addr.s6_addr[10] = 0xff;
|
|
addr6->sin6_addr.s6_addr[11] = 0xff;
|
|
memcpy(&addr6->sin6_addr.s6_addr[12],
|
|
&((struct sockaddr_in *)ifc->ifa_addr)->sin_addr,
|
|
sizeof(struct in_addr));
|
|
lwsl_debug("%s: uplevelling ipv4 bind to ipv6\n", __func__);
|
|
break;
|
|
}
|
|
|
|
sco = IP_SCORE_GLOBAL_NATIVE;
|
|
#endif
|
|
rc = LWS_ITOSA_USABLE;
|
|
memcpy(addr, (struct sockaddr_in *)ifc->ifa_addr,
|
|
sizeof(struct sockaddr_in));
|
|
break;
|
|
#if defined(LWS_WITH_IPV6)
|
|
case AF_INET6:
|
|
p = (const uint8_t *)
|
|
&((struct sockaddr_in6 *)ifc->ifa_addr)->sin6_addr;
|
|
ts = IP_SCORE_IPV6_SCOPE_BASE;
|
|
if (p[0] == 0xff)
|
|
ts = (unsigned long)(IP_SCORE_IPV6_SCOPE_BASE + (p[1] & 0xf));
|
|
|
|
if (sco >= ts)
|
|
break;
|
|
|
|
sco = ts;
|
|
rc = LWS_ITOSA_USABLE;
|
|
|
|
memcpy(&addr6->sin6_addr,
|
|
&((struct sockaddr_in6 *)ifc->ifa_addr)->sin6_addr,
|
|
sizeof(struct in6_addr));
|
|
break;
|
|
#endif
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
freeifaddrs(ifr);
|
|
|
|
if (rc &&
|
|
!lws_sa46_parse_numeric_address(ifname, (lws_sockaddr46 *)addr))
|
|
rc = LWS_ITOSA_USABLE;
|
|
|
|
return rc;
|
|
}
|
|
|
|
|
|
const char *
|
|
lws_plat_inet_ntop(int af, const void *src, char *dst, socklen_t cnt)
|
|
{
|
|
return inet_ntop(af, src, dst, cnt);
|
|
}
|
|
|
|
int
|
|
lws_plat_inet_pton(int af, const char *src, void *dst)
|
|
{
|
|
return inet_pton(af, src, dst);
|
|
}
|
|
|
|
int
|
|
lws_plat_ifname_to_hwaddr(int fd, const char *ifname, uint8_t *hwaddr, int len)
|
|
{
|
|
#if defined(__linux__)
|
|
struct ifreq i;
|
|
|
|
memset(&i, 0, sizeof(i));
|
|
lws_strncpy(i.ifr_name, ifname, sizeof(i.ifr_name));
|
|
|
|
if (ioctl(fd, SIOCGIFHWADDR, &i) < 0)
|
|
return -1;
|
|
|
|
memcpy(hwaddr, &i.ifr_hwaddr.sa_data, 6);
|
|
|
|
return 6;
|
|
#else
|
|
lwsl_err("%s: UNIMPLEMENTED on this platform\n", __func__);
|
|
|
|
return -1;
|
|
#endif
|
|
}
|
|
|
|
int
|
|
lws_plat_rawudp_broadcast(uint8_t *p, const uint8_t *canned, size_t canned_len,
|
|
size_t n, int fd, const char *iface)
|
|
{
|
|
#if defined(__linux__)
|
|
struct sockaddr_ll sll;
|
|
uint16_t *p16 = (uint16_t *)p;
|
|
uint32_t ucs = 0;
|
|
|
|
memcpy(p, canned, canned_len);
|
|
|
|
p[2] = (uint8_t)(n >> 8);
|
|
p[3] = (uint8_t)(n);
|
|
|
|
while (p16 < (uint16_t *)(p + 20))
|
|
ucs = ucs + (uint32_t)(ntohs((uint16_t)(*p16++)));
|
|
|
|
ucs += ucs >> 16;
|
|
ucs ^= 0xffff;
|
|
|
|
p[10] = (uint8_t)(ucs >> 8);
|
|
p[11] = (uint8_t)(ucs);
|
|
p[24] = (uint8_t)((n - 20) >> 8);
|
|
p[25] = (uint8_t)((n - 20));
|
|
|
|
memset(&sll, 0, sizeof(sll));
|
|
sll.sll_family = AF_PACKET;
|
|
sll.sll_protocol = (uint32_t)(htons((uint16_t)0x800));
|
|
sll.sll_halen = 6;
|
|
sll.sll_ifindex = (int)if_nametoindex(iface);
|
|
memset(sll.sll_addr, 0xff, 6);
|
|
|
|
return (int)sendto(fd, p, n, 0, (struct sockaddr *)&sll, sizeof(sll));
|
|
#else
|
|
lwsl_err("%s: UNIMPLEMENTED on this platform\n", __func__);
|
|
|
|
return -1;
|
|
#endif
|
|
}
|
|
|
|
int
|
|
lws_plat_if_up(const char *ifname, int fd, int up)
|
|
{
|
|
#if defined(__linux__)
|
|
struct ifreq ifr;
|
|
|
|
memset(&ifr, 0, sizeof(ifr));
|
|
lws_strncpy(ifr.ifr_name, ifname, IFNAMSIZ);
|
|
|
|
if (ioctl(fd, SIOCGIFFLAGS, &ifr) < 0) {
|
|
lwsl_err("%s: SIOCGIFFLAGS fail\n", __func__);
|
|
return 1;
|
|
}
|
|
|
|
if (up)
|
|
ifr.ifr_flags |= IFF_UP;
|
|
else
|
|
ifr.ifr_flags &= ~IFF_UP;
|
|
|
|
if (ioctl(fd, SIOCSIFFLAGS, &ifr) < 0) {
|
|
lwsl_err("%s: SIOCSIFFLAGS fail\n", __func__);
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
#else
|
|
lwsl_err("%s: UNIMPLEMENTED on this platform\n", __func__);
|
|
|
|
return -1;
|
|
#endif
|
|
}
|
|
|
|
int
|
|
lws_plat_BINDTODEVICE(lws_sockfd_type fd, const char *ifname)
|
|
{
|
|
#if defined(__linux__)
|
|
struct ifreq i;
|
|
|
|
memset(&i, 0, sizeof(i));
|
|
i.ifr_addr.sa_family = AF_INET;
|
|
lws_strncpy(i.ifr_ifrn.ifrn_name, ifname,
|
|
sizeof(i.ifr_ifrn.ifrn_name));
|
|
if (setsockopt(fd, SOL_SOCKET, SO_BINDTODEVICE, &i, sizeof(i)) < 0) {
|
|
lwsl_notice("%s: failed %d\n", __func__, LWS_ERRNO);
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
#else
|
|
lwsl_err("%s: UNIMPLEMENTED on this platform\n", __func__);
|
|
|
|
return -1;
|
|
#endif
|
|
}
|
|
|
|
int
|
|
lws_plat_ifconfig(int fd, lws_dhcpc_ifstate_t *is)
|
|
{
|
|
#if defined(__linux__)
|
|
struct rtentry route;
|
|
struct ifreq ifr;
|
|
|
|
memset(&ifr, 0, sizeof(ifr));
|
|
memset(&route, 0, sizeof(route));
|
|
|
|
lws_strncpy(ifr.ifr_name, is->ifname, IFNAMSIZ);
|
|
|
|
lws_plat_if_up(is->ifname, fd, 0);
|
|
|
|
memcpy(&ifr.ifr_addr, &is->sa46[LWSDH_SA46_IP], sizeof(struct sockaddr));
|
|
if (ioctl(fd, SIOCSIFADDR, &ifr) < 0) {
|
|
lwsl_err("%s: SIOCSIFADDR fail\n", __func__);
|
|
return 1;
|
|
}
|
|
|
|
if (is->sa46[LWSDH_SA46_IP].sa4.sin_family == AF_INET) {
|
|
struct sockaddr_in sin;
|
|
|
|
memset(&sin, 0, sizeof(sin));
|
|
sin.sin_family = AF_INET;
|
|
sin.sin_addr.s_addr = *(uint32_t *)&is->nums[LWSDH_IPV4_SUBNET_MASK];
|
|
memcpy(&ifr.ifr_addr, &sin, sizeof(struct sockaddr));
|
|
if (ioctl(fd, SIOCSIFNETMASK, &ifr) < 0) {
|
|
lwsl_err("%s: SIOCSIFNETMASK fail\n", __func__);
|
|
return 1;
|
|
}
|
|
|
|
lws_plat_if_up(is->ifname, fd, 1);
|
|
|
|
memcpy(&route.rt_gateway,
|
|
&is->sa46[LWSDH_SA46_IPV4_ROUTER].sa4,
|
|
sizeof(struct sockaddr));
|
|
|
|
sin.sin_addr.s_addr = 0;
|
|
memcpy(&route.rt_dst, &sin, sizeof(struct sockaddr));
|
|
memcpy(&route.rt_genmask, &sin, sizeof(struct sockaddr));
|
|
|
|
route.rt_flags = RTF_UP | RTF_GATEWAY;
|
|
route.rt_metric = 100;
|
|
route.rt_dev = (char *)is->ifname;
|
|
|
|
if (ioctl(fd, SIOCADDRT, &route) < 0) {
|
|
lwsl_err("%s: SIOCADDRT 0x%x fail: %d\n", __func__,
|
|
(unsigned int)htonl(*(uint32_t *)&is->
|
|
sa46[LWSDH_SA46_IPV4_ROUTER].
|
|
sa4.sin_addr.s_addr), LWS_ERRNO);
|
|
return 1;
|
|
}
|
|
} else
|
|
lws_plat_if_up(is->ifname, fd, 1);
|
|
|
|
return 0;
|
|
#else
|
|
lwsl_err("%s: UNIMPLEMENTED on this platform\n", __func__);
|
|
|
|
return -1;
|
|
#endif
|
|
}
|
|
|
|
int
|
|
lws_plat_vhost_tls_client_ctx_init(struct lws_vhost *vhost)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
#if defined(LWS_WITH_MBEDTLS)
|
|
int
|
|
lws_plat_mbedtls_net_send(void *ctx, const uint8_t *buf, size_t len)
|
|
{
|
|
int fd = ((mbedtls_net_context *) ctx)->MBEDTLS_PRIVATE_V30_ONLY(fd);
|
|
int ret;
|
|
|
|
if (fd < 0)
|
|
return MBEDTLS_ERR_NET_INVALID_CONTEXT;
|
|
|
|
ret = (int)write(fd, buf, len);
|
|
if (ret >= 0)
|
|
return ret;
|
|
|
|
if (errno == EAGAIN || errno == EWOULDBLOCK)
|
|
return MBEDTLS_ERR_SSL_WANT_WRITE;
|
|
|
|
if (errno == EPIPE || errno == ECONNRESET)
|
|
return MBEDTLS_ERR_NET_CONN_RESET;
|
|
|
|
if( errno == EINTR )
|
|
return MBEDTLS_ERR_SSL_WANT_WRITE;
|
|
|
|
return MBEDTLS_ERR_NET_SEND_FAILED;
|
|
}
|
|
|
|
int
|
|
lws_plat_mbedtls_net_recv(void *ctx, unsigned char *buf, size_t len)
|
|
{
|
|
int fd = ((mbedtls_net_context *) ctx)->MBEDTLS_PRIVATE_V30_ONLY(fd);
|
|
int ret;
|
|
|
|
if (fd < 0)
|
|
return MBEDTLS_ERR_NET_INVALID_CONTEXT;
|
|
|
|
ret = (int)read(fd, buf, len);
|
|
if (ret >= 0)
|
|
return ret;
|
|
|
|
if (errno == EAGAIN || errno == EWOULDBLOCK)
|
|
return MBEDTLS_ERR_SSL_WANT_READ;
|
|
|
|
if (errno == EPIPE || errno == ECONNRESET)
|
|
return MBEDTLS_ERR_NET_CONN_RESET;
|
|
|
|
if (errno == EINTR)
|
|
return MBEDTLS_ERR_SSL_WANT_READ;
|
|
|
|
return MBEDTLS_ERR_NET_RECV_FAILED;
|
|
}
|
|
#endif
|