• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * libwebsockets - small server side websockets and web server implementation
3  *
4  * Copyright (C) 2010 - 2019 Andy Green <andy@warmcat.com>
5  *
6  * Permission is hereby granted, free of charge, to any person obtaining a copy
7  * of this software and associated documentation files (the "Software"), to
8  * deal in the Software without restriction, including without limitation the
9  * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
10  * sell copies of the Software, and to permit persons to whom the Software is
11  * furnished to do so, subject to the following conditions:
12  *
13  * The above copyright notice and this permission notice shall be included in
14  * all copies or substantial portions of the Software.
15  *
16  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
22  * IN THE SOFTWARE.
23  */
24 
25 #if !defined(_GNU_SOURCE)
26 #define _GNU_SOURCE
27 #endif
28 #include "private-lib-core.h"
29 
30 #include <sys/ioctl.h>
31 
32 #if !defined(LWS_DETECTED_PLAT_IOS)
33 #include <net/route.h>
34 #endif
35 
36 #include <net/if.h>
37 
38 #include <pwd.h>
39 #include <grp.h>
40 
41 #if defined(LWS_WITH_MBEDTLS)
42 #if defined(LWS_HAVE_MBEDTLS_NET_SOCKETS)
43 #include "mbedtls/net_sockets.h"
44 #else
45 #include "mbedtls/net.h"
46 #endif
47 #endif
48 
49 #include <netinet/ip.h>
50 
51 int
lws_send_pipe_choked(struct lws * wsi)52 lws_send_pipe_choked(struct lws *wsi)
53 {
54 	struct lws_pollfd fds;
55 	struct lws *wsi_eff;
56 
57 #if !defined(LWS_WITHOUT_EXTENSIONS)
58 	if (wsi->ws && wsi->ws->tx_draining_ext)
59 		return 1;
60 #endif
61 
62 #if defined(LWS_WITH_HTTP2)
63 	wsi_eff = lws_get_network_wsi(wsi);
64 #else
65 	wsi_eff = wsi;
66 #endif
67 
68 	/* the fact we checked implies we avoided back-to-back writes */
69 	wsi_eff->could_have_pending = 0;
70 
71 	/* treat the fact we got a truncated send pending as if we're choked */
72 	if (lws_has_buffered_out(wsi_eff)
73 #if defined(LWS_WITH_HTTP_STREAM_COMPRESSION)
74 	    ||wsi->http.comp_ctx.buflist_comp ||
75 	    wsi->http.comp_ctx.may_have_more
76 #endif
77 	    )
78 		return 1;
79 
80 	fds.fd = wsi_eff->desc.sockfd;
81 	fds.events = POLLOUT;
82 	fds.revents = 0;
83 
84 	if (poll(&fds, 1, 0) != 1)
85 		return 1;
86 
87 	if ((fds.revents & POLLOUT) == 0)
88 		return 1;
89 
90 	/* okay to send another packet without blocking */
91 
92 	return 0;
93 }
94 
95 int
lws_plat_set_nonblocking(lws_sockfd_type fd)96 lws_plat_set_nonblocking(lws_sockfd_type fd)
97 {
98 	return fcntl(fd, F_SETFL, O_NONBLOCK) < 0;
99 }
100 
101 int
lws_plat_set_socket_options(struct lws_vhost * vhost,int fd,int unix_skt)102 lws_plat_set_socket_options(struct lws_vhost *vhost, int fd, int unix_skt)
103 {
104 #ifdef TCP_SYNCNT
105     int max_retry = 2;
106     if (setsockopt(fd, IPPROTO_TCP, TCP_SYNCNT, &max_retry, sizeof(max_retry)) < 0) {
107         return 1;
108     }
109 #endif
110 	int optval = 1;
111 	socklen_t optlen = sizeof(optval);
112 
113 #if defined(__APPLE__) || \
114     defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || \
115     defined(__NetBSD__) || \
116     defined(__OpenBSD__) || \
117     defined(__HAIKU__)
118 	struct protoent *tcp_proto;
119 #endif
120 
121 	(void)fcntl(fd, F_SETFD, FD_CLOEXEC);
122 
123 	if (!unix_skt && vhost->ka_time) {
124 		/* enable keepalive on this socket */
125 		optval = 1;
126 		if (setsockopt(fd, SOL_SOCKET, SO_KEEPALIVE,
127 			       (const void *)&optval, optlen) < 0)
128 			return 1;
129 
130 #if defined(__APPLE__) || \
131     defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || \
132     defined(__NetBSD__) || \
133     defined(__CYGWIN__) || defined(__OpenBSD__) || defined (__sun) || \
134     defined(__HAIKU__)
135 
136 		/*
137 		 * didn't find a way to set these per-socket, need to
138 		 * tune kernel systemwide values
139 		 */
140 #else
141 		/* set the keepalive conditions we want on it too */
142 
143 #if defined(LWS_HAVE_TCP_USER_TIMEOUT)
144 		optval = 1000 * (vhost->ka_time +
145 				 (vhost->ka_interval * vhost->ka_probes));
146 		if (setsockopt(fd, IPPROTO_TCP, TCP_USER_TIMEOUT,
147 			       (const void *)&optval, optlen) < 0)
148 			return 1;
149 #endif
150 		optval = vhost->ka_time;
151 		if (setsockopt(fd, IPPROTO_TCP, TCP_KEEPIDLE,
152 			       (const void *)&optval, optlen) < 0)
153 			return 1;
154 
155 		optval = vhost->ka_interval;
156 		if (setsockopt(fd, IPPROTO_TCP, TCP_KEEPINTVL,
157 			       (const void *)&optval, optlen) < 0)
158 			return 1;
159 
160 		optval = vhost->ka_probes;
161 		if (setsockopt(fd, IPPROTO_TCP, TCP_KEEPCNT,
162 			       (const void *)&optval, optlen) < 0)
163 			return 1;
164 #endif
165 	}
166 
167 #if defined(SO_BINDTODEVICE)
168 	if (!unix_skt && vhost->bind_iface && vhost->iface) {
169 		lwsl_info("binding listen skt to %s using SO_BINDTODEVICE\n", vhost->iface);
170 		if (setsockopt(fd, SOL_SOCKET, SO_BINDTODEVICE, vhost->iface,
171 				(socklen_t)strlen(vhost->iface)) < 0) {
172 			lwsl_warn("Failed to bind to device %s\n", vhost->iface);
173 			return 1;
174 		}
175 	}
176 #endif
177 
178 	/* Disable Nagle */
179 	optval = 1;
180 #if defined (__sun) || defined(__QNX__)
181 	if (!unix_skt && setsockopt(fd, IPPROTO_TCP, TCP_NODELAY, (const void *)&optval, optlen) < 0)
182 		return 1;
183 #elif !defined(__APPLE__) && \
184       !defined(__FreeBSD__) && !defined(__FreeBSD_kernel__) &&        \
185       !defined(__NetBSD__) && \
186       !defined(__OpenBSD__) && \
187       !defined(__HAIKU__)
188 	if (!unix_skt && setsockopt(fd, SOL_TCP, TCP_NODELAY, (const void *)&optval, optlen) < 0)
189 		return 1;
190 #else
191 	tcp_proto = getprotobyname("TCP");
192 	if (!unix_skt && setsockopt(fd, tcp_proto->p_proto, TCP_NODELAY, &optval, optlen) < 0)
193 		return 1;
194 #endif
195 
196 	return lws_plat_set_nonblocking(fd);
197 }
198 
199 static const int ip_opt_lws_flags[] = {
200 	LCCSCF_IP_LOW_LATENCY, LCCSCF_IP_HIGH_THROUGHPUT,
201 	LCCSCF_IP_HIGH_RELIABILITY
202 #if !defined(__OpenBSD__)
203 	, LCCSCF_IP_LOW_COST
204 #endif
205 }, ip_opt_val[] = {
206 	IPTOS_LOWDELAY, IPTOS_THROUGHPUT, IPTOS_RELIABILITY
207 #if !defined(__OpenBSD__) && !defined(__sun) && !defined(__QNX__)
208 	, IPTOS_MINCOST
209 #endif
210 };
211 #if !defined(LWS_WITH_NO_LOGS)
212 static const char *ip_opt_names[] = {
213 	"LOWDELAY", "THROUGHPUT", "RELIABILITY"
214 #if !defined(__OpenBSD__) && !defined(__sun)
215 	, "MINCOST"
216 #endif
217 };
218 #endif
219 
220 int
lws_plat_set_socket_options_ip(lws_sockfd_type fd,uint8_t pri,int lws_flags)221 lws_plat_set_socket_options_ip(lws_sockfd_type fd, uint8_t pri, int lws_flags)
222 {
223 	int optval = (int)pri, ret = 0, n;
224 	socklen_t optlen = sizeof(optval);
225 #if (_LWS_ENABLED_LOGS & LLL_WARN)
226 	int en;
227 #endif
228 
229 #if 0
230 #if defined(TCP_FASTOPEN_CONNECT)
231 	optval = 1;
232 	if (setsockopt(fd, IPPROTO_TCP, TCP_FASTOPEN_CONNECT, (void *)&optval,
233 		       sizeof(optval)))
234 		lwsl_warn("%s: FASTOPEN_CONNECT failed\n", __func__);
235 	optval = (int)pri;
236 #endif
237 #endif
238 
239 #if !defined(__APPLE__) && \
240       !defined(__FreeBSD__) && !defined(__FreeBSD_kernel__) &&        \
241       !defined(__NetBSD__) && \
242       !defined(__OpenBSD__) && \
243       !defined(__sun) && \
244       !defined(__HAIKU__) && \
245       !defined(__CYGWIN__) && \
246       !defined(__QNX__)
247 
248 	/* the BSDs don't have SO_PRIORITY */
249 
250 	if (pri) { /* 0 is the default already */
251 		if (setsockopt(fd, SOL_SOCKET, SO_PRIORITY,
252 				(const void *)&optval, optlen) < 0) {
253 #if (_LWS_ENABLED_LOGS & LLL_WARN)
254 			en = errno;
255 			lwsl_warn("%s: unable to set socket pri %d: errno %d\n",
256 				  __func__, (int)pri, en);
257 #endif
258 			ret = 1;
259 		} else
260 			lwsl_notice("%s: set pri %u\n", __func__, pri);
261 	}
262 #endif
263 
264 	for (n = 0; n < 4; n++) {
265 		if (!(lws_flags & ip_opt_lws_flags[n]))
266 			continue;
267 
268 		optval = (int)ip_opt_val[n];
269 		if (setsockopt(fd, IPPROTO_IP, IP_TOS, (const void *)&optval,
270 			       optlen) < 0) {
271 #if !defined(LWS_WITH_NO_LOGS)
272 			en = errno;
273 			lwsl_warn("%s: unable to set %s: errno %d\n", __func__,
274 				  ip_opt_names[n], en);
275 #endif
276 			ret = 1;
277 		} else
278 			lwsl_notice("%s: set ip flag %s\n", __func__,
279 				    ip_opt_names[n]);
280 	}
281 
282 	return ret;
283 }
284 
285 /* cast a struct sockaddr_in6 * into addr for ipv6 */
286 
287 enum {
288 	IP_SCORE_NONE,
289 	IP_SCORE_NONNATIVE,
290 	IP_SCORE_IPV6_SCOPE_BASE,
291 	/* ipv6 scopes */
292 	IP_SCORE_GLOBAL_NATIVE = 18
293 };
294 
295 int
lws_interface_to_sa(int ipv6,const char * ifname,struct sockaddr_in * addr,size_t addrlen)296 lws_interface_to_sa(int ipv6, const char *ifname, struct sockaddr_in *addr,
297 		    size_t addrlen)
298 {
299 	int rc = LWS_ITOSA_NOT_EXIST;
300 
301 	struct ifaddrs *ifr;
302 	struct ifaddrs *ifc;
303 #if defined(LWS_WITH_IPV6)
304 	struct sockaddr_in6 *addr6 = (struct sockaddr_in6 *)addr;
305 	unsigned long sco = IP_SCORE_NONE;
306 	unsigned long ts;
307 	const uint8_t *p;
308 #endif
309 
310 	if (getifaddrs(&ifr)) {
311 		lwsl_err("%s: unable to getifaddrs: errno %d\n", __func__, errno);
312 
313 		return LWS_ITOSA_USABLE;
314 	}
315 	for (ifc = ifr; ifc != NULL; ifc = ifc->ifa_next) {
316 		if (!ifc->ifa_addr || !ifc->ifa_name)
317 			continue;
318 
319 		lwsl_debug(" interface %s vs %s (fam %d) ipv6 %d\n",
320 			   ifc->ifa_name, ifname,
321 			   ifc->ifa_addr->sa_family, ipv6);
322 
323 		if (strcmp(ifc->ifa_name, ifname))
324 			continue;
325 
326 		switch (ifc->ifa_addr->sa_family) {
327 #if defined(AF_PACKET)
328 		case AF_PACKET:
329 			/* interface exists but is not usable */
330 			if (rc == LWS_ITOSA_NOT_EXIST)
331 				rc = LWS_ITOSA_NOT_USABLE;
332 			continue;
333 #endif
334 
335 		case AF_INET:
336 #if defined(LWS_WITH_IPV6)
337 			if (ipv6) {
338 				/* any existing solution is better than this */
339 				if (sco != IP_SCORE_NONE)
340 					break;
341 				sco = IP_SCORE_NONNATIVE;
342 				rc = LWS_ITOSA_USABLE;
343 				/* map IPv4 to IPv6 */
344 				memset((char *)&addr6->sin6_addr, 0,
345 						sizeof(struct in6_addr));
346 				addr6->sin6_addr.s6_addr[10] = 0xff;
347 				addr6->sin6_addr.s6_addr[11] = 0xff;
348 				memcpy(&addr6->sin6_addr.s6_addr[12],
349 				       &((struct sockaddr_in *)ifc->ifa_addr)->sin_addr,
350 							sizeof(struct in_addr));
351 				lwsl_debug("%s: uplevelling ipv4 bind to ipv6\n", __func__);
352 				break;
353 			}
354 
355 			sco = IP_SCORE_GLOBAL_NATIVE;
356 #endif
357 			rc = LWS_ITOSA_USABLE;
358 			memcpy(addr, (struct sockaddr_in *)ifc->ifa_addr,
359 						    sizeof(struct sockaddr_in));
360 			break;
361 #if defined(LWS_WITH_IPV6)
362 		case AF_INET6:
363 			p = (const uint8_t *)
364 				&((struct sockaddr_in6 *)ifc->ifa_addr)->sin6_addr;
365 			ts = IP_SCORE_IPV6_SCOPE_BASE;
366 			if (p[0] == 0xff)
367 				ts = (unsigned long)(IP_SCORE_IPV6_SCOPE_BASE + (p[1] & 0xf));
368 
369 			if (sco >= ts)
370 				break;
371 
372 			sco = ts;
373 			rc = LWS_ITOSA_USABLE;
374 
375 			memcpy(&addr6->sin6_addr,
376 			     &((struct sockaddr_in6 *)ifc->ifa_addr)->sin6_addr,
377 						       sizeof(struct in6_addr));
378 			break;
379 #endif
380 		default:
381 			break;
382 		}
383 	}
384 
385 	freeifaddrs(ifr);
386 
387 	if (rc &&
388 	    !lws_sa46_parse_numeric_address(ifname, (lws_sockaddr46 *)addr))
389 		rc = LWS_ITOSA_USABLE;
390 
391 	return rc;
392 }
393 
394 
395 const char *
lws_plat_inet_ntop(int af,const void * src,char * dst,socklen_t cnt)396 lws_plat_inet_ntop(int af, const void *src, char *dst, socklen_t cnt)
397 {
398 	return inet_ntop(af, src, dst, cnt);
399 }
400 
401 int
lws_plat_inet_pton(int af,const char * src,void * dst)402 lws_plat_inet_pton(int af, const char *src, void *dst)
403 {
404 	return inet_pton(af, src, dst);
405 }
406 
407 int
lws_plat_ifname_to_hwaddr(int fd,const char * ifname,uint8_t * hwaddr,int len)408 lws_plat_ifname_to_hwaddr(int fd, const char *ifname, uint8_t *hwaddr, int len)
409 {
410 #if defined(__linux__)
411 	struct ifreq i;
412 
413 	memset(&i, 0, sizeof(i));
414 	lws_strncpy(i.ifr_name, ifname, sizeof(i.ifr_name));
415 
416 	if (ioctl(fd, SIOCGIFHWADDR, &i) < 0)
417 		return -1;
418 
419 	memcpy(hwaddr, &i.ifr_hwaddr.sa_data, 6);
420 
421 	return 6;
422 #else
423 	lwsl_err("%s: UNIMPLEMENTED on this platform\n", __func__);
424 
425 	return -1;
426 #endif
427 }
428 
429 int
lws_plat_rawudp_broadcast(uint8_t * p,const uint8_t * canned,size_t canned_len,size_t n,int fd,const char * iface)430 lws_plat_rawudp_broadcast(uint8_t *p, const uint8_t *canned, size_t canned_len,
431 			  size_t n, int fd, const char *iface)
432 {
433 #if defined(__linux__)
434 	struct sockaddr_ll sll;
435 	uint16_t *p16 = (uint16_t *)p;
436 	uint32_t ucs = 0;
437 
438 	memcpy(p, canned, canned_len);
439 
440 	p[2] = (uint8_t)(n >> 8);
441 	p[3] = (uint8_t)(n);
442 
443 	while (p16 < (uint16_t *)(p + 20))
444 		ucs += ntohs(*p16++);
445 
446 	ucs += ucs >> 16;
447 	ucs ^= 0xffff;
448 
449 	p[10] = (uint8_t)(ucs >> 8);
450 	p[11] = (uint8_t)(ucs);
451 	p[24] = (uint8_t)((n - 20) >> 8);
452 	p[25] = (uint8_t)((n - 20));
453 
454 	memset(&sll, 0, sizeof(sll));
455 	sll.sll_family = AF_PACKET;
456 	sll.sll_protocol = htons(0x800);
457 	sll.sll_halen = 6;
458 	sll.sll_ifindex = (int)if_nametoindex(iface);
459 	memset(sll.sll_addr, 0xff, 6);
460 
461 	return (int)sendto(fd, p, n, 0, (struct sockaddr *)&sll, sizeof(sll));
462 #else
463 	lwsl_err("%s: UNIMPLEMENTED on this platform\n", __func__);
464 
465 	return -1;
466 #endif
467 }
468 
469 int
lws_plat_if_up(const char * ifname,int fd,int up)470 lws_plat_if_up(const char *ifname, int fd, int up)
471 {
472 #if defined(__linux__)
473 	struct ifreq ifr;
474 
475 	memset(&ifr, 0, sizeof(ifr));
476 	lws_strncpy(ifr.ifr_name, ifname, IFNAMSIZ);
477 
478 	if (ioctl(fd, SIOCGIFFLAGS, &ifr) < 0) {
479 		lwsl_err("%s: SIOCGIFFLAGS fail\n", __func__);
480 		return 1;
481 	}
482 
483 	if (up)
484 		ifr.ifr_flags |= IFF_UP;
485 	else
486 		ifr.ifr_flags &= ~IFF_UP;
487 
488 	if (ioctl(fd, SIOCSIFFLAGS, &ifr) < 0) {
489 		lwsl_err("%s: SIOCSIFFLAGS fail\n", __func__);
490 		return 1;
491 	}
492 
493 	return 0;
494 #else
495 	lwsl_err("%s: UNIMPLEMENTED on this platform\n", __func__);
496 
497 	return -1;
498 #endif
499 }
500 
501 int
lws_plat_BINDTODEVICE(lws_sockfd_type fd,const char * ifname)502 lws_plat_BINDTODEVICE(lws_sockfd_type fd, const char *ifname)
503 {
504 #if defined(__linux__)
505 	struct ifreq i;
506 
507 	memset(&i, 0, sizeof(i));
508 	i.ifr_addr.sa_family = AF_INET;
509 	lws_strncpy(i.ifr_ifrn.ifrn_name, ifname,
510 		    sizeof(i.ifr_ifrn.ifrn_name));
511 	if (setsockopt(fd, SOL_SOCKET, SO_BINDTODEVICE, &i, sizeof(i)) < 0) {
512 		lwsl_notice("%s: failed %d\n", __func__, LWS_ERRNO);
513 		return 1;
514 	}
515 
516 	return 0;
517 #else
518 	lwsl_err("%s: UNIMPLEMENTED on this platform\n", __func__);
519 
520 	return -1;
521 #endif
522 }
523 
524 int
lws_plat_ifconfig(int fd,lws_dhcpc_ifstate_t * is)525 lws_plat_ifconfig(int fd, lws_dhcpc_ifstate_t *is)
526 {
527 #if defined(__linux__)
528 	struct rtentry route;
529 	struct ifreq ifr;
530 
531 	memset(&ifr, 0, sizeof(ifr));
532 	memset(&route, 0, sizeof(route));
533 
534 	lws_strncpy(ifr.ifr_name, is->ifname, IFNAMSIZ);
535 
536 	lws_plat_if_up(is->ifname, fd, 0);
537 
538 	memcpy(&ifr.ifr_addr, &is->sa46[LWSDH_SA46_IP], sizeof(struct sockaddr));
539 	if (ioctl(fd, SIOCSIFADDR, &ifr) < 0) {
540 		lwsl_err("%s: SIOCSIFADDR fail\n", __func__);
541 		return 1;
542 	}
543 
544 	if (is->sa46[LWSDH_SA46_IP].sa4.sin_family == AF_INET) {
545 		struct sockaddr_in sin;
546 
547 		memset(&sin, 0, sizeof(sin));
548 		sin.sin_family = AF_INET;
549 		sin.sin_addr.s_addr = *(uint32_t *)&is->nums[LWSDH_IPV4_SUBNET_MASK];
550 		memcpy(&ifr.ifr_addr, &sin, sizeof(struct sockaddr));
551 		if (ioctl(fd, SIOCSIFNETMASK, &ifr) < 0) {
552 			lwsl_err("%s: SIOCSIFNETMASK fail\n", __func__);
553 			return 1;
554 		}
555 
556 		lws_plat_if_up(is->ifname, fd, 1);
557 
558 		memcpy(&route.rt_gateway,
559 		       &is->sa46[LWSDH_SA46_IPV4_ROUTER].sa4,
560 		       sizeof(struct sockaddr));
561 
562 		sin.sin_addr.s_addr = 0;
563 		memcpy(&route.rt_dst, &sin, sizeof(struct sockaddr));
564 		memcpy(&route.rt_genmask, &sin, sizeof(struct sockaddr));
565 
566 		route.rt_flags = RTF_UP | RTF_GATEWAY;
567 		route.rt_metric = 100;
568 		route.rt_dev = (char *)is->ifname;
569 
570 		if (ioctl(fd, SIOCADDRT, &route) < 0) {
571 			lwsl_err("%s: SIOCADDRT 0x%x fail: %d\n", __func__,
572 				(unsigned int)htonl(*(uint32_t *)&is->
573 					sa46[LWSDH_SA46_IPV4_ROUTER].
574 						sa4.sin_addr.s_addr), LWS_ERRNO);
575 			return 1;
576 		}
577 	} else
578 		lws_plat_if_up(is->ifname, fd, 1);
579 
580 	return 0;
581 #else
582 	lwsl_err("%s: UNIMPLEMENTED on this platform\n", __func__);
583 
584 	return -1;
585 #endif
586 }
587 
588 int
lws_plat_vhost_tls_client_ctx_init(struct lws_vhost * vhost)589 lws_plat_vhost_tls_client_ctx_init(struct lws_vhost *vhost)
590 {
591 	return 0;
592 }
593 
594 #if defined(LWS_WITH_MBEDTLS)
595 int
lws_plat_mbedtls_net_send(void * ctx,const uint8_t * buf,size_t len)596 lws_plat_mbedtls_net_send(void *ctx, const uint8_t *buf, size_t len)
597 {
598 	int fd = ((mbedtls_net_context *) ctx)->MBEDTLS_PRIVATE_V30_ONLY(fd);
599 	int ret;
600 
601 	if (fd < 0)
602 		return MBEDTLS_ERR_NET_INVALID_CONTEXT;
603 
604 	ret = (int)write(fd, buf, len);
605 	if (ret >= 0)
606 		return ret;
607 
608 	if (errno == EAGAIN || errno == EWOULDBLOCK)
609 		return MBEDTLS_ERR_SSL_WANT_WRITE;
610 
611 	if (errno == EPIPE || errno == ECONNRESET)
612 		return MBEDTLS_ERR_NET_CONN_RESET;
613 
614 	if( errno == EINTR )
615 		return MBEDTLS_ERR_SSL_WANT_WRITE;
616 
617 	return MBEDTLS_ERR_NET_SEND_FAILED;
618 }
619 
620 int
lws_plat_mbedtls_net_recv(void * ctx,unsigned char * buf,size_t len)621 lws_plat_mbedtls_net_recv(void *ctx, unsigned char *buf, size_t len)
622 {
623 	int fd = ((mbedtls_net_context *) ctx)->MBEDTLS_PRIVATE_V30_ONLY(fd);
624 	int ret;
625 
626 	if (fd < 0)
627 		return MBEDTLS_ERR_NET_INVALID_CONTEXT;
628 
629 	ret = (int)read(fd, buf, len);
630 	if (ret >= 0)
631 		return ret;
632 
633 	if (errno == EAGAIN || errno == EWOULDBLOCK)
634 		return MBEDTLS_ERR_SSL_WANT_READ;
635 
636 	if (errno == EPIPE || errno == ECONNRESET)
637 		return MBEDTLS_ERR_NET_CONN_RESET;
638 
639 	if (errno == EINTR)
640 		return MBEDTLS_ERR_SSL_WANT_READ;
641 
642 	return MBEDTLS_ERR_NET_RECV_FAILED;
643 }
644 #endif
645