libcoap 4.3.5-develop-1ba3158
Loading...
Searching...
No Matches
coap_io.c
Go to the documentation of this file.
1/* coap_io.c -- Default network I/O functions for libcoap
2 *
3 * Copyright (C) 2012,2014,2016-2025 Olaf Bergmann <bergmann@tzi.org> and others
4 *
5 * SPDX-License-Identifier: BSD-2-Clause
6 *
7 * This file is part of the CoAP library libcoap. Please see
8 * README for terms of use.
9 */
10
17
18#ifdef HAVE_STDIO_H
19# include <stdio.h>
20#endif
21
22#ifdef HAVE_SYS_SELECT_H
23# include <sys/select.h>
24#endif
25#ifdef HAVE_SYS_SOCKET_H
26# include <sys/socket.h>
27# define OPTVAL_T(t) (t)
28# define OPTVAL_GT(t) (t)
29#endif
30#ifdef HAVE_SYS_IOCTL_H
31#include <sys/ioctl.h>
32#endif
33#ifdef HAVE_NETINET_IN_H
34# include <netinet/in.h>
35#endif
36#ifdef HAVE_WS2TCPIP_H
37#include <ws2tcpip.h>
38# define OPTVAL_T(t) (const char*)(t)
39# define OPTVAL_GT(t) (char*)(t)
40# undef CMSG_DATA
41# define CMSG_DATA WSA_CMSG_DATA
42#endif
43#ifdef HAVE_SYS_UIO_H
44# include <sys/uio.h>
45#endif
46#ifdef HAVE_UNISTD_H
47# include <unistd.h>
48#endif
49#ifdef COAP_EPOLL_SUPPORT
50#include <sys/epoll.h>
51#include <sys/timerfd.h>
52#ifdef HAVE_LIMITS_H
53#include <limits.h>
54#endif
55#endif /* COAP_EPOLL_SUPPORT */
56
57#if !defined(WITH_CONTIKI) && !defined(RIOT_VERSION) && !defined(WITH_LWIP)
58/* define generic PKTINFO for IPv4 */
59#if defined(IP_PKTINFO)
60# define GEN_IP_PKTINFO IP_PKTINFO
61#elif defined(IP_RECVDSTADDR)
62# define GEN_IP_PKTINFO IP_RECVDSTADDR
63#else
64# error "Need IP_PKTINFO or IP_RECVDSTADDR to request ancillary data from OS."
65#endif /* IP_PKTINFO */
66
67/* define generic PKTINFO for IPv6 */
68#ifdef IPV6_RECVPKTINFO
69# define GEN_IPV6_PKTINFO IPV6_RECVPKTINFO
70#elif defined(IPV6_PKTINFO)
71# define GEN_IPV6_PKTINFO IPV6_PKTINFO
72#else
73# error "Need IPV6_PKTINFO or IPV6_RECVPKTINFO to request ancillary data from OS."
74#endif /* IPV6_RECVPKTINFO */
75#endif /* ! WITH_CONTIKI && ! RIOT_VERSION && ! WITH_LWIP */
76
77#if COAP_SERVER_SUPPORT
81}
82
83void
86}
87#endif /* COAP_SERVER_SUPPORT */
88
89#if defined(__MINGW32__)
90#if(_WIN32_WINNT >= 0x0600)
91#define CMSG_FIRSTHDR WSA_CMSG_FIRSTHDR
92#define CMSG_NXTHDR WSA_CMSG_NXTHDR
93#define CMSG_LEN WSA_CMSG_LEN
94#define CMSG_SPACE WSA_CMSG_SPACE
95#define cmsghdr _WSACMSGHDR
96#endif /* (_WIN32_WINNT>=0x0600) */
97#endif /* defined(__MINGW32__) */
98
99#if !defined(WITH_CONTIKI) && !defined(WITH_LWIP) && !defined(RIOT_VERSION)
100
101#if COAP_SERVER_SUPPORT
102int
104 const coap_address_t *listen_addr,
105 coap_address_t *bound_addr) {
106#ifndef RIOT_VERSION
107 int on = 1;
108#if COAP_IPV6_SUPPORT
109 int off = 0;
110#endif /* COAP_IPV6_SUPPORT */
111#else /* ! RIOT_VERSION */
112 struct timeval timeout = {0, 0};
113#endif /* ! RIOT_VERSION */
114#ifdef _WIN32
115 u_long u_on = 1;
116#endif
117
118 sock->fd = socket(listen_addr->addr.sa.sa_family, SOCK_DGRAM, 0);
119
120 if (sock->fd == COAP_INVALID_SOCKET) {
121 coap_log_warn("coap_socket_bind_udp: socket: %s\n", coap_socket_strerror());
122 goto error;
123 }
124#ifndef RIOT_VERSION
125#ifdef _WIN32
126 if (ioctlsocket(sock->fd, FIONBIO, &u_on) == COAP_SOCKET_ERROR)
127#else
128 if (ioctl(sock->fd, FIONBIO, &on) == COAP_SOCKET_ERROR)
129#endif
130 {
131 coap_log_warn("coap_socket_bind_udp: ioctl FIONBIO: %s\n", coap_socket_strerror());
132 }
133
134 if (setsockopt(sock->fd, SOL_SOCKET, SO_REUSEADDR, OPTVAL_T(&on), sizeof(on)) == COAP_SOCKET_ERROR)
135 coap_log_warn("coap_socket_bind_udp: setsockopt SO_REUSEADDR: %s\n",
137
138 switch (listen_addr->addr.sa.sa_family) {
139#if COAP_IPV4_SUPPORT
140 case AF_INET:
141 if (setsockopt(sock->fd, IPPROTO_IP, GEN_IP_PKTINFO, OPTVAL_T(&on),
142 sizeof(on)) == COAP_SOCKET_ERROR)
143 coap_log_alert("coap_socket_bind_udp: setsockopt IP_PKTINFO: %s\n",
145 break;
146#endif /* COAP_IPV4_SUPPORT */
147#if COAP_IPV6_SUPPORT
148 case AF_INET6:
149 /* Configure the socket as dual-stacked */
150 if (setsockopt(sock->fd, IPPROTO_IPV6, IPV6_V6ONLY, OPTVAL_T(&off),
151 sizeof(off)) == COAP_SOCKET_ERROR)
152 coap_log_alert("coap_socket_bind_udp: setsockopt IPV6_V6ONLY: %s\n",
154#if !defined(ESPIDF_VERSION)
155 if (setsockopt(sock->fd, IPPROTO_IPV6, GEN_IPV6_PKTINFO, OPTVAL_T(&on),
156 sizeof(on)) == COAP_SOCKET_ERROR)
157 coap_log_alert("coap_socket_bind_udp: setsockopt IPV6_PKTINFO: %s\n",
159#endif /* !defined(ESPIDF_VERSION) */
160#endif /* COAP_IPV6_SUPPORT */
161 setsockopt(sock->fd, IPPROTO_IP, GEN_IP_PKTINFO, OPTVAL_T(&on), sizeof(on));
162 /* ignore error, because likely cause is that IPv4 is disabled at the os
163 level */
164 break;
165#if COAP_AF_UNIX_SUPPORT
166 case AF_UNIX:
167 break;
168#endif /* COAP_AF_UNIX_SUPPORT */
169 default:
170 coap_log_alert("coap_socket_bind_udp: unsupported sa_family\n");
171 break;
172 }
173#else /* RIOT_VERSION */
174 if (setsockopt(sock->fd, SOL_SOCKET, SO_RCVTIMEO, OPTVAL_T(&timeout),
175 (socklen_t)sizeof(timeout)) == COAP_SOCKET_ERROR)
176 coap_log_alert("coap_socket_bind_udp: setsockopt SO_RCVTIMEO: %s\n",
178#endif /* RIOT_VERSION */
179
180 if (bind(sock->fd, &listen_addr->addr.sa,
182 listen_addr->addr.sa.sa_family == AF_INET ?
183 (socklen_t)sizeof(struct sockaddr_in) :
184#endif /* COAP_IPV4_SUPPORT */
185 (socklen_t)listen_addr->size) == COAP_SOCKET_ERROR) {
186 coap_log_warn("coap_socket_bind_udp: bind: %s\n",
188 goto error;
189 }
190
191 bound_addr->size = (socklen_t)sizeof(*bound_addr);
192 if (getsockname(sock->fd, &bound_addr->addr.sa, &bound_addr->size) < 0) {
193 coap_log_warn("coap_socket_bind_udp: getsockname: %s\n",
195 goto error;
196 }
197#if defined(RIOT_VERSION) && defined(COAP_SERVER_SUPPORT)
198 if (sock->endpoint &&
199 bound_addr->addr.sa.sa_family == AF_INET6) {
200 bound_addr->addr.sin6.sin6_scope_id =
201 listen_addr->addr.sin6.sin6_scope_id;
202 bound_addr->addr.sin6.sin6_flowinfo = 0;
203 }
204#endif /* RIOT_VERSION && COAP_SERVER_SUPPORT */
205
206 return 1;
207
208error:
209 coap_socket_close(sock);
210 return 0;
211}
212#endif /* COAP_SERVER_SUPPORT */
213
214#if COAP_CLIENT_SUPPORT
215int
217 const coap_address_t *local_if,
218 const coap_address_t *server,
219 int default_port,
220 coap_address_t *local_addr,
221 coap_address_t *remote_addr) {
222 int on = 1;
223#if COAP_IPV6_SUPPORT
224 int off = 0;
225#endif /* COAP_IPV6_SUPPORT */
226#ifdef _WIN32
227 u_long u_on = 1;
228#endif
229 coap_address_t connect_addr;
230 int is_mcast = coap_is_mcast(server);
231 coap_address_copy(&connect_addr, server);
232
234 sock->fd = socket(connect_addr.addr.sa.sa_family, SOCK_DGRAM, 0);
235
236 if (sock->fd == COAP_INVALID_SOCKET) {
237 coap_log_warn("coap_socket_connect_udp: socket: %s\n",
239 goto error;
240 }
241
242#ifdef _WIN32
243 if (ioctlsocket(sock->fd, FIONBIO, &u_on) == COAP_SOCKET_ERROR)
244#else
245 if (ioctl(sock->fd, FIONBIO, &on) == COAP_SOCKET_ERROR)
246#endif
247 {
248 coap_log_warn("coap_socket_connect_udp: ioctl FIONBIO: %s\n",
250 }
251
252 switch (connect_addr.addr.sa.sa_family) {
253#if COAP_IPV4_SUPPORT
254 case AF_INET:
255 if (connect_addr.addr.sin.sin_port == 0)
256 connect_addr.addr.sin.sin_port = htons(default_port);
257 break;
258#endif /* COAP_IPV4_SUPPORT */
259#if COAP_IPV6_SUPPORT
260 case AF_INET6:
261 if (connect_addr.addr.sin6.sin6_port == 0)
262 connect_addr.addr.sin6.sin6_port = htons(default_port);
263 /* Configure the socket as dual-stacked */
264 if (setsockopt(sock->fd, IPPROTO_IPV6, IPV6_V6ONLY, OPTVAL_T(&off),
265 sizeof(off)) == COAP_SOCKET_ERROR)
266 if (errno != ENOSYS) {
267 coap_log_warn("coap_socket_connect_udp: setsockopt IPV6_V6ONLY: %s\n",
269 }
270#endif /* COAP_IPV6_SUPPORT */
271 break;
272#if COAP_AF_UNIX_SUPPORT
273 case AF_UNIX:
274 break;
275#endif /* COAP_AF_UNIX_SUPPORT */
276 default:
277 coap_log_alert("coap_socket_connect_udp: unsupported sa_family %d\n",
278 connect_addr.addr.sa.sa_family);
279 goto error;;
280 }
281
282 if (local_if && local_if->addr.sa.sa_family) {
283 if (local_if->addr.sa.sa_family != connect_addr.addr.sa.sa_family) {
284 coap_log_warn("coap_socket_connect_udp: local address family != "
285 "remote address family\n");
286 goto error;
287 }
288 if (setsockopt(sock->fd, SOL_SOCKET, SO_REUSEADDR, OPTVAL_T(&on), sizeof(on)) == COAP_SOCKET_ERROR)
289 coap_log_warn("coap_socket_connect_udp: setsockopt SO_REUSEADDR: %s\n",
291 if (bind(sock->fd, &local_if->addr.sa,
293 local_if->addr.sa.sa_family == AF_INET ?
294 (socklen_t)sizeof(struct sockaddr_in) :
295#endif /* COAP_IPV4_SUPPORT */
296 (socklen_t)local_if->size) == COAP_SOCKET_ERROR) {
297 coap_log_warn("coap_socket_connect_udp: bind: %s\n",
299 goto error;
300 }
301#if COAP_AF_UNIX_SUPPORT
302 } else if (connect_addr.addr.sa.sa_family == AF_UNIX) {
303 /* Need to bind to a local address for clarity over endpoints */
304 coap_log_warn("coap_socket_connect_udp: local address required\n");
305 goto error;
306#endif /* COAP_AF_UNIX_SUPPORT */
307 }
308
309 /* special treatment for sockets that are used for multicast communication */
310 if (is_mcast) {
311 if (!(local_if && local_if->addr.sa.sa_family)) {
312 /* Bind to a (unused) port to simplify logging */
313 coap_address_t bind_addr;
314
315 coap_address_init(&bind_addr);
316 bind_addr.addr.sa.sa_family = connect_addr.addr.sa.sa_family;
317 if (bind(sock->fd, &bind_addr.addr.sa,
319 bind_addr.addr.sa.sa_family == AF_INET ?
320 (socklen_t)sizeof(struct sockaddr_in) :
321#endif /* COAP_IPV4_SUPPORT */
322 (socklen_t)bind_addr.size) == COAP_SOCKET_ERROR) {
323 coap_log_warn("coap_socket_connect_udp: bind: %s\n",
325 goto error;
326 }
327 }
328 if (getsockname(sock->fd, &local_addr->addr.sa, &local_addr->size) == COAP_SOCKET_ERROR) {
329 coap_log_warn("coap_socket_connect_udp: getsockname for multicast socket: %s\n",
331 }
332 coap_address_copy(remote_addr, &connect_addr);
333 coap_address_copy(&sock->mcast_addr, &connect_addr);
335 if (coap_is_bcast(server) &&
336 setsockopt(sock->fd, SOL_SOCKET, SO_BROADCAST, OPTVAL_T(&on),
337 sizeof(on)) == COAP_SOCKET_ERROR)
338 coap_log_warn("coap_socket_connect_udp: setsockopt SO_BROADCAST: %s\n",
340 return 1;
341 }
342
343 if (connect(sock->fd, &connect_addr.addr.sa, connect_addr.size) == COAP_SOCKET_ERROR) {
344#if COAP_AF_UNIX_SUPPORT
345 if (connect_addr.addr.sa.sa_family == AF_UNIX) {
346 coap_log_warn("coap_socket_connect_udp: connect: %s: %s\n",
347 connect_addr.addr.cun.sun_path, coap_socket_strerror());
348 } else
349#endif /* COAP_AF_UNIX_SUPPORT */
350 {
351 coap_log_warn("coap_socket_connect_udp: connect: %s (%d)\n",
352 coap_socket_strerror(), connect_addr.addr.sa.sa_family);
353 }
354 goto error;
355 }
356
357 if (getsockname(sock->fd, &local_addr->addr.sa, &local_addr->size) == COAP_SOCKET_ERROR) {
358 coap_log_warn("coap_socket_connect_udp: getsockname: %s\n",
360 }
361
362 if (getpeername(sock->fd, &remote_addr->addr.sa, &remote_addr->size) == COAP_SOCKET_ERROR) {
363 coap_log_warn("coap_socket_connect_udp: getpeername: %s\n",
365 }
366
368 return 1;
369
370error:
371 coap_socket_close(sock);
372 return 0;
373}
374#endif /* COAP_CLIENT_SUPPORT */
375
376void
378 if (sock->fd != COAP_INVALID_SOCKET) {
379#ifdef COAP_EPOLL_SUPPORT
380#if COAP_SERVER_SUPPORT
381 coap_context_t *context = sock->session ? sock->session->context :
382 sock->endpoint ? sock->endpoint->context : NULL;
383#else /* COAP_SERVER_SUPPORT */
384 coap_context_t *context = sock->session ? sock->session->context : NULL;
385#endif /* COAP_SERVER_SUPPORT */
386 if (context != NULL) {
387 int ret;
388 struct epoll_event event;
389
390 /* Kernels prior to 2.6.9 expect non NULL event parameter */
391 ret = epoll_ctl(context->epfd, EPOLL_CTL_DEL, sock->fd, &event);
392 if (ret == -1 && errno != ENOENT) {
393 coap_log_err("%s: epoll_ctl DEL failed: %s (%d)\n",
394 "coap_socket_close",
395 coap_socket_strerror(), errno);
396 }
397 }
398#if COAP_SERVER_SUPPORT
399#if COAP_AF_UNIX_SUPPORT
400 if (sock->endpoint &&
401 sock->endpoint->bind_addr.addr.sa.sa_family == AF_UNIX) {
402 /* Clean up Unix endpoint */
403 unlink(sock->endpoint->bind_addr.addr.cun.sun_path);
404 }
405#endif /* COAP_AF_UNIX_SUPPORT */
406 sock->endpoint = NULL;
407#endif /* COAP_SERVER_SUPPORT */
408#if COAP_CLIENT_SUPPORT
409#if COAP_AF_UNIX_SUPPORT
410 if (sock->session && sock->session->type == COAP_SESSION_TYPE_CLIENT &&
411 sock->session->addr_info.local.addr.sa.sa_family == AF_UNIX) {
412 /* Clean up Unix endpoint */
413 unlink(sock->session->addr_info.local.addr.cun.sun_path);
414 }
415#endif /* COAP_AF_UNIX_SUPPORT */
416#endif /* COAP_CLIENT_SUPPORT */
417 sock->session = NULL;
418#endif /* COAP_EPOLL_SUPPORT */
419 coap_closesocket(sock->fd);
420 sock->fd = COAP_INVALID_SOCKET;
421 }
422 sock->flags = COAP_SOCKET_EMPTY;
423}
424
425#ifdef COAP_EPOLL_SUPPORT
426void
428 uint32_t events,
429 const char *func) {
430 int ret;
431 struct epoll_event event;
432 coap_context_t *context;
433
434#if COAP_MAX_LOGGING_LEVEL < _COAP_LOG_ERR
435 (void)func;
436#endif
437
438 if (sock == NULL)
439 return;
440
441#if COAP_SERVER_SUPPORT
442 context = sock->session ? sock->session->context :
443 sock->endpoint ? sock->endpoint->context : NULL;
444#else /* ! COAP_SERVER_SUPPORT */
445 context = sock->session ? sock->session->context : NULL;
446#endif /* ! COAP_SERVER_SUPPORT */
447 if (context == NULL)
448 return;
449
450 /* Needed if running 32bit as ptr is only 32bit */
451 memset(&event, 0, sizeof(event));
452 event.events = events;
453 event.data.ptr = sock;
454
455 ret = epoll_ctl(context->epfd, EPOLL_CTL_ADD, sock->fd, &event);
456 if (ret == -1) {
457 coap_log_err("%s: epoll_ctl ADD failed: %s (%d)\n",
458 func,
459 coap_socket_strerror(), errno);
460 }
461}
462
463void
465 uint32_t events,
466 const char *func) {
467 int ret;
468 struct epoll_event event;
469 coap_context_t *context;
470
471#if COAP_MAX_LOGGING_LEVEL < _COAP_LOG_ERR
472 (void)func;
473#endif
474
475 if (sock == NULL)
476 return;
477
478#if COAP_SERVER_SUPPORT
479 context = sock->session ? sock->session->context :
480 sock->endpoint ? sock->endpoint->context : NULL;
481#else /* COAP_SERVER_SUPPORT */
482 context = sock->session ? sock->session->context : NULL;
483#endif /* COAP_SERVER_SUPPORT */
484 if (context == NULL)
485 return;
486
487 event.events = events;
488 event.data.ptr = sock;
489
490 ret = epoll_ctl(context->epfd, EPOLL_CTL_MOD, sock->fd, &event);
491 if (ret == -1) {
492#if (COAP_MAX_LOGGING_LEVEL < COAP_LOG_ERR)
493 (void)func;
494#endif
495 coap_log_err("%s: epoll_ctl MOD failed: %s (%d)\n",
496 func,
497 coap_socket_strerror(), errno);
498 }
499}
500#endif /* COAP_EPOLL_SUPPORT */
501
502#endif /* ! WITH_CONTIKI && ! WITH_LWIP && ! RIOT_VERSION*/
503
504#ifndef WITH_CONTIKI
505void
507#if COAP_EPOLL_SUPPORT
508 if (context->eptimerfd != -1) {
509 coap_tick_t now;
510
511 coap_ticks(&now);
512 if (context->next_timeout == 0 || context->next_timeout > now + delay) {
513 struct itimerspec new_value;
514 int ret;
515
516 context->next_timeout = now + delay;
517 memset(&new_value, 0, sizeof(new_value));
518 if (delay == 0) {
519 new_value.it_value.tv_nsec = 1; /* small but not zero */
520 } else {
521 new_value.it_value.tv_sec = delay / COAP_TICKS_PER_SECOND;
522 new_value.it_value.tv_nsec = (delay % COAP_TICKS_PER_SECOND) *
523 1000000;
524 }
525 ret = timerfd_settime(context->eptimerfd, 0, &new_value, NULL);
526 if (ret == -1) {
527 coap_log_err("%s: timerfd_settime failed: %s (%d)\n",
528 "coap_update_io_timer",
529 coap_socket_strerror(), errno);
530 }
531#ifdef COAP_DEBUG_WAKEUP_TIMES
532 else {
533 coap_log_debug("****** Next wakeup time %3ld.%09ld\n",
534 new_value.it_value.tv_sec, new_value.it_value.tv_nsec);
535 }
536#endif /* COAP_DEBUG_WAKEUP_TIMES */
537 }
538 }
539#else /* ! COAP_EPOLL_SUPPORT */
540 coap_tick_t now;
541
542 coap_ticks(&now);
543 if (context->next_timeout == 0 || context->next_timeout > now + delay) {
544 context->next_timeout = now + delay;
545 }
546#endif /* ! COAP_EPOLL_SUPPORT */
547}
548#endif /* ! WITH_CONTIKI */
549
550#if !defined(WITH_CONTIKI) && !defined(WITH_LWIP) && !defined(RIOT_VERSION)
551
552#ifdef _WIN32
553static void
554coap_win_error_to_errno(void) {
555 int w_error = WSAGetLastError();
556 switch (w_error) {
557 case WSA_NOT_ENOUGH_MEMORY:
558 errno = ENOMEM;
559 break;
560 case WSA_INVALID_PARAMETER:
561 errno = EINVAL;
562 break;
563 case WSAEINTR:
564 errno = EINTR;
565 break;
566 case WSAEBADF:
567 errno = EBADF;
568 break;
569 case WSAEACCES:
570 errno = EACCES;
571 break;
572 case WSAEFAULT:
573 errno = EFAULT;
574 break;
575 case WSAEINVAL:
576 errno = EINVAL;
577 break;
578 case WSAEMFILE:
579 errno = EMFILE;
580 break;
581 case WSAEWOULDBLOCK:
582 errno = EWOULDBLOCK;
583 break;
584 case WSAENETDOWN:
585 errno = ENETDOWN;
586 break;
587 case WSAENETUNREACH:
588 errno = ENETUNREACH;
589 break;
590 case WSAENETRESET:
591 errno = ENETRESET;
592 break;
593 case WSAECONNABORTED:
594 errno = ECONNABORTED;
595 break;
596 case WSAECONNRESET:
597 errno = ECONNRESET;
598 break;
599 case WSAENOBUFS:
600 errno = ENOBUFS;
601 break;
602 case WSAETIMEDOUT:
603 errno = ETIMEDOUT;
604 break;
605 case WSAECONNREFUSED:
606 errno = ECONNREFUSED;
607 break;
608 default:
609 coap_log_err("WSAGetLastError: %d mapping to errno failed - please fix\n",
610 w_error);
611 errno = EPERM;
612 break;
613 }
614}
615#endif /* _WIN32 */
616
617/*
618 * strm
619 * return +ve Number of bytes written.
620 * 0 No data written.
621 * -1 Error (error in errno).
622 */
623ssize_t
624coap_socket_write(coap_socket_t *sock, const uint8_t *data, size_t data_len) {
625 ssize_t r;
626
628#ifdef _WIN32
629 r = send(sock->fd, (const char *)data, (int)data_len, 0);
630#else
631#ifndef MSG_NOSIGNAL
632#define MSG_NOSIGNAL 0
633#endif /* MSG_NOSIGNAL */
634 r = send(sock->fd, data, data_len, MSG_NOSIGNAL);
635#endif
636 if (r == COAP_SOCKET_ERROR) {
637#ifdef _WIN32
638 coap_win_error_to_errno();
639#endif /* _WIN32 */
640 if (errno==EAGAIN ||
641#if EAGAIN != EWOULDBLOCK
642 errno == EWOULDBLOCK ||
643#endif
644 errno == EINTR) {
646#ifdef COAP_EPOLL_SUPPORT
648 EPOLLOUT |
649 ((sock->flags & COAP_SOCKET_WANT_READ) ?
650 EPOLLIN : 0),
651 __func__);
652#endif /* COAP_EPOLL_SUPPORT */
653 return 0;
654 }
655 if (errno == EPIPE || errno == ECONNRESET) {
656 coap_log_info("coap_socket_write: send: %s\n",
658 } else {
659 coap_log_warn("coap_socket_write: send: %s\n",
661 }
662 return -1;
663 }
664 if (r < (ssize_t)data_len) {
666#ifdef COAP_EPOLL_SUPPORT
668 EPOLLOUT |
669 ((sock->flags & COAP_SOCKET_WANT_READ) ?
670 EPOLLIN : 0),
671 __func__);
672#endif /* COAP_EPOLL_SUPPORT */
673 }
674 return r;
675}
676
677/*
678 * strm
679 * return >=0 Number of bytes read.
680 * -1 Error (error in errno).
681 */
682ssize_t
683coap_socket_read(coap_socket_t *sock, uint8_t *data, size_t data_len) {
684 ssize_t r;
685
686#ifdef _WIN32
687 r = recv(sock->fd, (char *)data, (int)data_len, 0);
688#else
689 r = recv(sock->fd, data, data_len, 0);
690#endif
691 if (r == 0) {
692 /* graceful shutdown */
693 sock->flags &= ~COAP_SOCKET_CAN_READ;
694 errno = ECONNRESET;
695 return -1;
696 } else if (r == COAP_SOCKET_ERROR) {
697 sock->flags &= ~COAP_SOCKET_CAN_READ;
698#ifdef _WIN32
699 coap_win_error_to_errno();
700#endif /* _WIN32 */
701 if (errno==EAGAIN ||
702#if EAGAIN != EWOULDBLOCK
703 errno == EWOULDBLOCK ||
704#endif
705 errno == EINTR) {
706 return 0;
707 }
708 if (errno != ECONNRESET) {
709 coap_log_warn("coap_socket_read: recv: %s\n",
711 }
712 return -1;
713 }
714 if (r < (ssize_t)data_len)
715 sock->flags &= ~COAP_SOCKET_CAN_READ;
716 return r;
717}
718
719#endif /* ! WITH_CONTIKI && ! WITH_LWIP && ! RIOT_VERSION */
720
721#if !defined(WITH_LWIP)
722#if (!defined(WITH_CONTIKI)) != ( defined(HAVE_NETINET_IN_H) || defined(HAVE_WS2TCPIP_H) )
723/* define struct in6_pktinfo and struct in_pktinfo if not available
724 FIXME: check with configure
725*/
726#if !defined(__MINGW32__) && !defined(RIOT_VERSION)
728 struct in6_addr ipi6_addr; /* src/dst IPv6 address */
729 unsigned int ipi6_ifindex; /* send/recv interface index */
730};
731
734 struct in_addr ipi_spec_dst;
735 struct in_addr ipi_addr;
736};
737#endif /* ! __MINGW32__ */
738#endif
739#endif /* ! WITH_LWIP */
740
741#if !defined(WITH_CONTIKI) && !defined(SOL_IP)
742/* Solaris expects level IPPROTO_IP for ancillary data. */
743#define SOL_IP IPPROTO_IP
744#endif
745#ifdef _WIN32
746#define COAP_SOL_IP IPPROTO_IP
747#else /* ! _WIN32 */
748#define COAP_SOL_IP SOL_IP
749#endif /* ! _WIN32 */
750
751#if defined(_WIN32)
752#include <mswsock.h>
753#if defined(__MINGW32__)
754static __thread LPFN_WSARECVMSG lpWSARecvMsg = NULL;
755#else /* ! __MINGW32__ */
756static __declspec(thread) LPFN_WSARECVMSG lpWSARecvMsg = NULL;
757#endif /* ! __MINGW32__ */
758/* Map struct WSABUF fields to their posix counterpart */
759#define msghdr _WSAMSG
760#define msg_name name
761#define msg_namelen namelen
762#define msg_iov lpBuffers
763#define msg_iovlen dwBufferCount
764#define msg_control Control.buf
765#define msg_controllen Control.len
766#define iovec _WSABUF
767#define iov_base buf
768#define iov_len len
769#define iov_len_t u_long
770#undef CMSG_DATA
771#define CMSG_DATA WSA_CMSG_DATA
772#define ipi_spec_dst ipi_addr
773#if !defined(__MINGW32__)
774#pragma warning( disable : 4116 )
775#endif /* ! __MINGW32__ */
776#else
777#define iov_len_t size_t
778#endif
779
780#if defined(_CYGWIN_ENV) || defined(__QNXNTO__)
781#define ipi_spec_dst ipi_addr
782#endif
783
784#if !defined(RIOT_VERSION) && !defined(WITH_LWIP) && !defined(WITH_CONTIKI)
785#if COAP_CLIENT_SUPPORT
786static uint32_t cid_track_counter;
787
788static void
789coap_test_cid_tuple_change(coap_session_t *session) {
790 if (session->type == COAP_SESSION_TYPE_CLIENT &&
791 session->negotiated_cid &&
793 session->proto == COAP_PROTO_DTLS && session->context->testing_cids) {
794 if ((++cid_track_counter) % session->context->testing_cids == 0) {
795 coap_address_t local_if = session->addr_info.local;
796 uint16_t port = coap_address_get_port(&local_if);
797
798 port++;
799 coap_address_set_port(&local_if, port);
800
801 coap_socket_close(&session->sock);
802 session->sock.session = session;
803 if (!coap_socket_connect_udp(&session->sock, &local_if, &session->addr_info.remote,
804 port,
805 &session->addr_info.local,
806 &session->addr_info.remote)) {
807 coap_log_err("Tuple change for CID failed\n");
808 return;
809#ifdef COAP_EPOLL_SUPPORT
810 } else {
811 coap_epoll_ctl_add(&session->sock,
812 EPOLLIN |
813 ((session->sock.flags & COAP_SOCKET_WANT_CONNECT) ?
814 EPOLLOUT : 0),
815 __func__);
816#endif /* COAP_EPOLL_SUPPORT */
817 }
819 }
820 }
821}
822#endif /* COAP_CLIENT_SUPPORT */
823
824/*
825 * dgram
826 * return +ve Number of bytes written.
827 * -1 Error error in errno).
828 */
829ssize_t
831 const uint8_t *data, size_t datalen) {
832 ssize_t bytes_written = 0;
833
834#if COAP_CLIENT_SUPPORT
835 coap_test_cid_tuple_change(session);
836#endif /* COAP_CLIENT_SUPPORT */
837
838 if (!coap_debug_send_packet()) {
839 bytes_written = (ssize_t)datalen;
840 } else if (sock->flags & COAP_SOCKET_CONNECTED) {
841#ifdef _WIN32
842 bytes_written = send(sock->fd, (const char *)data, (int)datalen, 0);
843#else
844 bytes_written = send(sock->fd, data, datalen, 0);
845#endif
846 } else {
847#if defined(_WIN32)
848 DWORD dwNumberOfBytesSent = 0;
849 int r;
850#endif /* _WIN32 && !__MINGW32__ */
851#ifdef HAVE_STRUCT_CMSGHDR
852 /* a buffer large enough to hold all packet info types, ipv6 is the largest */
853 char buf[CMSG_SPACE(sizeof(struct in6_pktinfo))];
854 struct msghdr mhdr;
855 struct iovec iov[1];
856 const void *addr = &session->addr_info.remote.addr;
857
858 assert(session);
859
860 memcpy(&iov[0].iov_base, &data, sizeof(iov[0].iov_base));
861 iov[0].iov_len = (iov_len_t)datalen;
862
863 memset(buf, 0, sizeof(buf));
864
865 memset(&mhdr, 0, sizeof(struct msghdr));
866 memcpy(&mhdr.msg_name, &addr, sizeof(mhdr.msg_name));
867 mhdr.msg_namelen = session->addr_info.remote.addr.sa.sa_family == AF_INET ?
868 (socklen_t)sizeof(struct sockaddr_in) :
869 session->addr_info.remote.size;
870
871 mhdr.msg_iov = iov;
872 mhdr.msg_iovlen = 1;
873
874 if (!coap_address_isany(&session->addr_info.local) &&
875 !coap_is_mcast(&session->addr_info.local)) {
876 switch (session->addr_info.local.addr.sa.sa_family) {
877#if COAP_IPV6_SUPPORT
878 case AF_INET6: {
879 struct cmsghdr *cmsg;
880
881#if COAP_IPV4_SUPPORT
882 if (IN6_IS_ADDR_V4MAPPED(&session->addr_info.local.addr.sin6.sin6_addr)) {
883#if defined(IP_PKTINFO)
884 struct in_pktinfo *pktinfo;
885 mhdr.msg_control = buf;
886 mhdr.msg_controllen = CMSG_SPACE(sizeof(struct in_pktinfo));
887
888 cmsg = CMSG_FIRSTHDR(&mhdr);
889 cmsg->cmsg_level = COAP_SOL_IP;
890 cmsg->cmsg_type = IP_PKTINFO;
891 cmsg->cmsg_len = CMSG_LEN(sizeof(struct in_pktinfo));
892
893 pktinfo = (struct in_pktinfo *)CMSG_DATA(cmsg);
894
895 pktinfo->ipi_ifindex = session->ifindex;
896 memcpy(&pktinfo->ipi_spec_dst,
897 session->addr_info.local.addr.sin6.sin6_addr.s6_addr + 12,
898 sizeof(pktinfo->ipi_spec_dst));
899#elif defined(IP_SENDSRCADDR)
900 mhdr.msg_control = buf;
901 mhdr.msg_controllen = CMSG_SPACE(sizeof(struct in_addr));
902
903 cmsg = CMSG_FIRSTHDR(&mhdr);
904 cmsg->cmsg_level = IPPROTO_IP;
905 cmsg->cmsg_type = IP_SENDSRCADDR;
906 cmsg->cmsg_len = CMSG_LEN(sizeof(struct in_addr));
907
908 memcpy(CMSG_DATA(cmsg),
909 session->addr_info.local.addr.sin6.sin6_addr.s6_addr + 12,
910 sizeof(struct in_addr));
911#endif /* IP_PKTINFO */
912 } else {
913#endif /* COAP_IPV4_SUPPORT */
914 struct in6_pktinfo *pktinfo;
915 mhdr.msg_control = buf;
916 mhdr.msg_controllen = CMSG_SPACE(sizeof(struct in6_pktinfo));
917
918 cmsg = CMSG_FIRSTHDR(&mhdr);
919 cmsg->cmsg_level = IPPROTO_IPV6;
920 cmsg->cmsg_type = IPV6_PKTINFO;
921 cmsg->cmsg_len = CMSG_LEN(sizeof(struct in6_pktinfo));
922
923 pktinfo = (struct in6_pktinfo *)CMSG_DATA(cmsg);
924
925 if (coap_is_mcast(&session->addr_info.remote)) {
926 pktinfo->ipi6_ifindex = session->addr_info.remote.addr.sin6.sin6_scope_id;
927 } else {
928 pktinfo->ipi6_ifindex = session->ifindex;
929 }
930 memcpy(&pktinfo->ipi6_addr,
931 &session->addr_info.local.addr.sin6.sin6_addr,
932 sizeof(pktinfo->ipi6_addr));
933#if COAP_IPV4_SUPPORT
934 }
935#endif /* COAP_IPV4_SUPPORT */
936 break;
937 }
938#endif /* COAP_IPV6_SUPPORT */
939#if COAP_IPV4_SUPPORT
940 case AF_INET: {
941#if defined(IP_PKTINFO)
942 struct cmsghdr *cmsg;
943 struct in_pktinfo *pktinfo;
944
945 mhdr.msg_control = buf;
946 mhdr.msg_controllen = CMSG_SPACE(sizeof(struct in_pktinfo));
947
948 cmsg = CMSG_FIRSTHDR(&mhdr);
949 cmsg->cmsg_level = COAP_SOL_IP;
950 cmsg->cmsg_type = IP_PKTINFO;
951 cmsg->cmsg_len = CMSG_LEN(sizeof(struct in_pktinfo));
952
953 pktinfo = (struct in_pktinfo *)CMSG_DATA(cmsg);
954
955 pktinfo->ipi_ifindex = session->ifindex;
956 memcpy(&pktinfo->ipi_spec_dst,
957 &session->addr_info.local.addr.sin.sin_addr,
958 sizeof(pktinfo->ipi_spec_dst));
959#elif defined(IP_SENDSRCADDR)
960 struct cmsghdr *cmsg;
961 mhdr.msg_control = buf;
962 mhdr.msg_controllen = CMSG_SPACE(sizeof(struct in_addr));
963
964 cmsg = CMSG_FIRSTHDR(&mhdr);
965 cmsg->cmsg_level = IPPROTO_IP;
966 cmsg->cmsg_type = IP_SENDSRCADDR;
967 cmsg->cmsg_len = CMSG_LEN(sizeof(struct in_addr));
968
969 memcpy(CMSG_DATA(cmsg),
970 &session->addr_info.local.addr.sin.sin_addr,
971 sizeof(struct in_addr));
972#endif /* IP_PKTINFO */
973 break;
974 }
975#endif /* COAP_IPV4_SUPPORT */
976#if COAP_AF_UNIX_SUPPORT
977 case AF_UNIX:
978 break;
979#endif /* COAP_AF_UNIX_SUPPORT */
980 default:
981 /* error */
982 coap_log_warn("protocol not supported\n");
983 return -1;
984 }
985 }
986#endif /* HAVE_STRUCT_CMSGHDR */
987
988#if defined(_WIN32)
989 r = WSASendMsg(sock->fd, &mhdr, 0 /*dwFlags*/, &dwNumberOfBytesSent, NULL /*lpOverlapped*/,
990 NULL /*lpCompletionRoutine*/);
991 if (r == 0)
992 bytes_written = (ssize_t)dwNumberOfBytesSent;
993 else {
994 bytes_written = -1;
995 coap_win_error_to_errno();
996 }
997#else /* !_WIN32 || __MINGW32__ */
998#ifdef HAVE_STRUCT_CMSGHDR
999 bytes_written = sendmsg(sock->fd, &mhdr, 0);
1000#else /* ! HAVE_STRUCT_CMSGHDR */
1001 bytes_written = sendto(sock->fd, (const void *)data, datalen, 0,
1002 &session->addr_info.remote.addr.sa,
1003 session->addr_info.remote.size);
1004#endif /* ! HAVE_STRUCT_CMSGHDR */
1005#endif /* !_WIN32 || __MINGW32__ */
1006 }
1007
1008 if (bytes_written < 0)
1009 coap_log_crit("coap_socket_send: %s\n", coap_socket_strerror());
1010
1011 return bytes_written;
1012}
1013#endif /* ! RIOT_VERSION && ! WITH_LWIP && ! WITH_CONTIKI */
1014
1015#define SIN6(A) ((struct sockaddr_in6 *)(A))
1016
1017void
1018coap_packet_get_memmapped(coap_packet_t *packet, unsigned char **address, size_t *length) {
1019 *address = packet->payload;
1020 *length = packet->length;
1021}
1022
1023#if !defined(WITH_LWIP) && !defined(WITH_CONTIKI) && !defined(RIOT_VERSION)
1024/*
1025 * dgram
1026 * return +ve Number of bytes written.
1027 * -1 Error error in errno).
1028 * -2 ICMP error response
1029 */
1030ssize_t
1032 ssize_t len = -1;
1033
1034 assert(sock);
1035 assert(packet);
1036
1037 if ((sock->flags & COAP_SOCKET_CAN_READ) == 0) {
1038 return -1;
1039 } else {
1040 /* clear has-data flag */
1041 sock->flags &= ~COAP_SOCKET_CAN_READ;
1042 }
1043
1044 if (sock->flags & COAP_SOCKET_CONNECTED) {
1045#ifdef _WIN32
1046 len = recv(sock->fd, (char *)packet->payload, COAP_RXBUFFER_SIZE, 0);
1047#else
1048 len = recv(sock->fd, packet->payload, COAP_RXBUFFER_SIZE, 0);
1049#endif
1050 if (len < 0) {
1051#ifdef _WIN32
1052 coap_win_error_to_errno();
1053#endif /* _WIN32 */
1054 if (errno == ECONNREFUSED || errno == EHOSTUNREACH || errno == ECONNRESET) {
1055 /* client-side ICMP destination unreachable, ignore it */
1056 coap_log_warn("** %s: coap_socket_recv: ICMP: %s\n",
1057 sock->session ?
1058 coap_session_str(sock->session) : "",
1060 return -2;
1061 }
1062 if (errno != EAGAIN) {
1063 coap_log_warn("** %s: coap_socket_recv: %s\n",
1064 sock->session ?
1065 coap_session_str(sock->session) : "",
1067 }
1068 goto error;
1069 } else if (len > 0) {
1070 packet->length = (size_t)len;
1071 }
1072 } else {
1073#if defined(_WIN32)
1074 DWORD dwNumberOfBytesRecvd = 0;
1075 int r;
1076#endif /* _WIN32 && !__MINGW32__ */
1077#ifdef HAVE_STRUCT_CMSGHDR
1078 /* a buffer large enough to hold all packet info types, ipv6 is the largest */
1079 char buf[CMSG_SPACE(sizeof(struct in6_pktinfo))];
1080 struct cmsghdr *cmsg;
1081 struct msghdr mhdr;
1082 struct iovec iov[1];
1083
1084#if defined(__MINGW32__)
1085 iov[0].iov_base = (char *) packet->payload;
1086#else
1087 iov[0].iov_base = packet->payload;
1088#endif /* defined(__MINGW32__) */
1089 iov[0].iov_len = (iov_len_t)COAP_RXBUFFER_SIZE;
1090
1091 memset(&mhdr, 0, sizeof(struct msghdr));
1092
1093 mhdr.msg_name = (struct sockaddr *)&packet->addr_info.remote.addr;
1094 mhdr.msg_namelen = sizeof(packet->addr_info.remote.addr);
1095
1096 mhdr.msg_iov = iov;
1097 mhdr.msg_iovlen = 1;
1098
1099 mhdr.msg_control = buf;
1100 mhdr.msg_controllen = sizeof(buf);
1101 /* set a big first length incase recvmsg() does not implement updating
1102 msg_control as well as preset the first cmsg with bad data */
1103 cmsg = (struct cmsghdr *)buf;
1104 cmsg->cmsg_len = CMSG_LEN(sizeof(buf));
1105 cmsg->cmsg_level = -1;
1106 cmsg->cmsg_type = -1;
1107
1108#if defined(_WIN32)
1109 if (!lpWSARecvMsg) {
1110 GUID wsaid = WSAID_WSARECVMSG;
1111 DWORD cbBytesReturned = 0;
1112 if (WSAIoctl(sock->fd, SIO_GET_EXTENSION_FUNCTION_POINTER, &wsaid, sizeof(wsaid), &lpWSARecvMsg,
1113 sizeof(lpWSARecvMsg), &cbBytesReturned, NULL, NULL) != 0) {
1114 coap_log_warn("coap_socket_recv: no WSARecvMsg\n");
1115 return -1;
1116 }
1117 }
1118 r = lpWSARecvMsg(sock->fd, &mhdr, &dwNumberOfBytesRecvd, NULL /* LPWSAOVERLAPPED */,
1119 NULL /* LPWSAOVERLAPPED_COMPLETION_ROUTINE */);
1120 if (r == 0)
1121 len = (ssize_t)dwNumberOfBytesRecvd;
1122 else if (r == COAP_SOCKET_ERROR)
1123 coap_win_error_to_errno();
1124#else
1125 len = recvmsg(sock->fd, &mhdr, 0);
1126#endif
1127
1128#else /* ! HAVE_STRUCT_CMSGHDR */
1129 len = recvfrom(sock->fd, (void *)packet->payload, COAP_RXBUFFER_SIZE, 0,
1130 &packet->addr_info.remote.addr.sa,
1131 &packet->addr_info.remote.size);
1132#endif /* ! HAVE_STRUCT_CMSGHDR */
1133
1134 if (len < 0) {
1135#ifdef _WIN32
1136 coap_win_error_to_errno();
1137#endif /* _WIN32 */
1138 if (errno == ECONNREFUSED || errno == EHOSTUNREACH || errno == ECONNRESET) {
1139 /* server-side ICMP destination unreachable, ignore it. The destination address is in msg_name. */
1140 coap_log_warn("** %s: coap_socket_recv: ICMP: %s\n",
1141 sock->session ?
1142 coap_session_str(sock->session) : "",
1144 return 0;
1145 }
1146 if (errno != EAGAIN) {
1147 coap_log_warn("coap_socket_recv: %s\n", coap_socket_strerror());
1148 }
1149 goto error;
1150 } else {
1151#ifdef HAVE_STRUCT_CMSGHDR
1152 int dst_found = 0;
1153
1154 packet->addr_info.remote.size = mhdr.msg_namelen;
1155 packet->length = (size_t)len;
1156
1157 /* Walk through ancillary data records until the local interface
1158 * is found where the data was received. */
1159 for (cmsg = CMSG_FIRSTHDR(&mhdr); cmsg; cmsg = CMSG_NXTHDR(&mhdr, cmsg)) {
1160
1161#if COAP_IPV6_SUPPORT
1162 /* get the local interface for IPv6 */
1163 if (cmsg->cmsg_level == IPPROTO_IPV6 && cmsg->cmsg_type == IPV6_PKTINFO) {
1164 union {
1165 uint8_t *c;
1166 struct in6_pktinfo *p;
1167 } u;
1168 u.c = CMSG_DATA(cmsg);
1169 packet->ifindex = (int)(u.p->ipi6_ifindex);
1170 memcpy(&packet->addr_info.local.addr.sin6.sin6_addr,
1171 &u.p->ipi6_addr, sizeof(struct in6_addr));
1172 dst_found = 1;
1173 break;
1174 }
1175#endif /* COAP_IPV6_SUPPORT */
1176
1177#if COAP_IPV4_SUPPORT
1178 /* local interface for IPv4 */
1179#if defined(IP_PKTINFO)
1180 if (cmsg->cmsg_level == COAP_SOL_IP && cmsg->cmsg_type == IP_PKTINFO) {
1181 union {
1182 uint8_t *c;
1183 struct in_pktinfo *p;
1184 } u;
1185 u.c = CMSG_DATA(cmsg);
1186 packet->ifindex = u.p->ipi_ifindex;
1187#if COAP_IPV6_SUPPORT
1188 if (packet->addr_info.local.addr.sa.sa_family == AF_INET6) {
1189 memset(packet->addr_info.local.addr.sin6.sin6_addr.s6_addr, 0, 10);
1190 packet->addr_info.local.addr.sin6.sin6_addr.s6_addr[10] = 0xff;
1191 packet->addr_info.local.addr.sin6.sin6_addr.s6_addr[11] = 0xff;
1192 memcpy(packet->addr_info.local.addr.sin6.sin6_addr.s6_addr + 12,
1193 &u.p->ipi_addr, sizeof(struct in_addr));
1194 } else
1195#endif /* COAP_IPV6_SUPPORT */
1196 {
1197 memcpy(&packet->addr_info.local.addr.sin.sin_addr,
1198 &u.p->ipi_addr, sizeof(struct in_addr));
1199 }
1200 dst_found = 1;
1201 break;
1202 }
1203#endif /* IP_PKTINFO */
1204#if defined(IP_RECVDSTADDR)
1205 if (cmsg->cmsg_level == IPPROTO_IP && cmsg->cmsg_type == IP_RECVDSTADDR) {
1206 packet->ifindex = (int)sock->fd;
1207 memcpy(&packet->addr_info.local.addr.sin.sin_addr,
1208 CMSG_DATA(cmsg), sizeof(struct in_addr));
1209 dst_found = 1;
1210 break;
1211 }
1212#endif /* IP_RECVDSTADDR */
1213#endif /* COAP_IPV4_SUPPORT */
1214 if (!dst_found) {
1215 /* cmsg_level / cmsg_type combination we do not understand
1216 (ignore preset case for bad recvmsg() not updating cmsg) */
1217 if (cmsg->cmsg_level != -1 && cmsg->cmsg_type != -1) {
1218 coap_log_debug("cmsg_level = %d and cmsg_type = %d not supported - fix\n",
1219 cmsg->cmsg_level, cmsg->cmsg_type);
1220 }
1221 }
1222 }
1223 if (!dst_found) {
1224 /* Not expected, but cmsg_level and cmsg_type don't match above and
1225 may need a new case */
1226 packet->ifindex = (int)sock->fd;
1227 if (getsockname(sock->fd, &packet->addr_info.local.addr.sa,
1228 &packet->addr_info.local.size) < 0) {
1229 coap_log_debug("Cannot determine local port\n");
1230 }
1231 }
1232#else /* ! HAVE_STRUCT_CMSGHDR */
1233 packet->length = (size_t)len;
1234 packet->ifindex = 0;
1235 if (getsockname(sock->fd, &packet->addr_info.local.addr.sa,
1236 &packet->addr_info.local.size) < 0) {
1237 coap_log_debug("Cannot determine local port\n");
1238 goto error;
1239 }
1240#endif /* ! HAVE_STRUCT_CMSGHDR */
1241 }
1242 }
1243
1244 if (len >= 0)
1245 return len;
1246error:
1247 return -1;
1248}
1249#endif /* ! WITH_LWIP && ! WITH_CONTIKI && ! RIOT_VERSION */
1250
1251COAP_API unsigned int
1253 unsigned int ret;
1254
1255 coap_lock_lock(ctx, return 0);
1256 ret = coap_io_prepare_epoll_lkd(ctx, now);
1257 coap_lock_unlock(ctx);
1258 return ret;
1259}
1260
1261unsigned int
1263#ifndef COAP_EPOLL_SUPPORT
1264 (void)ctx;
1265 (void)now;
1266 coap_log_emerg("coap_io_prepare_epoll() requires libcoap compiled for using epoll\n");
1267 return 0;
1268#else /* COAP_EPOLL_SUPPORT */
1269 coap_socket_t *sockets[1];
1270 unsigned int max_sockets = sizeof(sockets)/sizeof(sockets[0]);
1271 unsigned int num_sockets;
1272 unsigned int timeout;
1273
1275 /* Use the common logic */
1276 timeout = coap_io_prepare_io_lkd(ctx, sockets, max_sockets, &num_sockets, now);
1277 /* Save when the next expected I/O is to take place */
1278 ctx->next_timeout = timeout ? now + timeout : 0;
1279 if (ctx->eptimerfd != -1) {
1280 struct itimerspec new_value;
1281 int ret;
1282
1283 memset(&new_value, 0, sizeof(new_value));
1284 coap_ticks(&now);
1285 if (ctx->next_timeout != 0 && ctx->next_timeout > now) {
1286 coap_tick_t rem_timeout = ctx->next_timeout - now;
1287 /* Need to trigger an event on ctx->eptimerfd in the future */
1288 new_value.it_value.tv_sec = rem_timeout / COAP_TICKS_PER_SECOND;
1289 new_value.it_value.tv_nsec = (rem_timeout % COAP_TICKS_PER_SECOND) *
1290 1000000;
1291 }
1292#ifdef COAP_DEBUG_WAKEUP_TIMES
1293 coap_log_debug("****** Next wakeup time %3ld.%09ld\n",
1294 new_value.it_value.tv_sec, new_value.it_value.tv_nsec);
1295#endif /* COAP_DEBUG_WAKEUP_TIMES */
1296 /* reset, or specify a future time for eptimerfd to trigger */
1297 ret = timerfd_settime(ctx->eptimerfd, 0, &new_value, NULL);
1298 if (ret == -1) {
1299 coap_log_err("%s: timerfd_settime failed: %s (%d)\n",
1300 "coap_io_prepare_epoll",
1301 coap_socket_strerror(), errno);
1302 }
1303 }
1304 return timeout;
1305#endif /* COAP_EPOLL_SUPPORT */
1306}
1307
1308/*
1309 * return 0 No i/o pending
1310 * +ve millisecs to next i/o activity
1311 */
1312COAP_API unsigned int
1314 coap_socket_t *sockets[],
1315 unsigned int max_sockets,
1316 unsigned int *num_sockets,
1317 coap_tick_t now) {
1318 unsigned int ret;
1319
1320 coap_lock_lock(ctx, return 0);
1321 ret = coap_io_prepare_io_lkd(ctx, sockets, max_sockets, num_sockets, now);
1322 coap_lock_unlock(ctx);
1323 return ret;
1324}
1325
1326/*
1327 * return 0 No i/o pending
1328 * +ve millisecs to next i/o activity
1329 */
1330unsigned int
1332 coap_socket_t *sockets[],
1333 unsigned int max_sockets,
1334 unsigned int *num_sockets,
1335 coap_tick_t now) {
1336 coap_queue_t *nextpdu;
1337 coap_session_t *s, *stmp;
1338 coap_tick_t timeout = 0;
1339 coap_tick_t s_timeout;
1340#if COAP_SERVER_SUPPORT
1341 int check_dtls_timeouts = 0;
1342#endif /* COAP_SERVER_SUPPORT */
1343#if defined(COAP_EPOLL_SUPPORT) || defined(WITH_LWIP) || defined(RIOT_VERSION)
1344 (void)sockets;
1345 (void)max_sockets;
1346#endif /* COAP_EPOLL_SUPPORT || WITH_LWIP || RIOT_VERSION*/
1347
1349 *num_sockets = 0;
1350
1351#if COAP_SERVER_SUPPORT
1352 /* Check to see if we need to send off any Observe requests */
1354
1355#if COAP_ASYNC_SUPPORT
1356 /* Check to see if we need to send off any Async requests */
1357 timeout = coap_check_async(ctx, now);
1358#endif /* COAP_ASYNC_SUPPORT */
1359#endif /* COAP_SERVER_SUPPORT */
1360
1361 /* Check to see if we need to send off any retransmit request */
1362 nextpdu = coap_peek_next(ctx);
1363 while (nextpdu && now >= ctx->sendqueue_basetime &&
1364 nextpdu->t <= now - ctx->sendqueue_basetime) {
1365 coap_retransmit(ctx, coap_pop_next(ctx));
1366 nextpdu = coap_peek_next(ctx);
1367 }
1368 if (nextpdu && (timeout == 0 ||
1369 nextpdu->t - (now - ctx->sendqueue_basetime) < timeout))
1370 timeout = nextpdu->t - (now - ctx->sendqueue_basetime);
1371
1372 /* Check for DTLS timeouts */
1373 if (ctx->dtls_context) {
1376 if (tls_timeout > 0) {
1377 if (tls_timeout < now + COAP_TICKS_PER_SECOND / 10)
1378 tls_timeout = now + COAP_TICKS_PER_SECOND / 10;
1379 coap_log_debug("** DTLS global timeout set to %dms\n",
1380 (int)((tls_timeout - now) * 1000 / COAP_TICKS_PER_SECOND));
1381 if (timeout == 0 || tls_timeout - now < timeout)
1382 timeout = tls_timeout - now;
1383 }
1384#if COAP_SERVER_SUPPORT
1385 } else {
1386 check_dtls_timeouts = 1;
1387#endif /* COAP_SERVER_SUPPORT */
1388 }
1389 }
1390#if COAP_PROXY_SUPPORT
1391 if (coap_proxy_check_timeouts(ctx, now, &s_timeout)) {
1392 if (timeout == 0 || s_timeout < timeout)
1393 timeout = s_timeout;
1394 }
1395#endif /* COAP_PROXY_SUPPORT */
1396#if COAP_SERVER_SUPPORT
1397 coap_endpoint_t *ep;
1398 coap_tick_t session_timeout;
1399
1400 if (ctx->session_timeout > 0)
1401 session_timeout = ctx->session_timeout * COAP_TICKS_PER_SECOND;
1402 else
1404
1405 LL_FOREACH(ctx->endpoint, ep) {
1406#if !defined(COAP_EPOLL_SUPPORT) && !defined(WITH_LWIP) && !defined(RIOT_VERSION)
1408 if (*num_sockets < max_sockets)
1409 sockets[(*num_sockets)++] = &ep->sock;
1410 }
1411#endif /* ! COAP_EPOLL_SUPPORT && ! WITH_LWIP && ! RIOT_VERSION */
1412 SESSIONS_ITER_SAFE(ep->sessions, s, stmp) {
1413 /* Check whether any idle server sessions should be released */
1414 if (s->type == COAP_SESSION_TYPE_SERVER && s->ref == 0 &&
1415 s->delayqueue == NULL &&
1416 (s->last_rx_tx + session_timeout <= now ||
1420 continue;
1421 } else {
1422 if (s->type == COAP_SESSION_TYPE_SERVER && s->ref == 0 &&
1423 s->delayqueue == NULL) {
1424 s_timeout = (s->last_rx_tx + session_timeout) - now;
1425 if (timeout == 0 || s_timeout < timeout)
1426 timeout = s_timeout;
1427 }
1428 /* Make sure the session object is not deleted in any callbacks */
1430 /* Check any DTLS timeouts and expire if appropriate */
1431 if (check_dtls_timeouts && s->state == COAP_SESSION_STATE_HANDSHAKE &&
1432 s->proto == COAP_PROTO_DTLS && s->tls) {
1433 coap_tick_t tls_timeout = coap_dtls_get_timeout(s, now);
1434 while (tls_timeout > 0 && tls_timeout <= now) {
1435 coap_log_debug("** %s: DTLS retransmit timeout\n",
1436 coap_session_str(s));
1438 goto release_1;
1439
1440 if (s->tls)
1441 tls_timeout = coap_dtls_get_timeout(s, now);
1442 else {
1443 tls_timeout = 0;
1444 timeout = 1;
1445 }
1446 }
1447 if (tls_timeout > 0 && (timeout == 0 || tls_timeout - now < timeout))
1448 timeout = tls_timeout - now;
1449 }
1450 /* Check if any server large receives are missing blocks */
1451 if (s->lg_srcv) {
1452 if (coap_block_check_lg_srcv_timeouts(s, now, &s_timeout)) {
1453 if (timeout == 0 || s_timeout < timeout)
1454 timeout = s_timeout;
1455 }
1456 }
1457 /* Check if any server large sending have timed out */
1458 if (s->lg_xmit) {
1459 if (coap_block_check_lg_xmit_timeouts(s, now, &s_timeout)) {
1460 if (timeout == 0 || s_timeout < timeout)
1461 timeout = s_timeout;
1462 }
1463 }
1464#if !defined(COAP_EPOLL_SUPPORT) && !defined(WITH_LWIP) && !defined(RIOT_VERSION)
1466 if (*num_sockets < max_sockets)
1467 sockets[(*num_sockets)++] = &s->sock;
1468 }
1469#endif /* ! COAP_EPOLL_SUPPORT && ! WITH_LWIP && ! RIOT_VERSION */
1470#if COAP_Q_BLOCK_SUPPORT
1471 /*
1472 * Check if any server large transmits have hit MAX_PAYLOAD and need
1473 * restarting
1474 */
1475 if (s->lg_xmit) {
1476 s_timeout = coap_block_check_q_block2_xmit(s, now);
1477 if (timeout == 0 || s_timeout < timeout)
1478 timeout = s_timeout;
1479 }
1480#endif /* COAP_Q_BLOCK_SUPPORT */
1481release_1:
1483 }
1484 if (s->type == COAP_SESSION_TYPE_SERVER &&
1486 s->ref_subscriptions &&
1487 ctx->ping_timeout > 0) {
1488 /* Only do this if this session is observing */
1489 if (s->last_rx_tx + ctx->ping_timeout * COAP_TICKS_PER_SECOND <= now) {
1490 /* Time to send a ping */
1492 /* Some issue - not safe to continue processing */
1493 s->last_rx_tx = now;
1494 continue;
1495 }
1496 if (s->last_ping > 0 && s->last_pong < s->last_ping) {
1498 /* check the next session */
1499 continue;
1500 }
1501 s->last_rx_tx = now;
1502 s->last_ping = now;
1503 }
1504 s_timeout = (s->last_rx_tx + ctx->ping_timeout * COAP_TICKS_PER_SECOND) - now;
1505 if (timeout == 0 || s_timeout < timeout)
1506 timeout = s_timeout;
1507 }
1508 }
1509 }
1510#endif /* COAP_SERVER_SUPPORT */
1511#if COAP_CLIENT_SUPPORT
1512 SESSIONS_ITER_SAFE(ctx->sessions, s, stmp) {
1513 if (s->type == COAP_SESSION_TYPE_CLIENT &&
1515 ctx->ping_timeout > 0) {
1516 if (s->last_rx_tx + ctx->ping_timeout * COAP_TICKS_PER_SECOND <= now) {
1517 /* Time to send a ping */
1519 /* Some issue - not safe to continue processing */
1520 s->last_rx_tx = now;
1521 continue;
1522 }
1523 if (s->last_ping > 0 && s->last_pong < s->last_ping) {
1525 }
1526 s->last_rx_tx = now;
1527 s->last_ping = now;
1528 }
1529 s_timeout = (s->last_rx_tx + ctx->ping_timeout * COAP_TICKS_PER_SECOND) - now;
1530 if (timeout == 0 || s_timeout < timeout)
1531 timeout = s_timeout;
1532 }
1533
1534#if !COAP_DISABLE_TCP
1536 s->state == COAP_SESSION_STATE_CSM && ctx->csm_timeout_ms > 0) {
1537 if (s->csm_tx == 0) {
1538 s->csm_tx = now;
1539 s_timeout = (ctx->csm_timeout_ms * COAP_TICKS_PER_SECOND) / 1000;
1540 } else if (s->csm_tx + (ctx->csm_timeout_ms * COAP_TICKS_PER_SECOND) / 1000 <= now) {
1541 /* timed out */
1542 s_timeout = 0;
1543 } else {
1544 s_timeout = (s->csm_tx + (ctx->csm_timeout_ms * COAP_TICKS_PER_SECOND) / 1000) - now;
1545 }
1546 if ((timeout == 0 || s_timeout < timeout) && s_timeout != 0)
1547 timeout = s_timeout;
1548 }
1549#endif /* !COAP_DISABLE_TCP */
1550
1551 /* Make sure the session object is not deleted in any callbacks */
1553 /* Check any DTLS timeouts and expire if appropriate */
1555 s->proto == COAP_PROTO_DTLS && s->tls) {
1556 coap_tick_t tls_timeout = coap_dtls_get_timeout(s, now);
1557 while (tls_timeout > 0 && tls_timeout <= now) {
1558 coap_log_debug("** %s: DTLS retransmit timeout\n", coap_session_str(s));
1560 goto release_2;
1561
1562 if (s->tls)
1563 tls_timeout = coap_dtls_get_timeout(s, now);
1564 else {
1565 tls_timeout = 0;
1566 timeout = 1;
1567 }
1568 }
1569 if (tls_timeout > 0 && (timeout == 0 || tls_timeout - now < timeout))
1570 timeout = tls_timeout - now;
1571 }
1572
1573 /* Check if any client large receives are missing blocks */
1574 if (s->lg_crcv) {
1575 if (coap_block_check_lg_crcv_timeouts(s, now, &s_timeout)) {
1576 if (timeout == 0 || s_timeout < timeout)
1577 timeout = s_timeout;
1578 }
1579 }
1580 /* Check if any client large sending have timed out */
1581 if (s->lg_xmit) {
1582 if (coap_block_check_lg_xmit_timeouts(s, now, &s_timeout)) {
1583 if (timeout == 0 || s_timeout < timeout)
1584 timeout = s_timeout;
1585 }
1586 }
1587#if COAP_Q_BLOCK_SUPPORT
1588 /*
1589 * Check if any client large transmits have hit MAX_PAYLOAD and need
1590 * restarting
1591 */
1592 if (s->lg_xmit) {
1593 s_timeout = coap_block_check_q_block1_xmit(s, now);
1594 if (timeout == 0 || s_timeout < timeout)
1595 timeout = s_timeout;
1596 }
1597#endif /* COAP_Q_BLOCK_SUPPORT */
1598
1599#if !defined(COAP_EPOLL_SUPPORT) && !defined(WITH_LWIP) && !defined(RIOT_VERSION)
1600 assert(s->ref > 1);
1601 if (s->sock.flags & (COAP_SOCKET_WANT_READ |
1604 if (*num_sockets < max_sockets)
1605 sockets[(*num_sockets)++] = &s->sock;
1606 }
1607#endif /* ! COAP_EPOLL_SUPPORT && ! WITH_LWIP && ! RIOT_VERSION */
1608release_2:
1610 }
1611#endif /* COAP_CLIENT_SUPPORT */
1612
1613 return (unsigned int)((timeout * 1000 + COAP_TICKS_PER_SECOND - 1) / COAP_TICKS_PER_SECOND);
1614}
1615
1616/*
1617 * return 0 Insufficient space to hold fds, or fds not supported
1618 * 1 All fds found
1619 */
1620COAP_API unsigned int
1622 coap_fd_t read_fds[],
1623 unsigned int *have_read_fds,
1624 unsigned int max_read_fds,
1625 coap_fd_t write_fds[],
1626 unsigned int *have_write_fds,
1627 unsigned int max_write_fds,
1628 unsigned int *rem_timeout_ms) {
1629 unsigned int ret;
1630
1631 coap_lock_lock(ctx, return 0);
1632 ret = coap_io_get_fds_lkd(ctx, read_fds, have_read_fds, max_read_fds, write_fds,
1633 have_write_fds, max_write_fds, rem_timeout_ms);
1634 coap_lock_unlock(ctx);
1635 return ret;
1636}
1637
1638#if !defined(WITH_LWIP) && !defined(WITH_CONTIKI)
1639static int
1640coap_add_fd(coap_fd_t fd, coap_fd_t this_fds[], unsigned int *have_this_fds,
1641 unsigned int max_this_fds) {
1642 if (*have_this_fds < max_this_fds) {
1643 this_fds[(*have_this_fds)++] = fd;
1644 return 1;
1645 }
1646 coap_log_warn("coap_io_get_fds: Insufficient space for new fd (%u >= %u)\n", *have_this_fds,
1647 max_this_fds);
1648 return 0;
1649}
1650
1651/*
1652 * return 0 Insufficient space to hold fds, or fds not supported
1653 * 1 All fds found
1654 */
1655unsigned int
1657 coap_fd_t read_fds[],
1658 unsigned int *have_read_fds,
1659 unsigned int max_read_fds,
1660 coap_fd_t write_fds[],
1661 unsigned int *have_write_fds,
1662 unsigned int max_write_fds,
1663 unsigned int *rem_timeout_ms) {
1664 *have_read_fds = 0;
1665 *have_write_fds = 0;
1666
1667#ifdef COAP_EPOLL_SUPPORT
1668 (void)write_fds;
1669 (void)max_write_fds;;
1670
1671 if (!coap_add_fd(ctx->epfd, read_fds, have_read_fds, max_read_fds))
1672 return 0;
1673 /* epoll is making use of timerfd, so no need to return any timeout */
1674 *rem_timeout_ms = 0;
1675 return 1;
1676#else /* ! COAP_EPOLL_SUPPORT */
1677 coap_session_t *s, *rtmp;
1678 coap_tick_t now;
1679 unsigned int timeout_ms;
1680#if COAP_SERVER_SUPPORT
1681 coap_endpoint_t *ep;
1682
1683 LL_FOREACH(ctx->endpoint, ep) {
1685 if (!coap_add_fd(ep->sock.fd, read_fds, have_read_fds, max_read_fds))
1686 return 0;
1687 }
1689 if (!coap_add_fd(ep->sock.fd, write_fds, have_write_fds, max_write_fds))
1690 return 0;
1691 }
1692 SESSIONS_ITER_SAFE(ep->sessions, s, rtmp) {
1694 if (!coap_add_fd(s->sock.fd, read_fds, have_read_fds, max_read_fds))
1695 return 0;
1696 }
1698 if (!coap_add_fd(s->sock.fd, write_fds, have_write_fds, max_write_fds))
1699 return 0;
1700 }
1701 }
1702 }
1703#endif /* COAP_SERVER_SUPPORT */
1704
1705#if COAP_CLIENT_SUPPORT
1706 SESSIONS_ITER_SAFE(ctx->sessions, s, rtmp) {
1708 if (!coap_add_fd(s->sock.fd, read_fds, have_read_fds, max_read_fds))
1709 return 0;
1710 }
1712 if (!coap_add_fd(s->sock.fd, write_fds, have_write_fds, max_write_fds))
1713 return 0;
1714 }
1715 }
1716#endif /* COAP_CLIENT_SUPPORT */
1717
1718 coap_ticks(&now);
1719 timeout_ms = (unsigned int)(ctx->next_timeout ? ctx->next_timeout > now ?
1720 ctx->next_timeout - now : 0 : 0) *
1721 1000 / COAP_TICKS_PER_SECOND;
1722 *rem_timeout_ms = timeout_ms;
1723 return 1;
1724#endif /* ! COAP_EPOLL_SUPPORT */
1725}
1726
1727#else /* WITH_LWIP || WITH_CONTIKI */
1728
1729/*
1730 * return 0 Insufficient space to hold fds, or fds not supported
1731 * 1 All fds found
1732 */
1733unsigned int
1735 coap_fd_t read_fds[],
1736 unsigned int *have_read_fds,
1737 unsigned int max_read_fds,
1738 coap_fd_t write_fds[],
1739 unsigned int *have_write_fds,
1740 unsigned int max_write_fds,
1741 unsigned int *rem_timeout_ms) {
1742 (void)ctx;
1743 (void)read_fds;
1744 (void)max_read_fds;
1745 (void)write_fds;
1746 (void)max_write_fds;
1747
1748 *have_read_fds = 0;
1749 *have_write_fds = 0;
1750 *rem_timeout_ms = 0;
1751
1752 coap_log_warn("coap_io_get_fds: Not supported\n");
1753 return 0;
1754}
1755#endif /* WITH_LWIP || WITH_CONTIKI */
1756
1757#if !defined(WITH_LWIP) && !defined(CONTIKI) && !defined(RIOT_VERSION)
1758COAP_API int
1759coap_io_process(coap_context_t *ctx, uint32_t timeout_ms) {
1760 int ret;
1761
1762 coap_lock_lock(ctx, return 0);
1763 ret = coap_io_process_lkd(ctx, timeout_ms);
1764 coap_lock_unlock(ctx);
1765 return ret;
1766}
1767
1768int
1769coap_io_process_lkd(coap_context_t *ctx, uint32_t timeout_ms) {
1770 return coap_io_process_with_fds_lkd(ctx, timeout_ms, 0, NULL, NULL, NULL);
1771}
1772
1773COAP_API int
1775 int enfds, fd_set *ereadfds, fd_set *ewritefds,
1776 fd_set *eexceptfds) {
1777 int ret;
1778
1779 coap_lock_lock(ctx, return 0);
1780 ret = coap_io_process_with_fds_lkd(ctx, timeout_ms, enfds, ereadfds, ewritefds,
1781 eexceptfds);
1782 coap_lock_unlock(ctx);
1783 return ret;
1784}
1785
1786#if !defined(COAP_EPOLL_SUPPORT) && COAP_THREAD_SAFE
1787static unsigned int
1788coap_io_prepare_fds(coap_context_t *ctx,
1789 int enfds, fd_set *ereadfds, fd_set *ewritefds,
1790 fd_set *eexceptfds) {
1791 coap_session_t *s, *stmp;
1792 unsigned int max_sockets = sizeof(ctx->sockets) / sizeof(ctx->sockets[0]);
1793 coap_fd_t nfds = 0;
1794 unsigned int i;
1795
1796 ctx->num_sockets = 0;
1797#if COAP_SERVER_SUPPORT
1798 coap_endpoint_t *ep;
1799
1800 LL_FOREACH(ctx->endpoint, ep) {
1802 if (ctx->num_sockets < max_sockets)
1803 ctx->sockets[ctx->num_sockets++] = &ep->sock;
1804 }
1805 SESSIONS_ITER(ep->sessions, s, stmp) {
1807 if (ctx->num_sockets < max_sockets)
1808 ctx->sockets[ctx->num_sockets++] = &s->sock;
1809 }
1810 }
1811 }
1812#endif /* COAP_SERVER_SUPPORT */
1813#if COAP_CLIENT_SUPPORT
1814 SESSIONS_ITER(ctx->sessions, s, stmp) {
1815 if (s->sock.flags & (COAP_SOCKET_WANT_READ |
1818 if (ctx->num_sockets < max_sockets)
1819 ctx->sockets[ctx->num_sockets++] = &s->sock;
1820 }
1821 }
1822#endif /* COAP_CLIENT_SUPPORT */
1823 if (ereadfds) {
1824 ctx->readfds = *ereadfds;
1825 nfds = enfds;
1826 } else {
1827 FD_ZERO(&ctx->readfds);
1828 }
1829 if (ewritefds) {
1830 ctx->writefds = *ewritefds;
1831 nfds = enfds;
1832 } else {
1833 FD_ZERO(&ctx->writefds);
1834 }
1835 if (eexceptfds) {
1836 ctx->exceptfds = *eexceptfds;
1837 nfds = enfds;
1838 } else {
1839 FD_ZERO(&ctx->exceptfds);
1840 }
1841 for (i = 0; i < ctx->num_sockets; i++) {
1842 if (ctx->sockets[i]->fd + 1 > nfds)
1843 nfds = ctx->sockets[i]->fd + 1;
1844 if (ctx->sockets[i]->flags & COAP_SOCKET_WANT_READ)
1845 FD_SET(ctx->sockets[i]->fd, &ctx->readfds);
1846 if (ctx->sockets[i]->flags & COAP_SOCKET_WANT_WRITE)
1847 FD_SET(ctx->sockets[i]->fd, &ctx->writefds);
1848#if !COAP_DISABLE_TCP
1849 if (ctx->sockets[i]->flags & COAP_SOCKET_WANT_ACCEPT)
1850 FD_SET(ctx->sockets[i]->fd, &ctx->readfds);
1851 if (ctx->sockets[i]->flags & COAP_SOCKET_WANT_CONNECT) {
1852 FD_SET(ctx->sockets[i]->fd, &ctx->writefds);
1853 FD_SET(ctx->sockets[i]->fd, &ctx->exceptfds);
1854 }
1855#endif /* !COAP_DISABLE_TCP */
1856 }
1857 return nfds;
1858}
1859#endif /* ! COAP_EPOLL_SUPPORT && COAP_THREAD_SAFE */
1860
1861int
1863 int enfds, fd_set *ereadfds, fd_set *ewritefds,
1864 fd_set *eexceptfds) {
1865 coap_fd_t nfds = 0;
1866 coap_tick_t before, now;
1867 unsigned int timeout;
1868#ifndef COAP_EPOLL_SUPPORT
1869 struct timeval tv;
1870 int result;
1871 unsigned int i;
1872#endif /* ! COAP_EPOLL_SUPPORT */
1873
1875 coap_ticks(&before);
1876
1877#ifndef COAP_EPOLL_SUPPORT
1878
1879 timeout = coap_io_prepare_io_lkd(ctx, ctx->sockets,
1880 (sizeof(ctx->sockets) / sizeof(ctx->sockets[0])),
1881 &ctx->num_sockets, before);
1882 ctx->next_timeout = timeout ? timeout + before : 0;
1883
1884 if (ereadfds) {
1885 ctx->readfds = *ereadfds;
1886 nfds = enfds;
1887 } else {
1888 FD_ZERO(&ctx->readfds);
1889 }
1890 if (ewritefds) {
1891 ctx->writefds = *ewritefds;
1892 nfds = enfds;
1893 } else {
1894 FD_ZERO(&ctx->writefds);
1895 }
1896 if (eexceptfds) {
1897 ctx->exceptfds = *eexceptfds;
1898 nfds = enfds;
1899 } else {
1900 FD_ZERO(&ctx->exceptfds);
1901 }
1902 for (i = 0; i < ctx->num_sockets; i++) {
1903 if (ctx->sockets[i]->fd + 1 > nfds)
1904 nfds = ctx->sockets[i]->fd + 1;
1905 if (ctx->sockets[i]->flags & COAP_SOCKET_WANT_READ)
1906 FD_SET(ctx->sockets[i]->fd, &ctx->readfds);
1907 if (ctx->sockets[i]->flags & COAP_SOCKET_WANT_WRITE)
1908 FD_SET(ctx->sockets[i]->fd, &ctx->writefds);
1909#if !COAP_DISABLE_TCP
1910 if (ctx->sockets[i]->flags & COAP_SOCKET_WANT_ACCEPT)
1911 FD_SET(ctx->sockets[i]->fd, &ctx->readfds);
1912 if (ctx->sockets[i]->flags & COAP_SOCKET_WANT_CONNECT) {
1913 FD_SET(ctx->sockets[i]->fd, &ctx->writefds);
1914 FD_SET(ctx->sockets[i]->fd, &ctx->exceptfds);
1915 }
1916#endif /* !COAP_DISABLE_TCP */
1917 }
1918
1919 if (timeout_ms == COAP_IO_NO_WAIT) {
1920 tv.tv_usec = 0;
1921 tv.tv_sec = 0;
1922 timeout = 1;
1923 } else if (timeout == 0 && timeout_ms == COAP_IO_WAIT) {
1924 ;
1925 } else {
1926 if (timeout == 0 || (timeout_ms != COAP_IO_WAIT && timeout_ms < timeout))
1927 timeout = timeout_ms;
1928 tv.tv_usec = (timeout % 1000) * 1000;
1929 tv.tv_sec = (long)(timeout / 1000);
1930 }
1931
1932 /* on Windows select will return an error if called without FDs */
1933 if (nfds > 0) {
1934 /* Unlock so that other threads can lock/update ctx */
1935 coap_lock_unlock(ctx);
1936
1937 result = select((int)nfds, &ctx->readfds, &ctx->writefds, &ctx->exceptfds,
1938 timeout > 0 ? &tv : NULL);
1939
1940 coap_lock_lock(ctx, return -1);
1941 } else {
1942 goto all_over;
1943 }
1944
1945 if (result < 0) { /* error */
1946#ifdef _WIN32
1947 coap_win_error_to_errno();
1948#endif
1949 if (errno != EINTR) {
1950#if COAP_THREAD_SAFE
1951 if (errno == EBADF) {
1952 coap_log_debug("select: %s\n", coap_socket_strerror());
1953 goto all_over;
1954 }
1955#endif /* COAP_THREAD_SAFE */
1956 coap_log_err("select: %s\n", coap_socket_strerror());
1957 return -1;
1958 }
1959 goto all_over;
1960 }
1961#if COAP_THREAD_SAFE
1962 /* Need to refresh what is available to read / write etc. */
1963 nfds = coap_io_prepare_fds(ctx, enfds, ereadfds, ewritefds, eexceptfds);
1964 tv.tv_usec = 0;
1965 tv.tv_sec = 0;
1966 result = select((int)nfds, &ctx->readfds, &ctx->writefds, &ctx->exceptfds, &tv);
1967 if (result < 0) { /* error */
1968#ifdef _WIN32
1969 coap_win_error_to_errno();
1970#endif
1971 if (errno != EINTR) {
1972 if (errno == EBADF) {
1973 coap_log_debug("select: %s\n", coap_socket_strerror());
1974 goto all_over;
1975 }
1976 coap_log_err("select: %s\n", coap_socket_strerror());
1977 return -1;
1978 }
1979 goto all_over;
1980 }
1981#endif /* COAP_THREAD_SAFE */
1982 if (ereadfds) {
1983 *ereadfds = ctx->readfds;
1984 }
1985 if (ewritefds) {
1986 *ewritefds = ctx->writefds;
1987 }
1988 if (eexceptfds) {
1989 *eexceptfds = ctx->exceptfds;
1990 }
1991
1992 if (result > 0) {
1993 for (i = 0; i < ctx->num_sockets; i++) {
1994 if ((ctx->sockets[i]->flags & COAP_SOCKET_WANT_READ) &&
1995 FD_ISSET(ctx->sockets[i]->fd, &ctx->readfds))
1997#if !COAP_DISABLE_TCP
1998 if ((ctx->sockets[i]->flags & COAP_SOCKET_WANT_ACCEPT) &&
1999 FD_ISSET(ctx->sockets[i]->fd, &ctx->readfds))
2001 if ((ctx->sockets[i]->flags & COAP_SOCKET_WANT_WRITE) &&
2002 FD_ISSET(ctx->sockets[i]->fd, &ctx->writefds))
2004 if ((ctx->sockets[i]->flags & COAP_SOCKET_WANT_CONNECT) &&
2005 (FD_ISSET(ctx->sockets[i]->fd, &ctx->writefds) ||
2006 FD_ISSET(ctx->sockets[i]->fd, &ctx->exceptfds)))
2008#endif /* !COAP_DISABLE_TCP */
2009 }
2010 }
2011
2012 coap_ticks(&now);
2013 coap_io_do_io_lkd(ctx, now);
2014 coap_ticks(&now);
2015 timeout = coap_io_prepare_io_lkd(ctx, ctx->sockets,
2016 (sizeof(ctx->sockets) / sizeof(ctx->sockets[0])),
2017 &ctx->num_sockets, now);
2018 ctx->next_timeout = timeout ? timeout + now : 0;
2019
2020#else /* COAP_EPOLL_SUPPORT */
2021 (void)ereadfds;
2022 (void)ewritefds;
2023 (void)eexceptfds;
2024 (void)enfds;
2025
2026 timeout = coap_io_prepare_epoll_lkd(ctx, before);
2027
2028 do {
2029 struct epoll_event events[COAP_MAX_EPOLL_EVENTS];
2030 int etimeout;
2031
2032 /* Potentially adjust based on what the caller wants */
2033 if (timeout_ms == COAP_IO_NO_WAIT) {
2034 /* Need to return immediately from epoll_wait() */
2035 etimeout = 0;
2036 } else if (timeout == 0 && timeout_ms == COAP_IO_WAIT) {
2037 /*
2038 * Nothing found in coap_io_prepare_epoll_lkd() and COAP_IO_WAIT set,
2039 * so wait forever in epoll_wait().
2040 */
2041 etimeout = -1;
2042 } else {
2043 etimeout = timeout;
2044 if (timeout == 0 || (timeout_ms != COAP_IO_WAIT && timeout_ms < timeout))
2045 etimeout = timeout_ms;
2046 if (etimeout < 0) {
2047 /*
2048 * If timeout > INT_MAX, epoll_wait() cannot wait longer than this as
2049 * it has int timeout parameter
2050 */
2051 etimeout = INT_MAX;
2052 }
2053 }
2054
2055 /* Unlock so that other threads can lock/update ctx */
2056 coap_lock_unlock(ctx);
2057
2058 nfds = epoll_wait(ctx->epfd, events, COAP_MAX_EPOLL_EVENTS, etimeout);
2059 if (nfds < 0) {
2060 if (errno != EINTR) {
2061 coap_log_err("epoll_wait: unexpected error: %s (%d)\n",
2062 coap_socket_strerror(), nfds);
2063 }
2064 coap_lock_lock(ctx, return -1);
2065 break;
2066 }
2067
2068 coap_lock_lock(ctx, return -1);
2069#if COAP_THREAD_SAFE
2070 /* Need to refresh what is available to read / write etc. */
2071 nfds = epoll_wait(ctx->epfd, events, COAP_MAX_EPOLL_EVENTS, 0);
2072 if (nfds < 0) {
2073 if (errno != EINTR) {
2074 coap_log_err("epoll_wait: unexpected error: %s (%d)\n",
2075 coap_socket_strerror(), nfds);
2076 }
2077 break;
2078 }
2079#endif /* COAP_THREAD_SAFE */
2080
2081 coap_io_do_epoll_lkd(ctx, events, nfds);
2082
2083 /*
2084 * reset to COAP_IO_NO_WAIT (which causes etimeout to become 0)
2085 * incase we have to do another iteration
2086 * (COAP_MAX_EPOLL_EVENTS insufficient)
2087 */
2088 timeout_ms = COAP_IO_NO_WAIT;
2089
2090 /* Keep retrying until less than COAP_MAX_EPOLL_EVENTS are returned */
2091 } while (nfds == COAP_MAX_EPOLL_EVENTS);
2092
2093#endif /* COAP_EPOLL_SUPPORT */
2094#if COAP_SERVER_SUPPORT
2096#endif /* COAP_SERVER_SUPPORT */
2097#if COAP_ASYNC_SUPPORT
2098 /* Check to see if we need to send off any Async requests as delay might
2099 have been updated */
2100 coap_ticks(&now);
2101 coap_check_async(ctx, now);
2102#endif /* COAP_ASYNC_SUPPORT */
2103
2104#ifndef COAP_EPOLL_SUPPORT
2105all_over:
2106#endif /* COAP_EPOLL_SUPPORT */
2107 coap_ticks(&now);
2108 return (int)(((now - before) * 1000) / COAP_TICKS_PER_SECOND);
2109}
2110
2111volatile int coap_thread_quit = 0;
2112
2113void
2118
2119COAP_API int
2121 coap_io_process_thread_t main_loop_code,
2122 void *main_loop_code_arg, uint32_t timeout_ms,
2123 uint32_t thread_count) {
2124 int ret;
2125
2126 if (!context)
2127 return 0;
2128 coap_lock_lock(context, return 0);
2129 ret = coap_io_process_loop_lkd(context, main_loop_code,
2130 main_loop_code_arg, timeout_ms,
2131 thread_count);
2132 coap_lock_unlock(context);
2133 return ret;
2134}
2135
2136int
2138 coap_io_process_thread_t main_loop_code,
2139 void *main_loop_code_arg, uint32_t timeout_ms,
2140 uint32_t thread_count) {
2141 int ret = 0;;
2142
2143#if COAP_THREAD_SAFE
2144 if (thread_count > 1) {
2145 if (!coap_io_process_configure_threads(context, thread_count - 1))
2146 return 0;
2147 }
2148#else /* COAP_THREAD_SAFE */
2149 thread_count = 1;
2150#endif /* COAP_THREAD_SAFE */
2151 while (!coap_thread_quit) {
2152 if (main_loop_code) {
2153 coap_tick_t begin, end;
2154 uint32_t used_ms;
2155
2156 coap_ticks(&begin);
2157 /*
2158 * main_loop_codecode should not be blocking for any time, and not calling
2159 * coap_io_process().
2160 */
2161 coap_lock_callback_release(context, main_loop_code(main_loop_code_arg),
2162 /* On re-lock failure */
2163 ret = 0; break);
2164 /*
2165 * Need to delay for the remainder of timeout_ms. In case main_loop_code()
2166 * is time sensitive (e.g Observe subscription to /time), delay to the
2167 * start of the a second boundary
2168 */
2169 coap_ticks(&end);
2170 used_ms = (uint32_t)(end - begin) * 1000 / COAP_TICKS_PER_SECOND;
2171 if (timeout_ms == COAP_IO_NO_WAIT || timeout_ms == COAP_IO_WAIT) {
2172 ret = coap_io_process_lkd(context, timeout_ms);
2173 } else if (timeout_ms > used_ms) {
2174 /* Wait for remaining time rounded up to next second start */
2175 coap_tick_t next_time = end + (timeout_ms - used_ms) * COAP_TICKS_PER_SECOND / 1000;
2176 unsigned int next_sec_us;
2177 unsigned int next_sec_ms;
2178
2179 next_sec_us = (timeout_ms - used_ms) * 1000000 / COAP_TICKS_PER_SECOND + 1000000 -
2180 (coap_ticks_to_rt_us(next_time) % 1000000);
2181 next_sec_ms = (next_sec_us + 999) / 1000;
2182 if (next_sec_ms > timeout_ms && next_sec_ms > 1000)
2183 next_sec_ms -= 1000;
2184 ret = coap_io_process_lkd(context, next_sec_ms ? next_sec_ms : 1);
2185 } else {
2186 /* timeout_ms has expired */
2187 ret = coap_io_process_lkd(context, COAP_IO_NO_WAIT);
2188 }
2189
2190 if (thread_count == 1) {
2191 /*
2192 * Need to delay if only one thread until the remainder of
2193 * timeout_ms is used up. Otherwise, another thread will be
2194 * waiting on coap_io_process() to do any input / timeout work.
2195 */
2196 coap_ticks(&end);
2197 used_ms = (uint32_t)(end - begin) * 1000 / COAP_TICKS_PER_SECOND;
2198 if (timeout_ms > 0 && timeout_ms < used_ms) {
2199 ret = coap_io_process_lkd(context, used_ms - timeout_ms);
2200 } else {
2201 ret = coap_io_process_lkd(context, COAP_IO_NO_WAIT);
2202 }
2203 }
2204 } else {
2205 ret = coap_io_process_lkd(context, timeout_ms);
2206 }
2207 /* coap_io_process_lkd() can return 0 */
2208 if (ret >= 0)
2209 ret = 1;
2210
2211 if (ret < 0) {
2212 ret = 0;
2213 break;
2214 }
2215 }
2216#if COAP_THREAD_SAFE
2218#endif /* COAP_THREAD_SAFE */
2219 coap_thread_quit = 0;
2220 return ret;
2221}
2222
2223#endif /* ! WITH_LWIP && ! WITH_CONTIKI && ! RIOT_VERSION*/
2224
2225COAP_API int
2227 int ret;
2228
2229 coap_lock_lock(context, return 0);
2230 ret = coap_io_pending_lkd(context);
2231 coap_lock_unlock(context);
2232 return ret;
2233}
2234
2235/*
2236 * return 1 I/O pending
2237 * 0 No I/O pending
2238 */
2239int
2241 coap_session_t *s, *rtmp;
2242#if COAP_SERVER_SUPPORT
2243 coap_endpoint_t *ep;
2244#endif /* COAP_SERVER_SUPPORT */
2245
2246 if (!context)
2247 return 0;
2248 coap_lock_check_locked(context);
2249 if (coap_io_process_lkd(context, COAP_IO_NO_WAIT) < 0)
2250 return 0;
2251
2252 if (context->sendqueue)
2253 return 1;
2254#if COAP_SERVER_SUPPORT
2255 LL_FOREACH(context->endpoint, ep) {
2256 SESSIONS_ITER(ep->sessions, s, rtmp) {
2257 if (s->delayqueue)
2258 return 1;
2259 if (s->lg_xmit)
2260 return 1;
2261 if (s->lg_srcv)
2262 return 1;
2263 }
2264 }
2265#endif /* COAP_SERVER_SUPPORT */
2266#if COAP_CLIENT_SUPPORT
2267 SESSIONS_ITER(context->sessions, s, rtmp) {
2268 if (s->delayqueue)
2269 return 1;
2270 if (s->lg_xmit)
2271 return 1;
2272 if (s->lg_crcv)
2273 return 1;
2274 }
2275#endif /* COAP_CLIENT_SUPPORT */
2276 return 0;
2277}
2278
2279const char *
2281 return strerror(error);
2282}
2283#ifdef _WIN32
2284const char *
2286 coap_win_error_to_errno();
2287 return coap_socket_format_errno(errno);
2288}
2289#else /* _WIN32 */
2290const char *
2292 return coap_socket_format_errno(errno);
2293}
2294#endif /* _WIN32 */
2295
2298#if !defined(WITH_LWIP) && !defined(WITH_CONTIKI)
2299 return sock->fd;
2300#else
2301 (void)(sock);
2302 return COAP_INVALID_SOCKET;
2303#endif
2304}
2305
2308 return sock->flags;
2309}
2310
2311COAP_API void
2313 sock->flags = flags;
2314}
2315
2316#undef SIN6
void coap_address_set_port(coap_address_t *addr, uint16_t port)
Set the port field of addr to port (in host byte order).
int coap_is_bcast(const coap_address_t *a)
Checks if given address a denotes a broadcast address.
void coap_address_init(coap_address_t *addr)
Resets the given coap_address_t object addr to its default values.
int coap_is_mcast(const coap_address_t *a)
Checks if given address a denotes a multicast address.
uint16_t coap_address_get_port(const coap_address_t *addr)
Returns the port from addr in host byte order.
void coap_address_copy(coap_address_t *dst, const coap_address_t *src)
COAP_STATIC_INLINE int coap_address_isany(const coap_address_t *a)
Checks if given address object a denotes the wildcard address.
int coap_debug_send_packet(void)
Check to see whether a packet should be sent or not.
#define COAP_IPV4_SUPPORT
const char * coap_socket_format_errno(int error)
Definition coap_io.c:2280
ssize_t coap_socket_read(coap_socket_t *sock, uint8_t *data, size_t data_len)
Function interface for data stream receiving off a socket.
Definition coap_io.c:683
static int coap_add_fd(coap_fd_t fd, coap_fd_t this_fds[], unsigned int *have_this_fds, unsigned int max_this_fds)
Definition coap_io.c:1640
void coap_socket_close(coap_socket_t *sock)
Function interface to close off a socket.
Definition coap_io.c:377
const char * coap_socket_strerror(void)
Definition coap_io.c:2291
ssize_t coap_socket_recv(coap_socket_t *sock, coap_packet_t *packet)
Function interface for reading data.
Definition coap_io.c:1031
ssize_t coap_socket_send(coap_socket_t *sock, coap_session_t *session, const uint8_t *data, size_t datalen)
Function interface for data transmission.
Definition coap_io.c:830
void coap_packet_get_memmapped(coap_packet_t *packet, unsigned char **address, size_t *length)
Given a packet, set msg and msg_len to an address and length of the packet's data in memory.
Definition coap_io.c:1018
ssize_t coap_socket_write(coap_socket_t *sock, const uint8_t *data, size_t data_len)
Function interface for data stream sending off a socket.
Definition coap_io.c:624
volatile int coap_thread_quit
Definition coap_io.c:2111
void coap_update_io_timer(coap_context_t *context, coap_tick_t delay)
Update when to continue with I/O processing, unless packets come in in the meantime.
Definition coap_io.c:506
#define MSG_NOSIGNAL
#define iov_len_t
Definition coap_io.c:777
#define COAP_SOL_IP
Definition coap_io.c:748
#define coap_closesocket
Definition coap_io.h:48
#define COAP_MAX_EPOLL_EVENTS
Definition coap_io.h:38
uint16_t coap_socket_flags_t
Definition coap_io.h:53
#define COAP_RXBUFFER_SIZE
Definition coap_io.h:29
#define COAP_SOCKET_ERROR
Definition coap_io.h:49
int coap_fd_t
Definition coap_io.h:47
#define COAP_INVALID_SOCKET
Definition coap_io.h:50
#define COAP_SOCKET_MULTICAST
socket is used for multicast communication
void coap_epoll_ctl_add(coap_socket_t *sock, uint32_t events, const char *func)
Epoll specific function to add the state of events that epoll is to track for the appropriate file de...
int coap_socket_connect_udp(coap_socket_t *sock, const coap_address_t *local_if, const coap_address_t *server, int default_port, coap_address_t *local_addr, coap_address_t *remote_addr)
#define COAP_SOCKET_WANT_ACCEPT
non blocking server socket is waiting for accept
#define COAP_SOCKET_NOT_EMPTY
the socket is not empty
#define COAP_SOCKET_CAN_WRITE
non blocking socket can now write without blocking
#define COAP_SOCKET_BOUND
the socket is bound
#define COAP_SOCKET_WANT_READ
non blocking socket is waiting for reading
#define COAP_SOCKET_CAN_ACCEPT
non blocking server socket can now accept without blocking
int coap_socket_bind_udp(coap_socket_t *sock, const coap_address_t *listen_addr, coap_address_t *bound_addr)
#define COAP_SOCKET_WANT_WRITE
non blocking socket is waiting for writing
#define COAP_SOCKET_CAN_CONNECT
non blocking client socket can now connect without blocking
coap_endpoint_t * coap_malloc_endpoint(void)
void coap_epoll_ctl_mod(coap_socket_t *sock, uint32_t events, const char *func)
Epoll specific function to modify the state of events that epoll is tracking on the appropriate file ...
#define COAP_SOCKET_WANT_CONNECT
non blocking client socket is waiting for connect
void coap_mfree_endpoint(coap_endpoint_t *ep)
#define COAP_SOCKET_CAN_READ
non blocking socket can now read without blocking
#define COAP_SOCKET_CONNECTED
the socket is connected
#define COAP_SOCKET_EMPTY
coap_socket_flags_t values
Library specific build wrapper for coap_internal.h.
#define COAP_API
@ COAP_ENDPOINT
Definition coap_mem.h:45
void * coap_malloc_type(coap_memory_tag_t type, size_t size)
Allocates a chunk of size bytes and returns a pointer to the newly allocated memory.
void coap_free_type(coap_memory_tag_t type, void *p)
Releases the memory that was allocated by coap_malloc_type().
coap_tick_t coap_dtls_get_timeout(coap_session_t *session COAP_UNUSED, coap_tick_t now COAP_UNUSED)
Definition coap_notls.c:229
coap_tick_t coap_dtls_get_context_timeout(void *dtls_context COAP_UNUSED)
Definition coap_notls.c:224
int coap_dtls_handle_timeout(coap_session_t *session COAP_UNUSED)
Definition coap_notls.c:238
#define SESSIONS_ITER_SAFE(e, el, rtmp)
#define SESSIONS_ITER(e, el, rtmp)
#define COAP_DEFAULT_SESSION_TIMEOUT
int coap_proxy_check_timeouts(coap_context_t *context, coap_tick_t now, coap_tick_t *tim_rem)
Idle timeout inactive proxy sessions as well as return in tim_rem the time to remaining to timeout th...
void coap_io_do_epoll_lkd(coap_context_t *ctx, struct epoll_event *events, size_t nevents)
Process all the epoll events.
Definition coap_net.c:2641
int coap_io_process_loop_lkd(coap_context_t *context, coap_io_process_thread_t main_loop_code, void *main_loop_code_arg, uint32_t timeout_ms, uint32_t thread_count)
Do the coap_io_process() across thread_count threads.
Definition coap_io.c:2137
int coap_io_pending_lkd(coap_context_t *context)
Check to see if there is any i/o pending for the context.
Definition coap_io.c:2240
void coap_io_do_io_lkd(coap_context_t *ctx, coap_tick_t now)
Processes any outstanding read, write, accept or connect I/O as indicated in the coap_socket_t struct...
Definition coap_net.c:2576
unsigned int coap_io_get_fds_lkd(coap_context_t *ctx, coap_fd_t read_fds[], unsigned int *have_read_fds, unsigned int max_read_fds, coap_fd_t write_fds[], unsigned int *have_write_fds, unsigned int max_write_fds, unsigned int *rem_timeout_ms)
Definition coap_io.c:1656
int coap_io_process_lkd(coap_context_t *ctx, uint32_t timeout_ms)
The main I/O processing function.
Definition coap_io.c:1769
int coap_io_process_with_fds_lkd(coap_context_t *ctx, uint32_t timeout_ms, int enfds, fd_set *ereadfds, fd_set *ewritefds, fd_set *eexceptfds)
The main message processing loop with additional fds for internal select.
Definition coap_io.c:1862
unsigned int coap_io_prepare_io_lkd(coap_context_t *ctx, coap_socket_t *sockets[], unsigned int max_sockets, unsigned int *num_sockets, coap_tick_t now)
Iterates through all the coap_socket_t structures embedded in endpoints or sessions associated with t...
Definition coap_io.c:1331
unsigned int coap_io_prepare_epoll_lkd(coap_context_t *ctx, coap_tick_t now)
Any now timed out delayed packet is transmitted, along with any packets associated with requested obs...
Definition coap_io.c:1262
COAP_API int coap_io_process(coap_context_t *ctx, uint32_t timeout_ms)
The main I/O processing function.
Definition coap_io.c:1759
COAP_API int coap_io_pending(coap_context_t *context)
Check to see if there is any i/o pending for the context.
Definition coap_io.c:2226
int coap_io_process_configure_threads(coap_context_t *context, uint32_t thread_count)
Configure a defined number of threads to do the alternate coap_io_process() work with traffic load ba...
void coap_io_process_terminate_loop(void)
Terminate all the additional threads created by coap_io_process_loop() and break out of the main thre...
Definition coap_io.c:2114
void coap_io_process_remove_threads(coap_context_t *context)
Release the coap_io_process() worker threads.
COAP_API int coap_io_process_loop(coap_context_t *context, coap_io_process_thread_t main_loop_code, void *main_loop_code_arg, uint32_t timeout_ms, uint32_t thread_count)
Do the coap_io_process() across thread_count threads.
Definition coap_io.c:2120
COAP_API unsigned int coap_io_get_fds(coap_context_t *ctx, coap_fd_t read_fds[], unsigned int *have_read_fds, unsigned int max_read_fds, coap_fd_t write_fds[], unsigned int *have_write_fds, unsigned int max_write_fds, unsigned int *rem_timeout_ms)
Definition coap_io.c:1621
COAP_API unsigned int coap_io_prepare_io(coap_context_t *ctx, coap_socket_t *sockets[], unsigned int max_sockets, unsigned int *num_sockets, coap_tick_t now)
Iterates through all the coap_socket_t structures embedded in endpoints or sessions associated with t...
Definition coap_io.c:1313
#define COAP_IO_NO_WAIT
Definition coap_net.h:704
COAP_API void coap_socket_set_flags(coap_socket_t *sock, coap_socket_flags_t flags)
Set the libcoap internal flags for a socket.
Definition coap_io.c:2312
COAP_API unsigned int coap_io_prepare_epoll(coap_context_t *ctx, coap_tick_t now)
Any now timed out delayed packet is transmitted, along with any packets associated with requested obs...
Definition coap_io.c:1252
COAP_API coap_fd_t coap_socket_get_fd(coap_socket_t *sock)
Get the libcoap internal file descriptor for a socket.
Definition coap_io.c:2297
COAP_API int coap_io_process_with_fds(coap_context_t *ctx, uint32_t timeout_ms, int enfds, fd_set *ereadfds, fd_set *ewritefds, fd_set *eexceptfds)
The main message processing loop with additional fds for internal select.
Definition coap_io.c:1774
#define COAP_IO_WAIT
Definition coap_net.h:703
void(* coap_io_process_thread_t)(void *arg)
Main thread coap_io_process_loop activity.
Definition coap_net.h:885
COAP_API coap_socket_flags_t coap_socket_get_flags(coap_socket_t *sock)
Get the libcoap internal flags for a socket.
Definition coap_io.c:2307
int coap_block_check_lg_crcv_timeouts(coap_session_t *session, coap_tick_t now, coap_tick_t *tim_rem)
int coap_block_check_lg_srcv_timeouts(coap_session_t *session, coap_tick_t now, coap_tick_t *tim_rem)
int coap_block_check_lg_xmit_timeouts(coap_session_t *session, coap_tick_t now, coap_tick_t *tim_rem)
void coap_expire_cache_entries(coap_context_t *context)
Expire coap_cache_entry_t entries.
uint64_t coap_tick_t
This data type represents internal timer ticks with COAP_TICKS_PER_SECOND resolution.
Definition coap_time.h:143
#define COAP_TICKS_PER_SECOND
Use ms resolution on POSIX systems.
Definition coap_time.h:158
uint64_t coap_ticks_to_rt_us(coap_tick_t t)
Helper function that converts coap ticks to POSIX wallclock time in us.
int coap_handle_event_lkd(coap_context_t *context, coap_event_t event, coap_session_t *session)
Invokes the event handler of context for the given event and data.
Definition coap_net.c:4707
coap_queue_t * coap_peek_next(coap_context_t *context)
Returns the next pdu to send without removing from sendqeue.
Definition coap_net.c:270
coap_queue_t * coap_pop_next(coap_context_t *context)
Returns the next pdu to send and removes it from the sendqeue.
Definition coap_net.c:278
coap_mid_t coap_retransmit(coap_context_t *context, coap_queue_t *node)
Handles retransmissions of confirmable messages.
Definition coap_net.c:2160
void coap_send_recv_terminate(void)
Terminate any active coap_send_recv() sessions.
Definition coap_net.c:2019
void coap_ticks(coap_tick_t *)
Returns the current value of an internal tick counter.
int coap_dtls_is_context_timeout(void)
Check if timeout is handled per CoAP session or per CoAP context.
Definition coap_notls.c:219
@ COAP_EVENT_SERVER_SESSION_DEL
Called in the CoAP IO loop if a server session is deleted (e.g., due to inactivity or because the max...
Definition coap_event.h:94
@ COAP_EVENT_KEEPALIVE_FAILURE
Triggered when no response to a keep alive (ping) packet.
Definition coap_event.h:132
#define coap_lock_callback_release(c, func, failed)
Dummy for no thread-safe code.
#define coap_lock_unlock(c)
Dummy for no thread-safe code.
#define coap_lock_lock(c, failed)
Dummy for no thread-safe code.
#define coap_lock_check_locked(c)
Dummy for no thread-safe code.
#define coap_log_debug(...)
Definition coap_debug.h:120
#define coap_log_alert(...)
Definition coap_debug.h:84
#define coap_log_emerg(...)
Definition coap_debug.h:81
const char * coap_session_str(const coap_session_t *session)
Get session description.
#define coap_log_info(...)
Definition coap_debug.h:108
#define coap_log_warn(...)
Definition coap_debug.h:102
#define coap_log_err(...)
Definition coap_debug.h:96
#define coap_log_crit(...)
Definition coap_debug.h:90
#define COAP_INVALID_MID
Indicates an invalid message id.
Definition coap_pdu.h:266
@ COAP_PROTO_DTLS
Definition coap_pdu.h:315
void coap_session_server_keepalive_failed(coap_session_t *session)
Clear down a session following a keepalive failure.
coap_mid_t coap_session_send_ping_lkd(coap_session_t *session)
Send a ping message for the session.
void coap_session_free(coap_session_t *session)
void coap_session_release_lkd(coap_session_t *session)
Decrement reference counter on a session.
coap_session_t * coap_session_reference_lkd(coap_session_t *session)
Increment reference counter on a session.
#define COAP_PROTO_RELIABLE(p)
@ COAP_SESSION_TYPE_SERVER
server-side
@ COAP_SESSION_TYPE_CLIENT
client-side
@ COAP_SESSION_STATE_HANDSHAKE
@ COAP_SESSION_STATE_CSM
@ COAP_SESSION_STATE_ESTABLISHED
@ COAP_SESSION_STATE_NONE
void coap_check_notify_lkd(coap_context_t *context)
Checks all known resources to see if they are dirty and then notifies subscribed observers.
coap_address_t remote
remote address and port
Definition coap_io.h:56
coap_address_t local
local address and port
Definition coap_io.h:57
Multi-purpose address abstraction.
socklen_t size
size of addr
struct sockaddr_in sin
struct coap_sockaddr_un cun
struct sockaddr_in6 sin6
struct sockaddr sa
union coap_address_t::@0 addr
The CoAP stack's global state is stored in a coap_context_t object.
coap_tick_t sendqueue_basetime
The time stamp in the first element of the sendqeue is relative to sendqueue_basetime.
coap_socket_t * sockets[64]
Track different socket information in coap_io_process_with_fds_lkd()
unsigned int num_sockets
Number of sockets being tracked.
coap_session_t * sessions
client sessions
fd_set exceptfds
Used for select call in coap_io_process_with_fds_lkd()
unsigned int ping_timeout
Minimum inactivity time before sending a ping message.
coap_queue_t * sendqueue
uint8_t testing_cids
Change client's source port every testing_cids.
coap_endpoint_t * endpoint
the endpoints used for listening
uint32_t csm_timeout_ms
Timeout for waiting for a CSM from the remote side.
coap_tick_t next_timeout
When the next timeout is to occur.
unsigned int session_timeout
Number of seconds of inactivity after which an unused session will be closed.
Abstraction of virtual endpoint that can be attached to coap_context_t.
coap_context_t * context
endpoint's context
coap_session_t * sessions
hash table or list of active sessions
coap_address_t bind_addr
local interface address
coap_socket_t sock
socket object for the interface, if any
size_t length
length of payload
coap_addr_tuple_t addr_info
local and remote addresses
unsigned char * payload
payload
int ifindex
the interface index
Queue entry.
coap_tick_t t
when to send PDU for the next time
Abstraction of virtual session that can be attached to coap_context_t (client) or coap_endpoint_t (se...
coap_lg_xmit_t * lg_xmit
list of large transmissions
unsigned ref_subscriptions
reference count of current subscriptions
coap_socket_t sock
socket object for the session, if any
coap_session_state_t state
current state of relationship with peer
coap_addr_tuple_t addr_info
remote/local address info
coap_proto_t proto
protocol used
unsigned ref
reference count from queues
uint8_t negotiated_cid
Set for a client if CID negotiated.
void * tls
security parameters
coap_queue_t * delayqueue
list of delayed messages waiting to be sent
coap_mid_t last_ping_mid
the last keepalive message id that was used in this session
coap_lg_srcv_t * lg_srcv
Server list of expected large receives.
coap_lg_crcv_t * lg_crcv
Client list of expected large receives.
coap_session_type_t type
client or server side socket
coap_context_t * context
session's context
int ifindex
interface index
char sun_path[COAP_UNIX_PATH_MAX]
coap_session_t * session
Used to determine session owner.
coap_endpoint_t * endpoint
Used by the epoll logic for a listening endpoint.
coap_address_t mcast_addr
remote address and port (multicast track)
coap_socket_flags_t flags
1 or more of COAP_SOCKET* flag values
struct in6_addr ipi6_addr
Definition coap_io.c:728
unsigned int ipi6_ifindex
Definition coap_io.c:729
struct in_addr ipi_spec_dst
Definition coap_io.c:734
struct in_addr ipi_addr
Definition coap_io.c:735
int ipi_ifindex
Definition coap_io.c:733