18#if COAP_SERVER_SUPPORT
21#ifdef COAP_EPOLL_SUPPORT
23#include <sys/timerfd.h>
27#define min(a,b) ((a) < (b) ? (a) : (b))
38#define PRINT_WITH_OFFSET(Buf,Offset,Char) \
39 if ((Offset) == 0) { \
40 (*(Buf)++) = (Char); \
48#define PRINT_COND_WITH_OFFSET(Buf,Bufend,Offset,Char,Result) { \
49 if ((Buf) < (Bufend)) { \
50 PRINT_WITH_OFFSET(Buf,Offset,Char); \
60#define COPY_COND_WITH_OFFSET(Buf,Bufend,Offset,Str,Length,Result) { \
62 for (i = 0; i < (Length); i++) { \
63 PRINT_COND_WITH_OFFSET((Buf), (Bufend), (Offset), (Str)[i], (Result)); \
69 int match_prefix,
int match_substring) {
76 if (match_substring) {
77 const uint8_t *next_token = text->
s;
78 size_t remaining_length = text->
length;
79 while (remaining_length) {
81 const uint8_t *token = next_token;
82 next_token = (
unsigned char *)memchr(token,
' ', remaining_length);
85 token_length = next_token - token;
86 remaining_length -= (token_length + 1);
89 token_length = remaining_length;
93 if ((match_prefix || pattern->
length == token_length) &&
94 memcmp(token, pattern->
s, pattern->
length) == 0)
100 return (match_prefix || pattern->
length == text->
length) &&
101 memcmp(text->
s, pattern->
s, pattern->
length) == 0;
106 size_t *buflen,
size_t offset,
110 result = coap_print_wellknown_lkd(context, buf, buflen, offset, query_filter);
121coap_print_wellknown_lkd(
coap_context_t *context,
unsigned char *buf,
122 size_t *buflen,
size_t offset,
125 unsigned char *p = buf;
126 const uint8_t *bufend = buf + *buflen;
127 size_t left, written = 0;
129 const size_t old_offset = offset;
130 int subsequent_resource = 0;
131#ifdef WITHOUT_QUERY_FILTER
136#define MATCH_URI 0x01
137#define MATCH_PREFIX 0x02
138#define MATCH_SUBSTRING 0x04
140 {2, (
const uint8_t *)
"rt"},
141 {2, (
const uint8_t *)
"if"},
142 {3, (
const uint8_t *)
"rel"},
148#ifndef WITHOUT_QUERY_FILTER
151 resource_param.
s = query_filter->
s;
153 resource_param.
s[resource_param.
length] !=
'=')
158 if (resource_param.
length == 4 &&
159 memcmp(resource_param.
s,
"href", 4) == 0)
162 for (rt_attributes = _rt_attributes; rt_attributes->
s; rt_attributes++) {
164 memcmp(resource_param.
s, rt_attributes->
s, rt_attributes->
length) == 0) {
165 flags |= MATCH_SUBSTRING;
172 query_filter->
s + resource_param.
length + 1;
174 assert((resource_param.
length + 1) <= query_filter->
length);
175 query_pattern.length =
178 if (query_pattern.length &&
179 (query_pattern.s[0] ==
'/') && ((flags & MATCH_URI) == MATCH_URI)) {
181 query_pattern.length--;
184 if (query_pattern.length &&
185 query_pattern.s[query_pattern.length-1] ==
'*') {
186 query_pattern.length--;
187 flags |= MATCH_PREFIX;
193 RESOURCES_ITER(context->resources, r) {
202#ifndef WITHOUT_QUERY_FILTER
203 if (resource_param.
length) {
205 if (flags & MATCH_URI) {
206 if (!match(r->uri_path, &query_pattern, (flags & MATCH_PREFIX) != 0,
207 (flags & MATCH_SUBSTRING) != 0))
213 if (!attr || !attr->value)
215 unquoted_val = *attr->value;
216 if (attr->value->s[0] ==
'"') {
220 if (!(match(&unquoted_val, &query_pattern,
221 (flags & MATCH_PREFIX) != 0,
222 (flags & MATCH_SUBSTRING) != 0)))
228 if (!subsequent_resource) {
229 subsequent_resource = 1;
231 PRINT_COND_WITH_OFFSET(p, bufend, offset,
',', written);
256 if (result + old_offset - offset < *buflen) {
282 }
else if (!uri_path) {
288 r->uri_path = uri_path;
299static const uint8_t coap_unknown_resource_uri[] =
314 r->uri_path =
coap_new_str_const(coap_unknown_resource_uri,
sizeof(coap_unknown_resource_uri)-1);
315 r->flags = flags & ~COAP_RESOURCE_FLAGS_RELEASE_URI;
329static const uint8_t coap_proxy_resource_uri[] =
334 size_t host_name_count,
335 const char *host_name_list[],
int flags) {
347 r->uri_path =
coap_new_str_const(coap_proxy_resource_uri,
sizeof(coap_proxy_resource_uri)-1);
349 for (i = 0; i < (
sizeof(r->handler) /
sizeof(r->handler[0])); i++) {
350 r->handler[i] = handler;
352 if (host_name_count) {
355 if (r->proxy_name_list) {
356 for (i = 0; i < host_name_count; i++) {
357 r->proxy_name_list[i] =
359 strlen(host_name_list[i]));
360 if (!r->proxy_name_list[i]) {
361 coap_log_err(
"coap_resource_proxy_uri_init: unable to add host name\n");
364 r->proxy_name_list =
NULL;
369 r->proxy_name_count = i;
372 r->flags = flags & ~COAP_RESOURCE_FLAGS_RELEASE_URI;
374 coap_log_debug(
"coap_resource_proxy_uri_init2: no memory left\n");
382 size_t host_name_count,
const char *host_name_list[]) {
387static const uint8_t coap_rev_proxy_resource_uri[] =
401 r->is_reverse_proxy = 1;
404 sizeof(coap_rev_proxy_resource_uri)-1);
405 r->flags = flags & ~COAP_RESOURCE_FLAGS_RELEASE_URI;
428 if (!resource || !name)
449 LL_PREPEND(resource->link_attr, attr);
462 if (!resource || !name)
465 LL_FOREACH(resource->link_attr, attr) {
466 if (attr->name->length == name->
length &&
467 memcmp(attr->name->s, name->
s, name->
length) == 0)
494 coap_deleting_resource_t deleting);
497coap_free_resource(
coap_resource_t *resource, coap_deleting_resource_t deleting) {
504 context = resource->context;
506 if (!context->observe_no_clear) {
507 coap_resource_notify_observers_lkd(resource, deleting);
508 coap_notify_observers(context, resource, deleting);
511 if (context->resource_deleted_cb)
514 context->observe_user_data));
516 if (context->release_userdata_cb && resource->user_data) {
522 LL_FOREACH_SAFE(resource->link_attr, attr, tmp) coap_delete_attr(attr);
528 LL_FOREACH_SAFE(resource->subscribers, obs, otmp) {
529 coap_delete_observer_internal(resource, obs->session, obs);
531 if (resource->proxy_name_count && resource->proxy_name_list) {
534 for (i = 0; i < resource->proxy_name_count; i++) {
546 coap_add_resource_lkd(context, resource);
553 if (resource->is_unknown) {
554 if (context->unknown_resource)
555 coap_free_resource(context->unknown_resource, COAP_DELETING_RESOURCE);
556 context->unknown_resource = resource;
557 }
else if (resource->is_proxy_uri) {
558 if (context->proxy_uri_resource)
559 coap_free_resource(context->proxy_uri_resource, COAP_DELETING_RESOURCE);
560 context->proxy_uri_resource = resource;
566 coap_log_warn(
"coap_add_resource: Duplicate uri_path '%*.*s', old resource deleted\n",
567 (
int)resource->uri_path->length, (
int)resource->uri_path->length,
568 resource->uri_path->s);
569 coap_delete_resource_lkd(r);
571 RESOURCES_ADD(context->resources, resource);
572#if COAP_WITH_OBSERVE_PERSIST
573 if (context->unknown_pdu && context->dyn_resource_save_file &&
574 context->dyn_resource_added_cb && resource->observable) {
577 raw_packet.
s = context->unknown_pdu->token -
578 context->unknown_pdu->hdr_size;
579 raw_packet.
length = context->unknown_pdu->used_size +
580 context->unknown_pdu->hdr_size;
584 context->observe_user_data));
588 assert(resource->context ==
NULL);
589 resource->context = context;
601 ret = coap_delete_resource_lkd(resource);
612 coap_deleting_resource_t deleting;
617 context = resource->context;
624 if (context && context->context_going_away) {
625 deleting = COAP_DELETING_RESOURCE_ON_EXIT;
627 deleting = COAP_DELETING_RESOURCE;
629 if (resource->is_unknown) {
630 if (context && context->unknown_resource == resource) {
631 context->unknown_resource =
NULL;
633 }
else if (resource->is_proxy_uri) {
634 if (context && context->proxy_uri_resource == resource) {
635 context->proxy_uri_resource =
NULL;
637 }
else if (context) {
639 RESOURCES_DELETE(context->resources, resource);
641 if (resource->is_dynamic) {
649 coap_free_resource(resource, deleting);
659 RESOURCE_ITER_SAFE(context->resources, r, tmp) {
660 coap_delete_resource_lkd(r);
663 context->resources =
NULL;
665 if (context->unknown_resource) {
666 coap_delete_resource_lkd(context->unknown_resource);
667 context->unknown_resource =
NULL;
669 if (context->proxy_uri_resource) {
670 coap_delete_resource_lkd(context->proxy_uri_resource);
671 context->proxy_uri_resource =
NULL;
680 result = coap_get_resource_from_uri_path_lkd(context, uri_path);
693 RESOURCES_FIND(context->resources, uri_path, result);
700 unsigned char *buf,
size_t *len,
size_t *offset) {
701 unsigned char *p = buf;
702 const uint8_t *bufend = buf + *len;
706 const size_t old_offset = *offset;
709 PRINT_COND_WITH_OFFSET(p, bufend, *offset,
'<', *len);
710 PRINT_COND_WITH_OFFSET(p, bufend, *offset,
'/', *len);
712 COPY_COND_WITH_OFFSET(p, bufend, *offset,
713 resource->uri_path->s, resource->uri_path->length, *len);
715 PRINT_COND_WITH_OFFSET(p, bufend, *offset,
'>', *len);
717 LL_FOREACH(resource->link_attr, attr) {
719 PRINT_COND_WITH_OFFSET(p, bufend, *offset,
';', *len);
721 COPY_COND_WITH_OFFSET(p, bufend, *offset,
722 attr->name->s, attr->name->length, *len);
724 if (attr->value && attr->value->s) {
725 PRINT_COND_WITH_OFFSET(p, bufend, *offset,
'=', *len);
727 COPY_COND_WITH_OFFSET(p, bufend, *offset,
728 attr->value->s, attr->value->length, *len);
732 if (resource->observable) {
733 COPY_COND_WITH_OFFSET(p, bufend, *offset,
";obs", 4, *len);
736#if COAP_OSCORE_SUPPORT
739 COPY_COND_WITH_OFFSET(p, bufend, *offset,
";osc", 4, *len);
750 if (result + old_offset - *offset < *len) {
769 assert(method > 0 && (
size_t)(method-1) <
771 resource->handler[method-1] = handler;
782 LL_FOREACH(resource->subscribers, s) {
783 if (s->session == session &&
799 LL_FOREACH(resource->subscribers, s) {
800 if (s->session == session
825 s = coap_find_observer(resource, session, token);
834 cache_ignore_options,
835 sizeof(cache_ignore_options)/
sizeof(cache_ignore_options[0]));
837 s = coap_find_observer_cache_key(resource, session, cache_key);
840 coap_delete_observer(resource, session, &s->pdu->actual_token);
852#if (COAP_RESOURCE_MAX_SUBSCRIBER > 0)
853 uint32_t subscriber_count = 0;
854 LL_COUNT(resource->subscribers, s, subscriber_count);
855 if (subscriber_count >= COAP_RESOURCE_MAX_SUBSCRIBER) {
868 coap_subscription_init(s);
871 if (s->pdu ==
NULL) {
878 s->pdu->max_size = 0;
881 if (cache_key ==
NULL) {
884 cache_ignore_options,
885 sizeof(cache_ignore_options)/
sizeof(cache_ignore_options[0]));
886 if (cache_key ==
NULL) {
893 s->cache_key = cache_key;
895 session->ref_subscriptions++;
898 LL_PREPEND(resource->subscribers, s);
900 coap_log_debug(
"create new subscription %p key 0x%02x%02x%02x%02x\n",
901 (
void *)s, s->cache_key->key[0], s->cache_key->key[1],
902 s->cache_key->key[2], s->cache_key->key[3]);
908#if COAP_OSCORE_SUPPORT
911 if (session->recipient_ctx && session->recipient_ctx->recipient_id) {
925 uint8_t info_buffer[60];
926 uint8_t *info_buf = info_buffer;
927 size_t info_len =
sizeof(info_buffer);
934 session->recipient_ctx->recipient_id->s,
935 session->recipient_ctx->recipient_id->length);
936 if (session->recipient_ctx->osc_ctx &&
937 session->recipient_ctx->osc_ctx->id_context) {
940 session->recipient_ctx->osc_ctx->id_context->s,
941 session->recipient_ctx->osc_ctx->id_context->length);
947 if (association->
aad) {
963 if (association->
nonce) {
975 if (ret >
sizeof(info_buffer)) {
978 ret =
sizeof(info_buffer);
985 memcpy(s->pdu->token - request->
hdr_size,
987 raw_packet.
s = s->pdu->token - request->
hdr_size;
990 &session->endpoint->bind_addr,
994 session->
context->observe_user_data));
995#if COAP_OSCORE_SUPPORT
999 if (resource->context->track_observe_value_cb) {
1001 coap_lock_callback(resource->context->track_observe_value_cb(resource->context,resource->uri_path,
1003 resource->context->observe_user_data));
1014 RESOURCES_ITER(context->resources, r) {
1015 s = coap_find_observer(r, session, token);
1029 char outbuf[2 * 8 + 1] =
"";
1034 for (i = 0; i < s->pdu->actual_token.length; i++) {
1035 size_t size = strlen(outbuf);
1037 snprintf(&outbuf[size],
sizeof(outbuf)-size,
"%02x",
1038 s->pdu->actual_token.s[i]);
1042 coap_log_debug(
"removed subscription '/%*.*s%s%*.*s' (%p) with token '%s' key 0x%02x%02x%02x%02x\n",
1043 uri_path ? (
int)uri_path->
length : 0, uri_path ? (int)uri_path->length : 0,
1044 uri_path ? (char *)uri_path->s :
"",
1045 uri_query ?
"?" :
"",
1046 uri_query ? (int)uri_query->length : 0, uri_query ? (int)uri_query->length : 0,
1047 uri_query ? (char *)uri_query->s :
"",
1048 (void *)s, outbuf, s->cache_key->key[0], s->cache_key->key[1],
1049 s->cache_key->key[2], s-> cache_key->key[3]);
1053 if (session->
context->observe_deleted_cb)
1055 session->
context->observe_user_data));
1057 if (resource->subscribers) {
1058 LL_DELETE(resource->subscribers, s);
1059 assert(session->ref_subscriptions > 0);
1060 session->ref_subscriptions--;
1075 s = coap_find_observer(resource, session, token);
1077 coap_delete_observer_internal(resource, session, s);
1088 s = coap_find_observer(resource, session, token);
1098 cache_ignore_options,
1099 sizeof(cache_ignore_options)/
sizeof(cache_ignore_options[0]));
1101 s = coap_find_observer_cache_key(resource, session, cache_key);
1104 ret = coap_delete_observer(resource, session, &s->pdu->actual_token);
1109 coap_delete_observer_internal(resource, session, s);
1117 RESOURCES_ITER(context->resources, resource) {
1119 LL_FOREACH_SAFE(resource->subscribers, s, tmp) {
1120 if (s->session == session) {
1121 if (context->observe_deleted_cb)
1122 coap_lock_callback(context->observe_deleted_cb(session, s, context->observe_user_data));
1123 assert(resource->subscribers);
1124 LL_DELETE(resource->subscribers, s);
1136 coap_deleting_resource_t deleting) {
1147 if (r->observable && (r->dirty || r->partiallydirty)) {
1148 if (r->list_being_traversed)
1150 r->list_being_traversed = 1;
1152 coap_resource_reference_lkd(r);
1154 r->partiallydirty = 0;
1156 LL_FOREACH_SAFE(r->subscribers, obs, otmp) {
1161 if ((r->dirty == 0 && obs->dirty == 0) || obs->session->is_rate_limiting) {
1166 context->observe_pending = 1;
1174 obs_session = obs->session;
1179 if (obs->session->con_active >=
COAP_NSTART(obs->session) &&
1181 (obs->non_cnt >= COAP_OBS_MAX_NON))) {
1186 if (obs->session->lg_xmit && obs->session->lg_xmit->last_all_sent == 0 &&
1187 obs->session->lg_xmit->last_obs &&
1198 coap_log_debug(
"coap_check_notify: pdu init failed, resource stays "
1199 "partially dirty\n");
1200 goto next_one_fail_no_pending;
1204 obs->pdu->actual_token.s)) {
1205 coap_log_debug(
"coap_check_notify: cannot add token, resource stays "
1206 "partially dirty\n");
1208 goto next_one_fail_no_pending;
1216 obs->non_cnt < COAP_OBS_MAX_NON)) {
1222 case COAP_NOT_DELETING_RESOURCE:
1238#if COAP_Q_BLOCK_SUPPORT
1251 h = r->handler[obs->pdu->code - 1];
1258 (
int)r->uri_path->length, (
int)r->uri_path->length,
1269 r->list_being_traversed = 0;
1270 coap_resource_release_lkd(r);
1274 h(r, obs->session, obs->pdu, query, response),
1280 r->list_being_traversed = 0;
1281 coap_resource_release_lkd(r);
1287 coap_log_warn(
"handle_request: Invalid PDU response code (%d.%02d)\n",
1289 response->
code & 0x1f);
1294 r->list_being_traversed = 0;
1295 coap_resource_release_lkd(r);
1306 coap_delete_observer(r, obs_session, &obs_pdu->
actual_token);
1310 case COAP_DELETING_RESOURCE_ON_EXIT:
1319 case COAP_DELETING_RESOURCE:
1333 LL_FOREACH(r->subscribers, s) {
1349#if COAP_Q_BLOCK_SUPPORT
1355 mid = coap_send_q_block2(obs_session, r, query, obs_pdu->
code,
1356 block, response, 1);
1364#if COAP_Q_BLOCK_SUPPORT
1368 coap_log_debug(
"* %s: coap_check_notify: sending failed, resource stays "
1376 LL_FOREACH(r->subscribers, s) {
1386 r->partiallydirty = 1;
1391 context->observe_pending = 1;
1392next_one_fail_no_pending:
1393 r->partiallydirty = 1;
1400 r->list_being_traversed = 0;
1402 coap_resource_release_lkd(r);
1415 ret = coap_resource_notify_observers_lkd(r, COAP_NOT_DELETING_RESOURCE);
1427 ret = coap_resource_notify_observers_lkd(r, COAP_NOT_DELETING_RESOURCE);
1434 coap_deleting_resource_t deleting) {
1438 if (!r->subscribers)
1443 r->observe = (r->observe + 1) & 0xFFFFFF;
1447 if (r->context->track_observe_value_cb) {
1449 if ((r->observe % r->context->observe_save_freq) == 0)
1452 r->context->observe_user_data));
1455 coap_notify_observers(r->context, r, deleting);
1461 resource->flags = (resource->flags &
1468 resource->user_data = data;
1473 return resource->user_data;
1479 context->release_userdata_cb = callback;
1484 if (resource->is_unknown || resource->is_proxy_uri) {
1486 coap_log_debug(
"coap_resource_set_get_observable: Not supported for Unknown or Proxy URIs\n");
1487 resource->observable = 0;
1489 resource->observable = mode ? 1 : 0;
1496 return resource->uri_path;
1503 coap_check_notify_lkd(context);
1511 if (context->observe_pending) {
1512 context->observe_pending = 0;
1513 RESOURCES_ITER(context->resources, r) {
1514 coap_notify_observers(context, r, COAP_NOT_DELETING_RESOURCE);
1521 uint32_t start_observe_no) {
1525 resource->observe = start_observe_no & 0xffffff;
1545 LL_FOREACH_SAFE(resource->subscribers, obs, otmp) {
1546 if (obs->session == session &&
1551 if (obs->fail_cnt >= COAP_OBS_MAX_FAIL) {
1553 &obs->pdu->actual_token);
1554 coap_delete_observer(resource, session, token);
1566 RESOURCES_ITER(context->resources, r) {
1567 coap_remove_failed_observers(context, r, session, token);
int coap_is_af_unix(const coap_address_t *a)
Checks if given address a denotes a AF_UNIX address.
struct coap_cache_key_t coap_cache_key_t
struct coap_attr_t coap_attr_t
struct coap_subscription_t coap_subscription_t
struct coap_resource_t coap_resource_t
Library specific build wrapper for coap_internal.h.
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_DEPRECATED COAP_API int coap_resource_set_dirty(coap_resource_t *r, const coap_string_t *query)
void coap_check_code_lg_xmit(const coap_session_t *session, const coap_pdu_t *request, coap_pdu_t *response, const coap_resource_t *resource, const coap_string_t *query)
The function checks that the code in a newly formed lg_xmit created by coap_add_data_large_response_l...
int coap_get_block_b(const coap_session_t *session, const coap_pdu_t *pdu, coap_option_num_t number, coap_block_b_t *block)
Initializes block from pdu.
void coap_delete_cache_key(coap_cache_key_t *cache_key)
Delete the cache-key.
coap_cache_key_t * coap_cache_derive_key_w_ignore(const coap_session_t *session, const coap_pdu_t *pdu, coap_cache_session_based_t session_based, const uint16_t *ignore_options, size_t ignore_count)
Calculates a cache-key for the given CoAP PDU.
@ COAP_CACHE_IS_SESSION_BASED
uint64_t coap_tick_t
This data type represents internal timer ticks with COAP_TICKS_PER_SECOND resolution.
#define COAP_TICKS_PER_SECOND
Use ms resolution on POSIX systems.
COAP_API coap_resource_t * coap_get_resource_from_uri_path(coap_context_t *context, coap_str_const_t *uri_path)
Returns the resource identified by the unique string uri_path.
#define COAP_RESOURCE_FLAGS_NOTIFY_NON
Observe Notifications will be sent non-confirmable by default.
coap_resource_t * coap_resource_proxy_uri_init(coap_method_handler_t handler, size_t host_name_count, const char *host_name_list[])
Creates a new resource object for handling proxy URIs.
coap_attr_t * coap_add_attr(coap_resource_t *resource, coap_str_const_t *name, coap_str_const_t *value, int flags)
Registers a new attribute with the given resource.
#define COAP_ATTR_FLAGS_RELEASE_VALUE
coap_print_status_t coap_print_link(const coap_resource_t *resource, unsigned char *buf, size_t *len, size_t *offset)
Writes a description of this resource in link-format to given text buffer.
coap_resource_t * coap_resource_proxy_uri_init2(coap_method_handler_t handler, size_t host_name_count, const char *host_name_list[], int flags)
Creates a new resource object for handling proxy URIs with configurable control over multicast reques...
#define COAP_RESOURCE_FLAGS_NOTIFY_NON_ALWAYS
Observe Notifications will always be sent non-confirmable.
#define COAP_RESOURCE_HANDLE_WELLKNOWN_CORE
Define this when invoking coap_resource_unknown_init2() if .well-known/core is to be passed to the un...
#define COAP_ATTR_FLAGS_RELEASE_NAME
void coap_resource_set_mode(coap_resource_t *resource, int mode)
Sets the notification message type of resource resource to given mode.
#define COAP_RESOURCE_SAFE_REQUEST_HANDLER
Don't lock this resource when calling app call-back handler for requests as handler will not be manip...
#define COAP_PRINT_STATUS_TRUNC
void(* coap_method_handler_t)(coap_resource_t *resource, coap_session_t *session, const coap_pdu_t *request, const coap_string_t *query, coap_pdu_t *response)
Definition of message handler function.
void coap_resource_release_userdata_handler(coap_context_t *context, coap_resource_release_userdata_handler_t callback)
Defines the context wide callback to use to when the resource is deleted to release the data held in ...
#define COAP_RESOURCE_FLAGS_OSCORE_ONLY
Define this resource as an OSCORE enabled access only.
COAP_API coap_print_status_t coap_print_wellknown(coap_context_t *context, unsigned char *buf, size_t *buflen, size_t offset, const coap_string_t *query_filter)
Prints the names of all known resources for context to buf.
#define COAP_RESOURCE_FLAGS_NOTIFY_CON
Observe Notifications will be sent confirmable.
coap_resource_t * coap_resource_reverse_proxy_init(coap_method_handler_t handler, int flags)
Creates a new resource object for the reverse-proxy resource handler with control over multicast requ...
COAP_API int coap_delete_resource(coap_context_t *context, coap_resource_t *resource)
Deletes a resource identified by resource.
void coap_register_handler(coap_resource_t *resource, coap_request_t method, coap_method_handler_t handler)
Registers the specified handler as message handler for the request type method.
#define COAP_PRINT_STATUS_MAX
void(* coap_resource_release_userdata_handler_t)(void *user_data)
Definition of release resource user_data callback function.
void coap_register_request_handler(coap_resource_t *resource, coap_request_t method, coap_method_handler_t handler)
Registers the specified handler as message handler for the request type method.
coap_resource_t * coap_resource_init(coap_str_const_t *uri_path, int flags)
Creates a new resource object and initializes the link field to the string uri_path.
void coap_resource_set_userdata(coap_resource_t *resource, void *data)
Sets the user_data.
uint32_t coap_print_status_t
Status word to encode the result of conditional print or copy operations such as coap_print_link().
#define COAP_PRINT_STATUS_ERROR
coap_str_const_t * coap_resource_get_uri_path(coap_resource_t *resource)
Get the uri_path from a resource.
coap_resource_t * coap_resource_unknown_init2(coap_method_handler_t put_handler, int flags)
Creates a new resource object for the unknown resource handler with support for PUT and configurable ...
coap_str_const_t * coap_attr_get_value(coap_attr_t *attribute)
Returns attribute's value.
#define COAP_PRINT_OUTPUT_LENGTH(v)
coap_attr_t * coap_find_attr(coap_resource_t *resource, coap_str_const_t *name)
Returns resource's coap_attr_t object with given name if found, NULL otherwise.
void * coap_resource_get_userdata(coap_resource_t *resource)
Gets the user_data.
COAP_API void coap_add_resource(coap_context_t *context, coap_resource_t *resource)
Registers the given resource for context.
#define COAP_RESOURCE_FLAGS_RELEASE_URI
The URI passed to coap_resource_init() is free'd by coap_delete_resource().
coap_resource_t * coap_resource_unknown_init(coap_method_handler_t put_handler)
Creates a new resource object for the unknown resource handler with support for PUT.
#define COAP_RESOURCE_HIDE_WELLKNOWN_CORE
Hide this resource from .well-known/core.
uint16_t coap_new_message_id_lkd(coap_session_t *session)
Returns a new message id and updates session->tx_mid accordingly.
coap_mid_t coap_send_internal(coap_session_t *session, coap_pdu_t *pdu, coap_pdu_t *request_pdu)
Sends a CoAP message to given peer.
int coap_check_code_class(coap_session_t *session, coap_pdu_t *pdu)
Check whether the pdu contains a valid code class.
void coap_cancel_all_messages(coap_context_t *context, coap_session_t *session, coap_bin_const_t *token)
Cancels all outstanding messages for session session that have the specified token.
void coap_ticks(coap_tick_t *t)
Returns the current value of an internal tick counter.
unsigned int coap_encode_var_safe(uint8_t *buf, size_t length, unsigned int val)
Encodes multiple-length byte sequences.
#define coap_lock_specific_callback_release(lock, func, failed)
Dummy for no thread-safe code.
#define coap_lock_callback(func)
Dummy for no thread-safe code.
#define coap_lock_init(lock)
Dummy for no thread-safe code.
#define coap_lock_unlock()
Dummy for no thread-safe code.
#define coap_lock_check_locked()
Dummy for no thread-safe code.
#define coap_lock_callback_release(func, failed)
Dummy for no thread-safe code.
#define coap_lock_lock(failed)
Dummy for no thread-safe code.
#define coap_log_debug(...)
coap_log_t coap_get_log_level(void)
Get the current logging level.
void coap_show_pdu(coap_log_t level, const coap_pdu_t *pdu)
Display the contents of the specified pdu.
const char * coap_session_str(const coap_session_t *session)
Get session description.
#define coap_log_warn(...)
#define coap_log_err(...)
void coap_persist_set_observe_num(coap_resource_t *resource, uint32_t observe_num)
Sets the current observe number value.
void coap_resource_set_get_observable(coap_resource_t *resource, int mode)
Set whether a resource is observable.
COAP_API void coap_check_notify(coap_context_t *context)
Checks all known resources to see if they are dirty and then notifies subscribed observers.
COAP_API int coap_resource_notify_observers(coap_resource_t *resource, const coap_string_t *query)
Initiate the sending of an Observe packet for all observers of resource, optionally matching query if...
size_t oscore_cbor_put_nil(uint8_t **buffer, size_t *buf_size)
size_t oscore_cbor_put_bytes(uint8_t **buffer, size_t *buf_size, const uint8_t *bytes, size_t bytes_len)
size_t oscore_cbor_put_array(uint8_t **buffer, size_t *buf_size, size_t elements)
oscore_association_t * oscore_find_association(coap_session_t *session, coap_bin_const_t *token)
coap_pdu_t * coap_pdu_reference_lkd(coap_pdu_t *pdu)
Increment reference counter on a pdu to stop it prematurely getting freed off when coap_delete_pdu() ...
void coap_delete_pdu_lkd(coap_pdu_t *pdu)
Dispose of an CoAP PDU and free off associated storage.
int coap_remove_option(coap_pdu_t *pdu, coap_option_num_t number)
Removes (first) option of given number from the pdu.
coap_pdu_t * coap_pdu_duplicate_lkd(const coap_pdu_t *old_pdu, coap_session_t *session, size_t token_length, const uint8_t *token, coap_opt_filter_t *drop_options, coap_bool_t expand_opt_abb)
Duplicate an existing PDU.
COAP_STATIC_INLINE void coap_pdu_release_lkd(coap_pdu_t *pdu)
size_t coap_add_option_internal(coap_pdu_t *pdu, coap_option_num_t number, size_t len, const uint8_t *data)
Adds option of given number to pdu that is passed as first parameter.
#define COAP_OPTION_BLOCK2
int coap_mid_t
coap_mid_t is used to store the CoAP Message ID of a CoAP PDU.
coap_request_t
CoAP PDU Request methods.
#define COAP_RESPONSE_CODE(N)
#define COAP_RESPONSE_CLASS(C)
#define COAP_OPTION_OSCORE
int coap_add_token(coap_pdu_t *pdu, size_t len, const uint8_t *data)
Adds token of length len to pdu.
#define COAP_OPTION_Q_BLOCK2
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.
coap_pdu_t * coap_pdu_init(coap_pdu_type_t type, coap_pdu_code_t code, coap_mid_t mid, size_t size)
Creates a new CoAP PDU with at least enough storage space for the given size maximum message size.
#define COAP_INVALID_MID
Indicates an invalid message id.
#define COAP_OPTION_MAXAGE
#define COAP_OPTION_OBSERVE
#define COAP_DEFAULT_URI_WELLKNOWN
well-known resources URI
int coap_add_data(coap_pdu_t *pdu, size_t len, const uint8_t *data)
Adds given data to the pdu that is passed as first parameter.
size_t coap_session_max_pdu_size_lkd(const coap_session_t *session)
Get maximum acceptable PDU size.
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_NOT_RELIABLE(p)
void coap_delete_bin_const(coap_bin_const_t *s)
Deletes the given const binary data and releases any memory allocated.
void coap_delete_str_const(coap_str_const_t *s)
Deletes the given const string and releases any memory allocated.
coap_bin_const_t * coap_new_bin_const(const uint8_t *data, size_t size)
Take the specified byte array (text) and create a coap_bin_const_t * Returns a new const binary objec...
#define coap_binary_equal(binary1, binary2)
Compares the two binary data for equality.
#define coap_string_equal(string1, string2)
Compares the two strings for equality.
coap_str_const_t * coap_new_str_const(const uint8_t *data, size_t size)
Returns a new const string object with at least size+1 bytes storage allocated, and the provided data...
void coap_delete_string(coap_string_t *s)
Deletes the given string and releases any memory allocated.
coap_string_t * coap_get_uri_path(const coap_pdu_t *request)
Extract uri_path string from request PDU.
coap_string_t * coap_get_query(const coap_pdu_t *request)
Extract query string from request PDU according to escape rules in 6.5.8.
coap_address_t local
local address and port
CoAP binary data definition with const data.
size_t length
length of binary data
const uint8_t * s
read-only binary data
Structure of Block options with BERT support.
unsigned int aszx
block size (0-7 including BERT
unsigned int m
1 if more blocks follow, 0 otherwise
unsigned int szx
block size (0-6)
The CoAP stack's global state is stored in a coap_context_t object.
uint32_t dynamic_cur
Current number of dynamic resources.
uint8_t * token
first byte of token (or extended length bytes prefix), if any, or options
coap_pdu_code_t code
request method (value 1–31) or response code (value 64-255)
uint8_t hdr_size
actual size used for protocol-specific header (0 until header is encoded)
coap_bin_const_t actual_token
Actual token in pdu.
coap_mid_t mid
message id, if any, in regular host byte order
coap_pdu_type_t type
message type
Abstraction of virtual session that can be attached to coap_context_t (client) or coap_endpoint_t (se...
coap_addr_tuple_t addr_info
remote/local address info
coap_proto_t proto
protocol used
coap_context_t * context
session's context
CoAP string data definition with const data.
const uint8_t * s
read-only string data
size_t length
length of string
CoAP string data definition.
size_t length
length of string
coap_bin_const_t * partial_iv