libcoap 4.3.5-develop-1ba3158
Loading...
Searching...
No Matches
coap_debug.c
Go to the documentation of this file.
1/* coap_debug.c -- debug utilities
2 *
3 * Copyright (C) 2010--2012,2014--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#if defined(HAVE_STRNLEN) && defined(__GNUC__) && !defined(_GNU_SOURCE)
19#define _GNU_SOURCE 1
20#endif
21
22#include <stdarg.h>
23#include <stdio.h>
24#include <string.h>
25#include <ctype.h>
26
27#ifdef HAVE_ARPA_INET_H
28#include <arpa/inet.h>
29#endif
30#ifdef HAVE_WS2TCPIP_H
31#include <ws2tcpip.h>
32#endif
33
34#ifdef HAVE_TIME_H
35#include <time.h>
36#endif
37
38#ifdef WITH_LWIP
39# define fprintf(fd, ...) LWIP_PLATFORM_DIAG((__VA_ARGS__))
40# define fflush(...)
41#endif
42
43#ifdef WITH_CONTIKI
44# define fprintf(fd, ...) { (void)fd; printf(__VA_ARGS__); }
45# define fflush(...)
46# define vfprintf(fd, ...) { (void)fd; printf(__VA_ARGS__); }
47
48# ifndef LOG_CONF_LEVEL_COAP
49# define LOG_CONF_LEVEL_COAP 2 /* = LOG_LEVEL_WARN */
50# endif
51static coap_log_t maxlog = LOG_CONF_LEVEL_COAP == 0 ? /* = LOG_LEVEL_NONE */
53 (LOG_CONF_LEVEL_COAP == 1 ? /* = LOG_LEVEL_ERR */
55 (LOG_CONF_LEVEL_COAP == 2 ? /* = LOG_LEVEL_WARN */
57 (LOG_CONF_LEVEL_COAP == 3 ? /* = LOG_LEVEL_INFO */
60#else /* !WITH_CONTIKI */
61static coap_log_t maxlog = COAP_LOG_WARN; /* default maximum CoAP log level */
62#endif /* !WITH_CONTIKI */
63
64#ifdef RIOT_VERSION
65#include "flash_utils.h"
66#endif /* RIOT_VERSION */
67
68static int use_fprintf_for_show_pdu = 1; /* non zero to output with fprintf */
69static int enable_data_for_show_pdu = 1; /* By default show PDU data for coap_show_pdu() */
70
71const char *
73 return PACKAGE_NAME;
74}
75
76const char *
78 return PACKAGE_STRING;
79}
80
81const char *
83#ifdef LIBCOAP_PACKAGE_BUILD
84 return LIBCOAP_PACKAGE_BUILD;
85#else /* !LIBCOAP_PACKAGE_BUILD */
86 return PACKAGE_STRING;
87#endif /* !LIBCOAP_PACKAGE_BUILD */
88}
89
90void
91coap_set_show_pdu_output(int use_fprintf) {
92 use_fprintf_for_show_pdu = use_fprintf;
93}
94
95void
97 enable_data_for_show_pdu = enable_data;
98}
99
102 return maxlog;
103}
104
105void
107 if (level > COAP_MAX_LOGGING_LEVEL)
109 maxlog = level;
110}
111
112/* this array has the same order as the type coap_log_t with the (D)TLS
113 entries added to the list with a COAP_LOG_DTLS_BASE offset */
114static const char *loglevels[] = {
115 /* General logging */
116 "EMRG", "ALRT", "CRIT", "ERR ", "WARN", "NOTE", "INFO", "DEBG", "OSC ",
117 /* (D)TLS logging */
118 "Emrg", "Alrt", "Crit", "Err ", "Warn", "Note", "Info", "Debg"
119};
120
121const char *
123 static char bad[8];
124 if (level >= sizeof(loglevels)/sizeof(loglevels[0])) {
125 snprintf(bad, sizeof(bad), "%4d", level);
126 return bad;
127 } else {
128 return loglevels[level];
129 }
130}
131
132#ifdef WITH_CONTIKI
133void
134coap_print_contiki_prefix(coap_log_t level) {
135 printf("[%s: COAP ] ", coap_log_level_desc(level));
136}
137#endif /* WITH_CONTIKI */
138
139#ifdef HAVE_TIME_H
140
142print_timestamp(char *s, size_t len, coap_tick_t t) {
143 struct tm *tmp;
144 size_t lensofar;
145 time_t now = coap_ticks_to_rt(t);
146 tmp = localtime(&now);
147 lensofar = strftime(s, len, "%b %d %H:%M:%S", tmp);
148 if (len > lensofar + 4) {
149 lensofar += snprintf(&s[lensofar], len-lensofar, ".%03u",
150 (unsigned int)((coap_ticks_to_rt_us(t) % 1000000)/1000));
151 }
152 return lensofar;
153}
154
155#else /* alternative implementation: just print the timestamp */
156
158print_timestamp(char *s, size_t len, coap_tick_t t) {
159#ifdef HAVE_SNPRINTF
160 return snprintf(s, len, "%u.%03u",
161 (unsigned int)coap_ticks_to_rt(t),
162 (unsigned int)((coap_ticks_to_rt_us(t) % 1000000)/1000));
163#else /* HAVE_SNPRINTF */
164 /* @todo do manual conversion of timestamp */
165 return 0;
166#endif /* HAVE_SNPRINTF */
167}
168
169#endif /* HAVE_TIME_H */
170
171#if !defined(HAVE_STRNLEN) && !defined(__MINGW32__)
180static inline size_t
181strnlen(const char *s, size_t maxlen) {
182 size_t n = 0;
183 while (*s++ && n < maxlen)
184 ++n;
185 return n;
186}
187#endif /* HAVE_STRNLEN && !__MINGW32__ */
188
189static size_t
190print_readable(const uint8_t *data, size_t len,
191 unsigned char *result, size_t buflen, int encode_always) {
192 const uint8_t hex[] = "0123456789ABCDEF";
193 size_t cnt = 0;
194 assert(data || len == 0);
195
196 if (buflen == 0) { /* there is nothing we can do here but return */
197 return 0;
198 }
199
200 while (len) {
201 if (!encode_always && isprint(*data)) {
202 if (cnt+1 < buflen) { /* keep one byte for terminating zero */
203 *result++ = *data;
204 ++cnt;
205 } else {
206 break;
207 }
208 } else {
209 if (cnt+4 < buflen) { /* keep one byte for terminating zero */
210 *result++ = '\\';
211 *result++ = 'x';
212 *result++ = hex[(*data & 0xf0) >> 4];
213 *result++ = hex[*data & 0x0f];
214 cnt += 4;
215 } else
216 break;
217 }
218
219 ++data;
220 --len;
221 }
222
223 *result = '\0'; /* add a terminating zero */
224 return cnt;
225}
226
227#ifndef min
228#define min(a,b) ((a) < (b) ? (a) : (b))
229#endif
230
231#ifndef INET6_ADDRSTRLEN
232#define INET6_ADDRSTRLEN 46
233#endif
234/*
235 * Returned buf is always NULL terminated.
236 * Returned size is number of characters, not including NULL terminator.
237 */
238size_t
239coap_print_addr(const coap_address_t *addr, unsigned char *buf, size_t len) {
240#if (defined( HAVE_ARPA_INET_H ) || defined( HAVE_WS2TCPIP_H )) && !defined(RIOT_VERSION)
241 char scratch[INET6_ADDRSTRLEN];
242
243 assert(buf);
244 assert(len);
245 buf[0] = '\000';
246
247 switch (addr->addr.sa.sa_family) {
248#if COAP_IPV4_SUPPORT
249 case AF_INET:
250 snprintf((char *)buf, len, "%s:%d",
251 coap_print_ip_addr(addr, scratch, sizeof(scratch)),
253 break;
254#endif /* COAP_IPV4_SUPPORT */
255#if COAP_IPV6_SUPPORT
256 case AF_INET6:
257 snprintf((char *)buf, len, "[%s]:%d",
258 coap_print_ip_addr(addr, scratch, sizeof(scratch)),
260 break;
261#endif /* COAP_IPV6_SUPPORT */
262#if COAP_AF_UNIX_SUPPORT
263 case AF_UNIX:
264 snprintf((char *)buf, len, "%s", addr->addr.cun.sun_path);
265 break;
266#endif /* COAP_AF_UNIX_SUPPORT */
267 default:
268 /* Include trailing NULL if possible */
269 memcpy(buf, "(unknown address type)", min(22+1, len));
270 buf[len-1] = '\000';
271 break;
272 }
273 return strlen((char *)buf);
274
275#else /* HAVE_ARPA_INET_H */
276
277# if defined(RIOT_VERSION)
278 char scratch[INET6_ADDRSTRLEN];
279
280 assert(buf);
281 assert(len);
282 buf[0] = '\000';
283
284 switch (addr->riot.family) {
285#if COAP_IPV4_SUPPORT
286 case AF_INET:
287 snprintf((char *)buf, len, "%s:%d",
288 coap_print_ip_addr(addr, scratch, sizeof(scratch)),
290 break;
291#endif /* COAP_IPV4_SUPPORT */
292#if COAP_IPV6_SUPPORT
293 case AF_INET6:
294 snprintf((char *)buf, len, "[%s]:%d",
295 coap_print_ip_addr(addr, scratch, sizeof(scratch)),
297 break;
298#endif /* COAP_IPV6_SUPPORT */
299 default:
300 /* Include trailing NULL if possible */
301 memcpy(buf, "(unknown address type)", min(22+1, len));
302 buf[len-1] = '\000';
303 break;
304 }
305 return strlen((char *)buf);
306
307# elif WITH_CONTIKI
308
309 char scratch[INET6_ADDRSTRLEN];
310#ifdef HAVE_SNPRINTF
311
312 snprintf((char *)buf, len, "[%s]:%d",
313 coap_print_ip_addr(addr, scratch, sizeof(scratch)),
315 return strlen((char *)buf);
316#else /* HAVE_SNPRINTF */
317 unsigned char *p = buf;
318# if NETSTACK_CONF_WITH_IPV6
319
320 assert(buf);
321 assert(len);
322 buf[0] = '\000';
323 if (len < 40 + 2 + 6)
324 return 0;
325
326 *p++ = '[';
327 memcpy(p, coap_print_ip_addr(addr, scratch, sizeof(scratch)), 40);
328 p += 40 - 1;
329 *p++ = ']';
330# else /* WITH_UIP6 */
331# warning "IPv4 network addresses will not be included in debug output"
332
333 if (len < 21) {
334 *p = '\000';
335 return 0;
336 }
337# endif /* WITH_UIP6 */
338
339 *p++ = ':';
340 *p++ = '0' + (coap_address_get_port(addr) / 10000) % 10;
341 *p++ = '0' + (coap_address_get_port(addr) / 1000) % 10;
342 *p++ = '0' + (coap_address_get_port(addr) / 100) % 10;
343 *p++ = '0' + (coap_address_get_port(addr) / 10) % 10;
344 *p++ = '0' + coap_address_get_port(addr) % 10;
345 *p = '\000';
346
347 return strlen((char *)buf);
348#endif /* HAVE_SNPRINTF */
349
350# elif WITH_LWIP
351
352 char scratch[INET6_ADDRSTRLEN];
353#ifdef HAVE_SNPRINTF
354
355 snprintf((char *)buf, len, "[%s]:%d",
356 coap_print_ip_addr(addr, scratch, sizeof(scratch)),
357 addr->port);
358 return strlen((char *)buf);
359#else /* HAVE_SNPRINTF */
360 unsigned char *p = buf;
361
362 assert(buf);
363 assert(len);
364 buf[0] = '\000';
365
366 switch (IP_GET_TYPE(addr->addr)) {
367 case IPADDR_TYPE_V4:
368 if (len < IP4ADDR_STRLEN_MAX + 6)
369 return 0;
370 memcpy(buf, coap_print_ip_addr(addr, scratch, sizeof(scratch)), IP4ADDR_STRLEN_MAX);
371 p += strlen((char *)buf);
372 break;
373#if LWIP_IPV6
374 case IPADDR_TYPE_V6:
375 case IPADDR_TYPE_ANY:
376 if (len < 40 + 2 + 6)
377 return 0;
378 *p++ = '[';
379 memcpy(p, coap_print_ip_addr(addr, scratch, sizeof(scratch)), 40);
380 p += strlen((char *)buf);
381 *p++ = ']';
382 break;
383#endif /* LWIP_IPV6 */
384 }
385
386 *p++ = ':';
387 *p++ = '0' + (addr->port / 10000) % 10;
388 *p++ = '0' + (addr->port / 1000) % 10;
389 *p++ = '0' + (addr->port / 100) % 10;
390 *p++ = '0' + (addr->port / 10) % 10;
391 *p++ = '0' + addr->port % 10;
392 *p = '\000';
393
394 return strlen((char *)buf);
395#endif /* HAVE_SNPRINTF */
396
397# else /* ! WITH_CONTIKI && ! WITH_LWIP */
398
399 (void)addr;
400 (void)len;
401
402 /* TODO: output addresses manually */
403# warning "inet_ntop() not available, network addresses will not be included in debug output"
404# endif /* ! WITH_CONTIKI && ! WITH_LWIP */
405 buf[0] = '\000';
406 return 0;
407#endif
408}
409
410/*
411 * Returned buf is always NULL terminated with as much as possible of the
412 * IP address filled in.
413 */
414const char *
415coap_print_ip_addr(const coap_address_t *addr, char *buf, size_t len) {
416#if (defined( HAVE_ARPA_INET_H ) || defined( HAVE_WS2TCPIP_H )) && !defined(RIOT_VERSION)
417 const void *addrptr = NULL;
418
419 assert(buf);
420 assert(len);
421 buf[0] = '\000';
422
423 switch (addr->addr.sa.sa_family) {
424#if COAP_IPV4_SUPPORT
425 case AF_INET:
426 if (len < INET_ADDRSTRLEN)
427 return buf;
428 addrptr = &addr->addr.sin.sin_addr;
429 break;
430#endif /* COAP_IPV4_SUPPORT */
431#if COAP_IPV6_SUPPORT
432 case AF_INET6:
433 if (len < INET6_ADDRSTRLEN)
434 return buf;
435 addrptr = &addr->addr.sin6.sin6_addr;
436 break;
437#endif /* COAP_IPV6_SUPPORT */
438#if COAP_AF_UNIX_SUPPORT
439 case AF_UNIX:
440 snprintf(buf, len, "%s", addr->addr.cun.sun_path);
441 return buf;
442#endif /* COAP_AF_UNIX_SUPPORT */
443 default:
444 /* Include trailing NULL if possible */
445 memcpy(buf, "(unknown address type)", min(22+1, len));
446 buf[len-1] = '\000';
447 return buf;
448 }
449
450 /* Cast needed for Windows, since it doesn't have the correct API signature. */
451 if (inet_ntop(addr->addr.sa.sa_family, addrptr, (char *)buf, len) == 0) {
452 coap_log_err("coap_print_ip_addr: inet_ntop\n");
453 buf[0] = '\000';
454 return buf;
455 }
456 return buf;
457
458#else /* HAVE_ARPA_INET_H */
459
460# if defined(RIOT_VERSION)
461 assert(buf);
462 assert(len);
463 buf[0] = '\000';
464
465 switch (addr->riot.family) {
466#if COAP_IPV4_SUPPORT
467 case AF_INET:
468 if (ipv4_addr_to_str(buf, (ipv4_addr_t *)&addr->riot.addr.ipv4, (size_t)len) == NULL) {
469 goto error;
470 }
471 break;
472#endif /* COAP_IPV4_SUPPORT */
473#if COAP_IPV6_SUPPORT
474 case AF_INET6:
475 if (ipv6_addr_to_str(buf, (ipv6_addr_t *)&addr->riot.addr.ipv6, (size_t)len) == NULL) {
476 goto error;
477 }
478 break;
479#endif /* COAP_IPV6_SUPPORT */
480 default:
481 goto error;
482 }
483 return buf;
484
485error:
486 coap_log_err("coap_print_ip_addr: inet_ntop\n");
487 buf[0] = '\000';
488 return buf;
489
490# elif WITH_CONTIKI
491 char *p = buf;
492 uint8_t i;
493# if NETSTACK_CONF_WITH_IPV6
494 const char hex[] = "0123456789ABCDEF";
495
496 assert(buf);
497 assert(len);
498 buf[0] = '\000';
499 if (len < 40)
500 return 0;
501
502 for (i=0; i < 16; i += 2) {
503 if (i) {
504 *p++ = ':';
505 }
506 *p++ = hex[(addr->addr.u8[i] & 0xf0) >> 4];
507 *p++ = hex[(addr->addr.u8[i] & 0x0f)];
508 *p++ = hex[(addr->addr.u8[i+1] & 0xf0) >> 4];
509 *p++ = hex[(addr->addr.u8[i+1] & 0x0f)];
510 }
511 *p = '\000';
512# else /* WITH_UIP6 */
513# warning "IPv4 network addresses will not be included in debug output"
514
515 if (len < 21) {
516 return buf;
517 }
518# endif /* WITH_UIP6 */
519 return buf;
520
521# elif WITH_LWIP
522
523 assert(buf);
524 assert(len);
525 buf[0] = '\000';
526
527 switch (IP_GET_TYPE(&addr->addr)) {
528#if LWIP_IPV4
529 case IPADDR_TYPE_V4:
530 if (len < IP4ADDR_STRLEN_MAX)
531 return buf;
532 memcpy(buf, ip4addr_ntoa(ip_2_ip4(&addr->addr)), IP4ADDR_STRLEN_MAX);
533 break;
534#endif /* LWIP_IPV4 */
535#if LWIP_IPV6
536 case IPADDR_TYPE_V6:
537 case IPADDR_TYPE_ANY:
538 if (len < 40)
539 return buf;
540#if LWIP_IPV4
541 memcpy(buf, ip6addr_ntoa(&addr->addr.u_addr.ip6), 40);
542#else /* LWIP_IPV4 */
543 memcpy(buf, ip6addr_ntoa(&addr->addr), 40);
544#endif /* LWIP_IPV4 */
545 break;
546#endif /* LWIP_IPV6 */
547 }
548 return buf;
549
550# else /* ! WITH_CONTIKI && ! WITH_LWIP */
551
552 (void)addr;
553 (void)len;
554
555 /* TODO: output addresses manually */
556# warning "inet_ntop() not available, network addresses will not be included in debug output"
557# endif /* WITH_CONTIKI */
558 buf[0] = '\000';
559 return buf;
560#endif
561}
562
564static const char *
565msg_type_string(uint16_t t) {
566 static const char *types[] = { "CON", "NON", "ACK", "RST", "???" };
567
568 return types[min(t, sizeof(types)/sizeof(char *) - 1)];
569}
570
572static const char *
573msg_code_string(uint16_t c) {
574 static const char *methods[] = { "0.00", "GET", "POST", "PUT", "DELETE",
575 "FETCH", "PATCH", "iPATCH"
576 };
577 static const char *signals[] = { "7.00", "CSM", "Ping", "Pong", "Release",
578 "Abort"
579 };
580 static char buf[5];
581
582 if (c < sizeof(methods)/sizeof(const char *)) {
583 return methods[c];
584 } else if (c >= 224 && c - 224 < (int)(sizeof(signals)/sizeof(const char *))) {
585 return signals[c-224];
586 } else {
587 snprintf(buf, sizeof(buf), "%u.%02u", (c >> 5) & 0x7, c & 0x1f);
588 return buf;
589 }
590}
591
593static const char *
594msg_option_string(uint8_t code, uint16_t option_type) {
595 struct option_desc_t {
596 uint16_t type;
597 const char *name;
598 };
599
600 static struct option_desc_t options[] = {
601 { COAP_OPTION_IF_MATCH, "If-Match" },
602 { COAP_OPTION_URI_HOST, "Uri-Host" },
603 { COAP_OPTION_ETAG, "ETag" },
604 { COAP_OPTION_IF_NONE_MATCH, "If-None-Match" },
605 { COAP_OPTION_OBSERVE, "Observe" },
606 { COAP_OPTION_URI_PORT, "Uri-Port" },
607 { COAP_OPTION_LOCATION_PATH, "Location-Path" },
608 { COAP_OPTION_OSCORE, "Oscore" },
609 { COAP_OPTION_URI_PATH, "Uri-Path" },
610 { COAP_OPTION_CONTENT_FORMAT, "Content-Format" },
611 { COAP_OPTION_MAXAGE, "Max-Age" },
612 { COAP_OPTION_URI_QUERY, "Uri-Query" },
613 { COAP_OPTION_HOP_LIMIT, "Hop-Limit" },
614 { COAP_OPTION_ACCEPT, "Accept" },
615 { COAP_OPTION_LOCATION_QUERY, "Location-Query" },
616 { COAP_OPTION_BLOCK2, "Block2" },
617 { COAP_OPTION_BLOCK1, "Block1" },
618 { COAP_OPTION_SIZE2, "Size2" },
619 { COAP_OPTION_PROXY_URI, "Proxy-Uri" },
620 { COAP_OPTION_PROXY_SCHEME, "Proxy-Scheme" },
621 { COAP_OPTION_SIZE1, "Size1" },
622 { COAP_OPTION_ECHO, "Echo" },
623 { COAP_OPTION_NORESPONSE, "No-Response" },
624 { COAP_OPTION_RTAG, "Request-Tag" },
625 { COAP_OPTION_Q_BLOCK1, "Q-Block1" },
626 { COAP_OPTION_Q_BLOCK2, "Q-Block2" }
627 };
628
629 static struct option_desc_t options_csm[] = {
630 { COAP_SIGNALING_OPTION_MAX_MESSAGE_SIZE, "Max-Message-Size" },
631 { COAP_SIGNALING_OPTION_BLOCK_WISE_TRANSFER, "Block-Wise-Transfer" },
632 { COAP_SIGNALING_OPTION_EXTENDED_TOKEN_LENGTH, "Extended-Token-Length" }
633 };
634
635 static struct option_desc_t options_pingpong[] = {
636 { COAP_SIGNALING_OPTION_CUSTODY, "Custody" }
637 };
638
639 static struct option_desc_t options_release[] = {
640 { COAP_SIGNALING_OPTION_ALTERNATIVE_ADDRESS, "Alternative-Address" },
641 { COAP_SIGNALING_OPTION_HOLD_OFF, "Hold-Off" }
642 };
643
644 static struct option_desc_t options_abort[] = {
645 { COAP_SIGNALING_OPTION_BAD_CSM_OPTION, "Bad-CSM-Option" }
646 };
647
648 static char buf[6];
649 size_t i;
650
651 if (code == COAP_SIGNALING_CSM) {
652 for (i = 0; i < sizeof(options_csm)/sizeof(struct option_desc_t); i++) {
653 if (option_type == options_csm[i].type) {
654 return options_csm[i].name;
655 }
656 }
657 } else if (code == COAP_SIGNALING_PING || code == COAP_SIGNALING_PONG) {
658 for (i = 0; i < sizeof(options_pingpong)/sizeof(struct option_desc_t); i++) {
659 if (option_type == options_pingpong[i].type) {
660 return options_pingpong[i].name;
661 }
662 }
663 } else if (code == COAP_SIGNALING_RELEASE) {
664 for (i = 0; i < sizeof(options_release)/sizeof(struct option_desc_t); i++) {
665 if (option_type == options_release[i].type) {
666 return options_release[i].name;
667 }
668 }
669 } else if (code == COAP_SIGNALING_ABORT) {
670 for (i = 0; i < sizeof(options_abort)/sizeof(struct option_desc_t); i++) {
671 if (option_type == options_abort[i].type) {
672 return options_abort[i].name;
673 }
674 }
675 } else {
676 /* search option_type in list of known options */
677 for (i = 0; i < sizeof(options)/sizeof(struct option_desc_t); i++) {
678 if (option_type == options[i].type) {
679 return options[i].name;
680 }
681 }
682 }
683 /* unknown option type, just print to buf */
684 snprintf(buf, sizeof(buf), "%u", option_type);
685 return buf;
686}
687
688static unsigned int
689print_content_format(unsigned int format_type,
690 unsigned char *result, unsigned int buflen) {
691 struct desc_t {
692 unsigned int type;
693 const char *name;
694 };
695
696 static struct desc_t formats[] = {
697 { COAP_MEDIATYPE_TEXT_PLAIN, "text/plain" },
698 { COAP_MEDIATYPE_APPLICATION_LINK_FORMAT, "application/link-format" },
699 { COAP_MEDIATYPE_APPLICATION_XML, "application/xml" },
700 { COAP_MEDIATYPE_APPLICATION_OCTET_STREAM, "application/octet-stream" },
701 { COAP_MEDIATYPE_APPLICATION_RDF_XML, "application/rdf+xml" },
702 { COAP_MEDIATYPE_APPLICATION_EXI, "application/exi" },
703 { COAP_MEDIATYPE_APPLICATION_JSON, "application/json" },
704 { COAP_MEDIATYPE_APPLICATION_CBOR, "application/cbor" },
705 { COAP_MEDIATYPE_APPLICATION_CWT, "application/cwt" },
706 { COAP_MEDIATYPE_APPLICATION_COAP_GROUP_JSON, "application/coap-group+json" },
707 { COAP_MEDIATYPE_APPLICATION_COSE_SIGN, "application/cose; cose-type=\"cose-sign\"" },
708 { COAP_MEDIATYPE_APPLICATION_COSE_SIGN1, "application/cose; cose-type=\"cose-sign1\"" },
709 { COAP_MEDIATYPE_APPLICATION_COSE_ENCRYPT, "application/cose; cose-type=\"cose-encrypt\"" },
710 { COAP_MEDIATYPE_APPLICATION_COSE_ENCRYPT0, "application/cose; cose-type=\"cose-encrypt0\"" },
711 { COAP_MEDIATYPE_APPLICATION_COSE_MAC, "application/cose; cose-type=\"cose-mac\"" },
712 { COAP_MEDIATYPE_APPLICATION_COSE_MAC0, "application/cose; cose-type=\"cose-mac0\"" },
713 { COAP_MEDIATYPE_APPLICATION_COSE_KEY, "application/cose-key" },
714 { COAP_MEDIATYPE_APPLICATION_COSE_KEY_SET, "application/cose-key-set" },
715 { COAP_MEDIATYPE_APPLICATION_SENML_JSON, "application/senml+json" },
716 { COAP_MEDIATYPE_APPLICATION_SENSML_JSON, "application/sensml+json" },
717 { COAP_MEDIATYPE_APPLICATION_SENML_CBOR, "application/senml+cbor" },
718 { COAP_MEDIATYPE_APPLICATION_SENSML_CBOR, "application/sensml+cbor" },
719 { COAP_MEDIATYPE_APPLICATION_SENML_EXI, "application/senml-exi" },
720 { COAP_MEDIATYPE_APPLICATION_SENSML_EXI, "application/sensml-exi" },
721 { COAP_MEDIATYPE_APPLICATION_SENML_XML, "application/senml+xml" },
722 { COAP_MEDIATYPE_APPLICATION_SENSML_XML, "application/sensml+xml" },
723 { COAP_MEDIATYPE_APPLICATION_DOTS_CBOR, "application/dots+cbor" },
724 { COAP_MEDIATYPE_APPLICATION_ACE_CBOR, "application/ace+cbor" },
725 { COAP_MEDIATYPE_APPLICATION_MB_CBOR_SEQ, "application/missing-blocks+cbor-seq" },
726 { COAP_MEDIATYPE_APPLICATION_OSCORE, "application/oscore" },
727 { 75, "application/dcaf+cbor" }
728 };
729
730 size_t i;
731
732 /* search format_type in list of known content formats */
733 for (i = 0; i < sizeof(formats)/sizeof(struct desc_t); i++) {
734 if (format_type == formats[i].type) {
735 return snprintf((char *)result, buflen, "%s", formats[i].name);
736 }
737 }
738
739 /* unknown content format, just print numeric value to buf */
740 return snprintf((char *)result, buflen, "%d", format_type);
741}
742
749is_binary(int content_format) {
750 return !(content_format == -1 ||
751 content_format == COAP_MEDIATYPE_TEXT_PLAIN ||
752 content_format == COAP_MEDIATYPE_APPLICATION_LINK_FORMAT ||
753 content_format == COAP_MEDIATYPE_APPLICATION_XML ||
754 content_format == COAP_MEDIATYPE_APPLICATION_JSON);
755}
756
757#define COAP_DO_SHOW_OUTPUT_LINE \
758 do { \
759 if (use_fprintf_for_show_pdu) { \
760 fprintf(COAP_DEBUG_FD, "%s", outbuf); \
761 } \
762 else { \
763 coap_log(level, "%s", outbuf); \
764 } \
765 } while (0)
766
767/*
768 * It is possible to override the output debug buffer size and hence control
769 * the amount of information printed out about a CoAP PDU.
770 * Note: Adding a byte may be insufficient to output the next byte of the PDU.
771 *
772 * This is done by the adding of a -DCOAP_DEBUG_BUF_SIZE=nnnn option to the
773 * CPPFLAGS parameter that is optionally used on the ./configure command line.
774 *
775 * E.g. ./configure CPPFLAGS="-DCOAP_DEBUG_BUF_SIZE=4096"
776 *
777 */
778
779#if COAP_DEBUG_BUF_SIZE < 5
780#error "COAP_DEBUG_BUF_SIZE must be at least 5, should be >= 32 to be useful"
781#endif /* COAP_DEBUG_BUF_SIZE < 5 */
782
783void
785#if COAP_CONSTRAINED_STACK
786 /* Proxy-Uri: can be 1034 bytes long */
787 /* buf and outbuf can be protected by m_show_pdu if needed */
788 static unsigned char buf[min(COAP_DEBUG_BUF_SIZE, 1035)];
789 static char outbuf[COAP_DEBUG_BUF_SIZE];
790#else /* ! COAP_CONSTRAINED_STACK */
791 /* Proxy-Uri: can be 1034 bytes long */
792 unsigned char buf[min(COAP_DEBUG_BUF_SIZE, 1035)];
793 char outbuf[COAP_DEBUG_BUF_SIZE];
794#endif /* ! COAP_CONSTRAINED_STACK */
795 size_t buf_len = 0; /* takes the number of bytes written to buf */
796 int encode = 0, have_options = 0;
797 uint32_t i;
798 coap_opt_iterator_t opt_iter;
799 coap_opt_t *option;
800 int content_format = -1;
801 size_t data_len;
802 const uint8_t *data;
803 uint32_t opt_len;
804 const uint8_t *opt_val;
805 size_t outbuflen = 0;
806 int is_oscore_payload = 0;
807
808 /* Save time if not needed */
809 if (level > coap_get_log_level())
810 return;
811
812#if COAP_THREAD_SAFE
813 coap_mutex_lock(&m_show_pdu);
814#endif /* COAP_THREAD_SAFE */
815 if (!pdu->session || COAP_PROTO_NOT_RELIABLE(pdu->session->proto)) {
816 snprintf(outbuf, sizeof(outbuf), "v:%d t:%s c:%s i:%04x {",
818 msg_code_string(pdu->code), pdu->mid);
819 } else if (pdu->session->proto == COAP_PROTO_WS ||
820 pdu->session->proto == COAP_PROTO_WSS) {
821 if (pdu->type != COAP_MESSAGE_CON)
822 coap_log_alert("WebSocket: type != CON\n");
823 snprintf(outbuf, sizeof(outbuf), "v:WebSocket c:%s {",
824 msg_code_string(pdu->code));
825 } else {
826 if (pdu->type != COAP_MESSAGE_CON)
827 coap_log_alert("Reliable: type != CON\n");
828 snprintf(outbuf, sizeof(outbuf), "v:Reliable c:%s {",
829 msg_code_string(pdu->code));
830 }
831
832 for (i = 0; i < pdu->actual_token.length; i++) {
833 outbuflen = strlen(outbuf);
834 snprintf(&outbuf[outbuflen], sizeof(outbuf)-outbuflen,
835 "%02x", pdu->actual_token.s[i]);
836 }
837 outbuflen = strlen(outbuf);
838 snprintf(&outbuf[outbuflen], sizeof(outbuf)-outbuflen, "}");
839
840 /* show options, if any */
842
843 outbuflen = strlen(outbuf);
844 snprintf(&outbuf[outbuflen], sizeof(outbuf)-outbuflen, " [");
845 while ((option = coap_option_next(&opt_iter))) {
846 buf[0] = '\000';
847 if (!have_options) {
848 have_options = 1;
849 } else {
850 outbuflen = strlen(outbuf);
851 snprintf(&outbuf[outbuflen], sizeof(outbuf)-outbuflen, ",");
852 }
853
854 if (pdu->code == COAP_SIGNALING_CODE_CSM) {
855 switch (opt_iter.number) {
858 buf_len = snprintf((char *)buf, sizeof(buf), "%u",
860 coap_opt_length(option)));
861 break;
862 default:
863 buf_len = 0;
864 break;
865 }
866 } else if (pdu->code == COAP_SIGNALING_CODE_PING ||
868 buf_len = 0;
869 } else if (pdu->code == COAP_SIGNALING_CODE_RELEASE) {
870 switch (opt_iter.number) {
872 buf_len = print_readable(coap_opt_value(option),
873 coap_opt_length(option),
874 buf, sizeof(buf), 0);
875 break;
877 buf_len = snprintf((char *)buf, sizeof(buf), "%u",
879 coap_opt_length(option)));
880 break;
881 default:
882 buf_len = 0;
883 break;
884 }
885 } else if (pdu->code == COAP_SIGNALING_CODE_ABORT) {
886 switch (opt_iter.number) {
888 buf_len = snprintf((char *)buf, sizeof(buf), "%u",
890 coap_opt_length(option)));
891 break;
892 default:
893 buf_len = 0;
894 break;
895 }
896 } else {
897 switch (opt_iter.number) {
900 content_format = (int)coap_decode_var_bytes(coap_opt_value(option),
901 coap_opt_length(option));
902
903 buf_len = print_content_format(content_format, buf, sizeof(buf));
904 break;
905
910 /* split block option into number/more/size where more is the
911 * letter M if set, the _ otherwise */
912 if (COAP_OPT_BLOCK_SZX(option) == 7) {
913 if (coap_get_data(pdu, &data_len, &data))
914 buf_len = snprintf((char *)buf, sizeof(buf), "%u/%c/BERT(%zu)",
915 coap_opt_block_num(option), /* block number */
916 COAP_OPT_BLOCK_MORE(option) ? 'M' : '_', /* M bit */
917 data_len);
918 else
919 buf_len = snprintf((char *)buf, sizeof(buf), "%u/%c/BERT",
920 coap_opt_block_num(option), /* block number */
921 COAP_OPT_BLOCK_MORE(option) ? 'M' : '_'); /* M bit */
922 } else {
923 buf_len = snprintf((char *)buf, sizeof(buf), "%u/%c/%u",
924 coap_opt_block_num(option), /* block number */
925 COAP_OPT_BLOCK_MORE(option) ? 'M' : '_', /* M bit */
926 (1 << (COAP_OPT_BLOCK_SZX(option) + 4))); /* block size */
927 }
928
929 break;
930
932 opt_len = coap_opt_length(option);
933 buf[0] = '\000';
934 if (opt_len) {
935 size_t ofs = 1;
936 size_t cnt;
937
938 opt_val = coap_opt_value(option);
939 if (opt_val[0] & 0x20) {
940 /* Group Flag */
941 snprintf((char *)buf, sizeof(buf), "grp");
942 }
943 if (opt_val[0] & 0x07) {
944 /* Partial IV */
945 cnt = opt_val[0] & 0x07;
946 if (cnt > opt_len - ofs)
947 goto no_more;
948 buf_len = strlen((char *)buf);
949 snprintf((char *)&buf[buf_len], sizeof(buf)-buf_len, "%spIV=0x",
950 buf_len ? "," : "");
951 for (i = 0; (uint32_t)i < cnt; i++) {
952 buf_len = strlen((char *)buf);
953 snprintf((char *)&buf[buf_len], sizeof(buf)-buf_len,
954 "%02x", opt_val[ofs + i]);
955 }
956 ofs += cnt;
957 }
958 if (opt_val[0] & 0x10) {
959 /* kid context */
960 if (ofs >= opt_len)
961 goto no_more;
962 cnt = opt_val[ofs];
963 if (cnt > opt_len - ofs - 1)
964 goto no_more;
965 ofs++;
966 buf_len = strlen((char *)buf);
967 snprintf((char *)&buf[buf_len], sizeof(buf)-buf_len, "%skc=0x",
968 buf_len ? "," : "");
969 for (i = 0; (uint32_t)i < cnt; i++) {
970 buf_len = strlen((char *)buf);
971 snprintf((char *)&buf[buf_len], sizeof(buf)-buf_len,
972 "%02x", opt_val[ofs + i]);
973 }
974 ofs += cnt;
975 }
976 if (opt_val[0] & 0x08) {
977 /* kid */
978 if (ofs >= opt_len)
979 goto no_more;
980 cnt = opt_len - ofs;
981 buf_len = strlen((char *)buf);
982 snprintf((char *)&buf[buf_len], sizeof(buf)-buf_len, "%skid=0x",
983 buf_len ? "," : "");
984 for (i = 0; (uint32_t)i < cnt; i++) {
985 buf_len = strlen((char *)buf);
986 snprintf((char *)&buf[buf_len], sizeof(buf)-buf_len,
987 "%02x", opt_val[ofs + i]);
988 }
989 }
990 }
991no_more:
992 buf_len = strlen((char *)buf);
993 is_oscore_payload = 1;
994 break;
995
1000 case COAP_OPTION_SIZE2:
1002 if (coap_opt_length(option)) {
1003 /* show values as unsigned decimal value */
1004 buf_len = snprintf((char *)buf, sizeof(buf), "%u",
1006 coap_opt_length(option)));
1007 }
1008 break;
1009
1011 case COAP_OPTION_ETAG:
1012 case COAP_OPTION_ECHO:
1014 case COAP_OPTION_RTAG:
1015 opt_len = coap_opt_length(option);
1016 opt_val = coap_opt_value(option);
1017 snprintf((char *)buf, sizeof(buf), "0x");
1018 for (i = 0; (uint32_t)i < opt_len; i++) {
1019 buf_len = strlen((char *)buf);
1020 snprintf((char *)&buf[buf_len], sizeof(buf)-buf_len,
1021 "%02x", opt_val[i]);
1022 }
1023 buf_len = strlen((char *)buf);
1024 break;
1025 default:
1026 /* generic output function for all other option types */
1027 if (opt_iter.number == COAP_OPTION_URI_PATH ||
1028 opt_iter.number == COAP_OPTION_PROXY_URI ||
1029 opt_iter.number == COAP_OPTION_URI_HOST ||
1030 opt_iter.number == COAP_OPTION_LOCATION_PATH ||
1031 opt_iter.number == COAP_OPTION_LOCATION_QUERY ||
1032 opt_iter.number == COAP_OPTION_PROXY_SCHEME ||
1033 opt_iter.number == COAP_OPTION_URI_QUERY) {
1034 encode = 0;
1035 } else {
1036 encode = 1;
1037 }
1038 buf_len = print_readable(coap_opt_value(option),
1039 coap_opt_length(option),
1040 buf, sizeof(buf), encode);
1041 }
1042 }
1043 outbuflen = strlen(outbuf);
1044 snprintf(&outbuf[outbuflen], sizeof(outbuf)-outbuflen,
1045 " %s:%.*s", msg_option_string(pdu->code, opt_iter.number),
1046 (int)buf_len, buf);
1047 }
1048
1049 outbuflen = strlen(outbuf);
1050 snprintf(&outbuf[outbuflen], sizeof(outbuf)-outbuflen, " ]");
1051
1052 if (coap_get_data(pdu, &data_len, &data)) {
1054 /* Only output data if wanted */
1055 outbuflen = strlen(outbuf);
1056 snprintf(&outbuf[outbuflen], sizeof(outbuf)-outbuflen, " :: ");
1057 outbuflen = strlen(outbuf);
1058 snprintf(&outbuf[outbuflen], sizeof(outbuf)-outbuflen,
1059 "data length %zu (data suppressed)\n", data_len);
1061#if COAP_THREAD_SAFE
1062 coap_mutex_unlock(&m_show_pdu);
1063#endif /* COAP_THREAD_SAFE */
1064 return;
1065 }
1066
1067 outbuflen = strlen(outbuf);
1068 snprintf(&outbuf[outbuflen], sizeof(outbuf)-outbuflen, " :: ");
1069
1070 if (is_binary(content_format) || !isprint(data[0]) || is_oscore_payload) {
1071 size_t keep_data_len = data_len;
1072 const uint8_t *keep_data = data;
1073
1074 outbuflen = strlen(outbuf);
1075 snprintf(&outbuf[outbuflen], sizeof(outbuf)-outbuflen,
1076 "binary data length %zu\n", data_len);
1078 /*
1079 * Output hex dump of binary data as a continuous entry
1080 */
1081 outbuf[0] = '\000';
1082 snprintf(outbuf, sizeof(outbuf), "<<");
1083 while (data_len--) {
1084 outbuflen = strlen(outbuf);
1085 snprintf(&outbuf[outbuflen], sizeof(outbuf)-outbuflen,
1086 "%02x", *data++);
1087 }
1088 outbuflen = strlen(outbuf);
1089 snprintf(&outbuf[outbuflen], sizeof(outbuf)-outbuflen, ">>");
1090 data_len = keep_data_len;
1091 data = keep_data;
1092 outbuflen = strlen(outbuf);
1093 if (outbuflen == sizeof(outbuf)-1)
1094 outbuflen--;
1095 snprintf(&outbuf[outbuflen], sizeof(outbuf)-outbuflen, "\n");
1097 /*
1098 * Output ascii readable (if possible), immediately under the
1099 * hex value of the character output above to help binary debugging
1100 */
1101 outbuf[0] = '\000';
1102 snprintf(outbuf, sizeof(outbuf), "<<");
1103 while (data_len--) {
1104 outbuflen = strlen(outbuf);
1105 snprintf(&outbuf[outbuflen], sizeof(outbuf)-outbuflen,
1106 "%c ", isprint(*data) ? *data : '.');
1107 data++;
1108 }
1109 outbuflen = strlen(outbuf);
1110 snprintf(&outbuf[outbuflen], sizeof(outbuf)-outbuflen, ">>");
1111 } else {
1112 size_t max_length;
1113
1114 outbuflen = strlen(outbuf);
1115 max_length = sizeof(outbuf)-outbuflen;
1116 if (max_length > 1) {
1117 outbuf[outbuflen++] = '\'';
1118 outbuf[outbuflen] = '\000';
1119 max_length--;
1120 }
1121 if (max_length > 1) {
1122 outbuflen += print_readable(data, data_len,
1123 (unsigned char *)&outbuf[outbuflen],
1124 max_length, 0);
1125 }
1126 /* print_readable may be handling unprintables - hence headroom of 4 */
1127 if (outbuflen < sizeof(outbuf)-4-1) {
1128 outbuf[outbuflen++] = '\'';
1129 outbuf[outbuflen] = '\000';
1130 }
1131 }
1132 }
1133
1134 outbuflen = strlen(outbuf);
1135 if (outbuflen == sizeof(outbuf)-1)
1136 outbuflen--;
1137 snprintf(&outbuf[outbuflen], sizeof(outbuf)-outbuflen, "\n");
1139
1140#if COAP_THREAD_SAFE
1141 coap_mutex_unlock(&m_show_pdu);
1142#endif /* COAP_THREAD_SAFE */
1143}
1144
1145void
1147 char buffer[128];
1148 coap_string_tls_version(buffer, sizeof(buffer));
1149 coap_log(level, "%s\n", buffer);
1150}
1151
1152char *
1153coap_string_tls_version(char *buffer, size_t bufsize) {
1155 char beta[8];
1156 char sub[2];
1157 char b_beta[8];
1158 char b_sub[2];
1159
1160 switch (tls_version->type) {
1162 snprintf(buffer, bufsize, "TLS Library: None");
1163 break;
1165 snprintf(buffer, bufsize, "TLS Library: TinyDTLS - runtime %lu.%lu.%lu, "
1166 "libcoap built for %lu.%lu.%lu",
1167 (unsigned long)(tls_version->version >> 16),
1168 (unsigned long)((tls_version->version >> 8) & 0xff),
1169 (unsigned long)(tls_version->version & 0xff),
1170 (unsigned long)(tls_version->built_version >> 16),
1171 (unsigned long)((tls_version->built_version >> 8) & 0xff),
1172 (unsigned long)(tls_version->built_version & 0xff));
1173 break;
1175 switch (tls_version->version &0xf) {
1176 case 0:
1177 strcpy(beta, "-dev");
1178 break;
1179 case 0xf:
1180 strcpy(beta, "");
1181 break;
1182 default:
1183 strcpy(beta, "-beta");
1184 beta[5] = (tls_version->version &0xf) + '0';
1185 beta[6] = '\000';
1186 break;
1187 }
1188 sub[0] = ((tls_version->version >> 4) & 0xff) ?
1189 ((tls_version->version >> 4) & 0xff) + 'a' -1 : '\000';
1190 sub[1] = '\000';
1191 switch (tls_version->built_version &0xf) {
1192 case 0:
1193 strcpy(b_beta, "-dev");
1194 break;
1195 case 0xf:
1196 strcpy(b_beta, "");
1197 break;
1198 default:
1199 strcpy(b_beta, "-beta");
1200 b_beta[5] = (tls_version->built_version &0xf) + '0';
1201 b_beta[6] = '\000';
1202 break;
1203 }
1204 b_sub[0] = ((tls_version->built_version >> 4) & 0xff) ?
1205 ((tls_version->built_version >> 4) & 0xff) + 'a' -1 : '\000';
1206 b_sub[1] = '\000';
1207 snprintf(buffer, bufsize, "TLS Library: OpenSSL - runtime "
1208 "%lu.%lu.%lu%s%s, libcoap built for %lu.%lu.%lu%s%s",
1209 (unsigned long)(tls_version->version >> 28),
1210 (unsigned long)((tls_version->version >> 20) & 0xff),
1211 (unsigned long)((tls_version->version >> 12) & 0xff), sub, beta,
1212 (unsigned long)(tls_version->built_version >> 28),
1213 (unsigned long)((tls_version->built_version >> 20) & 0xff),
1214 (unsigned long)((tls_version->built_version >> 12) & 0xff),
1215 b_sub, b_beta);
1216 break;
1218 snprintf(buffer, bufsize, "TLS Library: GnuTLS - runtime %lu.%lu.%lu, "
1219 "libcoap built for %lu.%lu.%lu",
1220 (unsigned long)(tls_version->version >> 16),
1221 (unsigned long)((tls_version->version >> 8) & 0xff),
1222 (unsigned long)(tls_version->version & 0xff),
1223 (unsigned long)(tls_version->built_version >> 16),
1224 (unsigned long)((tls_version->built_version >> 8) & 0xff),
1225 (unsigned long)(tls_version->built_version & 0xff));
1226 break;
1228 snprintf(buffer, bufsize, "TLS Library: Mbed TLS - runtime %lu.%lu.%lu, "
1229 "libcoap built for %lu.%lu.%lu",
1230 (unsigned long)(tls_version->version >> 24),
1231 (unsigned long)((tls_version->version >> 16) & 0xff),
1232 (unsigned long)((tls_version->version >> 8) & 0xff),
1233 (unsigned long)(tls_version->built_version >> 24),
1234 (unsigned long)((tls_version->built_version >> 16) & 0xff),
1235 (unsigned long)((tls_version->built_version >> 8) & 0xff));
1236 break;
1238 snprintf(buffer, bufsize, "TLS Library: wolfSSL - runtime %lu.%lu.%lu, "
1239 "libcoap built for %lu.%lu.%lu",
1240 (unsigned long)(tls_version->version >> 24),
1241 (unsigned long)((tls_version->version >> 12) & 0xfff),
1242 (unsigned long)((tls_version->version >> 0) & 0xfff),
1243 (unsigned long)(tls_version->built_version >> 24),
1244 (unsigned long)((tls_version->built_version >> 12) & 0xfff),
1245 (unsigned long)((tls_version->built_version >> 0) & 0xfff));
1246 break;
1247 default:
1248 snprintf(buffer, bufsize, "Library type %d unknown", tls_version->type);
1249 break;
1250 }
1251 return buffer;
1252}
1253
1254char *
1255coap_string_tls_support(char *buffer, size_t bufsize) {
1256 const int have_tls = coap_tls_is_supported();
1257 const int have_dtls = coap_dtls_is_supported();
1258 const int have_psk = coap_dtls_psk_is_supported();
1259 const int have_pki = coap_dtls_pki_is_supported();
1260 const int have_pkcs11 = coap_dtls_pkcs11_is_supported();
1261 const int have_rpk = coap_dtls_rpk_is_supported();
1262 const int have_cid = coap_dtls_cid_is_supported();
1263 const int have_oscore = coap_oscore_is_supported();
1264 const int have_ws = coap_ws_is_supported();
1265
1266 if (have_dtls == 0 && have_tls == 0) {
1267 snprintf(buffer, bufsize, "(No DTLS or TLS support)");
1268 return buffer;
1269 }
1270 snprintf(buffer, bufsize,
1271 "(%sDTLS and %sTLS support; %sPSK, %sPKI, %sPKCS11, %sRPK and %sCID support)\n(%sOSCORE)\n(%sWebSockets)",
1272 have_dtls ? "" : "No ",
1273 have_tls ? "" : "no ",
1274 have_psk ? "" : "no ",
1275 have_pki ? "" : "no ",
1276 have_pkcs11 ? "" : "no ",
1277 have_rpk ? "" : "no ",
1278 have_cid ? "" : "no ",
1279 have_oscore ? "Have " : "No ",
1280 have_ws ? "Have " : "No ");
1281 return buffer;
1282}
1283
1285
1286void
1288 log_handler = handler;
1289}
1290
1291/* Visible to only this thread */
1292extern COAP_THREAD_LOCAL_VAR uint32_t thread_no;
1293/* Visible across all threads */
1294extern uint32_t max_thread_no;
1295
1296void
1297coap_log_impl(coap_log_t level, const char *format, ...) {
1298
1299#if COAP_THREAD_SAFE
1300 coap_mutex_lock(&m_log_impl);
1301#endif /* COAP_THREAD_SAFE */
1302
1303 if (log_handler) {
1304#if COAP_CONSTRAINED_STACK
1305 /* message can be protected by m_log_impl if needed */
1306 static char message[COAP_DEBUG_BUF_SIZE];
1307#else /* ! COAP_CONSTRAINED_STACK */
1308 char message[COAP_DEBUG_BUF_SIZE];
1309#endif /* ! COAP_CONSTRAINED_STACK */
1310 va_list ap;
1311 va_start(ap, format);
1312
1313#ifdef RIOT_VERSION
1314 flash_vsnprintf(message, sizeof(message), format, ap);
1315#else /* !RIOT_VERSION */
1316 vsnprintf(message, sizeof(message), format, ap);
1317#endif /* !RIOT_VERSION */
1318 va_end(ap);
1319 log_handler(level, message);
1320 } else {
1321 char timebuf[32];
1322 coap_tick_t now;
1323 va_list ap;
1324 FILE *log_fd;
1325 size_t len;
1326
1327 log_fd = level <= COAP_LOG_CRIT ? COAP_ERR_FD : COAP_DEBUG_FD;
1328
1329 coap_ticks(&now);
1330 len = print_timestamp(timebuf,sizeof(timebuf), now);
1331 if (len)
1332 fprintf(log_fd, "%.*s ", (int)len, timebuf);
1333
1334#if COAP_THREAD_NUM_LOGGING
1335 if (thread_no == 0) {
1337 }
1338 fprintf(log_fd, "%2d ", thread_no);
1339#endif /* COAP_THREAD_NUM_LOGGING */
1340
1341 fprintf(log_fd, "%s ", coap_log_level_desc(level));
1342
1343 va_start(ap, format);
1344#ifdef RIOT_VERSION
1345 flash_vfprintf(log_fd, format, ap);
1346#else /* !RIOT_VERSION */
1347 vfprintf(log_fd, format, ap);
1348#endif /* !RIOT_VERSION */
1349 va_end(ap);
1350 fflush(log_fd);
1351 }
1352
1353#if COAP_THREAD_SAFE
1354 coap_mutex_unlock(&m_log_impl);
1355#endif /* COAP_THREAD_SAFE */
1356}
1357
1363static uint16_t packet_loss_level = 0;
1364static int send_packet_count = 0;
1365
1366int
1367coap_debug_set_packet_loss(const char *loss_level) {
1368 const char *p = loss_level;
1369 char *end = NULL;
1370 int n = (int)strtol(p, &end, 10), i = 0;
1371 if (end == p || n < 0)
1372 return 0;
1373 if (*end == '%') {
1374 if (n > 100)
1375 n = 100;
1376 packet_loss_level = n * 65536 / 100;
1377 coap_log_debug("packet loss level set to %d%%\n", n);
1378 } else {
1379 if (n <= 0)
1380 return 0;
1381 while (i < 10) {
1383 if (*end == '-') {
1384 p = end + 1;
1385 n = (int)strtol(p, &end, 10);
1386 if (end == p || n <= 0)
1387 return 0;
1388 }
1389 packet_loss_intervals[i++].end = n;
1390 if (*end == 0)
1391 break;
1392 if (*end != ',')
1393 return 0;
1394 p = end + 1;
1395 n = (int)strtol(p, &end, 10);
1396 if (end == p || n <= 0)
1397 return 0;
1398 }
1399 if (i == 10)
1400 return 0;
1402 }
1404 return 1;
1405}
1406
1407int
1410 if (num_packet_loss_intervals > 0) {
1411 int i;
1412 for (i = 0; i < num_packet_loss_intervals; i++) {
1413 if (send_packet_count >= packet_loss_intervals[i].start &&
1415 coap_log_debug("Packet %u dropped\n", send_packet_count);
1416 return 0;
1417 }
1418 }
1419 }
1420 if (packet_loss_level > 0) {
1421 uint16_t r = 0;
1422 coap_prng_lkd((uint8_t *)&r, 2);
1423 if (r < packet_loss_level) {
1424 coap_log_debug("Packet %u dropped\n", send_packet_count);
1425 return 0;
1426 }
1427 }
1428 return 1;
1429}
1430
1431void
uint16_t coap_address_get_port(const coap_address_t *addr)
Returns the port from addr in host byte order.
static uint16_t packet_loss_level
static struct packet_num_interval packet_loss_intervals[10]
static int send_packet_count
int coap_debug_set_packet_loss(const char *loss_level)
Set the packet loss level for testing.
uint32_t max_thread_no
static coap_log_t maxlog
Definition coap_debug.c:61
static coap_log_handler_t log_handler
COAP_THREAD_LOCAL_VAR uint32_t thread_no
COAP_STATIC_INLINE int is_binary(int content_format)
Returns 1 if the given content_format is either unknown or known to carry binary data.
Definition coap_debug.c:749
static int enable_data_for_show_pdu
Definition coap_debug.c:69
static const char * msg_code_string(uint16_t c)
Returns a textual description of the method or response code.
Definition coap_debug.c:573
#define COAP_DO_SHOW_OUTPUT_LINE
Definition coap_debug.c:757
static size_t print_readable(const uint8_t *data, size_t len, unsigned char *result, size_t buflen, int encode_always)
Definition coap_debug.c:190
static const char * msg_option_string(uint8_t code, uint16_t option_type)
Returns a textual description of the option name.
Definition coap_debug.c:594
static size_t strnlen(const char *s, size_t maxlen)
A length-safe strlen() fake.
Definition coap_debug.c:181
static const char * loglevels[]
Definition coap_debug.c:114
COAP_STATIC_INLINE size_t print_timestamp(char *s, size_t len, coap_tick_t t)
Definition coap_debug.c:158
#define min(a, b)
Definition coap_debug.c:228
static const char * msg_type_string(uint16_t t)
Returns a textual description of the message type t.
Definition coap_debug.c:565
static int use_fprintf_for_show_pdu
Definition coap_debug.c:68
static int num_packet_loss_intervals
int coap_debug_send_packet(void)
Check to see whether a packet should be sent or not.
void coap_debug_reset(void)
Reset all the defined logging parameters.
static unsigned int print_content_format(unsigned int format_type, unsigned char *result, unsigned int buflen)
Definition coap_debug.c:689
#define INET6_ADDRSTRLEN
Definition coap_debug.c:232
Library specific build wrapper for coap_internal.h.
#define coap_mutex_unlock(a)
#define coap_mutex_lock(a)
uint8_t coap_opt_t
Use byte-oriented access methods here because sliding a complex struct coap_opt_t over the data buffe...
Definition coap_option.h:26
#define COAP_OPT_BLOCK_SZX(opt)
Returns the value of the SZX-field of a Block option opt.
Definition coap_block.h:90
#define COAP_OPT_BLOCK_MORE(opt)
Returns the value of the More-bit of a Block option opt.
Definition coap_block.h:86
unsigned int coap_opt_block_num(const coap_opt_t *block_opt)
Returns the value of field num in the given block option block_opt.
Definition coap_block.c:43
uint64_t coap_tick_t
This data type represents internal timer ticks with COAP_TICKS_PER_SECOND resolution.
Definition coap_time.h:143
coap_time_t coap_ticks_to_rt(coap_tick_t t)
Helper function that converts coap ticks to wallclock time.
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_prng_lkd(void *buf, size_t len)
Fills buf with len random bytes using the default pseudo random number generator.
Definition coap_prng.c:178
void coap_ticks(coap_tick_t *)
Returns the current value of an internal tick counter.
coap_tls_version_t * coap_get_tls_library_version(void)
Determine the type and version of the underlying (D)TLS library.
Definition coap_notls.c:100
@ COAP_TLS_LIBRARY_WOLFSSL
Using wolfSSL library.
Definition coap_dtls.h:76
@ COAP_TLS_LIBRARY_GNUTLS
Using GnuTLS library.
Definition coap_dtls.h:74
@ COAP_TLS_LIBRARY_TINYDTLS
Using TinyDTLS library.
Definition coap_dtls.h:72
@ COAP_TLS_LIBRARY_NOTLS
No DTLS library.
Definition coap_dtls.h:71
@ COAP_TLS_LIBRARY_OPENSSL
Using OpenSSL library.
Definition coap_dtls.h:73
@ COAP_TLS_LIBRARY_MBEDTLS
Using Mbed TLS library.
Definition coap_dtls.h:75
unsigned int coap_decode_var_bytes(const uint8_t *buf, size_t len)
Decodes multiple-length byte sequences.
Definition coap_encode.c:38
void coap_set_log_handler(coap_log_handler_t handler)
Add a custom log callback handler.
const char * coap_log_level_desc(coap_log_t level)
Get the current logging description.
Definition coap_debug.c:122
#define coap_log_debug(...)
Definition coap_debug.h:120
coap_log_t coap_get_log_level(void)
Get the current logging level.
Definition coap_debug.c:101
char * coap_string_tls_version(char *buffer, size_t bufsize)
Build a string containing the current (D)TLS library linked with and built for version.
#define coap_log_alert(...)
Definition coap_debug.h:84
void coap_show_pdu(coap_log_t level, const coap_pdu_t *pdu)
Display the contents of the specified pdu.
Definition coap_debug.c:784
void coap_enable_pdu_data_output(int enable_data)
Defines whether the data is to be output or not for the coap_show_pdu() function.
Definition coap_debug.c:96
char * coap_string_tls_support(char *buffer, size_t bufsize)
Build a string containing the current (D)TLS library support.
#define COAP_MAX_LOGGING_LEVEL
Definition coap_debug.h:42
#define COAP_ERR_FD
Used for output for COAP_LOG_CRIT to COAP_LOG_EMERG.
Definition coap_debug.h:38
coap_log_t
Logging type.
Definition coap_debug.h:50
const char * coap_package_version(void)
Get the library package version.
Definition coap_debug.c:77
void coap_set_log_level(coap_log_t level)
Sets the log level to the specified value.
Definition coap_debug.c:106
void coap_log_impl(coap_log_t level, const char *format,...)
Writes the given text to COAP_ERR_FD (for level <= COAP_LOG_CRIT) or COAP_DEBUG_FD (for level >= COAP...
size_t coap_print_addr(const coap_address_t *addr, unsigned char *buf, size_t len)
Print the address into the defined buffer.
Definition coap_debug.c:239
#define COAP_DEBUG_FD
Used for output for COAP_LOG_OSCORE to COAP_LOG_ERR.
Definition coap_debug.h:31
const char * coap_package_build(void)
Get the library package build.
Definition coap_debug.c:82
const char * coap_print_ip_addr(const coap_address_t *addr, char *buf, size_t len)
Print the IP address into the defined buffer.
Definition coap_debug.c:415
void coap_show_tls_version(coap_log_t level)
Display the current (D)TLS library linked with and built for version.
void coap_set_show_pdu_output(int use_fprintf)
Defines the output mode for the coap_show_pdu() function.
Definition coap_debug.c:91
#define coap_log_err(...)
Definition coap_debug.h:96
const char * coap_package_name(void)
Get the library package name.
Definition coap_debug.c:72
void(* coap_log_handler_t)(coap_log_t level, const char *message)
Logging callback handler definition.
Definition coap_debug.h:209
#define coap_log(level,...)
Logging function.
Definition coap_debug.h:284
@ COAP_LOG_INFO
Definition coap_debug.h:57
@ COAP_LOG_EMERG
Definition coap_debug.h:51
@ COAP_LOG_DEBUG
Definition coap_debug.h:58
@ COAP_LOG_CRIT
Definition coap_debug.h:53
@ COAP_LOG_ERR
Definition coap_debug.h:54
@ COAP_LOG_WARN
Definition coap_debug.h:55
coap_opt_t * coap_option_next(coap_opt_iterator_t *oi)
Updates the iterator oi to point to the next option.
uint32_t coap_opt_length(const coap_opt_t *opt)
Returns the length of the given option.
coap_opt_iterator_t * coap_option_iterator_init(const coap_pdu_t *pdu, coap_opt_iterator_t *oi, const coap_opt_filter_t *filter)
Initializes the given option iterator oi to point to the beginning of the pdu's option list.
#define COAP_OPT_ALL
Pre-defined filter that includes all options.
const uint8_t * coap_opt_value(const coap_opt_t *opt)
Returns a pointer to the value of the given option.
#define COAP_DEFAULT_VERSION
#define COAP_OPTION_HOP_LIMIT
Definition coap_pdu.h:133
#define COAP_OPTION_NORESPONSE
Definition coap_pdu.h:145
#define COAP_OPTION_URI_HOST
Definition coap_pdu.h:120
#define COAP_OPTION_IF_MATCH
Definition coap_pdu.h:119
#define COAP_MEDIATYPE_APPLICATION_COSE_MAC
Definition coap_pdu.h:231
#define COAP_MEDIATYPE_APPLICATION_SENSML_EXI
Definition coap_pdu.h:243
#define COAP_MEDIATYPE_APPLICATION_CWT
Definition coap_pdu.h:221
#define COAP_OPTION_BLOCK2
Definition coap_pdu.h:137
#define COAP_MEDIATYPE_APPLICATION_MB_CBOR_SEQ
Definition coap_pdu.h:254
#define COAP_MEDIATYPE_APPLICATION_COSE_ENCRYPT
Definition coap_pdu.h:229
#define COAP_MEDIATYPE_APPLICATION_RDF_XML
Definition coap_pdu.h:217
#define COAP_OPTION_CONTENT_FORMAT
Definition coap_pdu.h:128
#define COAP_SIGNALING_OPTION_ALTERNATIVE_ADDRESS
Definition coap_pdu.h:205
#define COAP_OPTION_SIZE2
Definition coap_pdu.h:139
#define COAP_MEDIATYPE_APPLICATION_SENSML_XML
Definition coap_pdu.h:245
#define COAP_MEDIATYPE_APPLICATION_COSE_SIGN
Definition coap_pdu.h:227
#define COAP_OPTION_BLOCK1
Definition coap_pdu.h:138
#define COAP_OPTION_Q_BLOCK1
Definition coap_pdu.h:135
#define COAP_OPTION_PROXY_SCHEME
Definition coap_pdu.h:142
#define COAP_MEDIATYPE_APPLICATION_COSE_KEY_SET
Definition coap_pdu.h:235
#define COAP_MEDIATYPE_APPLICATION_OSCORE
Definition coap_pdu.h:257
#define COAP_MEDIATYPE_APPLICATION_SENML_CBOR
Definition coap_pdu.h:240
#define COAP_OPTION_URI_QUERY
Definition coap_pdu.h:132
#define COAP_MEDIATYPE_APPLICATION_COSE_KEY
Definition coap_pdu.h:234
#define COAP_MEDIATYPE_APPLICATION_COSE_SIGN1
Definition coap_pdu.h:228
#define COAP_OPTION_IF_NONE_MATCH
Definition coap_pdu.h:122
#define COAP_MEDIATYPE_APPLICATION_OCTET_STREAM
Definition coap_pdu.h:216
#define COAP_OPTION_LOCATION_PATH
Definition coap_pdu.h:125
#define COAP_OPTION_URI_PATH
Definition coap_pdu.h:127
#define COAP_SIGNALING_OPTION_EXTENDED_TOKEN_LENGTH
Definition coap_pdu.h:199
#define COAP_MEDIATYPE_APPLICATION_SENML_EXI
Definition coap_pdu.h:242
#define COAP_OPTION_OSCORE
Definition coap_pdu.h:126
#define COAP_MEDIATYPE_APPLICATION_CBOR
Definition coap_pdu.h:220
#define COAP_OPTION_SIZE1
Definition coap_pdu.h:143
#define COAP_MEDIATYPE_APPLICATION_COSE_ENCRYPT0
Definition coap_pdu.h:230
#define COAP_SIGNALING_OPTION_BLOCK_WISE_TRANSFER
Definition coap_pdu.h:198
#define COAP_MEDIATYPE_TEXT_PLAIN
Definition coap_pdu.h:213
#define COAP_MEDIATYPE_APPLICATION_JSON
Definition coap_pdu.h:219
#define COAP_OPTION_LOCATION_QUERY
Definition coap_pdu.h:136
#define COAP_MEDIATYPE_APPLICATION_COSE_MAC0
Definition coap_pdu.h:232
#define COAP_MEDIATYPE_APPLICATION_ACE_CBOR
Definition coap_pdu.h:251
#define COAP_OPTION_Q_BLOCK2
Definition coap_pdu.h:140
#define COAP_MEDIATYPE_APPLICATION_SENML_JSON
Definition coap_pdu.h:238
#define COAP_MEDIATYPE_APPLICATION_EXI
Definition coap_pdu.h:218
#define COAP_SIGNALING_OPTION_CUSTODY
Definition coap_pdu.h:202
int coap_get_data(const coap_pdu_t *pdu, size_t *len, const uint8_t **data)
Retrieves the length and data pointer of specified PDU.
Definition coap_pdu.c:872
#define COAP_OPTION_RTAG
Definition coap_pdu.h:146
#define COAP_MEDIATYPE_APPLICATION_COAP_GROUP_JSON
Definition coap_pdu.h:224
#define COAP_OPTION_URI_PORT
Definition coap_pdu.h:124
#define COAP_MEDIATYPE_APPLICATION_SENML_XML
Definition coap_pdu.h:244
#define COAP_OPTION_ACCEPT
Definition coap_pdu.h:134
#define COAP_MEDIATYPE_APPLICATION_DOTS_CBOR
Definition coap_pdu.h:248
#define COAP_OPTION_MAXAGE
Definition coap_pdu.h:131
#define COAP_OPTION_ETAG
Definition coap_pdu.h:121
#define COAP_MEDIATYPE_APPLICATION_SENSML_JSON
Definition coap_pdu.h:239
#define COAP_OPTION_PROXY_URI
Definition coap_pdu.h:141
#define COAP_OPTION_OBSERVE
Definition coap_pdu.h:123
#define COAP_SIGNALING_OPTION_HOLD_OFF
Definition coap_pdu.h:206
#define COAP_OPTION_ECHO
Definition coap_pdu.h:144
#define COAP_MEDIATYPE_APPLICATION_LINK_FORMAT
Definition coap_pdu.h:214
#define COAP_SIGNALING_OPTION_BAD_CSM_OPTION
Definition coap_pdu.h:209
#define COAP_SIGNALING_OPTION_MAX_MESSAGE_SIZE
Definition coap_pdu.h:197
#define COAP_MEDIATYPE_APPLICATION_XML
Definition coap_pdu.h:215
#define COAP_MEDIATYPE_APPLICATION_SENSML_CBOR
Definition coap_pdu.h:241
@ COAP_PROTO_WS
Definition coap_pdu.h:318
@ COAP_PROTO_WSS
Definition coap_pdu.h:319
@ COAP_SIGNALING_CODE_ABORT
Definition coap_pdu.h:369
@ COAP_SIGNALING_CODE_CSM
Definition coap_pdu.h:365
@ COAP_SIGNALING_CODE_PING
Definition coap_pdu.h:366
@ COAP_SIGNALING_CODE_PONG
Definition coap_pdu.h:367
@ COAP_SIGNALING_CODE_RELEASE
Definition coap_pdu.h:368
@ COAP_MESSAGE_CON
Definition coap_pdu.h:69
@ COAP_SIGNALING_RELEASE
Definition coap_pdu.h:192
@ COAP_SIGNALING_CSM
Definition coap_pdu.h:189
@ COAP_SIGNALING_PONG
Definition coap_pdu.h:191
@ COAP_SIGNALING_PING
Definition coap_pdu.h:190
@ COAP_SIGNALING_ABORT
Definition coap_pdu.h:193
#define COAP_PROTO_NOT_RELIABLE(p)
int coap_dtls_cid_is_supported(void)
Check whether (D)TLS CID is available.
Definition coap_notls.c:86
int coap_dtls_psk_is_supported(void)
Check whether (D)TLS PSK is available.
Definition coap_notls.c:50
int coap_tls_is_supported(void)
Check whether TLS is available.
Definition coap_notls.c:41
int coap_ws_is_supported(void)
Check whether WebSockets is available.
Definition coap_ws.c:933
int coap_oscore_is_supported(void)
Check whether OSCORE is available.
int coap_dtls_is_supported(void)
Check whether DTLS is available.
Definition coap_notls.c:36
int coap_dtls_pki_is_supported(void)
Check whether (D)TLS PKI is available.
Definition coap_notls.c:59
int coap_dtls_rpk_is_supported(void)
Check whether (D)TLS RPK is available.
Definition coap_notls.c:77
int coap_dtls_pkcs11_is_supported(void)
Check whether (D)TLS PKCS11 is available.
Definition coap_notls.c:68
#define COAP_STATIC_INLINE
Definition libcoap.h:53
#define COAP_THREAD_LOCAL_VAR
Definition libcoap.h:80
Multi-purpose address abstraction.
struct sockaddr_in sin
struct coap_sockaddr_un cun
struct sockaddr_in6 sin6
struct sockaddr sa
union coap_address_t::@0 addr
size_t length
length of binary data
Definition coap_str.h:65
const uint8_t * s
read-only binary data
Definition coap_str.h:66
Iterator to run through PDU options.
coap_option_num_t number
decoded option number
structure for CoAP PDUs
coap_pdu_code_t code
request method (value 1–31) or response code (value 64-255)
coap_bin_const_t actual_token
Actual token in pdu.
coap_mid_t mid
message id, if any, in regular host byte order
coap_session_t * session
Session responsible for PDU or NULL.
coap_pdu_type_t type
message type
coap_proto_t proto
protocol used
char sun_path[COAP_UNIX_PATH_MAX]
The structure used for returning the underlying (D)TLS library information.
Definition coap_dtls.h:83
uint64_t built_version
(D)TLS Built against Library Version
Definition coap_dtls.h:86
coap_tls_library_t type
Library type.
Definition coap_dtls.h:85
uint64_t version
(D)TLS runtime Library Version
Definition coap_dtls.h:84