libcoap 4.3.5-develop-1ba3158
Loading...
Searching...
No Matches
coap_block.c
Go to the documentation of this file.
1/* coap_block.c -- block transfer
2 *
3 * Copyright (C) 2010--2012,2015-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#ifndef min
19#define min(a,b) ((a) < (b) ? (a) : (b))
20#endif
21
22/* Can be 1 - 8 bytes long */
23#ifndef COAP_ETAG_MAX_BYTES
24#define COAP_ETAG_MAX_BYTES 4
25#endif
26#if COAP_ETAG_MAX_BYTES < 1 || COAP_ETAG_MAX_BYTES > 8
27#error COAP_ETAG_MAX_BYTES byte size invalid
28#endif
29
30#if COAP_Q_BLOCK_SUPPORT
31int
33 return 1;
34}
35#else /* ! COAP_Q_BLOCK_SUPPORT */
36int
38 return 0;
39}
40#endif /* ! COAP_Q_BLOCK_SUPPORT */
41
42unsigned int
43coap_opt_block_num(const coap_opt_t *block_opt) {
44 unsigned int num = 0;
45 uint16_t len;
46
47 len = coap_opt_length(block_opt);
48
49 if (len == 0) {
50 return 0;
51 }
52
53 if (len > 1) {
55 coap_opt_length(block_opt) - 1);
56 }
57
58 return (num << 4) | ((COAP_OPT_BLOCK_END_BYTE(block_opt) & 0xF0) >> 4);
59}
60
61int
62coap_get_block_b(const coap_session_t *session, const coap_pdu_t *pdu,
63 coap_option_num_t number, coap_block_b_t *block) {
64 coap_opt_iterator_t opt_iter;
65 coap_opt_t *option;
66
67 assert(block);
68 memset(block, 0, sizeof(coap_block_b_t));
69
70 if (pdu && (option = coap_check_option(pdu, number, &opt_iter)) != NULL) {
71 uint32_t num;
72
73 if (COAP_OPT_BLOCK_MORE(option))
74 block->m = 1;
75 block->aszx = block->szx = COAP_OPT_BLOCK_SZX(option);
76 if (block->szx == 7) {
77 size_t length;
78 const uint8_t *data;
79
80 if (session == NULL || COAP_PROTO_NOT_RELIABLE(session->proto) ||
81 !(session->csm_bert_rem_support && session->csm_bert_loc_support))
82 /* No BERT support */
83 return 0;
84
85 block->szx = 6; /* BERT is 1024 block chunks */
86 block->bert = 1;
87 if (coap_get_data(pdu, &length, &data)) {
88 if (block->m && (length % 1024) != 0) {
89 coap_log_debug("block: Oversized packet - reduced to %zu from %zu\n",
90 length - (length % 1024), length);
91 length -= length % 1024;
92 }
93 block->chunk_size = (uint32_t)length;
94 } else
95 block->chunk_size = 0;
96 } else {
97 block->chunk_size = (size_t)1 << (block->szx + 4);
98 }
99 block->defined = 1;
100
101 /* The block number is at most 20 bits, so values above 2^20 - 1
102 * are illegal. */
103 num = coap_opt_block_num(option);
104 if (num > 0xFFFFF) {
105 return 0;
106 }
107 block->num = num;
108 return 1;
109 }
110
111 return 0;
112}
113
114int
116 coap_block_t *block) {
117 coap_block_b_t block_b;
118
119 assert(block);
120 memset(block, 0, sizeof(coap_block_t));
121
122 if (coap_get_block_b(NULL, pdu, number, &block_b)) {
123 block->num = block_b.num;
124 block->m = block_b.m;
125 block->szx = block_b.szx;
126 return 1;
127 }
128 return 0;
129}
130
131static int
133 unsigned int num,
134 unsigned int blk_size, size_t total) {
135 size_t token_options = pdu->data ? (size_t)(pdu->data - pdu->token) : pdu->used_size;
136 size_t avail = pdu->max_size - token_options;
137 unsigned int start = num << (blk_size + 4);
138 unsigned int can_use_bert = block->defined == 0 || block->bert;
139
140 assert(start <= total);
141 memset(block, 0, sizeof(*block));
142 block->num = num;
143 block->szx = block->aszx = blk_size;
144 if (can_use_bert && blk_size == 6 && avail >= 1024 && session != NULL &&
145 COAP_PROTO_RELIABLE(session->proto) &&
146 session->csm_bert_rem_support && session->csm_bert_loc_support) {
147 block->bert = 1;
148 block->aszx = 7;
149 block->chunk_size = (uint32_t)((avail / 1024) * 1024);
150 } else {
151 block->chunk_size = (size_t)1 << (blk_size + 4);
152 if (avail < block->chunk_size && (total - start) >= avail) {
153 /* Need to reduce block size */
154 unsigned int szx;
155 int new_blk_size;
156
157 if (avail < 16) { /* bad luck, this is the smallest block size */
158 coap_log_debug("not enough space, even the smallest block does not fit (1)\n");
159 return 0;
160 }
161 new_blk_size = coap_flsll((long long)avail) - 5;
162 coap_log_debug("decrease block size for %zu to %d\n", avail, new_blk_size);
163 szx = block->szx;
164 block->szx = new_blk_size;
165 block->num <<= szx - block->szx;
166 block->chunk_size = (size_t)1 << (new_blk_size + 4);
167 }
168 }
169 block->m = block->chunk_size < total - start;
170 return 1;
171}
172
173int
175 coap_pdu_t *pdu, size_t data_length) {
176 size_t start;
177 unsigned char buf[4];
178 coap_block_b_t block_b;
179
180 assert(pdu);
181
182 start = block->num << (block->szx + 4);
183 if (block->num != 0 && data_length <= start) {
184 coap_log_debug("illegal block requested\n");
185 return -2;
186 }
187
188 assert(pdu->max_size > 0);
189
190 block_b.defined = 1;
191 block_b.bert = 0;
192 if (!setup_block_b(NULL, pdu, &block_b, block->num,
193 block->szx, data_length))
194 return -3;
195
196 /* to re-encode the block option */
197 coap_update_option(pdu, number, coap_encode_var_safe(buf, sizeof(buf),
198 ((block_b.num << 4) |
199 (block_b.m << 3) |
200 block_b.szx)),
201 buf);
202
203 return 1;
204}
205
206int
208 coap_option_num_t number,
209 coap_pdu_t *pdu, size_t data_length) {
210 size_t start;
211 unsigned char buf[4];
212
213 assert(pdu);
214
215 start = block->num << (block->szx + 4);
216 if (block->num != 0 && data_length <= start) {
217 coap_log_debug("illegal block requested\n");
218 return -2;
219 }
220
221 assert(pdu->max_size > 0);
222
223 if (!setup_block_b(session, pdu, block, block->num,
224 block->szx, data_length))
225 return -3;
226
227 /* to re-encode the block option */
228 coap_update_option(pdu, number, coap_encode_var_safe(buf, sizeof(buf),
229 ((block->num << 4) |
230 (block->m << 3) |
231 block->aszx)),
232 buf);
233
234 return 1;
235}
236
237int
238coap_add_block(coap_pdu_t *pdu, size_t len, const uint8_t *data,
239 unsigned int block_num, unsigned char block_szx) {
240 unsigned int start;
241 start = block_num << (block_szx + 4);
242
243 if (len <= start)
244 return 0;
245
246 return coap_add_data(pdu,
247 min(len - start, ((size_t)1 << (block_szx + 4))),
248 data + start);
249}
250
251int
252coap_add_block_b_data(coap_pdu_t *pdu, size_t len, const uint8_t *data,
253 coap_block_b_t *block) {
254 unsigned int start = block->num << (block->szx + 4);
255 size_t max_size;
256
257 if (len <= start)
258 return 0;
259
260 if (block->bert) {
261 size_t token_options = pdu->data ? (size_t)(pdu->data - pdu->token) : pdu->used_size;
262 max_size = ((pdu->max_size - token_options) / 1024) * 1024;
263 } else {
264 max_size = (size_t)1 << (block->szx + 4);
265 }
266 block->chunk_size = (uint32_t)max_size;
267
268 return coap_add_data(pdu,
269 min(len - start, max_size),
270 data + start);
271}
272
273/*
274 * Note that the COAP_OPTION_ have to be added in the correct order
275 */
276void
278 coap_pdu_t *response,
279 uint16_t media_type,
280 int maxage,
281 size_t length,
282 const uint8_t *data
283 ) {
284 unsigned char buf[4];
285 coap_block_t block2;
286 int block2_requested = 0;
287#if COAP_SERVER_SUPPORT
288 uint64_t etag = 0;
289 coap_digest_t digest;
290 coap_digest_ctx_t *dctx = NULL;
291#endif /* COAP_SERVER_SUPPORT */
292
293 memset(&block2, 0, sizeof(block2));
294 /*
295 * Need to check that a valid block is getting asked for so that the
296 * correct options are put into the PDU.
297 */
298 if (request) {
299 if (coap_get_block(request, COAP_OPTION_BLOCK2, &block2)) {
300 block2_requested = 1;
301 if (block2.num != 0 && length <= (block2.num << (block2.szx + 4))) {
302 coap_log_debug("Illegal block requested (%d > last = %zu)\n",
303 block2.num,
304 length >> (block2.szx + 4));
305 response->code = COAP_RESPONSE_CODE(400);
306 goto error;
307 }
308 }
309 }
310 response->code = COAP_RESPONSE_CODE(205);
311
312#if COAP_SERVER_SUPPORT
313 /* add ETag for the resource data */
314 if (length) {
315 dctx = coap_digest_setup();
316 if (!dctx)
317 goto error;
318 if (!coap_digest_update(dctx, data, length))
319 goto error;
320 if (!coap_digest_final(dctx, &digest))
321 goto error;
322 dctx = NULL;
323 memcpy(&etag, digest.key, sizeof(etag));
324#if COAP_ETAG_MAX_BYTES != 8
325 etag = etag >> 8*(8 - COAP_ETAG_MAX_BYTES);
326#endif
327 if (!etag)
328 etag = 1;
329 coap_update_option(response,
331 coap_encode_var_safe8(buf, sizeof(buf), etag),
332 buf);
333 }
334#endif /* COAP_SERVER_SUPPORT */
335
337 coap_encode_var_safe(buf, sizeof(buf),
338 media_type),
339 buf);
340
341 if (maxage >= 0) {
342 coap_insert_option(response,
344 coap_encode_var_safe(buf, sizeof(buf), maxage), buf);
345 }
346
347 if (block2_requested) {
348 int res;
349
350 res = coap_write_block_opt(&block2, COAP_OPTION_BLOCK2, response, length);
351
352 switch (res) {
353 case -2: /* illegal block (caught above) */
354 response->code = COAP_RESPONSE_CODE(400);
355 goto error;
356 case -1: /* should really not happen */
357 assert(0);
358 /* fall through if assert is a no-op */
359 case -3: /* cannot handle request */
360 response->code = COAP_RESPONSE_CODE(500);
361 goto error;
362 default: /* everything is good */
363 ;
364 }
365
368 coap_encode_var_safe8(buf, sizeof(buf), length),
369 buf);
370
371 coap_add_block(response, length, data,
372 block2.num, block2.szx);
373 return;
374 }
375
376 /*
377 * Block2 not requested
378 */
379 if (!coap_add_data(response, length, data)) {
380 /*
381 * Insufficient space to add in data - use block mode
382 * set initial block size, will be lowered by
383 * coap_write_block_opt() automatically
384 */
385 block2.num = 0;
386 block2.szx = 6;
387 coap_write_block_opt(&block2, COAP_OPTION_BLOCK2, response, length);
388
391 coap_encode_var_safe8(buf, sizeof(buf), length),
392 buf);
393
394 coap_add_block(response, length, data,
395 block2.num, block2.szx);
396 }
397 return;
398
399error:
400#if COAP_SERVER_SUPPORT
401 coap_digest_free(dctx);
402#endif /* COAP_SERVER_SUPPORT */
403 coap_add_data(response,
404 strlen(coap_response_phrase(response->code)),
405 (const unsigned char *)coap_response_phrase(response->code));
406}
407
408COAP_API void
410 uint32_t block_mode) {
411 coap_lock_lock(context, return);
412 coap_context_set_block_mode_lkd(context, block_mode);
413 coap_lock_unlock(context);
414}
415
416void
418 uint32_t block_mode) {
419 coap_lock_check_locked(context);
420 if (!(block_mode & COAP_BLOCK_USE_LIBCOAP))
421 block_mode = 0;
422 context->block_mode &= ~COAP_BLOCK_SET_MASK;
423 context->block_mode |= block_mode & COAP_BLOCK_SET_MASK;
424#if ! COAP_Q_BLOCK_SUPPORT
426 coap_log_debug("Q-Block support not compiled in - ignored\n");
427#endif /* ! COAP_Q_BLOCK_SUPPORT */
428}
429
430COAP_API int
432 size_t max_block_size) {
433 int ret;
434
435 coap_lock_lock(context, return 0);
436 ret = coap_context_set_max_block_size_lkd(context, max_block_size);
437 coap_lock_unlock(context);
438 return ret;
439}
440
441int
442coap_context_set_max_block_size_lkd(coap_context_t *context, size_t max_block_size) {
443 switch (max_block_size) {
444 case 0:
445 case 16:
446 case 32:
447 case 64:
448 case 128:
449 case 256:
450 case 512:
451 case 1024:
452 break;
453 default:
454 coap_log_info("coap_context_set_max_block_size: Invalid max block size (%zu)\n",
455 max_block_size);
456 return 0;
457 }
458 coap_lock_check_locked(context);
459 max_block_size = (coap_fls((uint32_t)max_block_size >> 4) - 1) & 0x07;
460 context->block_mode &= ~COAP_BLOCK_MAX_SIZE_MASK;
461 context->block_mode |= COAP_BLOCK_MAX_SIZE_SET((uint32_t)max_block_size);
462 return 1;
463}
464
466full_match(const uint8_t *a, size_t alen,
467 const uint8_t *b, size_t blen) {
468 return alen == blen && (alen == 0 || memcmp(a, b, alen) == 0);
469}
470
471#if COAP_CLIENT_SUPPORT
472
473COAP_API int
475 coap_pdu_type_t type) {
476 int ret;
477
478 coap_lock_lock(session->context, return 0);
479 ret = coap_cancel_observe_lkd(session, token, type);
480 coap_lock_unlock(session->context);
481 return ret;
482}
483
484int
486 coap_pdu_type_t type) {
487 coap_lg_crcv_t *lg_crcv, *q;
488
489 assert(session);
490 if (!session)
491 return 0;
492
494 if (!(session->block_mode & COAP_BLOCK_USE_LIBCOAP)) {
495 coap_log_debug("** %s: coap_cancel_observe: COAP_BLOCK_USE_LIBCOAP not enabled\n",
496 coap_session_str(session));
497 return 0;
498 }
499
500 LL_FOREACH_SAFE(session->lg_crcv, lg_crcv, q) {
501 if (lg_crcv->observe_set) {
502 if ((!token && !lg_crcv->app_token->length) || (token &&
503 coap_binary_equal(token, lg_crcv->app_token))) {
504 uint8_t buf[8];
505 coap_mid_t mid;
506 size_t size;
507 const uint8_t *data;
508#if COAP_Q_BLOCK_SUPPORT
509 coap_block_b_t block;
510 int using_q_block1 = coap_get_block_b(session, &lg_crcv->pdu,
511 COAP_OPTION_Q_BLOCK1, &block);
512#endif /* COAP_Q_BLOCK_SUPPORT */
513 coap_bin_const_t *otoken = lg_crcv->obs_token ?
514 lg_crcv->obs_token[0] ?
515 lg_crcv->obs_token[0] :
516 (coap_bin_const_t *)lg_crcv->app_token :
517 (coap_bin_const_t *)lg_crcv->app_token;
518 coap_pdu_t *pdu = coap_pdu_duplicate_lkd(&lg_crcv->pdu,
519 session,
520 otoken->length,
521 otoken->s,
522 NULL);
523
524 lg_crcv->observe_set = 0;
525 if (pdu == NULL)
526 return 0;
527 /* Need to make sure that this is the correct requested type */
528 pdu->type = type;
529
531 coap_encode_var_safe(buf, sizeof(buf),
533 buf);
534 if (lg_crcv->o_block_option) {
536 coap_encode_var_safe(buf, sizeof(buf),
537 lg_crcv->o_blk_size),
538 buf);
539 }
540 if (coap_get_data(&lg_crcv->pdu, &size, &data))
541 coap_add_data_large_request_lkd(session, pdu, size, data, NULL, NULL);
542
543 /*
544 * Need to fix lg_xmit stateless token as using tokens from
545 * observe setup
546 */
547 if (pdu->lg_xmit)
548 pdu->lg_xmit->b.b1.state_token = lg_crcv->state_token;
549
550#if COAP_Q_BLOCK_SUPPORT
551 /* See if large xmit using Q-Block1 (but not testing Q-Block1) */
552 if (using_q_block1) {
553 mid = coap_send_q_block1(session, block, pdu, COAP_SEND_INC_PDU);
554 } else {
555 mid = coap_send_internal(session, pdu, NULL);
556 }
557#else /* ! COAP_Q_BLOCK_SUPPORT */
558 mid = coap_send_internal(session, pdu, NULL);
559#endif /* ! COAP_Q_BLOCK_SUPPORT */
560 if (mid != COAP_INVALID_MID)
561 return 1;
562 break;
563 }
564 }
565 }
566 return 0;
567}
568
569#if COAP_OSCORE_SUPPORT
572 coap_pdu_t *pdu,
573 coap_opt_t *echo) {
574 coap_lg_crcv_t *lg_crcv;
575 uint64_t token_match =
577 pdu->actual_token.length));
578 uint8_t ltoken[8];
579 size_t ltoken_len;
580 uint64_t token;
581 const uint8_t *data;
582 size_t data_len;
583 coap_pdu_t *resend_pdu;
584 coap_block_b_t block;
585
586 LL_FOREACH(session->lg_crcv, lg_crcv) {
587 if (token_match != STATE_TOKEN_BASE(lg_crcv->state_token) &&
588 !coap_binary_equal(&pdu->actual_token, lg_crcv->app_token)) {
589 /* try out the next one */
590 continue;
591 }
592
593 /* lg_crcv found */
594
595 /* Re-send request with new token */
596 token = STATE_TOKEN_FULL(lg_crcv->state_token,
597 ++lg_crcv->retry_counter);
598 ltoken_len = coap_encode_var_safe8(ltoken, sizeof(token), token);
599 /* There could be a Block option in pdu */
600 resend_pdu = coap_pdu_duplicate_lkd(pdu, session, ltoken_len,
601 ltoken, NULL);
602 if (!resend_pdu)
603 goto error;
604 if (echo) {
606 coap_opt_value(echo));
607 }
608 if (coap_get_data(&lg_crcv->pdu, &data_len, &data)) {
609 if (coap_get_block_b(session, resend_pdu, COAP_OPTION_BLOCK1, &block)) {
610 if (data_len > block.chunk_size && block.chunk_size != 0) {
611 data_len = block.chunk_size;
612 }
613 }
614 coap_add_data(resend_pdu, data_len, data);
615 }
616
617 return coap_send_internal(session, resend_pdu, NULL);
618 }
619error:
620 return COAP_INVALID_MID;
621}
622#endif /* COAP_OSCORE_SUPPORT */
623#endif /* COAP_CLIENT_SUPPORT */
624
625#if COAP_SERVER_SUPPORT
626/*
627 * Find the response lg_xmit
628 */
631 const coap_pdu_t *request,
632 const coap_resource_t *resource,
633 const coap_string_t *query) {
634 coap_lg_xmit_t *lg_xmit;
635 coap_opt_iterator_t opt_iter;
636 coap_opt_t *rtag_opt = coap_check_option(request,
638 &opt_iter);
639 size_t rtag_length = rtag_opt ? coap_opt_length(rtag_opt) : 0;
640 const uint8_t *rtag = rtag_opt ? coap_opt_value(rtag_opt) : NULL;
641
642 LL_FOREACH(session->lg_xmit, lg_xmit) {
643 static coap_string_t empty = { 0, NULL};
644
645 if (COAP_PDU_IS_REQUEST(&lg_xmit->pdu) ||
646 resource != lg_xmit->b.b2.resource ||
647 request->code != lg_xmit->b.b2.request_method ||
648 !coap_string_equal(query ? query : &empty,
649 lg_xmit->b.b2.query ?
650 lg_xmit->b.b2.query : &empty)) {
651 /* try out the next one */
652 continue;
653 }
654 /* lg_xmit is a response */
655 if (rtag_opt || lg_xmit->b.b2.rtag_set == 1) {
656 if (!(rtag_opt && lg_xmit->b.b2.rtag_set == 1))
657 continue;
658 if (lg_xmit->b.b2.rtag_length != rtag_length ||
659 memcmp(lg_xmit->b.b2.rtag, rtag, rtag_length) != 0)
660 continue;
661 }
662 return lg_xmit;
663 }
664 return NULL;
665}
666#endif /* COAP_SERVER_SUPPORT */
667
668static int
670 const coap_pdu_t *request,
671 coap_pdu_t *pdu,
672 coap_resource_t *resource,
673 const coap_string_t *query,
674 int maxage,
675 uint64_t etag,
676 size_t length,
677 const uint8_t *data,
678 coap_release_large_data_t release_func,
679 void *app_ptr,
680 int single_request, coap_pdu_code_t request_method) {
681
682 ssize_t avail;
683 coap_block_b_t block;
684#if COAP_Q_BLOCK_SUPPORT
685 coap_block_b_t alt_block;
686#endif /* COAP_Q_BLOCK_SUPPORT */
687 size_t chunk;
688 coap_lg_xmit_t *lg_xmit = NULL;
689 uint8_t buf[8];
690 int have_block_defined = 0;
691 uint8_t blk_size;
692 uint8_t max_blk_size;
693 uint16_t option;
694 size_t token_options;
695 coap_opt_t *opt;
696 coap_opt_iterator_t opt_iter;
697#if COAP_Q_BLOCK_SUPPORT
698 uint16_t alt_option;
699#endif /* COAP_Q_BLOCK_SUPPORT */
700
701#if !COAP_SERVER_SUPPORT
702 (void)etag;
703#endif /* COAP_SERVER_SUPPORT */
704
705 assert(pdu);
706 if (pdu->data) {
707 coap_log_warn("coap_add_data_large: PDU already contains data\n");
708 if (release_func) {
709 coap_lock_callback(session->context, release_func(session, app_ptr));
710 }
711 return 0;
712 }
713
714 if (!(session->block_mode & COAP_BLOCK_USE_LIBCOAP)) {
715 coap_log_debug("** %s: coap_add_data_large: COAP_BLOCK_USE_LIBCOAP not enabled\n",
716 coap_session_str(session));
717 goto add_data;
718 }
719
720 /* A lot of the reliable code assumes type is CON */
721 if (COAP_PROTO_RELIABLE(session->proto) && pdu->type == COAP_MESSAGE_NON)
722 pdu->type = COAP_MESSAGE_CON;
723
724 /* Block NUM max 20 bits and block size is "2**(SZX + 4)"
725 and using SZX max of 6 gives maximum size = 1,073,740,800
726 CSM Max-Message-Size theoretical maximum = 4,294,967,295
727 So, if using blocks, we are limited to 1,073,740,800.
728 */
729#define MAX_BLK_LEN (((1UL << 20) - 1) * (1 << (6 + 4)))
730
731#if UINT_MAX > MAX_BLK_LEN
732 if (length > MAX_BLK_LEN) {
733 coap_log_warn("Size of large buffer restricted to 0x%lx bytes\n", MAX_BLK_LEN);
734 length = MAX_BLK_LEN;
735 }
736#endif /* UINT_MAX > MAX_BLK_LEN */
737
738#if COAP_SERVER_SUPPORT
739 if (COAP_PDU_IS_RESPONSE(pdu) && length) {
740 coap_opt_t *etag_opt = coap_check_option(pdu, COAP_OPTION_ETAG, &opt_iter);
741
742 if (etag_opt) {
743 /* Have to use ETag as supplied in the response PDU */
744 etag = coap_decode_var_bytes8(coap_opt_value(etag_opt),
745 coap_opt_length(etag_opt));
746 } else {
747 if (!etag) {
748 /* calculate ETag for the response */
749 coap_digest_t digest;
751
752 if (dctx) {
753 if (coap_digest_update(dctx, data, length)) {
754 if (coap_digest_final(dctx, &digest)) {
755 memcpy(&etag, digest.key, sizeof(etag));
756#if COAP_ETAG_MAX_BYTES != 8
757 etag = etag >> 8*(8 - COAP_ETAG_MAX_BYTES);
758#endif
759 dctx = NULL;
760 }
761 }
762 coap_digest_free(dctx);
763 }
764 if (!etag)
765 etag = 1;
766 }
769 coap_encode_var_safe8(buf, sizeof(buf), etag),
770 buf);
771 }
772 if (request) {
773 etag_opt = coap_check_option(request, COAP_OPTION_ETAG, &opt_iter);
774 if (etag_opt) {
775 /* There may be multiple ETag - need to check each one */
776 coap_option_iterator_init(request, &opt_iter, COAP_OPT_ALL);
777 while ((etag_opt = coap_option_next(&opt_iter))) {
778 if (opt_iter.number == COAP_OPTION_ETAG) {
779 uint64_t etag_r = coap_decode_var_bytes8(coap_opt_value(etag_opt),
780 coap_opt_length(etag_opt));
781
782 if (etag == etag_r) {
783 pdu->code = COAP_RESPONSE_CODE(203);
784 return 1;
785 }
786 }
787 }
788 }
789 }
790 }
791#endif /* COAP_SERVER_SUPPORT */
792
793 /* Determine the block size to use, adding in sensible options if needed */
794 if (COAP_PDU_IS_REQUEST(pdu)) {
796
797#if COAP_Q_BLOCK_SUPPORT
798 if (session->block_mode & (COAP_BLOCK_HAS_Q_BLOCK|COAP_BLOCK_TRY_Q_BLOCK)) {
799 option = COAP_OPTION_Q_BLOCK1;
800 alt_option = COAP_OPTION_BLOCK1;
801 } else {
802 option = COAP_OPTION_BLOCK1;
803 alt_option = COAP_OPTION_Q_BLOCK1;
804 }
805#else /* ! COAP_Q_BLOCK_SUPPORT */
806 if (coap_get_block_b(session, pdu, COAP_OPTION_Q_BLOCK1, &block)) {
808 }
809 option = COAP_OPTION_BLOCK1;
810#endif /* ! COAP_Q_BLOCK_SUPPORT */
811
812 /* See if this token is already in use for large bodies (unlikely) */
813 LL_FOREACH_SAFE(session->lg_xmit, lg_xmit, q) {
814 if (coap_binary_equal(&pdu->actual_token, lg_xmit->b.b1.app_token)) {
815 /* Unfortunately need to free this off as potential size change */
816 LL_DELETE(session->lg_xmit, lg_xmit);
817 coap_block_delete_lg_xmit(session, lg_xmit);
818 lg_xmit = NULL;
820 break;
821 }
822 }
823 } else {
824 /* Have to assume that it is a response even if code is 0.00 */
825 assert(resource);
826#if COAP_Q_BLOCK_SUPPORT
827 if (session->block_mode & COAP_BLOCK_HAS_Q_BLOCK) {
828 option = COAP_OPTION_Q_BLOCK2;
829 alt_option = COAP_OPTION_BLOCK2;
830 } else {
831 option = COAP_OPTION_BLOCK2;
832 alt_option = COAP_OPTION_Q_BLOCK2;
833 }
834#else /* ! COAP_Q_BLOCK_SUPPORT */
835 if (coap_get_block_b(session, pdu, COAP_OPTION_Q_BLOCK2, &block)) {
837 }
838 option = COAP_OPTION_BLOCK2;
839#endif /* ! COAP_Q_BLOCK_SUPPORT */
840#if COAP_SERVER_SUPPORT
841 /*
842 * Check if resource+query+rtag is already in use for large bodies
843 * (unlikely)
844 */
845 lg_xmit = coap_find_lg_xmit_response(session, request, resource, query);
846 if (lg_xmit) {
847 /* Unfortunately need to free this off as potential size change */
848 LL_DELETE(session->lg_xmit, lg_xmit);
849 coap_block_delete_lg_xmit(session, lg_xmit);
850 lg_xmit = NULL;
852 }
853#endif /* COAP_SERVER_SUPPORT */
854 }
855#if COAP_OSCORE_SUPPORT
856 if (session->oscore_encryption) {
857 /* Need to convert Proxy-Uri to Proxy-Scheme option if needed */
859 goto fail;
860 }
861#endif /* COAP_OSCORE_SUPPORT */
862
863 token_options = pdu->data ? (size_t)(pdu->data - pdu->token) : pdu->used_size;
864 avail = pdu->max_size - token_options;
865 /* There may be a response with Echo option */
867#if COAP_OSCORE_SUPPORT
868 avail -= coap_oscore_overhead(session, pdu);
869#endif /* COAP_OSCORE_SUPPORT */
870 /* May need token of length 8, so account for this */
871 avail -= (pdu->actual_token.length < 8) ? 8 - pdu->actual_token.length : 0;
872 blk_size = coap_flsll((long long)avail) - 4 - 1;
873 if (blk_size > 6)
874 blk_size = 6;
875
876 max_blk_size = COAP_BLOCK_MAX_SIZE_GET(session->block_mode);
877 if (max_blk_size && blk_size > max_blk_size)
878 blk_size = max_blk_size;
879
880 /* see if BlockX defined - if so update blk_size as given by app */
881 if (coap_get_block_b(session, pdu, option, &block)) {
882 if (block.szx < blk_size)
883 blk_size = block.szx;
884 have_block_defined = 1;
885 }
886#if COAP_Q_BLOCK_SUPPORT
887 /* see if alternate BlockX defined */
888 if (coap_get_block_b(session, pdu, alt_option, &alt_block)) {
889 if (have_block_defined) {
890 /* Cannot have both options set */
891 coap_log_warn("Both BlockX and Q-BlockX cannot be set at the same time\n");
892 coap_remove_option(pdu, alt_option);
893 } else {
894 block = alt_block;
895 if (block.szx < blk_size)
896 blk_size = block.szx;
897 have_block_defined = 1;
898 option = alt_option;
899 }
900 }
901#endif /* COAP_Q_BLOCK_SUPPORT */
902
903 if (avail < 16 && ((ssize_t)length > avail || have_block_defined)) {
904 /* bad luck, this is the smallest block size */
905 coap_log_debug("not enough space, even the smallest block does not fit (2)\n");
906 goto fail;
907 }
908
909 chunk = (size_t)1 << (blk_size + 4);
910 if ((have_block_defined && block.num != 0) || single_request ||
912 /* App is defining a single block to send or we are stateless */
913 size_t rem;
914
915 if (length >= block.num * chunk) {
916#if COAP_SERVER_SUPPORT
918 /* We are running server stateless */
921 coap_encode_var_safe(buf, sizeof(buf),
922 (unsigned int)length),
923 buf);
924 if (request) {
925 if (!coap_get_block_b(session, request, option, &block))
926 block.num = 0;
927 }
928 if (!setup_block_b(session, pdu, &block, block.num,
929 blk_size, length))
930 goto fail;
931
932 /* Add in with requested block num, more bit and block size */
934 option,
935 coap_encode_var_safe(buf, sizeof(buf),
936 (block.num << 4) | (block.m << 3) | block.aszx),
937 buf);
938 }
939#endif /* COAP_SERVER_SUPPORT */
940 rem = chunk;
941 if (chunk > length - block.num * chunk)
942 rem = length - block.num * chunk;
943 if (!coap_add_data(pdu, rem, &data[block.num * chunk]))
944 goto fail;
945 }
946 if (release_func) {
947 coap_lock_callback(session->context, release_func(session, app_ptr));
948 }
949 } else if ((have_block_defined && length > chunk) || (ssize_t)length > avail) {
950 /* Only add in lg_xmit if more than one block needs to be handled */
951 size_t rem;
952
953 lg_xmit = coap_malloc_type(COAP_LG_XMIT, sizeof(coap_lg_xmit_t));
954 if (!lg_xmit)
955 goto fail;
956
957 /* Set up for displaying all the data in the pdu */
958 pdu->body_data = data;
959 pdu->body_length = length;
960 coap_log_debug("PDU presented by app.\n");
962 pdu->body_data = NULL;
963 pdu->body_length = 0;
964
965 coap_log_debug("** %s: lg_xmit %p initialized\n",
966 coap_session_str(session), (void *)lg_xmit);
967 /* Update lg_xmit with large data information */
968 memset(lg_xmit, 0, sizeof(coap_lg_xmit_t));
969 lg_xmit->blk_size = blk_size;
970 lg_xmit->option = option;
971 lg_xmit->data = data;
972 lg_xmit->length = length;
973#if COAP_Q_BLOCK_SUPPORT
974 lg_xmit->non_timeout_random_ticks =
976#endif /* COAP_Q_BLOCK_SUPPORT */
977 lg_xmit->release_func = release_func;
978 lg_xmit->app_ptr = app_ptr;
979 pdu->lg_xmit = lg_xmit;
980 coap_ticks(&lg_xmit->last_obs);
981 coap_ticks(&lg_xmit->last_sent);
982 if (COAP_PDU_IS_REQUEST(pdu)) {
983 /* Need to keep original token for updating response PDUs */
984 lg_xmit->b.b1.app_token = coap_new_binary(pdu->actual_token.length);
985 if (!lg_xmit->b.b1.app_token)
986 goto fail;
987 memcpy(lg_xmit->b.b1.app_token->s, pdu->actual_token.s,
988 pdu->actual_token.length);
989 /*
990 * Need to set up new token for use during transmits
991 * RFC9177#section-5
992 */
993 lg_xmit->b.b1.count = 1;
994 lg_xmit->b.b1.state_token = STATE_TOKEN_FULL(++session->tx_token,
995 lg_xmit->b.b1.count);
996 /*
997 * Token will be updated in pdu later as original pdu may be needed in
998 * coap_send()
999 */
1002 coap_encode_var_safe(buf, sizeof(buf),
1003 (unsigned int)length),
1004 buf);
1005 if (!coap_check_option(pdu, COAP_OPTION_RTAG, &opt_iter))
1008 coap_encode_var_safe(buf, sizeof(buf),
1009 ++session->tx_rtag),
1010 buf);
1011 } else {
1012 /*
1013 * resource+query+rtag match is used for Block2 large body transmissions
1014 * token match is used for Block1 large body transmissions
1015 */
1016 lg_xmit->b.b2.resource = resource;
1017 if (query) {
1018 lg_xmit->b.b2.query = coap_new_string(query->length);
1019 if (lg_xmit->b.b2.query) {
1020 memcpy(lg_xmit->b.b2.query->s, query->s, query->length);
1021 }
1022 } else {
1023 lg_xmit->b.b2.query = NULL;
1024 }
1025 opt = coap_check_option(request, COAP_OPTION_RTAG, &opt_iter);
1026 if (opt) {
1027 lg_xmit->b.b2.rtag_length = (uint8_t)min(coap_opt_length(opt),
1028 sizeof(lg_xmit->b.b2.rtag));
1029 memcpy(lg_xmit->b.b2.rtag, coap_opt_value(opt), coap_opt_length(opt));
1030 lg_xmit->b.b2.rtag_set = 1;
1031 } else {
1032 lg_xmit->b.b2.rtag_set = 0;
1033 }
1034 lg_xmit->b.b2.etag = etag;
1035 lg_xmit->b.b2.request_method = request_method;
1036 if (maxage >= 0) {
1037 coap_tick_t now;
1038
1039 coap_ticks(&now);
1040 lg_xmit->b.b2.maxage_expire = coap_ticks_to_rt(now) + maxage;
1041 } else {
1042 lg_xmit->b.b2.maxage_expire = 0;
1043 }
1046 coap_encode_var_safe(buf, sizeof(buf),
1047 (unsigned int)length),
1048 buf);
1049 }
1050
1051 if (!setup_block_b(session, pdu, &block, block.num,
1052 blk_size, lg_xmit->length))
1053 goto fail;
1054
1055 /* Add in with requested block num, more bit and block size */
1057 lg_xmit->option,
1058 coap_encode_var_safe(buf, sizeof(buf),
1059 (block.num << 4) | (block.m << 3) | block.aszx),
1060 buf);
1061
1062 /* Set up skeletal PDU to use as a basis for all the subsequent blocks */
1063 memcpy(&lg_xmit->pdu, pdu, sizeof(lg_xmit->pdu));
1064 lg_xmit->pdu.token = coap_malloc_type(COAP_PDU_BUF,
1065 lg_xmit->pdu.used_size + lg_xmit->pdu.max_hdr_size);
1066 if (!lg_xmit->pdu.token)
1067 goto fail;
1068
1069 lg_xmit->pdu.alloc_size = lg_xmit->pdu.used_size;
1070 lg_xmit->pdu.token += lg_xmit->pdu.max_hdr_size;
1071 memcpy(lg_xmit->pdu.token, pdu->token, lg_xmit->pdu.used_size);
1072 if (pdu->data)
1073 lg_xmit->pdu.data = lg_xmit->pdu.token + (pdu->data - pdu->token);
1074 lg_xmit->pdu.actual_token.s = lg_xmit->pdu.token + pdu->e_token_length -
1075 pdu->actual_token.length;
1076 lg_xmit->pdu.actual_token.length = pdu->actual_token.length;
1077
1078 /* Check we still have space after adding in some options */
1079 token_options = pdu->data ? (size_t)(pdu->data - pdu->token) : pdu->used_size;
1080 avail = pdu->max_size - token_options;
1081 /* There may be a response with Echo option */
1083 /* May need token of length 8, so account for this */
1084 avail -= (pdu->actual_token.length < 8) ? 8 - pdu->actual_token.length : 0;
1085#if COAP_OSCORE_SUPPORT
1086 avail -= coap_oscore_overhead(session, pdu);
1087#endif /* COAP_OSCORE_SUPPORT */
1088 if (avail < (ssize_t)chunk) {
1089 /* chunk size change down */
1090 if (avail < 16) {
1091 coap_log_warn("not enough space, even the smallest block does not fit (3)\n");
1092 goto fail;
1093 }
1094 blk_size = coap_flsll((long long)avail) - 4 - 1;
1095 block.num = block.num << (lg_xmit->blk_size - blk_size);
1096 lg_xmit->blk_size = blk_size;
1097 chunk = (size_t)1 << (lg_xmit->blk_size + 4);
1098 block.chunk_size = (uint32_t)chunk;
1099 block.bert = 0;
1101 lg_xmit->option,
1102 coap_encode_var_safe(buf, sizeof(buf),
1103 (block.num << 4) | (block.m << 3) | lg_xmit->blk_size),
1104 buf);
1105 }
1106
1107 rem = block.chunk_size;
1108 if (rem > lg_xmit->length - block.num * chunk)
1109 rem = lg_xmit->length - block.num * chunk;
1110 if (!coap_add_data(pdu, rem, &data[block.num * chunk]))
1111 goto fail;
1112
1113 if (COAP_PDU_IS_REQUEST(pdu))
1114 lg_xmit->b.b1.bert_size = rem;
1115
1116 lg_xmit->last_block = -1;
1117
1118 /* Link the new lg_xmit in */
1119 LL_PREPEND(session->lg_xmit,lg_xmit);
1120 } else {
1121 /* No need to use blocks */
1122 if (have_block_defined) {
1124 option,
1125 coap_encode_var_safe(buf, sizeof(buf),
1126 (0 << 4) | (0 << 3) | blk_size), buf);
1127 }
1128add_data:
1129 if (!coap_add_data(pdu, length, data))
1130 goto fail;
1131
1132 if (release_func) {
1133 coap_lock_callback(session->context, release_func(session, app_ptr));
1134 }
1135 }
1136 return 1;
1137
1138fail:
1139 if (lg_xmit) {
1140 coap_block_delete_lg_xmit(session, lg_xmit);
1141 } else if (release_func) {
1142 coap_lock_callback(session->context, release_func(session, app_ptr));
1143 }
1144 return 0;
1145}
1146
1147#if COAP_CLIENT_SUPPORT
1148COAP_API int
1150 coap_pdu_t *pdu,
1151 size_t length,
1152 const uint8_t *data,
1153 coap_release_large_data_t release_func,
1154 void *app_ptr
1155 ) {
1156 int ret;
1157
1158 coap_lock_lock(session->context, return 0);
1159 ret = coap_add_data_large_request_lkd(session, pdu, length, data,
1160 release_func, app_ptr);
1161 coap_lock_unlock(session->context);
1162 return ret;
1163}
1164
1165int
1167 coap_pdu_t *pdu,
1168 size_t length,
1169 const uint8_t *data,
1170 coap_release_large_data_t release_func,
1171 void *app_ptr) {
1172 /*
1173 * Delay if session->doing_first is set.
1174 * E.g. Reliable and CSM not in yet for checking block support
1175 */
1176 if (coap_client_delay_first(session) == 0) {
1177 if (release_func) {
1178 coap_lock_callback(session->context, release_func(session, app_ptr));
1179 }
1180 return 0;
1181 }
1182 return coap_add_data_large_internal(session, NULL, pdu, NULL, NULL, -1, 0,
1183 length, data, release_func, app_ptr, 0, 0);
1184}
1185#endif /* ! COAP_CLIENT_SUPPORT */
1186
1187#if COAP_SERVER_SUPPORT
1188COAP_API int
1190 coap_session_t *session,
1191 const coap_pdu_t *request,
1192 coap_pdu_t *response,
1193 const coap_string_t *query,
1194 uint16_t media_type,
1195 int maxage,
1196 uint64_t etag,
1197 size_t length,
1198 const uint8_t *data,
1199 coap_release_large_data_t release_func,
1200 void *app_ptr
1201 ) {
1202 int ret;
1203
1204 coap_lock_lock(session->context, return 0);
1205 ret = coap_add_data_large_response_lkd(resource, session, request,
1206 response, query, media_type, maxage, etag,
1207 length, data, release_func, app_ptr);
1208 coap_lock_unlock(session->context);
1209 return ret;
1210}
1211
1212int
1214 coap_session_t *session,
1215 const coap_pdu_t *request,
1216 coap_pdu_t *response,
1217 const coap_string_t *query,
1218 uint16_t media_type,
1219 int maxage,
1220 uint64_t etag,
1221 size_t length,
1222 const uint8_t *data,
1223 coap_release_large_data_t release_func,
1224 void *app_ptr
1225 ) {
1226 unsigned char buf[4];
1227 coap_block_b_t block;
1228 int block_requested = 0;
1229 int single_request = 0;
1230#if COAP_Q_BLOCK_SUPPORT
1231 uint32_t block_opt = (session->block_mode & COAP_BLOCK_HAS_Q_BLOCK) ?
1233#else /* ! COAP_Q_BLOCK_SUPPORT */
1234 uint16_t block_opt = COAP_OPTION_BLOCK2;
1235#endif /* ! COAP_Q_BLOCK_SUPPORT */
1236
1237 memset(&block, 0, sizeof(block));
1238 /*
1239 * Need to check that a valid block is getting asked for so that the
1240 * correct options are put into the PDU.
1241 */
1242 if (request) {
1243 if (coap_get_block_b(session, request, COAP_OPTION_BLOCK2, &block)) {
1244 block_requested = 1;
1245 if (block.num != 0 && length <= (block.num << (block.szx + 4))) {
1246 coap_log_debug("Illegal block requested (%d > last = %zu)\n",
1247 block.num,
1248 length >> (block.szx + 4));
1249 response->code = COAP_RESPONSE_CODE(400);
1250 goto error;
1251 }
1252 }
1253#if COAP_Q_BLOCK_SUPPORT
1254 else if (coap_get_block_b(session, request, COAP_OPTION_Q_BLOCK2, &block)) {
1255 block_requested = 1;
1256 if (block.num != 0 && length <= (block.num << (block.szx + 4))) {
1257 coap_log_debug("Illegal block requested (%d > last = %zu)\n",
1258 block.num,
1259 length >> (block.szx + 4));
1260 response->code = COAP_RESPONSE_CODE(400);
1261 goto error;
1262 }
1263 if (!(session->block_mode & COAP_BLOCK_HAS_Q_BLOCK)) {
1264 set_block_mode_has_q(session->block_mode);
1265 block_opt = COAP_OPTION_Q_BLOCK2;
1266 }
1267 if (block.m == 0)
1268 single_request = 1;
1269 }
1270#endif /* COAP_Q_BLOCK_SUPPORT */
1271 }
1272
1274 coap_encode_var_safe(buf, sizeof(buf),
1275 media_type),
1276 buf);
1277
1278 if (maxage >= 0) {
1279 coap_insert_option(response,
1281 coap_encode_var_safe(buf, sizeof(buf), maxage), buf);
1282 }
1283
1284 if (block_requested) {
1285 int res;
1286
1287 res = coap_write_block_b_opt(session, &block, block_opt, response,
1288 length);
1289
1290 switch (res) {
1291 case -2: /* illegal block (caught above) */
1292 response->code = COAP_RESPONSE_CODE(400);
1293 goto error;
1294 case -1: /* should really not happen */
1295 assert(0);
1296 /* fall through if assert is a no-op */
1297 case -3: /* cannot handle request */
1298 response->code = COAP_RESPONSE_CODE(500);
1299 goto error;
1300 default: /* everything is good */
1301 ;
1302 }
1303 }
1304
1305 /* add data body */
1306 if (request &&
1307 !coap_add_data_large_internal(session, request, response, resource,
1308 query, maxage, etag, length, data,
1309 release_func, app_ptr, single_request,
1310 request->code)) {
1311 response->code = COAP_RESPONSE_CODE(500);
1312 goto error_released;
1313 }
1314
1315 return 1;
1316
1317error:
1318 if (release_func) {
1319 coap_lock_callback(session->context, release_func(session, app_ptr));
1320 }
1321error_released:
1322#if COAP_ERROR_PHRASE_LENGTH > 0
1323 coap_add_data(response,
1324 strlen(coap_response_phrase(response->code)),
1325 (const unsigned char *)coap_response_phrase(response->code));
1326#endif /* COAP_ERROR_PHRASE_LENGTH > 0 */
1327 return 0;
1328}
1329#endif /* ! COAP_SERVER_SUPPORT */
1330
1331/*
1332 * return 1 if there is a future expire time, else 0.
1333 * update tim_rem with remaining value if return is 1.
1334 */
1335int
1337 coap_tick_t *tim_rem) {
1338 coap_lg_xmit_t *lg_xmit;
1339 coap_lg_xmit_t *q;
1340#if COAP_Q_BLOCK_SUPPORT
1341 coap_tick_t idle_timeout = 4 * COAP_NON_TIMEOUT_TICKS(session);
1342#else /* ! COAP_Q_BLOCK_SUPPORT */
1343 coap_tick_t idle_timeout = 8 * COAP_TICKS_PER_SECOND;
1344#endif /* ! COAP_Q_BLOCK_SUPPORT */
1345 coap_tick_t partial_timeout = COAP_MAX_TRANSMIT_WAIT_TICKS(session);
1346 int ret = 0;
1347
1348 *tim_rem = -1;
1349
1350 LL_FOREACH_SAFE(session->lg_xmit, lg_xmit, q) {
1351 if (lg_xmit->last_all_sent) {
1352 if (lg_xmit->last_all_sent + idle_timeout <= now) {
1353 /* Expire this entry */
1354 LL_DELETE(session->lg_xmit, lg_xmit);
1355 coap_block_delete_lg_xmit(session, lg_xmit);
1356 } else {
1357 /* Delay until the lg_xmit needs to expire */
1358 if (*tim_rem > lg_xmit->last_all_sent + idle_timeout - now) {
1359 *tim_rem = lg_xmit->last_all_sent + idle_timeout - now;
1360 ret = 1;
1361 }
1362 }
1363 } else if (lg_xmit->last_sent) {
1364 if (lg_xmit->last_sent + partial_timeout <= now) {
1365 /* Expire this entry */
1366 LL_DELETE(session->lg_xmit, lg_xmit);
1367 coap_block_delete_lg_xmit(session, lg_xmit);
1369 } else {
1370 /* Delay until the lg_xmit needs to expire */
1371 if (*tim_rem > lg_xmit->last_sent + partial_timeout - now) {
1372 *tim_rem = lg_xmit->last_sent + partial_timeout - now;
1373 ret = 1;
1374 }
1375 }
1376 }
1377 }
1378 return ret;
1379}
1380
1381#if COAP_CLIENT_SUPPORT
1382#if COAP_Q_BLOCK_SUPPORT
1383static coap_pdu_t *
1384coap_build_missing_pdu(coap_session_t *session, coap_lg_crcv_t *lg_crcv) {
1385 coap_pdu_t *pdu;
1386 coap_opt_filter_t drop_options;
1387 uint64_t token = STATE_TOKEN_FULL(lg_crcv->state_token, ++lg_crcv->retry_counter);
1388 uint8_t buf[8];
1389 size_t len = coap_encode_var_safe8(buf, sizeof(token), token);
1390
1391 memset(&drop_options, 0, sizeof(coap_opt_filter_t));
1394 pdu = coap_pdu_duplicate_lkd(&lg_crcv->pdu, session, len, buf,
1395 &drop_options);
1396 if (!pdu)
1397 return NULL;
1398 pdu->type = lg_crcv->last_type;
1399 return pdu;
1400}
1401
1402static void
1403coap_request_missing_q_block2(coap_session_t *session, coap_lg_crcv_t *lg_crcv) {
1404 uint8_t buf[8];
1405 uint32_t i;
1406 int block = -1; /* Last one seen */
1407 size_t sofar;
1408 size_t block_size;
1409 coap_pdu_t *pdu = NULL;
1410 int block_payload_set = -1;
1411
1412 if (session->block_mode & COAP_BLOCK_USE_M_Q_BLOCK) {
1413 /*
1414 * See if it is safe to use the single 'M' block variant of request
1415 *
1416 * If any blocks seen, then missing blocks are after range[0].end and
1417 * terminate on the last block or before range[1].begin if set.
1418 * If not defined or range[1].begin is in a different payload set then
1419 * safe to use M bit.
1420 */
1421 if (lg_crcv->rec_blocks.used &&
1422 (lg_crcv->rec_blocks.used < 2 ||
1423 ((lg_crcv->rec_blocks.range[0].end + 1) / COAP_MAX_PAYLOADS(session) !=
1424 (lg_crcv->rec_blocks.range[1].begin -1) / COAP_MAX_PAYLOADS(session)))) {
1425 block = lg_crcv->rec_blocks.range[0].end + 1;
1426 block_size = (size_t)1 << (lg_crcv->szx + 4);
1427 sofar = block * block_size;
1428 if (sofar < lg_crcv->total_len) {
1429 /* Ask for missing blocks */
1430 if (pdu == NULL) {
1431 pdu = coap_build_missing_pdu(session, lg_crcv);
1432 if (!pdu)
1433 return;
1434 }
1436 coap_encode_var_safe(buf, sizeof(buf),
1437 (block << 4) | (1 << 3) | lg_crcv->szx),
1438 buf);
1439 block_payload_set = block / COAP_MAX_PAYLOADS(session);
1440 goto send_it;
1441 }
1442 }
1443 }
1444 block = -1;
1445 for (i = 0; i < lg_crcv->rec_blocks.used; i++) {
1446 if (block < (int)lg_crcv->rec_blocks.range[i].begin &&
1447 lg_crcv->rec_blocks.range[i].begin != 0) {
1448 /* Ask for missing blocks */
1449 if (pdu == NULL) {
1450 pdu = coap_build_missing_pdu(session, lg_crcv);
1451 if (!pdu)
1452 continue;
1453 }
1454 block++;
1455 if (block_payload_set == -1)
1456 block_payload_set = block / COAP_MAX_PAYLOADS(session);
1457 for (; block < (int)lg_crcv->rec_blocks.range[i].begin &&
1458 block_payload_set == (block / COAP_MAX_PAYLOADS(session)); block++) {
1460 coap_encode_var_safe(buf, sizeof(buf),
1461 (block << 4) | (0 << 3) | lg_crcv->szx),
1462 buf);
1463 }
1464 }
1465 if (block < (int)lg_crcv->rec_blocks.range[i].end) {
1466 block = lg_crcv->rec_blocks.range[i].end;
1467 }
1468 }
1469 block_size = (size_t)1 << (lg_crcv->szx + 4);
1470 sofar = (block + 1) * block_size;
1471 if (sofar < lg_crcv->total_len) {
1472 /* Ask for trailing missing blocks */
1473 if (pdu == NULL) {
1474 pdu = coap_build_missing_pdu(session, lg_crcv);
1475 if (!pdu)
1476 return;
1477 }
1478 sofar = (lg_crcv->total_len + block_size - 1)/block_size;
1479 block++;
1480 if (block_payload_set == -1)
1481 block_payload_set = block / COAP_MAX_PAYLOADS(session);
1482 for (; block < (ssize_t)sofar &&
1483 block_payload_set == (block / COAP_MAX_PAYLOADS(session)); block++) {
1485 coap_encode_var_safe(buf, sizeof(buf),
1486 (block << 4) | (0 << 3) | lg_crcv->szx),
1487 buf);
1488 }
1489 }
1490send_it:
1491 if (pdu)
1492 coap_send_internal(session, pdu, NULL);
1493 lg_crcv->rec_blocks.retry++;
1494 if (block_payload_set != -1)
1495 lg_crcv->rec_blocks.processing_payload_set = block_payload_set;
1496 coap_ticks(&lg_crcv->rec_blocks.last_seen);
1497}
1498#endif /* COAP_Q_BLOCK_SUPPORT */
1499
1500/*
1501 * return 1 if there is a future expire time, else 0.
1502 * update tim_rem with remaining value if return is 1.
1503 */
1504int
1506 coap_tick_t *tim_rem) {
1507 coap_lg_crcv_t *lg_crcv;
1508 coap_lg_crcv_t *q;
1509 coap_tick_t partial_timeout;
1510#if COAP_Q_BLOCK_SUPPORT
1511 coap_tick_t receive_timeout = COAP_NON_RECEIVE_TIMEOUT_TICKS(session);
1512#endif /* COAP_Q_BLOCK_SUPPORT */
1513 int ret = 0;
1514
1515 *tim_rem = -1;
1516#if COAP_Q_BLOCK_SUPPORT
1517 if (COAP_PROTO_NOT_RELIABLE(session->proto) &&
1518 session->block_mode & COAP_BLOCK_HAS_Q_BLOCK)
1519 partial_timeout = COAP_NON_PARTIAL_TIMEOUT_TICKS(session);
1520 else
1521#endif /* COAP_Q_BLOCK_SUPPORT */
1522 partial_timeout = COAP_MAX_TRANSMIT_WAIT_TICKS(session);
1523
1524 LL_FOREACH_SAFE(session->lg_crcv, lg_crcv, q) {
1525 if (COAP_PROTO_RELIABLE(session->proto) || lg_crcv->last_type != COAP_MESSAGE_NON)
1526 goto check_expire;
1527
1528#if COAP_Q_BLOCK_SUPPORT
1529 if (lg_crcv->block_option == COAP_OPTION_Q_BLOCK2 && lg_crcv->rec_blocks.used) {
1530 size_t scaled_timeout = receive_timeout *
1531 ((size_t)1 << lg_crcv->rec_blocks.retry);
1532
1533 if (lg_crcv->rec_blocks.retry >= COAP_NON_MAX_RETRANSMIT(session)) {
1534 /* Done NON_MAX_RETRANSMIT retries */
1535 coap_handle_nack(session, &lg_crcv->pdu,
1537 goto expire;
1538 }
1539 if (lg_crcv->rec_blocks.last_seen + scaled_timeout <= now) {
1540 coap_request_missing_q_block2(session, lg_crcv);
1541 } else {
1542 if (*tim_rem > lg_crcv->rec_blocks.last_seen + scaled_timeout - now) {
1543 *tim_rem = lg_crcv->rec_blocks.last_seen + scaled_timeout - now;
1544 ret = 1;
1545 }
1546 }
1547 }
1548#endif /* COAP_Q_BLOCK_SUPPORT */
1549 /* Used for Block2 and Q-Block2 */
1550check_expire:
1551 if (!lg_crcv->observe_set && lg_crcv->last_used &&
1552 lg_crcv->last_used + partial_timeout <= now) {
1553#if COAP_Q_BLOCK_SUPPORT
1554expire:
1555#endif /* COAP_Q_BLOCK_SUPPORT */
1556 /* Expire this entry */
1557 LL_DELETE(session->lg_crcv, lg_crcv);
1558 coap_block_delete_lg_crcv(session, lg_crcv);
1559 } else if (!lg_crcv->observe_set && lg_crcv->last_used) {
1560 /* Delay until the lg_crcv needs to expire */
1561 if (*tim_rem > lg_crcv->last_used + partial_timeout - now) {
1562 *tim_rem = lg_crcv->last_used + partial_timeout - now;
1563 ret = 1;
1564 }
1565 }
1566 }
1567 return ret;
1568}
1569#endif /* COAP_CLIENT_SUPPORT */
1570
1571#if COAP_SERVER_SUPPORT
1572#if COAP_Q_BLOCK_SUPPORT
1573static coap_pdu_t *
1574pdu_408_build(coap_session_t *session, coap_lg_srcv_t *lg_srcv) {
1575 coap_pdu_t *pdu;
1576 uint8_t buf[4];
1577
1579 COAP_RESPONSE_CODE(408),
1580 coap_new_message_id_lkd(session),
1582 if (!pdu)
1583 return NULL;
1584 if (lg_srcv->last_token)
1585 coap_add_token(pdu, lg_srcv->last_token->length, lg_srcv->last_token->s);
1587 coap_encode_var_safe(buf, sizeof(buf),
1589 buf);
1590 pdu->token[pdu->used_size++] = COAP_PAYLOAD_START;
1591 pdu->data = pdu->token + pdu->used_size;
1592 return pdu;
1593}
1594
1595static int
1596add_408_block(coap_pdu_t *pdu, int block) {
1597 size_t len;
1598 uint8_t val[8];
1599
1600 assert(block >= 0 && block < (1 << 20));
1601
1602 if (block < 0 || block >= (1 << 20)) {
1603 return 0;
1604 } else if (block < 24) {
1605 len = 1;
1606 val[0] = block;
1607 } else if (block < 0x100) {
1608 len = 2;
1609 val[0] = 24;
1610 val[1] = block;
1611 } else if (block < 0x10000) {
1612 len = 3;
1613 val[0] = 25;
1614 val[1] = block >> 8;
1615 val[2] = block & 0xff;
1616 } else { /* Largest block number is 2^^20 - 1 */
1617 len = 4;
1618 val[0] = 26;
1619 val[1] = block >> 16;
1620 val[2] = (block >> 8) & 0xff;
1621 val[3] = block & 0xff;
1622 }
1623 if (coap_pdu_check_resize(pdu, pdu->used_size + len)) {
1624 memcpy(&pdu->token[pdu->used_size], val, len);
1625 pdu->used_size += len;
1626 return 1;
1627 }
1628 return 0;
1629}
1630#endif /* COAP_Q_BLOCK_SUPPORT */
1631#endif /* COAP_SERVER_SUPPORT */
1632
1633static int
1634check_if_received_block(coap_rblock_t *rec_blocks, uint32_t block_num) {
1635 uint32_t i;
1636
1637 for (i = 0; i < rec_blocks->used; i++) {
1638 if (block_num < rec_blocks->range[i].begin)
1639 return 0;
1640 if (block_num <= rec_blocks->range[i].end)
1641 return 1;
1642 }
1643 return 0;
1644}
1645
1646#if COAP_SERVER_SUPPORT
1647static int
1648check_if_next_block(coap_rblock_t *rec_blocks, uint32_t block_num) {
1649 if (rec_blocks->used == 0) {
1650 return block_num == 0 ? 1 : 0;
1651 }
1652 if (rec_blocks->range[rec_blocks->used-1].end + 1 == block_num)
1653 return 1;
1654
1655 return 0;
1656}
1657#endif /* COAP_SERVER_SUPPORT */
1658
1659static int
1661 uint32_t i;
1662 uint32_t block = 0;
1663
1664 if (rec_blocks->total_blocks == 0) {
1665 /* Not seen block with More bit unset yet */
1666 return 0;
1667 }
1668
1669 for (i = 0; i < rec_blocks->used; i++) {
1670 if (block < rec_blocks->range[i].begin)
1671 return 0;
1672 if (block < rec_blocks->range[i].end)
1673 block = rec_blocks->range[i].end;
1674 }
1675 return 1;
1676}
1677
1678#if COAP_CLIENT_SUPPORT
1679#if COAP_Q_BLOCK_SUPPORT
1680static int
1681check_all_blocks_in_for_payload_set(coap_session_t *session,
1682 coap_rblock_t *rec_blocks) {
1683 if (rec_blocks->used &&
1684 (rec_blocks->range[0].end + 1) / COAP_MAX_PAYLOADS(session) >
1685 rec_blocks->processing_payload_set)
1686 return 1;
1687 return 0;
1688}
1689
1690static int
1691check_any_blocks_next_payload_set(coap_session_t *session,
1692 coap_rblock_t *rec_blocks) {
1693 if (rec_blocks->used > 1 &&
1694 rec_blocks->range[1].begin / COAP_MAX_PAYLOADS(session) ==
1695 rec_blocks->processing_payload_set)
1696 return 1;
1697 return 0;
1698}
1699#endif /* COAP_Q_BLOCK_SUPPORT */
1700#endif /* COAP_CLIENT_SUPPORT */
1701
1702#if COAP_SERVER_SUPPORT
1703/*
1704 * return 1 if there is a future expire time, else 0.
1705 * update tim_rem with remaining value if return is 1.
1706 */
1707int
1709 coap_tick_t *tim_rem) {
1710 coap_lg_srcv_t *lg_srcv;
1711 coap_lg_srcv_t *q;
1712 coap_tick_t partial_timeout;
1713#if COAP_Q_BLOCK_SUPPORT
1714 coap_tick_t receive_timeout = COAP_NON_RECEIVE_TIMEOUT_TICKS(session);
1715#endif /* COAP_Q_BLOCK_SUPPORT */
1716 int ret = 0;
1717
1718 *tim_rem = -1;
1719#if COAP_Q_BLOCK_SUPPORT
1720 if (COAP_PROTO_NOT_RELIABLE(session->proto) &&
1721 session->block_mode & COAP_BLOCK_HAS_Q_BLOCK)
1722 partial_timeout = COAP_NON_PARTIAL_TIMEOUT_TICKS(session);
1723 else
1724#endif /* COAP_Q_BLOCK_SUPPORT */
1725 partial_timeout = COAP_MAX_TRANSMIT_WAIT_TICKS(session);
1726
1727 LL_FOREACH_SAFE(session->lg_srcv, lg_srcv, q) {
1728 if (COAP_PROTO_RELIABLE(session->proto) || lg_srcv->last_type != COAP_MESSAGE_NON)
1729 goto check_expire;
1730
1731#if COAP_Q_BLOCK_SUPPORT
1732 if (lg_srcv->block_option == COAP_OPTION_Q_BLOCK1 && lg_srcv->rec_blocks.used) {
1733 size_t scaled_timeout = receive_timeout *
1734 ((size_t)1 << lg_srcv->rec_blocks.retry);
1735
1736 if (lg_srcv->rec_blocks.retry >= COAP_NON_MAX_RETRANSMIT(session)) {
1737 /* Done NON_MAX_RETRANSMIT retries */
1738 goto expire;
1739 }
1740 if (lg_srcv->rec_blocks.last_seen + scaled_timeout <= now) {
1741 uint32_t i;
1742 int block = -1; /* Last one seen */
1743 size_t block_size = (size_t)1 << (lg_srcv->szx + 4);
1744 size_t final_block = (lg_srcv->total_len + block_size - 1)/block_size - 1;
1745 size_t cur_payload;
1746 size_t last_payload_block;
1747 coap_pdu_t *pdu = NULL;
1748 size_t no_blocks = 0;
1749
1750 /* Need to count the number of missing blocks */
1751 for (i = 0; i < lg_srcv->rec_blocks.used; i++) {
1752 if (block < (int)lg_srcv->rec_blocks.range[i].begin &&
1753 lg_srcv->rec_blocks.range[i].begin != 0) {
1754 block++;
1755 no_blocks += lg_srcv->rec_blocks.range[i].begin - block;
1756 }
1757 if (block < (int)lg_srcv->rec_blocks.range[i].end) {
1758 block = lg_srcv->rec_blocks.range[i].end;
1759 }
1760 }
1761 if (no_blocks == 0 && block == (int)final_block)
1762 goto expire;
1763
1764 /* Include missing up to end of current payload or total amount */
1765 cur_payload = block / COAP_MAX_PAYLOADS(session);
1766 last_payload_block = (cur_payload + 1) * COAP_MAX_PAYLOADS(session) - 1;
1767 if (final_block > last_payload_block) {
1768 final_block = last_payload_block;
1769 }
1770 no_blocks += final_block - block;
1771 if (no_blocks == 0) {
1772 /* Add in the blocks out of the next payload */
1773 final_block = (lg_srcv->total_len + block_size - 1)/block_size - 1;
1774 last_payload_block += COAP_MAX_PAYLOADS(session);
1775 if (final_block > last_payload_block) {
1776 final_block = last_payload_block;
1777 }
1778 }
1779 /* Ask for the missing blocks */
1780 block = -1;
1781 for (i = 0; i < lg_srcv->rec_blocks.used; i++) {
1782 if (block < (int)lg_srcv->rec_blocks.range[i].begin &&
1783 lg_srcv->rec_blocks.range[i].begin != 0) {
1784 /* Report on missing blocks */
1785 if (pdu == NULL) {
1786 pdu = pdu_408_build(session, lg_srcv);
1787 if (!pdu)
1788 continue;
1789 }
1790 block++;
1791 for (; block < (int)lg_srcv->rec_blocks.range[i].begin; block++) {
1792 if (!add_408_block(pdu, block)) {
1793 break;
1794 }
1795 }
1796 }
1797 if (block < (int)lg_srcv->rec_blocks.range[i].end) {
1798 block = lg_srcv->rec_blocks.range[i].end;
1799 }
1800 }
1801 block++;
1802 for (; block <= (int)final_block; block++) {
1803 if (pdu == NULL) {
1804 pdu = pdu_408_build(session, lg_srcv);
1805 if (!pdu)
1806 continue;
1807 }
1808 if (!add_408_block(pdu, block)) {
1809 break;
1810 }
1811 }
1812 if (pdu)
1813 coap_send_internal(session, pdu, NULL);
1814 lg_srcv->rec_blocks.retry++;
1815 coap_ticks(&lg_srcv->rec_blocks.last_seen);
1816 }
1817 if (*tim_rem > lg_srcv->rec_blocks.last_seen + scaled_timeout - now) {
1818 *tim_rem = lg_srcv->rec_blocks.last_seen + scaled_timeout - now;
1819 ret = 1;
1820 }
1821 }
1822#endif /* COAP_Q_BLOCK_SUPPORT */
1823 /* Used for Block1 and Q-Block1 */
1824check_expire:
1825 if (lg_srcv->no_more_seen)
1826 partial_timeout = 10 * COAP_TICKS_PER_SECOND;
1827 if (lg_srcv->last_used && lg_srcv->last_used + partial_timeout <= now) {
1828#if COAP_Q_BLOCK_SUPPORT
1829expire:
1830#endif /* COAP_Q_BLOCK_SUPPORT */
1831 /* Expire this entry */
1832 if (lg_srcv->no_more_seen && lg_srcv->block_option == COAP_OPTION_BLOCK1) {
1833 /*
1834 * Need to send a separate 4.08 to indicate missing blocks
1835 * Using NON is permissable as per
1836 * https://datatracker.ietf.org/doc/html/rfc7252#section-5.2.3
1837 */
1838 coap_pdu_t *pdu;
1839
1841 COAP_RESPONSE_CODE(408),
1842 coap_new_message_id_lkd(session),
1844 if (pdu) {
1845 if (lg_srcv->last_token)
1846 coap_add_token(pdu, lg_srcv->last_token->length, lg_srcv->last_token->s);
1847 coap_add_data(pdu, sizeof("Missing interim block")-1,
1848 (const uint8_t *)"Missing interim block");
1849 coap_send_internal(session, pdu, NULL);
1850 }
1851 }
1852 LL_DELETE(session->lg_srcv, lg_srcv);
1853 coap_block_delete_lg_srcv(session, lg_srcv);
1854 } else if (lg_srcv->last_used) {
1855 /* Delay until the lg_srcv needs to expire */
1856 if (*tim_rem > lg_srcv->last_used + partial_timeout - now) {
1857 *tim_rem = lg_srcv->last_used + partial_timeout - now;
1858 ret = 1;
1859 }
1860 }
1861 }
1862 return ret;
1863}
1864#endif /* COAP_SERVER_SUPPORT */
1865
1866#if COAP_Q_BLOCK_SUPPORT
1867/*
1868 * pdu is always released before return IF COAP_SEND_INC_PDU
1869 */
1871coap_send_q_blocks(coap_session_t *session,
1872 coap_lg_xmit_t *lg_xmit,
1873 coap_block_b_t block,
1874 coap_pdu_t *pdu,
1875 coap_send_pdu_t send_pdu) {
1876 coap_pdu_t *block_pdu = NULL;
1877 coap_opt_filter_t drop_options;
1879 uint64_t token;
1880 const uint8_t *ptoken;
1881 uint8_t ltoken[8];
1882 size_t ltoken_length;
1883 uint32_t delayqueue_cnt = 0;
1884
1885 if (!lg_xmit) {
1886 if (send_pdu == COAP_SEND_INC_PDU)
1887 return coap_send_internal(session, pdu, NULL);
1888 return COAP_INVALID_MID;
1889 }
1890
1891 if (pdu->type == COAP_MESSAGE_CON) {
1892 coap_queue_t *delayqueue;
1893
1894 delayqueue_cnt = session->con_active +
1895 (send_pdu == COAP_SEND_INC_PDU ? 1 : 0);
1896 LL_FOREACH(session->delayqueue, delayqueue) {
1897 delayqueue_cnt++;
1898 }
1899 }
1900 pdu->lg_xmit = lg_xmit;
1901 if (block.m &&
1902 ((pdu->type == COAP_MESSAGE_NON &&
1903 ((block.num + 1) % COAP_MAX_PAYLOADS(session)) + 1 !=
1904 COAP_MAX_PAYLOADS(session)) ||
1905 (pdu->type == COAP_MESSAGE_ACK &&
1906 lg_xmit->option == COAP_OPTION_Q_BLOCK2) ||
1907 (pdu->type == COAP_MESSAGE_CON &&
1908 delayqueue_cnt < COAP_NSTART(session)) ||
1909 COAP_PROTO_RELIABLE(session->proto))) {
1910 /* Allocate next pdu if there is headroom */
1911 if (COAP_PDU_IS_RESPONSE(pdu)) {
1912 ptoken = pdu->actual_token.s;
1913 ltoken_length = pdu->actual_token.length;
1914 } else {
1915 token = STATE_TOKEN_FULL(lg_xmit->b.b1.state_token,++lg_xmit->b.b1.count);
1916 ltoken_length = coap_encode_var_safe8(ltoken, sizeof(token), token);
1917 ptoken = ltoken;
1918 }
1919
1920 memset(&drop_options, 0, sizeof(coap_opt_filter_t));
1921 coap_option_filter_set(&drop_options, lg_xmit->option);
1922 block_pdu = coap_pdu_duplicate_lkd(pdu, session,
1923 ltoken_length,
1924 ptoken, &drop_options);
1925 if (block_pdu->type == COAP_MESSAGE_ACK)
1926 block_pdu->type = COAP_MESSAGE_CON;
1927 }
1928
1929 /* Send initial pdu (which deletes 'pdu') */
1930 if (send_pdu == COAP_SEND_INC_PDU &&
1931 (mid = coap_send_internal(session, pdu, NULL)) == COAP_INVALID_MID) {
1932 /* Not expected, underlying issue somewhere */
1933 coap_delete_pdu_lkd(block_pdu);
1934 return COAP_INVALID_MID;
1935 }
1936
1937 while (block_pdu) {
1938 coap_pdu_t *t_pdu = NULL;
1939 uint8_t buf[8];
1940 size_t chunk = ((size_t)1 << (lg_xmit->blk_size + 4));
1941
1942 block.num++;
1943 lg_xmit->offset = block.num * chunk;
1944 block.m = lg_xmit->offset + chunk < lg_xmit->length;
1945 if (block.m && ((block_pdu->type == COAP_MESSAGE_NON &&
1946 (block.num % COAP_MAX_PAYLOADS(session)) + 1 !=
1947 COAP_MAX_PAYLOADS(session)) ||
1948 (block_pdu->type == COAP_MESSAGE_CON &&
1949 delayqueue_cnt + 1 < COAP_NSTART(session)) ||
1950 COAP_PROTO_RELIABLE(session->proto))) {
1951 /*
1952 * Send following block if
1953 * NON and more in MAX_PAYLOADS
1954 * CON and NSTART allows it (based on number in delayqueue)
1955 * Reliable transport
1956 */
1957 if (COAP_PDU_IS_RESPONSE(block_pdu)) {
1958 ptoken = block_pdu->actual_token.s;
1959 ltoken_length = block_pdu->actual_token.length;
1960 } else {
1961 token = STATE_TOKEN_FULL(lg_xmit->b.b1.state_token,++lg_xmit->b.b1.count);
1962 ltoken_length = coap_encode_var_safe8(ltoken, sizeof(token), token);
1963 ptoken = ltoken;
1964 }
1965 t_pdu = coap_pdu_duplicate_lkd(block_pdu, session,
1966 ltoken_length, ptoken, &drop_options);
1967 }
1968 if (!coap_update_option(block_pdu, lg_xmit->option,
1970 sizeof(buf),
1971 ((block.num) << 4) |
1972 (block.m << 3) |
1973 block.szx),
1974 buf)) {
1975 coap_log_warn("Internal update issue option\n");
1976 coap_delete_pdu_lkd(block_pdu);
1977 coap_delete_pdu_lkd(t_pdu);
1978 break;
1979 }
1980
1981 if (!coap_add_block(block_pdu,
1982 lg_xmit->length,
1983 lg_xmit->data,
1984 block.num,
1985 block.szx)) {
1986 coap_log_warn("Internal update issue data\n");
1987 coap_delete_pdu_lkd(block_pdu);
1988 coap_delete_pdu_lkd(t_pdu);
1989 break;
1990 }
1991 if (COAP_PDU_IS_RESPONSE(block_pdu)) {
1992 lg_xmit->last_block = block.num;
1993 }
1994 mid = coap_send_internal(session, block_pdu, NULL);
1995 if (mid == COAP_INVALID_MID) {
1996 /* Not expected, underlying issue somewhere */
1997 coap_delete_pdu_lkd(t_pdu);
1998 return COAP_INVALID_MID;
1999 }
2000 block_pdu = t_pdu;
2001 }
2002 if (!block.m) {
2003 lg_xmit->last_payload = 0;
2004 coap_ticks(&lg_xmit->last_all_sent);
2005 } else
2006 coap_ticks(&lg_xmit->last_payload);
2007 return mid;
2008}
2009
2010#if COAP_CLIENT_SUPPORT
2012coap_block_check_q_block1_xmit(coap_session_t *session, coap_tick_t now) {
2013 coap_lg_xmit_t *lg_xmit;
2014 coap_lg_xmit_t *q;
2015 coap_tick_t timed_out;
2016 coap_tick_t tim_rem = (coap_tick_t)-1;
2017
2018 LL_FOREACH_SAFE(session->lg_xmit, lg_xmit, q) {
2019 coap_tick_t non_timeout = lg_xmit->non_timeout_random_ticks;
2020
2021 if (now < non_timeout)
2022 return non_timeout - now;
2023 timed_out = now - non_timeout;
2024
2025 if (lg_xmit->last_payload) {
2026 if (lg_xmit->last_payload <= timed_out) {
2027 /* Send off the next MAX_PAYLOAD set */
2028 coap_block_b_t block;
2029 size_t chunk = (size_t)1 << (lg_xmit->blk_size + 4);
2030
2031 memset(&block, 0, sizeof(block));
2032 block.num = (uint32_t)(lg_xmit->offset / chunk);
2033 block.m = lg_xmit->offset + chunk < lg_xmit->length;
2034 block.szx = lg_xmit->blk_size;
2035 coap_send_q_blocks(session, lg_xmit, block, &lg_xmit->pdu, COAP_SEND_SKIP_PDU);
2036 if (tim_rem > non_timeout)
2037 tim_rem = non_timeout;
2038 } else {
2039 /* Delay until the next MAX_PAYLOAD needs to be sent off */
2040 if (tim_rem > lg_xmit->last_payload - timed_out)
2041 tim_rem = lg_xmit->last_payload - timed_out;
2042 }
2043 } else if (lg_xmit->last_all_sent) {
2044 non_timeout = COAP_NON_TIMEOUT_TICKS(session);
2045 if (lg_xmit->last_all_sent + 4 * non_timeout <= now) {
2046 /* Expire this entry */
2047 LL_DELETE(session->lg_xmit, lg_xmit);
2048 coap_block_delete_lg_xmit(session, lg_xmit);
2049 } else {
2050 /* Delay until the lg_xmit needs to expire */
2051 if (tim_rem > lg_xmit->last_all_sent + 4 * non_timeout - now)
2052 tim_rem = lg_xmit->last_all_sent + 4 * non_timeout - now;
2053 }
2054 }
2055 }
2056 return tim_rem;
2057}
2058#endif /* COAP_CLIENT_SUPPORT */
2059
2060#if COAP_SERVER_SUPPORT
2062coap_block_check_q_block2_xmit(coap_session_t *session, coap_tick_t now) {
2063 coap_lg_xmit_t *lg_xmit;
2064 coap_lg_xmit_t *q;
2065 coap_tick_t timed_out;
2066 coap_tick_t tim_rem = (coap_tick_t)-1;
2067
2068 LL_FOREACH_SAFE(session->lg_xmit, lg_xmit, q) {
2069 coap_tick_t non_timeout = lg_xmit->non_timeout_random_ticks;
2070
2071 if (now < non_timeout)
2072 return non_timeout - now;
2073 timed_out = now - non_timeout;
2074
2075 if (lg_xmit->last_payload) {
2076 if (lg_xmit->last_payload <= timed_out) {
2077 /* Send off the next MAX_PAYLOAD set */
2078 coap_block_b_t block;
2079 size_t chunk = (size_t)1 << (lg_xmit->blk_size + 4);
2080
2081 memset(&block, 0, sizeof(block));
2082 block.num = (uint32_t)(lg_xmit->offset / chunk);
2083 block.m = lg_xmit->offset + chunk < lg_xmit->length;
2084 block.szx = lg_xmit->blk_size;
2085 if (block.num == (uint32_t)lg_xmit->last_block)
2086 coap_send_q_blocks(session, lg_xmit, block, &lg_xmit->pdu, COAP_SEND_SKIP_PDU);
2087 if (tim_rem > non_timeout)
2088 tim_rem = non_timeout;
2089 } else {
2090 /* Delay until the next MAX_PAYLOAD needs to be sent off */
2091 if (tim_rem > lg_xmit->last_payload - timed_out)
2092 tim_rem = lg_xmit->last_payload - timed_out;
2093 }
2094 } else if (lg_xmit->last_all_sent) {
2095 non_timeout = COAP_NON_TIMEOUT_TICKS(session);
2096 if (lg_xmit->last_all_sent + 4 * non_timeout <= now) {
2097 /* Expire this entry */
2098 LL_DELETE(session->lg_xmit, lg_xmit);
2099 coap_block_delete_lg_xmit(session, lg_xmit);
2100 } else {
2101 /* Delay until the lg_xmit needs to expire */
2102 if (tim_rem > lg_xmit->last_all_sent + 4 * non_timeout - now)
2103 tim_rem = lg_xmit->last_all_sent + 4 * non_timeout - now;
2104 }
2105 }
2106 }
2107 return tim_rem;
2108}
2109#endif /* COAP_SERVER_SUPPORT */
2110#endif /* COAP_Q_BLOCK_SUPPORT */
2111
2112#if COAP_CLIENT_SUPPORT
2113/*
2114 * If Observe = 0, save the token away and return NULL
2115 * Else If Observe = 1, return the saved token for this block
2116 * Else, return NULL
2117 */
2118static coap_bin_const_t *
2119track_fetch_observe(coap_pdu_t *pdu, coap_lg_crcv_t *lg_crcv,
2120 uint32_t block_num, coap_bin_const_t *token) {
2121 /* Need to handle Observe for large FETCH */
2122 coap_opt_iterator_t opt_iter;
2124 &opt_iter);
2125
2126 if (opt && lg_crcv) {
2127 int observe_action = -1;
2128 coap_bin_const_t **tmp;
2129
2130 observe_action = coap_decode_var_bytes(coap_opt_value(opt),
2131 coap_opt_length(opt));
2132 if (observe_action == COAP_OBSERVE_ESTABLISH) {
2133 /* Save the token in lg_crcv */
2134 if (lg_crcv->obs_token_cnt <= block_num) {
2135 size_t i;
2136
2137 tmp = coap_realloc_type(COAP_STRING, lg_crcv->obs_token,
2138 (block_num + 1) * sizeof(lg_crcv->obs_token[0]));
2139 if (tmp == NULL)
2140 return NULL;
2141 lg_crcv->obs_token = tmp;
2142 for (i = lg_crcv->obs_token_cnt; i < block_num + 1; i++) {
2143 lg_crcv->obs_token[i] = NULL;
2144 }
2145 }
2146 coap_delete_bin_const(lg_crcv->obs_token[block_num]);
2147
2148 lg_crcv->obs_token_cnt = block_num + 1;
2149 lg_crcv->obs_token[block_num] = coap_new_bin_const(token->s,
2150 token->length);
2151 if (lg_crcv->obs_token[block_num] == NULL)
2152 return NULL;
2153 } else if (observe_action == COAP_OBSERVE_CANCEL) {
2154 /* Use the token in lg_crcv */
2155 if (block_num < lg_crcv->obs_token_cnt) {
2156 return lg_crcv->obs_token[block_num];
2157 }
2158 }
2159 }
2160 return NULL;
2161}
2162
2163#if COAP_Q_BLOCK_SUPPORT
2165coap_send_q_block1(coap_session_t *session,
2166 coap_block_b_t block,
2167 coap_pdu_t *request,
2168 coap_send_pdu_t send_request) {
2169 /* Need to send up to MAX_PAYLOAD blocks if this is a Q_BLOCK1 */
2170 coap_lg_xmit_t *lg_xmit;
2171 uint64_t token_match =
2173 request->actual_token.length));
2174
2175 LL_FOREACH(session->lg_xmit, lg_xmit) {
2176 if (lg_xmit->option == COAP_OPTION_Q_BLOCK1 &&
2177 (token_match == STATE_TOKEN_BASE(lg_xmit->b.b1.state_token) ||
2178 token_match ==
2180 lg_xmit->b.b1.app_token->length))))
2181 break;
2182 /* try out the next one */
2183 }
2184 return coap_send_q_blocks(session, lg_xmit, block, request, send_request);
2185}
2186#endif /* COAP_Q_BLOCK_SUPPORT */
2187#endif /* COAP_CLIENT_SUPPORT */
2188
2189#if COAP_SERVER_SUPPORT
2190#if COAP_Q_BLOCK_SUPPORT
2191/*
2192 * response is always released before return IF COAP_SEND_INC_PDU
2193 */
2195coap_send_q_block2(coap_session_t *session,
2196 coap_resource_t *resource,
2197 const coap_string_t *query,
2198 coap_pdu_code_t request_method,
2199 coap_block_b_t block,
2200 coap_pdu_t *response,
2201 coap_send_pdu_t send_response) {
2202 /* Need to send up to MAX_PAYLOAD blocks if this is a Q_BLOCK2 */
2203 coap_lg_xmit_t *lg_xmit;
2204 coap_string_t empty = { 0, NULL};
2205
2206 LL_FOREACH(session->lg_xmit, lg_xmit) {
2207 if (lg_xmit->option == COAP_OPTION_Q_BLOCK2 &&
2208 resource == lg_xmit->b.b2.resource &&
2209 request_method == lg_xmit->b.b2.request_method &&
2210 coap_string_equal(query ? query : &empty,
2211 lg_xmit->b.b2.query ? lg_xmit->b.b2.query : &empty))
2212 break;
2213 }
2214 return coap_send_q_blocks(session, lg_xmit, block, response, send_response);
2215}
2216#endif /* COAP_Q_BLOCK_SUPPORT */
2217#endif /* COAP_SERVER_SUPPORT */
2218
2219#if COAP_CLIENT_SUPPORT
2220#if COAP_Q_BLOCK_SUPPORT
2221/*
2222 * Send out a test PDU for Q-Block.
2223 */
2225coap_block_test_q_block(coap_session_t *session, coap_pdu_t *actual) {
2226 coap_pdu_t *pdu;
2227 uint8_t token[8];
2228 size_t token_len;
2229 uint8_t buf[4];
2230 coap_mid_t mid;
2231
2232#if NDEBUG
2233 (void)actual;
2234#endif /* NDEBUG */
2235 assert(session->block_mode & COAP_BLOCK_TRY_Q_BLOCK &&
2236 session->type == COAP_SESSION_TYPE_CLIENT &&
2237 COAP_PDU_IS_REQUEST(actual));
2238
2239 coap_log_debug("Testing for Q-Block support\n");
2240 /* RFC9177 Section 4.1 when checking if available */
2242 coap_new_message_id_lkd(session),
2244 if (!pdu) {
2245 return COAP_INVALID_MID;
2246 }
2247
2248 coap_session_new_token(session, &token_len, token);
2249 coap_add_token(pdu, token_len, token);
2250 /* Use a resource that the server MUST support (.well-known/core) */
2252 11, (const uint8_t *)".well-known");
2254 4, (const uint8_t *)"core");
2255 /*
2256 * M needs to be unset as 'asking' for only the first block using
2257 * Q-Block2 as a test for server support.
2258 * See RFC9177 Section 4.4 Using the Q-Block2 Option.
2259 *
2260 * As the client is asking for 16 byte chunks, it is unlikely that
2261 * the .well-known/core response will be 16 bytes or less, so
2262 * if the server supports Q-Block, it will be forced to respond with
2263 * a Q-Block2, so the client can detect the server Q-Block support.
2264 */
2266 coap_encode_var_safe(buf, sizeof(buf),
2267 (0 << 4) | (0 << 3) | 0),
2268 buf);
2269 set_block_mode_probe_q(session->block_mode);
2270 mid = coap_send_internal(session, pdu, NULL);
2271 if (mid == COAP_INVALID_MID)
2272 return COAP_INVALID_MID;
2273 session->remote_test_mid = mid;
2274 return mid;
2275}
2276#endif /* COAP_Q_BLOCK_SUPPORT */
2277
2280 coap_lg_xmit_t *lg_xmit) {
2281 coap_block_b_t block;
2282 coap_lg_crcv_t *lg_crcv;
2283 uint64_t state_token = STATE_TOKEN_FULL(++session->tx_token, 1);
2284 size_t token_options = pdu->data ? (size_t)(pdu->data - pdu->token) :
2285 pdu->used_size;
2286 size_t data_len;
2287 int observe_action = -1;
2288
2289 if (lg_xmit) {
2290 coap_opt_iterator_t opt_iter;
2291 coap_opt_t *opt;
2292
2293 opt = coap_check_option(pdu, COAP_OPTION_OBSERVE, &opt_iter);
2294
2295 if (opt) {
2296 observe_action = coap_decode_var_bytes(coap_opt_value(opt),
2297 coap_opt_length(opt));
2298 }
2299 }
2300 if (observe_action == COAP_OBSERVE_ESTABLISH) {
2301 data_len = lg_xmit->length;
2302 } else {
2303 lg_xmit = NULL;
2304 data_len = pdu->data ?
2305 pdu->used_size - (pdu->data - pdu->token) : 0;
2306 }
2307
2308 lg_crcv = coap_malloc_type(COAP_LG_CRCV, sizeof(coap_lg_crcv_t));
2309
2310 if (lg_crcv == NULL)
2311 return NULL;
2312
2313 coap_log_debug("** %s: lg_crcv %p initialized - stateless token xxxx%012llx\n",
2314 coap_session_str(session), (void *)lg_crcv,
2315 STATE_TOKEN_BASE(state_token));
2316 memset(lg_crcv, 0, sizeof(coap_lg_crcv_t));
2317 lg_crcv->initial = 1;
2318 coap_ticks(&lg_crcv->last_used);
2319 /* Set up skeletal PDU to use as a basis for all the subsequent blocks */
2320 memcpy(&lg_crcv->pdu, pdu, sizeof(lg_crcv->pdu));
2321 /* Make sure that there is space for increased token + option change */
2322 lg_crcv->pdu.max_size = token_options + data_len + 9;
2323 lg_crcv->pdu.used_size = token_options + data_len;
2324 lg_crcv->pdu.token = coap_malloc_type(COAP_PDU_BUF,
2325 token_options + data_len + lg_crcv->pdu.max_hdr_size);
2326 if (!lg_crcv->pdu.token) {
2327 coap_block_delete_lg_crcv(session, lg_crcv);
2328 return NULL;
2329 }
2330 lg_crcv->pdu.token += lg_crcv->pdu.max_hdr_size;
2331 memcpy(lg_crcv->pdu.token, pdu->token, token_options);
2332 if (lg_crcv->pdu.actual_token.length < COAP_TOKEN_EXT_1B_BIAS) {
2333 lg_crcv->pdu.actual_token.s = &lg_crcv->pdu.token[0];
2334 } else if (lg_crcv->pdu.actual_token.length < COAP_TOKEN_EXT_2B_BIAS) {
2335 lg_crcv->pdu.actual_token.s = &lg_crcv->pdu.token[1];
2336 } else if (lg_crcv->pdu.actual_token.length <= COAP_TOKEN_EXT_MAX) {
2337 lg_crcv->pdu.actual_token.s = &lg_crcv->pdu.token[2];
2338 }
2339
2340 if (lg_crcv->pdu.data) {
2341 lg_crcv->pdu.data = lg_crcv->pdu.token + token_options;
2342 assert(pdu->data);
2343 memcpy(lg_crcv->pdu.data, lg_xmit ? lg_xmit->data : pdu->data, data_len);
2344 }
2345
2346 /* Need to keep original token for updating response PDUs */
2347 lg_crcv->app_token = coap_new_binary(pdu->actual_token.length);
2348 if (!lg_crcv->app_token) {
2349 coap_block_delete_lg_crcv(session, lg_crcv);
2350 return NULL;
2351 }
2352 memcpy(lg_crcv->app_token->s, pdu->actual_token.s, pdu->actual_token.length);
2353
2354 /* Need to set up a base token for actual communications if retries needed */
2355 lg_crcv->retry_counter = 1;
2356 lg_crcv->state_token = state_token;
2357
2358 if (pdu->code == COAP_REQUEST_CODE_FETCH) {
2359 coap_bin_const_t *new_token;
2360
2361 /* Need to save/restore Observe Token for large FETCH */
2362 new_token = track_fetch_observe(pdu, lg_crcv, 0, &pdu->actual_token);
2363 if (new_token)
2364 coap_update_token(pdu, new_token->length, new_token->s);
2365 }
2366
2367 if (coap_get_block_b(session, pdu, COAP_OPTION_BLOCK1, &block)) {
2368 /* In case it is there - must not be in continuing request PDUs */
2369 lg_crcv->o_block_option = COAP_OPTION_BLOCK1;
2370 lg_crcv->o_blk_size = block.aszx;
2371 coap_remove_option(&lg_crcv->pdu, COAP_OPTION_BLOCK1);
2372 }
2373
2374 if (lg_xmit)
2375 lg_xmit->lg_crcv = lg_crcv;
2376
2377 return lg_crcv;
2378}
2379
2380void
2382 coap_lg_crcv_t *lg_crcv) {
2383 size_t i;
2384 coap_lg_xmit_t *lg_xmit;
2385
2386#if (COAP_MAX_LOGGING_LEVEL < _COAP_LOG_DEBUG)
2387 (void)session;
2388#endif
2389 if (lg_crcv == NULL)
2390 return;
2391
2392 LL_FOREACH(session->lg_xmit, lg_xmit) {
2393 if (lg_xmit->lg_crcv == lg_crcv) {
2394 lg_xmit->lg_crcv = NULL;
2395 break;
2396 }
2397 }
2398
2399 if (lg_crcv->pdu.token)
2400 coap_free_type(COAP_PDU_BUF, lg_crcv->pdu.token - lg_crcv->pdu.max_hdr_size);
2402 coap_log_debug("** %s: lg_crcv %p released\n",
2403 coap_session_str(session), (void *)lg_crcv);
2404 coap_delete_binary(lg_crcv->app_token);
2405 for (i = 0; i < lg_crcv->obs_token_cnt; i++) {
2406 coap_delete_bin_const(lg_crcv->obs_token[i]);
2407 }
2409 coap_free_type(COAP_LG_CRCV, lg_crcv);
2410}
2411#endif /* COAP_CLIENT_SUPPORT */
2412
2413#if COAP_SERVER_SUPPORT
2414void
2416 coap_lg_srcv_t *lg_srcv) {
2417#if (COAP_MAX_LOGGING_LEVEL < _COAP_LOG_DEBUG)
2418 (void)session;
2419#endif
2420 if (lg_srcv == NULL)
2421 return;
2422
2426 coap_log_debug("** %s: lg_srcv %p released\n",
2427 coap_session_str(session), (void *)lg_srcv);
2428 coap_free_type(COAP_LG_SRCV, lg_srcv);
2429}
2430#endif /* COAP_SERVER_SUPPORT */
2431
2432void
2434 coap_lg_xmit_t *lg_xmit) {
2435 if (lg_xmit == NULL)
2436 return;
2437
2438 if (lg_xmit->release_func) {
2439 coap_lock_callback(session->context, lg_xmit->release_func(session, lg_xmit->app_ptr));
2440 }
2441 if (lg_xmit->pdu.token) {
2442 coap_free_type(COAP_PDU_BUF, lg_xmit->pdu.token - lg_xmit->pdu.max_hdr_size);
2443 }
2444 if (COAP_PDU_IS_REQUEST(&lg_xmit->pdu))
2445 coap_delete_binary(lg_xmit->b.b1.app_token);
2446 else
2447 coap_delete_string(lg_xmit->b.b2.query);
2448
2449 coap_log_debug("** %s: lg_xmit %p released\n",
2450 coap_session_str(session), (void *)lg_xmit);
2451 coap_free_type(COAP_LG_XMIT, lg_xmit);
2452}
2453
2454#if COAP_SERVER_SUPPORT
2455typedef struct {
2456 uint32_t num;
2457 int is_continue;
2458} send_track;
2459
2460static int
2461add_block_send(uint32_t num, int is_continue, send_track *out_blocks,
2462 uint32_t *count, uint32_t max_count) {
2463 uint32_t i;
2464
2465 for (i = 0; i < *count && *count < max_count; i++) {
2466 if (num == out_blocks[i].num)
2467 return 0;
2468 else if (num < out_blocks[i].num) {
2469 if (*count - i > 1)
2470 memmove(&out_blocks[i], &out_blocks[i+1], *count - i -1);
2471 out_blocks[i].num = num;
2472 out_blocks[i].is_continue = is_continue;
2473 (*count)++;
2474 return 1;
2475 }
2476 }
2477 if (*count < max_count) {
2478 out_blocks[i].num = num;
2479 out_blocks[i].is_continue = is_continue;
2480 (*count)++;
2481 return 1;
2482 }
2483 return 0;
2484}
2485
2486/*
2487 * Need to see if this is a request for the next block of a large body
2488 * transfer. If so, need to initiate the response with the next blocks
2489 * and not trouble the application.
2490 *
2491 * If additional responses needed, then these are expicitly sent out and
2492 * 'response' is updated to be the last response to be sent. There can be
2493 * multiple Q-Block2 in the request, as well as the 'Continue' Q-Block2
2494 * request.
2495 *
2496 * This is set up using coap_add_data_large_response_lkd()
2497 *
2498 * Server is sending a large data response to GET / observe (Block2)
2499 *
2500 * Return: 0 Call application handler
2501 * 1 Do not call application handler - just send the built response
2502 */
2503int
2505 coap_pdu_t *pdu,
2506 coap_pdu_t *response,
2507 coap_resource_t *resource,
2508 coap_string_t *query) {
2509 coap_lg_xmit_t *lg_xmit = NULL;
2510 coap_block_b_t block;
2511 coap_block_b_t alt_block;
2512 uint16_t block_opt = 0;
2513 send_track *out_blocks = NULL;
2514 const char *error_phrase;
2515 coap_opt_iterator_t opt_iter;
2516 size_t chunk;
2517 coap_opt_iterator_t opt_b_iter;
2518 coap_opt_t *option;
2519 uint32_t request_cnt, i;
2520 coap_opt_t *etag_opt = NULL;
2521 coap_pdu_t *out_pdu = response;
2522#if COAP_Q_BLOCK_SUPPORT
2523 size_t max_block;
2524
2525 /* Is client indicating that it supports Q_BLOCK2 ? */
2526 if (coap_get_block_b(session, pdu, COAP_OPTION_Q_BLOCK2, &block)) {
2527 if (!(session->block_mode & COAP_BLOCK_HAS_Q_BLOCK))
2528 set_block_mode_has_q(session->block_mode);
2529 block_opt = COAP_OPTION_Q_BLOCK2;
2530 }
2531#endif /* COAP_Q_BLOCK_SUPPORT */
2532 if (coap_get_block_b(session, pdu, COAP_OPTION_BLOCK2, &alt_block)) {
2533 if (block_opt) {
2534 coap_log_warn("Block2 and Q-Block2 cannot be in the same request\n");
2535 coap_add_data(response, sizeof("Both Block2 and Q-Block2 invalid")-1,
2536 (const uint8_t *)"Both Block2 and Q-Block2 invalid");
2537 response->code = COAP_RESPONSE_CODE(400);
2538 goto skip_app_handler;
2539 }
2540 block = alt_block;
2541 block_opt = COAP_OPTION_BLOCK2;
2542 }
2543 if (block_opt == 0)
2544 return 0;
2545 if (block.num == 0) {
2546 /* Get a fresh copy of the data */
2547 return 0;
2548 }
2549 lg_xmit = coap_find_lg_xmit_response(session, pdu, resource, query);
2550 if (lg_xmit == NULL)
2551 return 0;
2552
2553#if COAP_Q_BLOCK_SUPPORT
2554 out_blocks = coap_malloc_type(COAP_STRING, sizeof(send_track) * COAP_MAX_PAYLOADS(session));
2555#else /* ! COAP_Q_BLOCK_SUPPORT */
2556 out_blocks = coap_malloc_type(COAP_STRING, sizeof(send_track));
2557#endif /* ! COAP_Q_BLOCK_SUPPORT */
2558 if (!out_blocks) {
2559 goto internal_issue;
2560 }
2561
2562 /* lg_xmit (response) found */
2563
2564 etag_opt = coap_check_option(pdu, COAP_OPTION_ETAG, &opt_iter);
2565 if (etag_opt) {
2566 /* There may be multiple ETag - need to check each one */
2567 coap_option_iterator_init(pdu, &opt_iter, COAP_OPT_ALL);
2568 while ((etag_opt = coap_option_next(&opt_iter))) {
2569 if (opt_iter.number == COAP_OPTION_ETAG) {
2570 uint64_t etag = coap_decode_var_bytes8(coap_opt_value(etag_opt),
2571 coap_opt_length(etag_opt));
2572 if (etag == lg_xmit->b.b2.etag) {
2573 break;
2574 }
2575 }
2576 }
2577 if (!etag_opt) {
2578 /* Not a match - pass up to a higher level */
2579 return 0;
2580 }
2581 }
2582 out_pdu->code = lg_xmit->pdu.code;
2583 coap_ticks(&lg_xmit->last_obs);
2584 lg_xmit->last_all_sent = 0;
2585
2586 chunk = (size_t)1 << (lg_xmit->blk_size + 4);
2587 if (block_opt) {
2588 if (block.bert) {
2589 coap_log_debug("found Block option, block is BERT, block nr. %u, M %d\n",
2590 block.num, block.m);
2591 } else {
2592 coap_log_debug("found Block option, block size is %u, block nr. %u, M %d\n",
2593 1 << (block.szx + 4), block.num, block.m);
2594 }
2595 if (block.bert == 0 && block.szx != lg_xmit->blk_size) {
2596 if (block.num == 0) {
2597 if ((lg_xmit->offset + chunk) % ((size_t)1 << (block.szx + 4)) == 0) {
2598 /*
2599 * Recompute the block number of the previous packet given
2600 * the new block size
2601 */
2602 block.num = (uint32_t)(((lg_xmit->offset + chunk) >> (block.szx + 4)) - 1);
2603 lg_xmit->blk_size = block.szx;
2604 chunk = (size_t)1 << (lg_xmit->blk_size + 4);
2605 lg_xmit->offset = block.num * chunk;
2606 coap_log_debug("new Block size is %u, block number %u completed\n",
2607 1 << (block.szx + 4), block.num);
2608 } else {
2609 coap_log_debug("ignoring request to increase Block size, "
2610 "next block is not aligned on requested block size "
2611 "boundary. (%zu x %u mod %u = %zu (which is not 0)\n",
2612 lg_xmit->offset/chunk + 1, (1 << (lg_xmit->blk_size + 4)),
2613 (1 << (block.szx + 4)),
2614 (lg_xmit->offset + chunk) % ((size_t)1 << (block.szx + 4)));
2615 }
2616 } else {
2617 coap_log_debug("ignoring request to change Block size from %u to %u\n",
2618 (1 << (lg_xmit->blk_size + 4)), (1 << (block.szx + 4)));
2619 block.szx = block.aszx = lg_xmit->blk_size;
2620 }
2621 }
2622 }
2623
2624 /*
2625 * Need to check if there are multiple Q-Block2 requests. If so, they
2626 * need to be sent out in order of requests with the final request being
2627 * handled as per singular Block 2 request.
2628 */
2629 request_cnt = 0;
2630#if COAP_Q_BLOCK_SUPPORT
2631 max_block = (lg_xmit->length + chunk - 1)/chunk;
2632#endif /* COAP_Q_BLOCK_SUPPORT */
2633 coap_option_iterator_init(pdu, &opt_b_iter, COAP_OPT_ALL);
2634 while ((option = coap_option_next(&opt_b_iter))) {
2635 uint32_t num;
2636 if (opt_b_iter.number != lg_xmit->option)
2637 continue;
2638 num = coap_opt_block_num(option);
2639 if (num > 0xFFFFF) /* 20 bits max for num */
2640 continue;
2641 if (block.aszx != COAP_OPT_BLOCK_SZX(option)) {
2642 coap_add_data(response,
2643 sizeof("Changing blocksize during request invalid")-1,
2644 (const uint8_t *)"Changing blocksize during request invalid");
2645 response->code = COAP_RESPONSE_CODE(400);
2646 goto skip_app_handler;
2647 }
2648#if COAP_Q_BLOCK_SUPPORT
2649 if (COAP_OPT_BLOCK_MORE(option) && lg_xmit->option == COAP_OPTION_Q_BLOCK2) {
2650 if ((num % COAP_MAX_PAYLOADS(session)) == 0) {
2651 if (num == 0) {
2652 /* This is a repeat request for everything - hmm */
2653 goto call_app_handler;
2654 }
2655 /* 'Continue' request */
2656 for (i = 0; i < COAP_MAX_PAYLOADS(session) &&
2657 num + i < max_block; i++) {
2658 add_block_send(num + i, 1, out_blocks, &request_cnt,
2659 COAP_MAX_PAYLOADS(session));
2660 lg_xmit->last_block = num + i;
2661 }
2662 } else {
2663 /* Requesting remaining payloads in this MAX_PAYLOADS */
2664 for (i = 0; i < COAP_MAX_PAYLOADS(session) -
2665 num % COAP_MAX_PAYLOADS(session) &&
2666 num + i < max_block; i++) {
2667 add_block_send(num + i, 0, out_blocks, &request_cnt,
2668 COAP_MAX_PAYLOADS(session));
2669 }
2670 }
2671 } else
2672 add_block_send(num, 0, out_blocks, &request_cnt,
2673 COAP_MAX_PAYLOADS(session));
2674#else /* ! COAP_Q_BLOCK_SUPPORT */
2675 add_block_send(num, 0, out_blocks, &request_cnt, 1);
2676 break;
2677#endif /* ! COAP_Q_BLOCK_SUPPORT */
2678 }
2679 if (request_cnt == 0) {
2680 /* Block2 or Q-Block2 not found - give them the first block */
2681 block.szx = lg_xmit->blk_size;
2682 lg_xmit->offset = 0;
2683 out_blocks[0].num = 0;
2684 out_blocks[0].is_continue = 0;
2685 request_cnt = 1;
2686 }
2687
2688 for (i = 0; i < request_cnt; i++) {
2689 uint8_t buf[8];
2690
2691 block.num = out_blocks[i].num;
2692 lg_xmit->offset = block.num * chunk;
2693
2694 if (i + 1 < request_cnt) {
2695 /* Need to set up a copy of the pdu to send */
2696 coap_opt_filter_t drop_options;
2697
2698 memset(&drop_options, 0, sizeof(coap_opt_filter_t));
2699 if (block.num != 0)
2701 if (out_blocks[i].is_continue) {
2702 out_pdu = coap_pdu_duplicate_lkd(&lg_xmit->pdu, session, lg_xmit->pdu.actual_token.length,
2703 lg_xmit->pdu.actual_token.s, &drop_options);
2704 } else {
2705 out_pdu = coap_pdu_duplicate_lkd(&lg_xmit->pdu, session, pdu->actual_token.length,
2706 pdu->actual_token.s, &drop_options);
2707 }
2708 if (!out_pdu) {
2709 goto internal_issue;
2710 }
2711 } else {
2712 if (out_blocks[i].is_continue)
2713 coap_update_token(response, lg_xmit->pdu.actual_token.length,
2714 lg_xmit->pdu.actual_token.s);
2715 /*
2716 * Copy the options across and then fix the block option
2717 *
2718 * Need to drop Observe option if Block2 and block.num != 0
2719 */
2720 coap_option_iterator_init(&lg_xmit->pdu, &opt_iter, COAP_OPT_ALL);
2721 while ((option = coap_option_next(&opt_iter))) {
2722 if (opt_iter.number == COAP_OPTION_OBSERVE && block.num != 0)
2723 continue;
2724 if (!coap_insert_option(response, opt_iter.number,
2725 coap_opt_length(option),
2726 coap_opt_value(option))) {
2727 goto internal_issue;
2728 }
2729 }
2730 out_pdu = response;
2731 }
2732 if (pdu->type == COAP_MESSAGE_NON)
2733 out_pdu->type = COAP_MESSAGE_NON;
2734 if (block.bert) {
2735 size_t token_options = pdu->data ? (size_t)(pdu->data - pdu->token) : pdu->used_size;
2736 block.m = (lg_xmit->length - lg_xmit->offset) >
2737 ((out_pdu->max_size - token_options) /1024) * 1024;
2738 } else {
2739 block.m = (lg_xmit->offset + chunk) < lg_xmit->length;
2740 }
2741 if (!coap_update_option(out_pdu, lg_xmit->option,
2743 sizeof(buf),
2744 (block.num << 4) |
2745 (block.m << 3) |
2746 block.aszx),
2747 buf)) {
2748 goto internal_issue;
2749 }
2750 if (!(lg_xmit->offset + chunk < lg_xmit->length)) {
2751 /* Last block - keep in cache for 4 * ACK_TIMOUT */
2752 coap_ticks(&lg_xmit->last_all_sent);
2753 }
2754 if (lg_xmit->b.b2.maxage_expire) {
2755 coap_tick_t now;
2756 coap_time_t rem;
2757
2758 if (!(lg_xmit->offset + chunk < lg_xmit->length)) {
2759 /* Last block - keep in cache for 4 * ACK_TIMOUT */
2760 coap_ticks(&lg_xmit->last_all_sent);
2761 }
2762 coap_ticks(&now);
2763 rem = coap_ticks_to_rt(now);
2764 if (lg_xmit->b.b2.maxage_expire > rem) {
2765 rem = lg_xmit->b.b2.maxage_expire - rem;
2766 } else {
2767 rem = 0;
2768 /* Entry needs to be expired */
2769 coap_ticks(&lg_xmit->last_all_sent);
2770 }
2773 sizeof(buf),
2774 rem),
2775 buf)) {
2776 goto internal_issue;
2777 }
2778 }
2779
2780 if (!coap_add_block_b_data(out_pdu,
2781 lg_xmit->length,
2782 lg_xmit->data,
2783 &block)) {
2784 goto internal_issue;
2785 }
2786 if (i + 1 < request_cnt) {
2787 coap_ticks(&lg_xmit->last_sent);
2788 coap_send_internal(session, out_pdu, NULL);
2789 }
2790 }
2791 coap_ticks(&lg_xmit->last_payload);
2792 goto skip_app_handler;
2793#if COAP_Q_BLOCK_SUPPORT
2794call_app_handler:
2795 coap_free_type(COAP_STRING, out_blocks);
2796 return 0;
2797#endif /* COAP_Q_BLOCK_SUPPORT */
2798
2799internal_issue:
2800 response->code = COAP_RESPONSE_CODE(500);
2801 error_phrase = coap_response_phrase(response->code);
2802 coap_add_data(response, strlen(error_phrase),
2803 (const uint8_t *)error_phrase);
2804 /* Keep in cache for 4 * ACK_TIMOUT incase of retry */
2805 if (lg_xmit)
2806 coap_ticks(&lg_xmit->last_all_sent);
2807
2808skip_app_handler:
2809 coap_free_type(COAP_STRING, out_blocks);
2810 return 1;
2811}
2812#endif /* COAP_SERVER_SUPPORT */
2813
2814static int
2815update_received_blocks(coap_rblock_t *rec_blocks, uint32_t block_num, uint32_t block_m) {
2816 uint32_t i;
2817
2818 if (rec_blocks->total_blocks && block_num + 1 > rec_blocks->total_blocks) {
2819 /* received block number greater than Block No defined when More bit unset */
2820 return 0;
2821 }
2822
2823 /* Reset as there is activity */
2824 rec_blocks->retry = 0;
2825
2826 for (i = 0; i < rec_blocks->used; i++) {
2827 if (block_num >= rec_blocks->range[i].begin &&
2828 block_num <= rec_blocks->range[i].end)
2829 break;
2830
2831 if (block_num < rec_blocks->range[i].begin) {
2832 if (block_num + 1 == rec_blocks->range[i].begin) {
2833 rec_blocks->range[i].begin = block_num;
2834 } else {
2835 /* Need to insert a new range */
2836 if (rec_blocks->used == COAP_RBLOCK_CNT -1)
2837 /* Too many losses */
2838 return 0;
2839 memmove(&rec_blocks->range[i+1], &rec_blocks->range[i],
2840 (rec_blocks->used - i) * sizeof(rec_blocks->range[0]));
2841 rec_blocks->range[i].begin = rec_blocks->range[i].end = block_num;
2842 rec_blocks->used++;
2843 }
2844 break;
2845 }
2846 if (block_num == rec_blocks->range[i].end + 1) {
2847 rec_blocks->range[i].end = block_num;
2848 if (i + 1 < rec_blocks->used) {
2849 if (rec_blocks->range[i+1].begin == block_num + 1) {
2850 /* Merge the 2 ranges */
2851 rec_blocks->range[i].end = rec_blocks->range[i+1].end;
2852 if (i+2 < rec_blocks->used) {
2853 memmove(&rec_blocks->range[i+1], &rec_blocks->range[i+2],
2854 (rec_blocks->used - (i+2)) * sizeof(rec_blocks->range[0]));
2855 }
2856 rec_blocks->used--;
2857 }
2858 }
2859 break;
2860 }
2861 }
2862 if (i == rec_blocks->used) {
2863 if (rec_blocks->used == COAP_RBLOCK_CNT -1)
2864 /* Too many losses */
2865 return 0;
2866 rec_blocks->range[i].begin = rec_blocks->range[i].end = block_num;
2867 rec_blocks->used++;
2868 }
2869 if (!block_m)
2870 rec_blocks->total_blocks = block_num + 1;
2871
2872 coap_ticks(&rec_blocks->last_seen);
2873 return 1;
2874}
2875
2876#if COAP_SERVER_SUPPORT
2877/*
2878 * Need to check if this is a large PUT / POST etc. using multiple blocks
2879 *
2880 * Server receiving PUT/POST etc. of a large amount of data (Block1)
2881 *
2882 * Return: 0 Call application handler
2883 * 1 Do not call application handler - just send the built response
2884 */
2885int
2887 coap_session_t *session,
2888 coap_pdu_t *pdu,
2889 coap_pdu_t *response,
2890 coap_resource_t *resource,
2891 coap_string_t *uri_path,
2892 coap_opt_t *observe,
2893 int *added_block,
2894 coap_lg_srcv_t **pfree_lg_srcv) {
2895 size_t length = 0;
2896 const uint8_t *data = NULL;
2897 size_t offset = 0;
2898 size_t total = 0;
2899 coap_block_b_t block;
2900 coap_opt_iterator_t opt_iter;
2901 uint16_t block_option = 0;
2902 coap_lg_srcv_t *lg_srcv;
2903 coap_opt_t *size_opt;
2904 coap_opt_t *fmt_opt;
2905 uint16_t fmt;
2906 coap_opt_t *rtag_opt;
2907 size_t rtag_length;
2908 const uint8_t *rtag;
2909 uint32_t max_block_szx;
2910 int update_data;
2911 unsigned int saved_num;
2912 size_t saved_offset;
2913
2914 *added_block = 0;
2915 *pfree_lg_srcv = NULL;
2916 coap_get_data_large(pdu, &length, &data, &offset, &total);
2917 pdu->body_offset = 0;
2918 pdu->body_total = length;
2919
2920 if (coap_get_block_b(session, pdu, COAP_OPTION_BLOCK1, &block)) {
2921 block_option = COAP_OPTION_BLOCK1;
2922#if COAP_Q_BLOCK_SUPPORT
2923 if (coap_check_option(pdu, COAP_OPTION_Q_BLOCK1, &opt_iter)) {
2924 /* Cannot handle Q-Block1 as well */
2925 coap_add_data(response, sizeof("Block1 + Q-Block1 together")-1,
2926 (const uint8_t *)"Block1 + Q-Block1 together");
2927 response->code = COAP_RESPONSE_CODE(402);
2928 goto skip_app_handler;
2929 }
2930#endif /* COAP_Q_BLOCK_SUPPORT */
2931 }
2932#if COAP_Q_BLOCK_SUPPORT
2933 else if (coap_get_block_b(session, pdu, COAP_OPTION_Q_BLOCK1, &block)) {
2934 block_option = COAP_OPTION_Q_BLOCK1;
2935 set_block_mode_has_q(session->block_mode);
2936 }
2937#endif /* COAP_Q_BLOCK_SUPPORT */
2938 if (!block_option ||
2939 (block_option == COAP_OPTION_BLOCK1 && block.num == 0 && block.m == 0)) {
2940 /* Not blocked, or a single block */
2941 goto call_app_handler;
2942 }
2943
2944 size_opt = coap_check_option(pdu,
2946 &opt_iter);
2947 fmt_opt = coap_check_option(pdu,
2949 &opt_iter);
2950 fmt = fmt_opt ? coap_decode_var_bytes(coap_opt_value(fmt_opt),
2951 coap_opt_length(fmt_opt)) :
2953 rtag_opt = coap_check_option(pdu,
2955 &opt_iter);
2956 rtag_length = rtag_opt ? coap_opt_length(rtag_opt) : 0;
2957 rtag = rtag_opt ? coap_opt_value(rtag_opt) : NULL;
2958
2959 if (length > block.chunk_size) {
2960 coap_log_debug("block: Oversized packet - reduced to %"PRIu32" from %zu\n",
2961 block.chunk_size, length);
2962 length = block.chunk_size;
2963 } else if (!block.bert && block.m && length != block.chunk_size) {
2964 coap_log_info("block: Undersized packet chunk %"PRIu32" got %zu\n",
2965 block.chunk_size, length);
2966 response->code = COAP_RESPONSE_CODE(400);
2967 goto skip_app_handler;
2968 }
2969 total = size_opt ? coap_decode_var_bytes(coap_opt_value(size_opt),
2970 coap_opt_length(size_opt)) : 0;
2971 offset = block.num << (block.szx + 4);
2972
2973 if (!(session->block_mode &
2974#if COAP_Q_BLOCK_SUPPORT
2976#else /* COAP_Q_BLOCK_SUPPORT */
2978#endif /* COAP_Q_BLOCK_SUPPORT */
2979 && !block.bert) {
2980 uint8_t buf[4];
2981
2982 /* Ask for the next block */
2983 coap_insert_option(response, block_option,
2984 coap_encode_var_safe(buf, sizeof(buf),
2985 (block.num << 4) |
2986 (block.m << 3) |
2987 block.aszx),
2988 buf);
2989 /* Not re-assembling or checking for receipt order */
2990 pdu->body_data = data;
2991 pdu->body_length = length;
2992 pdu->body_offset = offset;
2993 if (total < (length + offset + (block.m ? 1 : 0)))
2994 total = length + offset + (block.m ? 1 : 0);
2995 pdu->body_total = total;
2996 *added_block = block.m;
2997 /* The application is responsible for returning the correct 2.01/2.04/2.31 etc. */
2998 goto call_app_handler;
2999 }
3000
3001 /*
3002 * locate the lg_srcv
3003 */
3004 LL_FOREACH(session->lg_srcv, lg_srcv) {
3005 if (rtag_opt || lg_srcv->rtag_set == 1) {
3006 if (!(rtag_opt && lg_srcv->rtag_set == 1))
3007 continue;
3008 if (lg_srcv->rtag_length != rtag_length ||
3009 memcmp(lg_srcv->rtag, rtag, rtag_length) != 0)
3010 continue;
3011 }
3012 if (resource == lg_srcv->resource) {
3013 break;
3014 }
3015 if ((lg_srcv->resource == context->unknown_resource ||
3016 resource == context->proxy_uri_resource) &&
3017 coap_string_equal(uri_path, lg_srcv->uri_path))
3018 break;
3019 }
3020
3021 if (!lg_srcv && block.num != 0 && session->block_mode & COAP_BLOCK_NOT_RANDOM_BLOCK1) {
3022 coap_add_data(response, sizeof("Missing block 0")-1,
3023 (const uint8_t *)"Missing block 0");
3024 response->code = COAP_RESPONSE_CODE(408);
3025 goto skip_app_handler;
3026 }
3027
3028 if (!lg_srcv) {
3029 /* Allocate lg_srcv to use for tracking */
3030 lg_srcv = coap_malloc_type(COAP_LG_SRCV, sizeof(coap_lg_srcv_t));
3031 if (lg_srcv == NULL) {
3032 coap_add_data(response, sizeof("Memory issue")-1,
3033 (const uint8_t *)"Memory issue");
3034 response->code = COAP_RESPONSE_CODE(500);
3035 goto skip_app_handler;
3036 }
3037 coap_log_debug("** %s: lg_srcv %p initialized\n",
3038 coap_session_str(session), (void *)lg_srcv);
3039 memset(lg_srcv, 0, sizeof(coap_lg_srcv_t));
3040 coap_ticks(&lg_srcv->last_used);
3041 lg_srcv->resource = resource;
3042 if (resource == context->unknown_resource ||
3043 resource == context->proxy_uri_resource)
3044 lg_srcv->uri_path = coap_new_str_const(uri_path->s, uri_path->length);
3045 lg_srcv->content_format = fmt;
3046 lg_srcv->total_len = total;
3047 max_block_szx = COAP_BLOCK_MAX_SIZE_GET(session->block_mode);
3048 if (!block.bert && block.num == 0 && max_block_szx != 0 &&
3049 max_block_szx < block.szx) {
3050 lg_srcv->szx = max_block_szx;
3051 } else {
3052 lg_srcv->szx = block.szx;
3053 }
3054 lg_srcv->block_option = block_option;
3055 if (observe) {
3056 lg_srcv->observe_length = min(coap_opt_length(observe), 3);
3057 memcpy(lg_srcv->observe, coap_opt_value(observe), lg_srcv->observe_length);
3058 lg_srcv->observe_set = 1;
3059 }
3060 if (rtag_opt) {
3061 lg_srcv->rtag_length = coap_opt_length(rtag_opt);
3062 memcpy(lg_srcv->rtag, coap_opt_value(rtag_opt), lg_srcv->rtag_length);
3063 lg_srcv->rtag_set = 1;
3064 }
3065 lg_srcv->body_data = NULL;
3066 LL_PREPEND(session->lg_srcv, lg_srcv);
3067 }
3068
3069 if (block_option == COAP_OPTION_BLOCK1 &&
3071 !check_if_next_block(&lg_srcv->rec_blocks, block.num)) {
3072 coap_add_data(response, sizeof("Missing interim block")-1,
3073 (const uint8_t *)"Missing interim block");
3074 response->code = COAP_RESPONSE_CODE(408);
3075 goto skip_app_handler;
3076 }
3077
3078 if (fmt != lg_srcv->content_format) {
3079 coap_add_data(response, sizeof("Content-Format mismatch")-1,
3080 (const uint8_t *)"Content-Format mismatch");
3081 response->code = COAP_RESPONSE_CODE(408);
3082 goto free_lg_srcv;
3083 }
3084
3085#if COAP_Q_BLOCK_SUPPORT
3086 if (block_option == COAP_OPTION_Q_BLOCK1) {
3087 if (total != lg_srcv->total_len) {
3088 coap_add_data(response, sizeof("Size1 mismatch")-1,
3089 (const uint8_t *)"Size1 mismatch");
3090 response->code = COAP_RESPONSE_CODE(408);
3091 goto free_lg_srcv;
3092 }
3095 pdu->actual_token.length);
3096 }
3097#endif /* COAP_Q_BLOCK_SUPPORT */
3098
3099 lg_srcv->last_mid = pdu->mid;
3100 lg_srcv->last_type = pdu->type;
3101
3102 update_data = 0;
3103 saved_num = block.num;
3104 saved_offset = offset;
3105
3106 while (offset < saved_offset + length) {
3107 if (!check_if_received_block(&lg_srcv->rec_blocks, block.num)) {
3108 /* Update list of blocks received */
3109 if (!update_received_blocks(&lg_srcv->rec_blocks, block.num, block.m)) {
3111 coap_add_data(response, sizeof("Too many missing blocks")-1,
3112 (const uint8_t *)"Too many missing blocks");
3113 response->code = COAP_RESPONSE_CODE(408);
3114 goto free_lg_srcv;
3115 }
3116 update_data = 1;
3117 }
3118 block.num++;
3119 offset = block.num << (block.szx + 4);
3120 }
3121 block.num--;
3122
3123 if (update_data) {
3124 /* Update saved data */
3125#if COAP_Q_BLOCK_SUPPORT
3126 lg_srcv->rec_blocks.processing_payload_set =
3127 block.num / COAP_MAX_PAYLOADS(session);
3128#endif /* COAP_Q_BLOCK_SUPPORT */
3129 if (lg_srcv->total_len < saved_offset + length) {
3130 lg_srcv->total_len = saved_offset + length;
3131 }
3132 lg_srcv->body_data = coap_block_build_body(lg_srcv->body_data, length, data,
3133 saved_offset, lg_srcv->total_len);
3134 if (!lg_srcv->body_data) {
3135 coap_add_data(response, sizeof("Memory issue")-1,
3136 (const uint8_t *)"Memory issue");
3137 response->code = COAP_RESPONSE_CODE(500);
3138 goto skip_app_handler;
3139 }
3140 }
3141
3142 if (block.m ||
3143 !check_all_blocks_in(&lg_srcv->rec_blocks)) {
3144 /* Not all the payloads of the body have arrived */
3145 if (block.m) {
3146 uint8_t buf[4];
3147
3148#if COAP_Q_BLOCK_SUPPORT
3149 if (block_option == COAP_OPTION_Q_BLOCK1) {
3150 if (check_all_blocks_in(&lg_srcv->rec_blocks)) {
3151 goto give_app_data;
3152 }
3153 if (lg_srcv->rec_blocks.used == 1 &&
3154 (lg_srcv->rec_blocks.range[0].end % COAP_MAX_PAYLOADS(session)) + 1
3155 == COAP_MAX_PAYLOADS(session)) {
3156 /* Blocks could arrive in wrong order */
3157 block.num = lg_srcv->rec_blocks.range[0].end;
3158 } else {
3159 /* The remote end will be sending the next one unless this
3160 is a MAX_PAYLOADS and all previous have been received */
3161 goto skip_app_handler;
3162 }
3163 if (COAP_PROTO_RELIABLE(session->proto) ||
3164 pdu->type != COAP_MESSAGE_NON)
3165 goto skip_app_handler;
3166 }
3167#endif /* COAP_Q_BLOCK_SUPPORT */
3168
3169 /* Check to see if block size is getting forced down */
3170 max_block_szx = COAP_BLOCK_MAX_SIZE_GET(session->block_mode);
3171 if (!block.bert && saved_num == 0 && max_block_szx != 0 &&
3172 max_block_szx < block.aszx) {
3173 block.aszx = max_block_szx;
3174 }
3175
3176 /*
3177 * If the last block has been seen, packets are coming in in
3178 * random order. If all blocks are now in, then need to send
3179 * complete payload to application and acknowledge this current
3180 * block.
3181 */
3182 if ((total == 0 && block.m) || !check_all_blocks_in(&lg_srcv->rec_blocks)) {
3183 /* Ask for the next block */
3184 coap_insert_option(response, block_option,
3185 coap_encode_var_safe(buf, sizeof(buf),
3186 (saved_num << 4) |
3187 (block.m << 3) |
3188 block.aszx),
3189 buf);
3190 response->code = COAP_RESPONSE_CODE(231);
3191 } else {
3192 /* Need to separately respond to this request */
3193 coap_pdu_t *tmp_pdu = coap_pdu_duplicate_lkd(response,
3194 session,
3195 response->actual_token.length,
3196 response->actual_token.s,
3197 NULL);
3198 if (tmp_pdu) {
3199 tmp_pdu->code = COAP_RESPONSE_CODE(231);
3200 coap_send_internal(session, tmp_pdu, NULL);
3201 }
3202 if (lg_srcv->last_token) {
3203 coap_update_token(response, lg_srcv->last_token->length, lg_srcv->last_token->s);
3204 coap_update_token(pdu, lg_srcv->last_token->length, lg_srcv->last_token->s);
3205 }
3206 /* Pass the assembled pdu and body to the application */
3207 goto give_app_data;
3208 }
3209 } else {
3210 /* block.m Block More option not set. Some outstanding blocks */
3211#if COAP_Q_BLOCK_SUPPORT
3212 if (block_option != COAP_OPTION_Q_BLOCK1) {
3213#endif /* COAP_Q_BLOCK_SUPPORT */
3214 /* Last chunk - but not all in */
3215 coap_ticks(&lg_srcv->last_used);
3216 lg_srcv->no_more_seen = 1;
3219 pdu->actual_token.length);
3220
3221 /*
3222 * Need to just ACK (no response code) to handle client's NSTART.
3223 * When final missing block comes in, we will pass all the data
3224 * for processing so a 2.01, 2.04 etc. code can be generated
3225 * and responded to as a separate response "RFC7252 5.2.2. Separate"
3226 * If missing block(s) do not come in, then will generate a 4.08
3227 * when lg_srcv times out.
3228 * Fall through to skip_app_handler.
3229 */
3230#if COAP_Q_BLOCK_SUPPORT
3231 }
3232#endif /* COAP_Q_BLOCK_SUPPORT */
3233 }
3234 goto skip_app_handler;
3235 }
3236
3237 /*
3238 * Entire payload received.
3239 * Remove the Block1 option as passing all of the data to
3240 * application layer. Add back in observe option if appropriate.
3241 * Adjust all other information.
3242 */
3243give_app_data:
3244 if (lg_srcv->observe_set) {
3246 lg_srcv->observe_length, lg_srcv->observe);
3247 }
3248 coap_remove_option(pdu, block_option);
3249 if (lg_srcv->body_data) {
3250 pdu->body_data = lg_srcv->body_data->s;
3251 pdu->body_length = lg_srcv->total_len;
3252 } else {
3253 pdu->body_data = NULL;
3254 pdu->body_length = 0;
3255 }
3256 pdu->body_offset = 0;
3257 pdu->body_total = lg_srcv->total_len;
3258 coap_log_debug("Server app version of updated PDU\n");
3260 *pfree_lg_srcv = lg_srcv;
3261
3262call_app_handler:
3263 return 0;
3264
3265free_lg_srcv:
3266 LL_DELETE(session->lg_srcv, lg_srcv);
3267 coap_block_delete_lg_srcv(session, lg_srcv);
3268
3269skip_app_handler:
3270 return 1;
3271}
3272#endif /* COAP_SERVER_SUPPORT */
3273
3274#if COAP_CLIENT_SUPPORT
3275#if COAP_Q_BLOCK_SUPPORT
3276static uint32_t
3277derive_cbor_value(const uint8_t **bp, size_t rem_len) {
3278 uint32_t value = **bp & 0x1f;
3279 (*bp)++;
3280 if (value < 24) {
3281 return value;
3282 } else if (value == 24) {
3283 if (rem_len < 2)
3284 return (uint32_t)-1;
3285 value = **bp;
3286 (*bp)++;
3287 return value;
3288 } else if (value == 25) {
3289 if (rem_len < 3)
3290 return (uint32_t)-1;
3291 value = **bp << 8;
3292 (*bp)++;
3293 value |= **bp;
3294 (*bp)++;
3295 return value;
3296 }
3297 if (rem_len < 4)
3298 return (uint32_t)-1;
3299 value = **bp << 24;
3300 (*bp)++;
3301 value |= **bp << 16;
3302 (*bp)++;
3303 value |= **bp << 8;
3304 (*bp)++;
3305 value |= **bp;
3306 (*bp)++;
3307 return value;
3308}
3309#endif /* COAP_Q_BLOCK_SUPPORT */
3310
3311static int
3312check_freshness(coap_session_t *session, coap_pdu_t *rcvd, coap_pdu_t *sent,
3313 coap_lg_xmit_t *lg_xmit, coap_lg_crcv_t *lg_crcv) {
3314 /* Check for Echo option for freshness */
3315 coap_opt_iterator_t opt_iter;
3316 coap_opt_t *opt = coap_check_option(rcvd, COAP_OPTION_ECHO, &opt_iter);
3317
3318 if (opt) {
3319 if (sent || lg_xmit || lg_crcv) {
3320 /* Need to retransmit original request with Echo option added */
3321 coap_pdu_t *echo_pdu;
3322 coap_mid_t mid;
3323 const uint8_t *data;
3324 size_t data_len;
3325 int have_data = 0;
3326 uint8_t ltoken[8];
3327 size_t ltoken_len;
3328 uint64_t token;
3329
3330 if (sent) {
3331 if (coap_get_data(sent, &data_len, &data))
3332 have_data = 1;
3333 } else if (lg_xmit) {
3334 sent = &lg_xmit->pdu;
3335 if (lg_xmit->length) {
3336 size_t blk_size = (size_t)1 << (lg_xmit->blk_size + 4);
3337 size_t offset = (lg_xmit->last_block + 1) * blk_size;
3338 have_data = 1;
3339 data = &lg_xmit->data[offset];
3340 data_len = (lg_xmit->length - offset) > blk_size ? blk_size :
3341 lg_xmit->length - offset;
3342 }
3343 } else { /* lg_crcv */
3344 sent = &lg_crcv->pdu;
3345 if (coap_get_data(sent, &data_len, &data))
3346 have_data = 1;
3347 }
3348 if (lg_xmit) {
3349 token = STATE_TOKEN_FULL(lg_xmit->b.b1.state_token,
3350 ++lg_xmit->b.b1.count);
3351 } else {
3352 token = STATE_TOKEN_FULL(lg_crcv->state_token,
3353 ++lg_crcv->retry_counter);
3354 }
3355 ltoken_len = coap_encode_var_safe8(ltoken, sizeof(token), token);
3356 echo_pdu = coap_pdu_duplicate_lkd(sent, session, ltoken_len, ltoken, NULL);
3357 if (!echo_pdu)
3358 return 0;
3359 if (!coap_insert_option(echo_pdu, COAP_OPTION_ECHO,
3360 coap_opt_length(opt), coap_opt_value(opt)))
3361 goto not_sent;
3362 if (have_data) {
3363 coap_add_data(echo_pdu, data_len, data);
3364 }
3365 /* Need to track Observe token change if Observe */
3366 track_fetch_observe(echo_pdu, lg_crcv, 0, &echo_pdu->actual_token);
3367#if COAP_OSCORE_SUPPORT
3368 if (session->oscore_encryption &&
3369 (opt = coap_check_option(echo_pdu, COAP_OPTION_OBSERVE, &opt_iter)) &&
3371 /* Need to update the base PDU's Token for closing down Observe */
3372 if (lg_xmit) {
3373 lg_xmit->b.b1.state_token = token;
3374 } else {
3375 lg_crcv->state_token = token;
3376 }
3377 }
3378#endif /* COAP_OSCORE_SUPPORT */
3379 mid = coap_send_internal(session, echo_pdu, NULL);
3380 if (mid == COAP_INVALID_MID)
3381 goto not_sent;
3382 return 1;
3383 } else {
3384 /* Need to save Echo option value to add to next reansmission */
3385not_sent:
3386 coap_delete_bin_const(session->echo);
3387 session->echo = coap_new_bin_const(coap_opt_value(opt),
3388 coap_opt_length(opt));
3389 }
3390 }
3391 return 0;
3392}
3393
3394static void
3395track_echo(coap_session_t *session, coap_pdu_t *rcvd) {
3396 coap_opt_iterator_t opt_iter;
3397 coap_opt_t *opt = coap_check_option(rcvd, COAP_OPTION_ECHO, &opt_iter);
3398
3399 if (opt) {
3400 coap_delete_bin_const(session->echo);
3401 session->echo = coap_new_bin_const(coap_opt_value(opt),
3402 coap_opt_length(opt));
3403 }
3404}
3405
3406/*
3407 * Need to see if this is a response to a large body request transfer. If so,
3408 * need to initiate the request containing the next block and not trouble the
3409 * application. Note that Token must unique per request/response.
3410 *
3411 * Client receives large data acknowledgement from server (Block1)
3412 *
3413 * This is set up using coap_add_data_large_request_lkd()
3414 *
3415 * Client is using GET etc.
3416 *
3417 * Return: 0 Call application handler
3418 * 1 Do not call application handler - just send the built response
3419 */
3420int
3422 coap_pdu_t *rcvd) {
3423 coap_lg_xmit_t *lg_xmit;
3424 coap_lg_xmit_t *q;
3425 uint64_t token_match =
3427 rcvd->actual_token.length));
3428 coap_lg_crcv_t *lg_crcv = NULL;
3429
3430 LL_FOREACH_SAFE(session->lg_xmit, lg_xmit, q) {
3431 if (!COAP_PDU_IS_REQUEST(&lg_xmit->pdu) ||
3432 (token_match != STATE_TOKEN_BASE(lg_xmit->b.b1.state_token) &&
3433 !coap_binary_equal(&rcvd->actual_token, lg_xmit->b.b1.app_token))) {
3434 /* try out the next one */
3435 continue;
3436 }
3437 /* lg_xmit found */
3438 size_t chunk = (size_t)1 << (lg_xmit->blk_size + 4);
3439 coap_block_b_t block;
3440
3441 lg_crcv = lg_xmit->lg_crcv;
3442 if (lg_crcv)
3443 coap_ticks(&lg_crcv->last_used);
3444
3445 if (COAP_RESPONSE_CLASS(rcvd->code) == 2 &&
3446 coap_get_block_b(session, rcvd, lg_xmit->option, &block)) {
3447
3448 if (block.bert) {
3449 coap_log_debug("found Block option, block is BERT, block nr. %u (%zu)\n",
3450 block.num, lg_xmit->b.b1.bert_size);
3451 } else {
3452 coap_log_debug("found Block option, block size is %u, block nr. %u\n",
3453 1 << (block.szx + 4), block.num);
3454 }
3455 if (block.szx != lg_xmit->blk_size) {
3456 if (block.szx > lg_xmit->blk_size) {
3457 coap_log_info("ignoring request to increase Block size, "
3458 "(%u > %u)\n",
3459 1 << (block.szx + 4), 1 << (lg_xmit->blk_size + 4));
3460 } else if ((lg_xmit->offset + chunk) % ((size_t)1 << (block.szx + 4)) == 0) {
3461 /*
3462 * Recompute the block number of the previous packet given the
3463 * new block size
3464 */
3465 block.num = (uint32_t)(((lg_xmit->offset + chunk) >> (block.szx + 4)) - 1);
3466 lg_xmit->blk_size = block.szx;
3467 chunk = (size_t)1 << (lg_xmit->blk_size + 4);
3468 lg_xmit->offset = block.num * chunk;
3469 coap_log_debug("new Block size is %u, block number %u completed\n",
3470 1 << (block.szx + 4), block.num);
3471 block.bert = 0;
3472 block.aszx = block.szx;
3473 } else {
3474 coap_log_debug("ignoring request to increase Block size, "
3475 "next block is not aligned on requested block size boundary. "
3476 "(%zu x %u mod %u = %zu != 0)\n",
3477 lg_xmit->offset/chunk + 1, (1 << (lg_xmit->blk_size + 4)),
3478 (1 << (block.szx + 4)),
3479 (lg_xmit->offset + chunk) % ((size_t)1 << (block.szx + 4)));
3480 }
3481 }
3482 track_echo(session, rcvd);
3483 if (lg_xmit->last_block == (int)block.num &&
3484 lg_xmit->option != COAP_OPTION_Q_BLOCK1) {
3485 /*
3486 * Duplicate Block1 ACK
3487 *
3488 * RFCs not clear here, but on a lossy connection, there could
3489 * be multiple Block1 ACKs, causing the client to retransmit the
3490 * same block multiple times, or the server retransmitting the
3491 * same ACK.
3492 *
3493 * Once a block has been ACKd, there is no need to retransmit it.
3494 */
3495 return 1;
3496 }
3497 if (block.bert)
3498 block.num += (unsigned int)(lg_xmit->b.b1.bert_size / 1024 - 1);
3499 lg_xmit->last_block = block.num;
3500 lg_xmit->offset = (block.num + 1) * chunk;
3501 if (lg_xmit->offset < lg_xmit->length) {
3502 /* Build the next PDU request based off the skeletal PDU */
3503 uint8_t buf[8];
3504 coap_pdu_t *pdu;
3505 uint64_t token = STATE_TOKEN_FULL(lg_xmit->b.b1.state_token, ++lg_xmit->b.b1.count);
3506 size_t len = coap_encode_var_safe8(buf, sizeof(token), token);
3507
3508 if (lg_xmit->pdu.code == COAP_REQUEST_CODE_FETCH) {
3509 /* Need to handle Observe for large FETCH */
3510 if (lg_crcv) {
3511 if (coap_binary_equal(lg_xmit->b.b1.app_token, lg_crcv->app_token)) {
3512 coap_bin_const_t *new_token;
3513 coap_bin_const_t ctoken = { len, buf };
3514
3515 /* Need to save/restore Observe Token for large FETCH */
3516 new_token = track_fetch_observe(&lg_xmit->pdu, lg_crcv, block.num + 1,
3517 &ctoken);
3518 if (new_token) {
3519 assert(len <= sizeof(buf));
3520 len = new_token->length;
3521 memcpy(buf, new_token->s, len);
3522 }
3523 }
3524 }
3525 }
3526 pdu = coap_pdu_duplicate_lkd(&lg_xmit->pdu, session, len, buf, NULL);
3527 if (!pdu)
3528 goto fail_body;
3529
3530 block.num++;
3531 if (block.bert) {
3532 size_t token_options = pdu->data ? (size_t)(pdu->data - pdu->token) :
3533 pdu->used_size;
3534 block.m = (lg_xmit->length - lg_xmit->offset) >
3535 ((pdu->max_size - token_options) /1024) * 1024;
3536 } else {
3537 block.m = (lg_xmit->offset + chunk) < lg_xmit->length;
3538 }
3539 coap_update_option(pdu, lg_xmit->option,
3540 coap_encode_var_safe(buf, sizeof(buf),
3541 (block.num << 4) |
3542 (block.m << 3) |
3543 block.aszx),
3544 buf);
3545
3546 if (!coap_add_block_b_data(pdu,
3547 lg_xmit->length,
3548 lg_xmit->data,
3549 &block))
3550 goto fail_body;
3551 lg_xmit->b.b1.bert_size = block.chunk_size;
3552 coap_ticks(&lg_xmit->last_sent);
3553#if COAP_Q_BLOCK_SUPPORT
3554 if (lg_xmit->option == COAP_OPTION_Q_BLOCK1 &&
3555 pdu->type == COAP_MESSAGE_NON) {
3556 if (coap_send_q_block1(session, block, pdu,
3557 COAP_SEND_INC_PDU) == COAP_INVALID_MID)
3558 goto fail_body;
3559 return 1;
3560 } else if (coap_send_internal(session, pdu, NULL) == COAP_INVALID_MID)
3561 goto fail_body;
3562#else /* ! COAP_Q_BLOCK_SUPPORT */
3563 if (coap_send_internal(session, pdu, NULL) == COAP_INVALID_MID)
3564 goto fail_body;
3565#endif /* ! COAP_Q_BLOCK_SUPPORT */
3566 return 1;
3567 }
3568 } else if (COAP_RESPONSE_CLASS(rcvd->code) == 2) {
3569 /*
3570 * Not a block response asking for the next block.
3571 * Could be an Observe response overlapping with block FETCH doing
3572 * Observe cancellation.
3573 */
3574 coap_opt_iterator_t opt_iter;
3575 coap_opt_t *obs_opt;
3576 int observe_action = -1;
3577
3578 if (lg_xmit->pdu.code != COAP_REQUEST_CODE_FETCH) {
3579 goto lg_xmit_finished;
3580 }
3581 obs_opt = coap_check_option(&lg_xmit->pdu,
3583 &opt_iter);
3584 if (obs_opt) {
3585 observe_action = coap_decode_var_bytes(coap_opt_value(obs_opt),
3586 coap_opt_length(obs_opt));
3587 }
3588 if (observe_action != COAP_OBSERVE_CANCEL) {
3589 goto lg_xmit_finished;
3590 }
3591 obs_opt = coap_check_option(rcvd,
3593 &opt_iter);
3594 if (obs_opt) {
3595 return 0;
3596 }
3597 goto lg_xmit_finished;
3598 } else if (rcvd->code == COAP_RESPONSE_CODE(401)) {
3599 if (check_freshness(session, rcvd, sent, lg_xmit, NULL))
3600 return 1;
3601#if COAP_Q_BLOCK_SUPPORT
3602 } else if (rcvd->code == COAP_RESPONSE_CODE(402)) {
3603 /* Q-Block1 or Q-Block2 not present in p - duplicate error ? */
3604 if (coap_get_block_b(session, rcvd, COAP_OPTION_Q_BLOCK2, &block) ||
3605 coap_get_block_b(session, rcvd, COAP_OPTION_Q_BLOCK1, &block))
3606 return 1;
3607 } else if (rcvd->code == COAP_RESPONSE_CODE(408) &&
3608 lg_xmit->option == COAP_OPTION_Q_BLOCK1) {
3609 size_t length;
3610 const uint8_t *data;
3611 coap_opt_iterator_t opt_iter;
3612 coap_opt_t *fmt_opt = coap_check_option(rcvd,
3614 &opt_iter);
3615 uint16_t fmt = fmt_opt ?
3617 coap_opt_length(fmt_opt)) :
3619
3621 goto fail_body;
3622
3623 if (COAP_PROTO_RELIABLE(session->proto) ||
3624 rcvd->type != COAP_MESSAGE_NON) {
3625 coap_log_debug("Unexpected 4.08 - protocol violation - ignore\n");
3626 return 1;
3627 }
3628
3629 if (coap_get_data(rcvd, &length, &data)) {
3630 /* Need to decode CBOR to work out what blocks to re-send */
3631 const uint8_t *bp = data;
3632 uint32_t i;
3633 uint8_t buf[8];
3634 coap_pdu_t *pdu;
3635 uint64_t token;
3636 uint8_t ltoken[8];
3637 size_t ltoken_length;
3638
3639 for (i = 0; (bp < data + length) &&
3640 i < COAP_MAX_PAYLOADS(session); i++) {
3641 if ((*bp & 0xc0) != 0x00) /* uint(value) */
3642 goto fail_cbor;
3643 block.num = derive_cbor_value(&bp, data + length - bp);
3644 coap_log_debug("Q-Block1: Missing block %d\n", block.num);
3645 if (block.num > (1 << 20) -1)
3646 goto fail_cbor;
3647 block.m = (block.num + 1) * chunk < lg_xmit->length;
3648 block.szx = lg_xmit->blk_size;
3649
3650 /* Build the next PDU request based off the skeletal PDU */
3651 token = STATE_TOKEN_FULL(lg_xmit->b.b1.state_token,++lg_xmit->b.b1.count);
3652 ltoken_length = coap_encode_var_safe8(ltoken, sizeof(token), token);
3653 pdu = coap_pdu_duplicate_lkd(&lg_xmit->pdu, session, ltoken_length,
3654 ltoken, NULL);
3655 if (!pdu)
3656 goto fail_body;
3657
3658 coap_update_option(pdu, lg_xmit->option,
3659 coap_encode_var_safe(buf, sizeof(buf),
3660 (block.num << 4) |
3661 (block.m << 3) |
3662 block.szx),
3663 buf);
3664
3665 if (!coap_add_block(pdu,
3666 lg_xmit->length,
3667 lg_xmit->data,
3668 block.num,
3669 block.szx))
3670 goto fail_body;
3671 if (coap_send_internal(session, pdu, NULL) == COAP_INVALID_MID)
3672 goto fail_body;
3673 }
3674 return 1;
3675 }
3676fail_cbor:
3677 coap_log_info("Invalid application/missing-blocks+cbor-seq\n");
3678#endif /* COAP_Q_BLOCK_SUPPORT */
3679 }
3680 goto lg_xmit_finished;
3681 } /* end of LL_FOREACH_SAFE */
3682 return 0;
3683
3684fail_body:
3686 /* There has been an internal error of some sort */
3687 rcvd->code = COAP_RESPONSE_CODE(500);
3688lg_xmit_finished:
3689 if (lg_crcv) {
3690 if (STATE_TOKEN_BASE(lg_xmit->b.b1.state_token) ==
3691 STATE_TOKEN_BASE(lg_crcv->state_token)) {
3692 /* In case of observe */
3693 lg_crcv->state_token = lg_xmit->b.b1.state_token;
3694 lg_crcv->retry_counter = lg_xmit->b.b1.count;
3695 }
3696 }
3697 if (!lg_crcv) {
3698 /* need to put back original token into rcvd */
3699 if (lg_xmit->b.b1.app_token)
3700 coap_update_token(rcvd, lg_xmit->b.b1.app_token->length,
3701 lg_xmit->b.b1.app_token->s);
3702 coap_log_debug("Client app version of updated PDU\n");
3704 } else {
3705 lg_crcv->pdu.lg_xmit = 0;
3706 }
3707
3708 if (sent) {
3709 /* need to put back original token into sent */
3710 if (lg_xmit->b.b1.app_token)
3711 coap_update_token(sent, lg_xmit->b.b1.app_token->length,
3712 lg_xmit->b.b1.app_token->s);
3713 if (sent->lg_xmit)
3714 coap_remove_option(sent, sent->lg_xmit->option);
3715 sent->lg_xmit = NULL;
3716 }
3717 LL_DELETE(session->lg_xmit, lg_xmit);
3718 coap_block_delete_lg_xmit(session, lg_xmit);
3719 return 0;
3720}
3721#endif /* COAP_CLIENT_SUPPORT */
3722
3723/*
3724 * Re-assemble payloads into a body
3725 */
3727coap_block_build_body(coap_binary_t *body_data, size_t length,
3728 const uint8_t *data, size_t offset, size_t total) {
3729 if (data == NULL)
3730 return NULL;
3731 if (body_data == NULL && total) {
3732 body_data = coap_new_binary(total);
3733 }
3734 if (body_data == NULL)
3735 return NULL;
3736
3737 /* Update saved data */
3738 if (offset + length <= total && body_data->length >= total) {
3739 memcpy(&body_data->s[offset], data, length);
3740 } else {
3741 /*
3742 * total may be inaccurate as per
3743 * https://rfc-editor.org/rfc/rfc7959#section-4
3744 * o In a request carrying a Block1 Option, to indicate the current
3745 * estimate the client has of the total size of the resource
3746 * representation, measured in bytes ("size indication").
3747 * o In a response carrying a Block2 Option, to indicate the current
3748 * estimate the server has of the total size of the resource
3749 * representation, measured in bytes ("size indication").
3750 */
3751 coap_binary_t *new = coap_resize_binary(body_data, offset + length);
3752
3753 if (new) {
3754 body_data = new;
3755 memcpy(&body_data->s[offset], data, length);
3756 } else {
3757 coap_delete_binary(body_data);
3758 return NULL;
3759 }
3760 }
3761 return body_data;
3762}
3763
3764#if COAP_CLIENT_SUPPORT
3765/*
3766 * Need to see if this is a large body response to a request. If so,
3767 * need to initiate the request for the next block and not trouble the
3768 * application. Note that Token must be unique per request/response.
3769 *
3770 * This is set up using coap_send()
3771 * Client receives large data from server ((Q-)Block2)
3772 *
3773 * Return: 0 Call application handler
3774 * 1 Do not call application handler - just sent the next request
3775 */
3776int
3778 coap_session_t *session,
3779 coap_pdu_t *sent,
3780 coap_pdu_t *rcvd,
3781 coap_recurse_t recursive) {
3782 coap_lg_crcv_t *lg_crcv;
3783 coap_block_b_t block;
3784#if COAP_Q_BLOCK_SUPPORT
3785 coap_block_b_t qblock;
3786#endif /* COAP_Q_BLOCK_SUPPORT */
3787 int have_block = 0;
3788 uint16_t block_opt = 0;
3789 size_t offset;
3790 int ack_rst_sent = 0;
3791 uint64_t token_match =
3793 rcvd->actual_token.length));
3794
3795 coap_lock_check_locked(context);
3796 memset(&block, 0, sizeof(block));
3797#if COAP_Q_BLOCK_SUPPORT
3798 memset(&qblock, 0, sizeof(qblock));
3799#endif /* COAP_Q_BLOCK_SUPPORT */
3800 LL_FOREACH(session->lg_crcv, lg_crcv) {
3801 size_t chunk = 0;
3802 uint8_t buf[8];
3803 coap_opt_iterator_t opt_iter;
3804
3805 if (token_match != STATE_TOKEN_BASE(lg_crcv->state_token) &&
3806 !coap_binary_equal(&rcvd->actual_token, lg_crcv->app_token)) {
3807 /* try out the next one */
3808 continue;
3809 }
3810
3811 /* lg_crcv found */
3812
3813 if (COAP_RESPONSE_CLASS(rcvd->code) == 2) {
3814 size_t length;
3815 const uint8_t *data;
3817 &opt_iter);
3818 size_t size2 = size_opt ?
3820 coap_opt_length(size_opt)) : 0;
3821
3822 /* length and data are cleared on error */
3823 (void)coap_get_data(rcvd, &length, &data);
3824 rcvd->body_offset = 0;
3825 rcvd->body_total = length;
3826 if (coap_get_block_b(session, rcvd, COAP_OPTION_BLOCK2, &block)) {
3827 have_block = 1;
3828 block_opt = COAP_OPTION_BLOCK2;
3829 }
3830#if COAP_Q_BLOCK_SUPPORT
3831 if (coap_get_block_b(session, rcvd, COAP_OPTION_Q_BLOCK2, &qblock)) {
3832 if (have_block) {
3833 coap_log_warn("Both Block1 and Q-Block1 not supported in a response\n");
3834 }
3835 have_block = 1;
3836 block_opt = COAP_OPTION_Q_BLOCK2;
3837 block = qblock;
3838 /* server indicating that it supports Q_BLOCK */
3839 if (!(session->block_mode & COAP_BLOCK_HAS_Q_BLOCK)) {
3840 set_block_mode_has_q(session->block_mode);
3841 }
3842 }
3843#endif /* COAP_Q_BLOCK_SUPPORT */
3844 track_echo(session, rcvd);
3845 if (have_block && (block.m || length)) {
3846 coap_opt_t *fmt_opt = coap_check_option(rcvd,
3848 &opt_iter);
3849 uint16_t fmt = fmt_opt ?
3851 coap_opt_length(fmt_opt)) :
3853 coap_opt_t *etag_opt = coap_check_option(rcvd,
3855 &opt_iter);
3856 size_t saved_offset;
3857 int updated_block;
3858
3859 if (length > block.chunk_size) {
3860 coap_log_debug("block: Oversized packet - reduced to %"PRIu32" from %zu\n",
3861 block.chunk_size, length);
3862 length = block.chunk_size;
3863 }
3864 if (block.m && length != block.chunk_size) {
3865 coap_log_warn("block: Undersized packet - expected %"PRIu32", got %zu\n",
3866 block.chunk_size, length);
3867 /* Unclear how to properly handle this */
3868 rcvd->code = COAP_RESPONSE_CODE(402);
3869 goto expire_lg_crcv;
3870 }
3871 /* Possibility that Size2 not sent, or is too small */
3872 chunk = (size_t)1 << (block.szx + 4);
3873 offset = block.num * chunk;
3874 if (size2 < (offset + length)) {
3875 if (block.m)
3876 size2 = offset + length + 1;
3877 else
3878 size2 = offset + length;
3879 }
3880 saved_offset = offset;
3881
3882 if (lg_crcv->initial) {
3883#if COAP_Q_BLOCK_SUPPORT
3884reinit:
3885#endif /* COAP_Q_BLOCK_SUPPORT */
3886 lg_crcv->initial = 0;
3887 if (lg_crcv->body_data) {
3889 lg_crcv->body_data = NULL;
3890 }
3891 if (etag_opt) {
3892 lg_crcv->etag_length = coap_opt_length(etag_opt);
3893 memcpy(lg_crcv->etag, coap_opt_value(etag_opt), lg_crcv->etag_length);
3894 lg_crcv->etag_set = 1;
3895 } else {
3896 lg_crcv->etag_set = 0;
3897 }
3898 lg_crcv->total_len = size2;
3899 lg_crcv->content_format = fmt;
3900 lg_crcv->szx = block.szx;
3901 lg_crcv->block_option = block_opt;
3902 lg_crcv->last_type = rcvd->type;
3903 lg_crcv->rec_blocks.used = 0;
3904 lg_crcv->rec_blocks.total_blocks = 0;
3905#if COAP_Q_BLOCK_SUPPORT
3906 lg_crcv->rec_blocks.processing_payload_set = 0;
3907#endif /* COAP_Q_BLOCK_SUPPORT */
3908 }
3909 if (lg_crcv->total_len < size2)
3910 lg_crcv->total_len = size2;
3911
3912 if (etag_opt) {
3913 if (!full_match(coap_opt_value(etag_opt),
3914 coap_opt_length(etag_opt),
3915 lg_crcv->etag, lg_crcv->etag_length)) {
3916 /* body of data has changed - need to restart request */
3917 coap_pdu_t *pdu;
3918 uint64_t token = STATE_TOKEN_FULL(lg_crcv->state_token,
3919 ++lg_crcv->retry_counter);
3920 size_t len = coap_encode_var_safe8(buf, sizeof(token), token);
3921 coap_opt_filter_t drop_options;
3922
3923#if COAP_Q_BLOCK_SUPPORT
3924 if (block_opt == COAP_OPTION_Q_BLOCK2)
3925 goto reinit;
3926#endif /* COAP_Q_BLOCK_SUPPORT */
3927
3928 coap_log_warn("Data body updated during receipt - new request started\n");
3929 if (!(session->block_mode & COAP_BLOCK_SINGLE_BODY))
3931
3932 lg_crcv->initial = 1;
3934 lg_crcv->body_data = NULL;
3935
3936 coap_session_new_token(session, &len, buf);
3937 memset(&drop_options, 0, sizeof(coap_opt_filter_t));
3939 pdu = coap_pdu_duplicate_lkd(&lg_crcv->pdu, session, len, buf, &drop_options);
3940 if (!pdu)
3941 goto fail_resp;
3942
3943 coap_update_option(pdu, block_opt,
3944 coap_encode_var_safe(buf, sizeof(buf),
3945 (0 << 4) | (0 << 3) | block.aszx),
3946 buf);
3947
3948 if (coap_send_internal(session, pdu, NULL) == COAP_INVALID_MID)
3949 goto fail_resp;
3950
3951 goto skip_app_handler;
3952 }
3953 } else if (lg_crcv->etag_set) {
3954 /* Cannot handle this change in ETag to not being there */
3955 coap_log_warn("Not all blocks have ETag option\n");
3956 goto fail_resp;
3957 }
3958
3959 if (fmt != lg_crcv->content_format) {
3960 coap_log_warn("Content-Format option mismatch\n");
3961 goto fail_resp;
3962 }
3963#if COAP_Q_BLOCK_SUPPORT
3964 if (block_opt == COAP_OPTION_Q_BLOCK2 && size2 != lg_crcv->total_len) {
3965 coap_log_warn("Size2 option mismatch\n");
3966 goto fail_resp;
3967 }
3968#endif /* COAP_Q_BLOCK_SUPPORT */
3969 if (block.num == 0) {
3970 coap_opt_t *obs_opt = coap_check_option(rcvd,
3972 &opt_iter);
3973 if (obs_opt) {
3974 lg_crcv->observe_length = min(coap_opt_length(obs_opt), 3);
3975 memcpy(lg_crcv->observe, coap_opt_value(obs_opt), lg_crcv->observe_length);
3976 lg_crcv->observe_set = 1;
3977 } else {
3978 lg_crcv->observe_set = 0;
3979 }
3980 }
3981 updated_block = 0;
3982 while (offset < saved_offset + length) {
3983 if (!check_if_received_block(&lg_crcv->rec_blocks, block.num)) {
3984#if COAP_Q_BLOCK_SUPPORT
3985 uint32_t this_payload_set = block.num / COAP_MAX_PAYLOADS(session);
3986#endif /* COAP_Q_BLOCK_SUPPORT */
3987
3988 coap_log_debug("found Block option, block size is %u, block nr. %u\n",
3989 1 << (block.szx + 4), block.num);
3990#if COAP_Q_BLOCK_SUPPORT
3991 if (block_opt == COAP_OPTION_Q_BLOCK2 && lg_crcv->rec_blocks.used &&
3992 this_payload_set > lg_crcv->rec_blocks.processing_payload_set &&
3993 this_payload_set != lg_crcv->rec_blocks.latest_payload_set) {
3994 coap_request_missing_q_block2(session, lg_crcv);
3995 }
3996 lg_crcv->rec_blocks.latest_payload_set = this_payload_set;
3997#endif /* COAP_Q_BLOCK_SUPPORT */
3998 /* Update list of blocks received */
3999 if (!update_received_blocks(&lg_crcv->rec_blocks, block.num, block.m)) {
4001 goto fail_resp;
4002 }
4003 updated_block = 1;
4004 }
4005 block.num++;
4006 offset = block.num << (block.szx + 4);
4007 if (!block.bert && block_opt != COAP_OPTION_Q_BLOCK2)
4008 break;
4009 }
4010 block.num--;
4011 /* Only process if not duplicate block */
4012 if (updated_block) {
4013 if ((session->block_mode & COAP_SINGLE_BLOCK_OR_Q) || block.bert) {
4014 if (size2 < saved_offset + length) {
4015 size2 = saved_offset + length;
4016 }
4017 lg_crcv->body_data = coap_block_build_body(lg_crcv->body_data, length, data,
4018 saved_offset, size2);
4019 if (lg_crcv->body_data == NULL) {
4020 goto fail_resp;
4021 }
4022 }
4023 if (block.m || !check_all_blocks_in(&lg_crcv->rec_blocks)) {
4024 /* Not all the payloads of the body have arrived */
4025 size_t len;
4026 coap_pdu_t *pdu;
4027 uint64_t token;
4028
4029 if (block.m) {
4030#if COAP_Q_BLOCK_SUPPORT
4031 if (block_opt == COAP_OPTION_Q_BLOCK2) {
4032 /* Blocks could arrive in wrong order */
4033 if (check_all_blocks_in(&lg_crcv->rec_blocks)) {
4034 goto give_to_app;
4035 }
4036 if (check_all_blocks_in_for_payload_set(session,
4037 &lg_crcv->rec_blocks)) {
4038 block.num = lg_crcv->rec_blocks.range[0].end;
4039 /* Now requesting next payload */
4040 lg_crcv->rec_blocks.processing_payload_set =
4041 block.num / COAP_MAX_PAYLOADS(session) + 1;
4042 if (check_any_blocks_next_payload_set(session,
4043 &lg_crcv->rec_blocks)) {
4044 /* Need to ask for them individually */
4045 coap_request_missing_q_block2(session, lg_crcv);
4046 goto skip_app_handler;
4047 }
4048 } else {
4049 /* The remote end will be sending the next one unless this
4050 is a MAX_PAYLOADS and all previous have been received */
4051 goto skip_app_handler;
4052 }
4053 if (COAP_PROTO_RELIABLE(session->proto) ||
4054 rcvd->type != COAP_MESSAGE_NON)
4055 goto skip_app_handler;
4056
4057 } else
4058#endif /* COAP_Q_BLOCK_SUPPORT */
4059 block.m = 0;
4060
4061 /* Ask for the next block */
4062 token = STATE_TOKEN_FULL(lg_crcv->state_token, ++lg_crcv->retry_counter);
4063 len = coap_encode_var_safe8(buf, sizeof(token), token);
4064 pdu = coap_pdu_duplicate_lkd(&lg_crcv->pdu, session, len, buf, NULL);
4065 if (!pdu)
4066 goto fail_resp;
4067
4068 if (rcvd->type == COAP_MESSAGE_NON)
4069 pdu->type = COAP_MESSAGE_NON; /* Server is using NON */
4070
4071 /* Only sent with the first block */
4073
4074 coap_update_option(pdu, block_opt,
4075 coap_encode_var_safe(buf, sizeof(buf),
4076 ((block.num + 1) << 4) |
4077 (block.m << 3) | block.aszx),
4078 buf);
4079
4081 (void)coap_get_data(&lg_crcv->pdu, &length, &data);
4082 coap_add_data_large_internal(session, NULL, pdu, NULL, NULL, -1, 0, length, data, NULL, NULL, 0, 0);
4083 }
4084 if (coap_send_internal(session, pdu, NULL) == COAP_INVALID_MID)
4085 goto fail_resp;
4086 }
4087 if ((session->block_mode & COAP_SINGLE_BLOCK_OR_Q) || block.bert)
4088 goto skip_app_handler;
4089
4090 /* need to put back original token into rcvd */
4091 coap_update_token(rcvd, lg_crcv->app_token->length, lg_crcv->app_token->s);
4092 rcvd->body_offset = saved_offset;
4093#if COAP_Q_BLOCK_SUPPORT
4094 rcvd->body_total = block_opt == COAP_OPTION_Q_BLOCK2 ?
4095 lg_crcv->total_len : size2;
4096#else /* ! COAP_Q_BLOCK_SUPPORT */
4097 rcvd->body_total = size2;
4098#endif /* ! COAP_Q_BLOCK_SUPPORT */
4099 coap_log_debug("Client app version of updated PDU\n");
4101
4102 if (sent) {
4103 /* need to put back original token into sent */
4104 if (lg_crcv->app_token)
4105 coap_update_token(sent, lg_crcv->app_token->length,
4106 lg_crcv->app_token->s);
4107 coap_remove_option(sent, lg_crcv->block_option);
4108 }
4109 goto call_app_handler;
4110 }
4111#if COAP_Q_BLOCK_SUPPORT
4112give_to_app:
4113#endif /* COAP_Q_BLOCK_SUPPORT */
4114 if ((session->block_mode & COAP_SINGLE_BLOCK_OR_Q) || block.bert) {
4115 /* Pretend that there is no block */
4116 coap_remove_option(rcvd, block_opt);
4117 if (lg_crcv->observe_set) {
4119 lg_crcv->observe_length, lg_crcv->observe);
4120 }
4121 rcvd->body_data = lg_crcv->body_data->s;
4122#if COAP_Q_BLOCK_SUPPORT
4123 rcvd->body_length = block_opt == COAP_OPTION_Q_BLOCK2 ?
4124 lg_crcv->total_len : saved_offset + length;
4125#else /* ! COAP_Q_BLOCK_SUPPORT */
4126 rcvd->body_length = saved_offset + length;
4127#endif /* ! COAP_Q_BLOCK_SUPPORT */
4128 rcvd->body_offset = 0;
4129 rcvd->body_total = rcvd->body_length;
4130 } else {
4131 rcvd->body_offset = saved_offset;
4132#if COAP_Q_BLOCK_SUPPORT
4133 rcvd->body_total = block_opt == COAP_OPTION_Q_BLOCK2 ?
4134 lg_crcv->total_len : size2;
4135#else /* ! COAP_Q_BLOCK_SUPPORT */
4136 rcvd->body_total = size2;
4137#endif /* ! COAP_Q_BLOCK_SUPPORT */
4138 }
4139 if (context->response_handler || session->doing_send_recv) {
4140 coap_response_t ret;
4141
4142 /* need to put back original token into rcvd */
4143 if (!coap_binary_equal(&rcvd->actual_token, lg_crcv->app_token)) {
4144 coap_update_token(rcvd, lg_crcv->app_token->length, lg_crcv->app_token->s);
4145 coap_log_debug("Client app version of updated PDU\n");
4147 }
4148 if (sent) {
4149 /* need to put back original token into sent */
4150 if (lg_crcv->app_token)
4151 coap_update_token(sent, lg_crcv->app_token->length,
4152 lg_crcv->app_token->s);
4153 coap_remove_option(sent, lg_crcv->block_option);
4154 }
4155 if (!session->doing_send_recv || !session->req_token ||
4156 !coap_binary_equal(session->req_token, &rcvd->actual_token)) {
4157 if (context->response_handler) {
4159 context->response_handler(session, sent, rcvd,
4160 rcvd->mid),
4161 /* context is being freed off */
4162 assert(0));
4163 } else {
4164 ret = COAP_RESPONSE_OK;
4165 }
4166 if (ret == COAP_RESPONSE_FAIL && rcvd->type != COAP_MESSAGE_ACK) {
4167 coap_send_rst_lkd(session, rcvd);
4169 } else {
4170 coap_send_ack_lkd(session, rcvd);
4172 }
4173 } else {
4174 /* processing coap_send_recv() call */
4175 session->resp_pdu = rcvd;
4177 /* Will get freed off when PDU is freed off */
4178 rcvd->data_free = lg_crcv->body_data;
4179 lg_crcv->body_data = NULL;
4180 coap_send_ack_lkd(session, rcvd);
4182 }
4183 } else {
4184 coap_send_ack_lkd(session, rcvd);
4186 }
4187 ack_rst_sent = 1;
4188 if (lg_crcv->observe_set == 0) {
4189 /* Expire this entry */
4190 LL_DELETE(session->lg_crcv, lg_crcv);
4191 coap_block_delete_lg_crcv(session, lg_crcv);
4192 goto skip_app_handler;
4193 }
4194 /* Set up for the next data body as observing */
4195 lg_crcv->initial = 1;
4196 if (lg_crcv->body_data) {
4198 lg_crcv->body_data = NULL;
4199 }
4200 }
4201 coap_ticks(&lg_crcv->last_used);
4202 goto skip_app_handler;
4203 } else {
4204 coap_opt_t *obs_opt = coap_check_option(rcvd,
4206 &opt_iter);
4207 if (obs_opt) {
4208 lg_crcv->observe_length = min(coap_opt_length(obs_opt), 3);
4209 memcpy(lg_crcv->observe, coap_opt_value(obs_opt), lg_crcv->observe_length);
4210 lg_crcv->observe_set = 1;
4211 } else {
4212 lg_crcv->observe_set = 0;
4213 if (!coap_binary_equal(&rcvd->actual_token, lg_crcv->app_token)) {
4214 /* need to put back original token into rcvd */
4215 coap_update_token(rcvd, lg_crcv->app_token->length, lg_crcv->app_token->s);
4217 coap_log_debug("PDU presented to app.\n");
4219 }
4220 /* Expire this entry */
4221 goto expire_lg_crcv;
4222 }
4223 }
4224 coap_ticks(&lg_crcv->last_used);
4225 } else if (rcvd->code == COAP_RESPONSE_CODE(401)) {
4226#if COAP_OSCORE_SUPPORT
4227 if (check_freshness(session, rcvd,
4228 (session->oscore_encryption == 0) ? sent : NULL,
4229 NULL, lg_crcv))
4230#else /* !COAP_OSCORE_SUPPORT */
4231 if (check_freshness(session, rcvd, sent, NULL, lg_crcv))
4232#endif /* !COAP_OSCORE_SUPPORT */
4233 goto skip_app_handler;
4234 goto expire_lg_crcv;
4235 } else {
4236 /* Not 2.xx or 4.01 - assume it is a failure of some sort */
4237 goto expire_lg_crcv;
4238 }
4239 if (!block.m && !lg_crcv->observe_set) {
4240fail_resp:
4241 /* lg_crcv no longer required - cache it for 1 sec */
4242 coap_ticks(&lg_crcv->last_used);
4243 lg_crcv->last_used = lg_crcv->last_used - COAP_MAX_TRANSMIT_WAIT_TICKS(session) +
4245 }
4246 /* need to put back original token into rcvd */
4247 if (!coap_binary_equal(&rcvd->actual_token, lg_crcv->app_token)) {
4248 coap_update_token(rcvd, lg_crcv->app_token->length, lg_crcv->app_token->s);
4249 coap_log_debug("Client app version of updated PDU (3)\n");
4251 }
4252 break;
4253 } /* LL_FOREACH() */
4254
4255 /* Check if receiving a block response and if blocks can be set up */
4256 if (recursive == COAP_RECURSE_OK && !lg_crcv) {
4257 if (!sent) {
4258 if (coap_get_block_b(session, rcvd, COAP_OPTION_BLOCK2, &block)
4259#if COAP_Q_BLOCK_SUPPORT
4260 ||
4261 coap_get_block_b(session, rcvd, COAP_OPTION_Q_BLOCK2, &block)
4262#endif /* COAP_Q_BLOCK_SUPPORT */
4263 ) {
4264 coap_log_debug("** %s: large body receive internal issue\n",
4265 coap_session_str(session));
4266 goto skip_app_handler;
4267 }
4268 } else if (COAP_RESPONSE_CLASS(rcvd->code) == 2) {
4269 if (coap_get_block_b(session, rcvd, COAP_OPTION_BLOCK2, &block)) {
4270#if COAP_Q_BLOCK_SUPPORT
4271 if (session->block_mode & COAP_BLOCK_PROBE_Q_BLOCK) {
4272 set_block_mode_drop_q(session->block_mode);
4273 coap_log_debug("Q-Block support disabled\n");
4274 }
4275#endif /* COAP_Q_BLOCK_SUPPORT */
4276 have_block = 1;
4277 if (block.num != 0) {
4278 /* Assume random access and just give the single response to app */
4279 size_t length;
4280 const uint8_t *data;
4281 size_t chunk = (size_t)1 << (block.szx + 4);
4282
4283 coap_get_data(rcvd, &length, &data);
4284 rcvd->body_offset = block.num*chunk;
4285 rcvd->body_total = block.num*chunk + length + (block.m ? 1 : 0);
4286 goto call_app_handler;
4287 }
4288 }
4289#if COAP_Q_BLOCK_SUPPORT
4290 else if (coap_get_block_b(session, rcvd, COAP_OPTION_Q_BLOCK2, &block)) {
4291 have_block = 1;
4292 /* server indicating that it supports Q_BLOCK2 */
4293 if (!(session->block_mode & COAP_BLOCK_HAS_Q_BLOCK)) {
4294 set_block_mode_has_q(session->block_mode);
4295 }
4296 }
4297#endif /* COAP_Q_BLOCK_SUPPORT */
4298 if (have_block) {
4299 lg_crcv = coap_block_new_lg_crcv(session, sent, NULL);
4300
4301 if (lg_crcv) {
4302 LL_PREPEND(session->lg_crcv, lg_crcv);
4303 return coap_handle_response_get_block(context, session, sent, rcvd,
4305 }
4306 }
4307 track_echo(session, rcvd);
4308 } else if (rcvd->code == COAP_RESPONSE_CODE(401)) {
4309 lg_crcv = coap_block_new_lg_crcv(session, sent, NULL);
4310
4311 if (lg_crcv) {
4312 LL_PREPEND(session->lg_crcv, lg_crcv);
4313 return coap_handle_response_get_block(context, session, sent, rcvd,
4315 }
4316 }
4317 }
4318 return 0;
4319
4320expire_lg_crcv:
4321 /* need to put back original token into rcvd */
4322 if (!coap_binary_equal(&rcvd->actual_token, lg_crcv->app_token)) {
4323 coap_update_token(rcvd, lg_crcv->app_token->length, lg_crcv->app_token->s);
4324 coap_log_debug("Client app version of updated PDU\n");
4326 }
4327
4328 if (sent) {
4329 /* need to put back original token into sent */
4330 if (lg_crcv->app_token)
4331 coap_update_token(sent, lg_crcv->app_token->length,
4332 lg_crcv->app_token->s);
4333 coap_remove_option(sent, lg_crcv->block_option);
4334 }
4335 /* Expire this entry */
4336 LL_DELETE(session->lg_crcv, lg_crcv);
4337 coap_block_delete_lg_crcv(session, lg_crcv);
4338
4339call_app_handler:
4340 return 0;
4341
4342skip_app_handler:
4343 if (!ack_rst_sent)
4344 coap_send_ack_lkd(session, rcvd);
4345 return 1;
4346}
4347#endif /* COAP_CLIENT_SUPPORT */
4348
4349#if COAP_SERVER_SUPPORT
4350/* Check if lg_xmit generated and update PDU code if so */
4351void
4353 const coap_pdu_t *request,
4354 coap_pdu_t *response, const coap_resource_t *resource,
4355 const coap_string_t *query) {
4356 coap_lg_xmit_t *lg_xmit;
4357
4358 if (response->code == 0)
4359 return;
4360 lg_xmit = coap_find_lg_xmit_response(session, request, resource, query);
4361 if (lg_xmit && lg_xmit->pdu.code == 0) {
4362 lg_xmit->pdu.code = response->code;
4363 return;
4364 }
4365}
4366#endif /* COAP_SERVER_SUPPORT */
4367
4368#if COAP_CLIENT_SUPPORT
4369void
4371 uint64_t token_match =
4373 pdu->actual_token.length));
4374 coap_lg_xmit_t *lg_xmit;
4375 coap_lg_crcv_t *lg_crcv;
4376
4377 if (session->lg_crcv) {
4378 LL_FOREACH(session->lg_crcv, lg_crcv) {
4379 if (coap_binary_equal(&pdu->actual_token, lg_crcv->app_token))
4380 return;
4381 if (token_match == STATE_TOKEN_BASE(lg_crcv->state_token)) {
4382 coap_update_token(pdu, lg_crcv->app_token->length,
4383 lg_crcv->app_token->s);
4384 coap_log_debug("Client app version of updated PDU\n");
4386 return;
4387 }
4388 }
4389 }
4390 if (COAP_PDU_IS_REQUEST(pdu) && session->lg_xmit) {
4391 LL_FOREACH(session->lg_xmit, lg_xmit) {
4392 if (coap_binary_equal(&pdu->actual_token, lg_xmit->b.b1.app_token))
4393 return;
4394 if (token_match == STATE_TOKEN_BASE(lg_xmit->b.b1.state_token)) {
4395 coap_update_token(pdu, lg_xmit->b.b1.app_token->length,
4396 lg_xmit->b.b1.app_token->s);
4397 coap_log_debug("Client app version of updated PDU\n");
4399 return;
4400 }
4401 }
4402 }
4403}
4404#endif /* ! COAP_CLIENT_SUPPORT */
#define COAP_ETAG_MAX_BYTES
Definition coap_block.c:24
COAP_STATIC_INLINE int full_match(const uint8_t *a, size_t alen, const uint8_t *b, size_t blen)
Definition coap_block.c:466
#define MAX_BLK_LEN
static int update_received_blocks(coap_rblock_t *rec_blocks, uint32_t block_num, uint32_t block_m)
static int check_all_blocks_in(coap_rblock_t *rec_blocks)
static int coap_add_data_large_internal(coap_session_t *session, const coap_pdu_t *request, coap_pdu_t *pdu, coap_resource_t *resource, const coap_string_t *query, int maxage, uint64_t etag, size_t length, const uint8_t *data, coap_release_large_data_t release_func, void *app_ptr, int single_request, coap_pdu_code_t request_method)
Definition coap_block.c:669
static int setup_block_b(coap_session_t *session, coap_pdu_t *pdu, coap_block_b_t *block, unsigned int num, unsigned int blk_size, size_t total)
Definition coap_block.c:132
#define min(a, b)
Definition coap_block.c:19
static int check_if_received_block(coap_rblock_t *rec_blocks, uint32_t block_num)
int coap_flsll(long long j)
Definition coap_encode.c:28
int coap_fls(unsigned int i)
Definition coap_encode.c:21
#define PRIu32
@ COAP_NACK_TOO_MANY_RETRIES
Definition coap_io.h:63
Library specific build wrapper for coap_internal.h.
#define COAP_API
@ COAP_LG_XMIT
Definition coap_mem.h:55
@ COAP_LG_CRCV
Definition coap_mem.h:56
@ COAP_LG_SRCV
Definition coap_mem.h:57
@ COAP_STRING
Definition coap_mem.h:39
@ COAP_PDU_BUF
Definition coap_mem.h:47
void * coap_realloc_type(coap_memory_tag_t type, void *p, size_t size)
Reallocates a chunk p of bytes created by coap_malloc_type() or coap_realloc_type() and returns a poi...
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().
uint16_t coap_option_num_t
Definition coap_option.h:20
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
coap_mid_t coap_send_rst_lkd(coap_session_t *session, const coap_pdu_t *request)
Sends an RST message with code 0 for the specified request to dst.
Definition coap_net.c:1029
coap_mid_t coap_send_ack_lkd(coap_session_t *session, const coap_pdu_t *request)
Sends an ACK message with code 0 for the specified request to dst.
Definition coap_net.c:1044
int coap_add_data_large_response_lkd(coap_resource_t *resource, coap_session_t *session, const coap_pdu_t *request, coap_pdu_t *response, const coap_string_t *query, uint16_t media_type, int maxage, uint64_t etag, size_t length, const uint8_t *data, coap_release_large_data_t release_func, void *app_ptr)
Associates given data with the response pdu that is passed as fourth parameter.
int coap_context_set_max_block_size_lkd(coap_context_t *context, size_t max_block_size)
Set the context level maximum block size that the server supports when sending or receiving packets w...
Definition coap_block.c:442
void coap_block_delete_lg_srcv(coap_session_t *session, coap_lg_srcv_t *lg_srcv)
void coap_context_set_block_mode_lkd(coap_context_t *context, uint32_t block_mode)
Set the context level CoAP block handling bits for handling RFC7959.
Definition coap_block.c:417
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)
#define COAP_BLOCK_MAX_SIZE_SET(a)
#define COAP_RBLOCK_CNT
void coap_block_delete_lg_crcv(coap_session_t *session, coap_lg_crcv_t *lg_crcv)
int coap_handle_response_get_block(coap_context_t *context, coap_session_t *session, coap_pdu_t *sent, coap_pdu_t *rcvd, coap_recurse_t recursive)
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_handle_response_send_block(coap_session_t *session, coap_pdu_t *sent, coap_pdu_t *rcvd)
coap_mid_t coap_retransmit_oscore_pdu(coap_session_t *session, coap_pdu_t *pdu, coap_opt_t *echo)
int coap_add_data_large_request_lkd(coap_session_t *session, coap_pdu_t *pdu, size_t length, const uint8_t *data, coap_release_large_data_t release_func, void *app_ptr)
Associates given data with the pdu that is passed as second parameter.
void coap_check_update_token(coap_session_t *session, coap_pdu_t *pdu)
The function checks if the token needs to be updated before PDU is presented to the application (only...
void coap_block_delete_lg_xmit(coap_session_t *session, coap_lg_xmit_t *lg_xmit)
#define STATE_TOKEN_FULL(t, r)
#define COAP_SINGLE_BLOCK_OR_Q
int coap_handle_request_put_block(coap_context_t *context, coap_session_t *session, coap_pdu_t *pdu, coap_pdu_t *response, coap_resource_t *resource, coap_string_t *uri_path, coap_opt_t *observe, int *added_block, coap_lg_srcv_t **free_lg_srcv)
#define STATE_TOKEN_BASE(t)
#define COAP_BLOCK_SET_MASK
coap_lg_xmit_t * coap_find_lg_xmit_response(const coap_session_t *session, const coap_pdu_t *request, const coap_resource_t *resource, const coap_string_t *query)
coap_lg_crcv_t * coap_block_new_lg_crcv(coap_session_t *session, coap_pdu_t *pdu, coap_lg_xmit_t *lg_xmit)
int coap_handle_request_send_block(coap_session_t *session, coap_pdu_t *pdu, coap_pdu_t *response, coap_resource_t *resource, coap_string_t *query)
#define COAP_BLOCK_MAX_SIZE_GET(a)
int coap_block_check_lg_xmit_timeouts(coap_session_t *session, coap_tick_t now, coap_tick_t *tim_rem)
@ COAP_RECURSE_OK
@ COAP_RECURSE_NO
COAP_API void coap_context_set_block_mode(coap_context_t *context, uint32_t block_mode)
Set the context level CoAP block handling bits for handling RFC7959.
Definition coap_block.c:409
COAP_API int coap_add_data_large_response(coap_resource_t *resource, coap_session_t *session, const coap_pdu_t *request, coap_pdu_t *response, const coap_string_t *query, uint16_t media_type, int maxage, uint64_t etag, size_t length, const uint8_t *data, coap_release_large_data_t release_func, void *app_ptr)
Associates given data with the response pdu that is passed as fourth parameter.
#define COAP_BLOCK_USE_M_Q_BLOCK
Definition coap_block.h:64
#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_BLOCK_STLESS_BLOCK2
Definition coap_block.h:67
COAP_API int coap_add_data_large_request(coap_session_t *session, coap_pdu_t *pdu, size_t length, const uint8_t *data, coap_release_large_data_t release_func, void *app_ptr)
Associates given data with the pdu that is passed as second parameter.
#define COAP_BLOCK_TRY_Q_BLOCK
Definition coap_block.h:63
#define COAP_BLOCK_STLESS_FETCH
Definition coap_block.h:66
COAP_API int coap_context_set_max_block_size(coap_context_t *context, size_t max_block_size)
Set the context level maximum block size that the server supports when sending or receiving packets w...
Definition coap_block.c:431
int coap_add_block_b_data(coap_pdu_t *pdu, size_t len, const uint8_t *data, coap_block_b_t *block)
Adds the appropriate payload data of the body to the pdu.
Definition coap_block.c:252
#define COAP_BLOCK_SINGLE_BODY
Definition coap_block.h:62
int coap_write_block_b_opt(coap_session_t *session, coap_block_b_t *block, coap_option_num_t number, coap_pdu_t *pdu, size_t data_length)
Writes a block option of type number to message pdu.
Definition coap_block.c:207
int coap_add_block(coap_pdu_t *pdu, size_t len, const uint8_t *data, unsigned int block_num, unsigned char block_szx)
Adds the block_num block of size 1 << (block_szx + 4) from source data to pdu.
Definition coap_block.c:238
void(* coap_release_large_data_t)(coap_session_t *session, void *app_ptr)
Callback handler for de-allocating the data based on app_ptr provided to coap_add_data_large_*() func...
Definition coap_block.h:287
void coap_add_data_blocked_response(const coap_pdu_t *request, coap_pdu_t *response, uint16_t media_type, int maxage, size_t length, const uint8_t *data)
Adds the appropriate part of data to the response pdu.
Definition coap_block.c:277
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.
Definition coap_block.c:62
#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
int coap_get_block(const coap_pdu_t *pdu, coap_option_num_t number, coap_block_t *block)
Initializes block from pdu.
Definition coap_block.c:115
#define COAP_BLOCK_NOT_RANDOM_BLOCK1
Definition coap_block.h:68
#define COAP_OPT_BLOCK_END_BYTE(opt)
Returns the value of the last byte of opt.
Definition coap_block.h:81
int coap_write_block_opt(coap_block_t *block, coap_option_num_t number, coap_pdu_t *pdu, size_t data_length)
Writes a block option of type number to message pdu.
Definition coap_block.c:174
coap_binary_t * coap_block_build_body(coap_binary_t *body_data, size_t length, const uint8_t *data, size_t offset, size_t total)
Re-assemble payloads into a body.
#define COAP_BLOCK_USE_LIBCOAP
Definition coap_block.h:61
void coap_digest_free(coap_digest_ctx_t *digest_ctx)
Free off coap_digest_ctx_t.
int coap_digest_final(coap_digest_ctx_t *digest_ctx, coap_digest_t *digest_buffer)
Finalize the coap_digest information into the provided digest_buffer.
int coap_digest_update(coap_digest_ctx_t *digest_ctx, const uint8_t *data, size_t data_len)
Update the coap_digest information with the next chunk of data.
void coap_digest_ctx_t
coap_digest_ctx_t * coap_digest_setup(void)
Initialize a coap_digest.
time_t coap_time_t
CoAP time in seconds since epoch.
Definition coap_time.h:148
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.
#define COAP_TICKS_PER_SECOND
Use ms resolution on POSIX systems.
Definition coap_time.h:158
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
uint16_t coap_new_message_id_lkd(coap_session_t *session)
Returns a new message id and updates session->tx_mid accordingly.
int coap_client_delay_first(coap_session_t *session)
Delay the sending of the first client request until some other negotiation has completed.
Definition coap_net.c:1284
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.
Definition coap_net.c:1769
coap_response_t
Definition coap_net.h:48
void coap_ticks(coap_tick_t *)
Returns the current value of an internal tick counter.
@ COAP_RESPONSE_FAIL
Response not liked - send CoAP RST packet.
Definition coap_net.h:49
@ COAP_RESPONSE_OK
Response is fine.
Definition coap_net.h:50
unsigned int coap_encode_var_safe(uint8_t *buf, size_t length, unsigned int val)
Encodes multiple-length byte sequences.
Definition coap_encode.c:47
unsigned int coap_decode_var_bytes(const uint8_t *buf, size_t len)
Decodes multiple-length byte sequences.
Definition coap_encode.c:38
uint64_t coap_decode_var_bytes8(const uint8_t *buf, size_t len)
Decodes multiple-length byte sequences.
Definition coap_encode.c:67
unsigned int coap_encode_var_safe8(uint8_t *buf, size_t length, uint64_t val)
Encodes multiple-length byte sequences.
Definition coap_encode.c:77
@ COAP_EVENT_PARTIAL_BLOCK
Triggered when not all of a large body has been received.
Definition coap_event.h:71
@ COAP_EVENT_XMIT_BLOCK_FAIL
Triggered when not all of a large body has been transmitted.
Definition coap_event.h:73
#define coap_lock_callback_ret_release(r, 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_callback(c, func)
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
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
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
@ COAP_LOG_DEBUG
Definition coap_debug.h:58
#define COAP_OBSERVE_CANCEL
The value COAP_OBSERVE_CANCEL in a GET/FETCH request option COAP_OPTION_OBSERVE indicates that the ob...
#define COAP_OBSERVE_ESTABLISH
The value COAP_OBSERVE_ESTABLISH in a GET/FETCH request option COAP_OPTION_OBSERVE indicates a new ob...
COAP_API int coap_cancel_observe(coap_session_t *session, coap_binary_t *token, coap_pdu_type_t message_type)
Cancel an observe that is being tracked by the client large receive logic.
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.
size_t coap_opt_encode_size(uint16_t delta, size_t length)
Compute storage bytes needed for an option with given delta and length.
#define COAP_OPT_ALL
Pre-defined filter that includes all options.
coap_opt_t * coap_check_option(const coap_pdu_t *pdu, coap_option_num_t number, coap_opt_iterator_t *oi)
Retrieves the first option of number number from pdu.
const uint8_t * coap_opt_value(const coap_opt_t *opt)
Returns a pointer to the value of the given option.
int coap_option_filter_set(coap_opt_filter_t *filter, coap_option_num_t option)
Sets the corresponding entry for number in filter.
int coap_rebuild_pdu_for_proxy(coap_pdu_t *pdu)
Convert PDU to use Proxy-Scheme option if Proxy-Uri option is present.
size_t coap_oscore_overhead(coap_session_t *session, coap_pdu_t *pdu)
Determine the additional data size requirements for adding in OSCORE.
#define COAP_PDU_IS_RESPONSE(pdu)
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() ...
Definition coap_pdu.c:1623
void coap_delete_pdu_lkd(coap_pdu_t *pdu)
Dispose of an CoAP PDU and free off associated storage.
Definition coap_pdu.c:190
size_t coap_insert_option(coap_pdu_t *pdu, coap_option_num_t number, size_t len, const uint8_t *data)
Inserts option of given number in the pdu with the appropriate data.
Definition coap_pdu.c:626
int coap_remove_option(coap_pdu_t *pdu, coap_option_num_t number)
Removes (first) option of given number from the pdu.
Definition coap_pdu.c:489
int coap_update_token(coap_pdu_t *pdu, size_t len, const uint8_t *data)
Updates token in pdu with length len and data.
Definition coap_pdu.c:413
#define COAP_TOKEN_EXT_1B_BIAS
#define COAP_TOKEN_EXT_2B_BIAS
size_t coap_update_option(coap_pdu_t *pdu, coap_option_num_t number, size_t len, const uint8_t *data)
Updates existing first option of given number in the pdu with the new data.
Definition coap_pdu.c:720
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)
Duplicate an existing PDU.
Definition coap_pdu.c:230
#define COAP_PAYLOAD_START
int coap_pdu_check_resize(coap_pdu_t *pdu, size_t size)
Dynamically grows the size of pdu to new_size if needed.
Definition coap_pdu.c:339
#define COAP_PDU_IS_REQUEST(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.
Definition coap_pdu.c:776
#define COAP_OPTION_BLOCK2
Definition coap_pdu.h:137
const char * coap_response_phrase(unsigned char code)
Returns a human-readable response phrase for the specified CoAP response code.
Definition coap_pdu.c:947
#define COAP_MEDIATYPE_APPLICATION_MB_CBOR_SEQ
Definition coap_pdu.h:254
#define COAP_OPTION_CONTENT_FORMAT
Definition coap_pdu.h:128
#define COAP_OPTION_SIZE2
Definition coap_pdu.h:139
#define COAP_OPTION_BLOCK1
Definition coap_pdu.h:138
#define COAP_OPTION_Q_BLOCK1
Definition coap_pdu.h:135
int coap_mid_t
coap_mid_t is used to store the CoAP Message ID of a CoAP PDU.
Definition coap_pdu.h:263
#define COAP_TOKEN_EXT_MAX
Definition coap_pdu.h:60
#define COAP_OPTION_URI_PATH
Definition coap_pdu.h:127
#define COAP_RESPONSE_CODE(N)
Definition coap_pdu.h:160
#define COAP_RESPONSE_CLASS(C)
Definition coap_pdu.h:163
coap_pdu_code_t
Set of codes available for a PDU.
Definition coap_pdu.h:326
#define COAP_OPTION_SIZE1
Definition coap_pdu.h:143
coap_pdu_type_t
CoAP PDU message type definitions.
Definition coap_pdu.h:68
#define COAP_MEDIATYPE_TEXT_PLAIN
Definition coap_pdu.h:213
int coap_add_token(coap_pdu_t *pdu, size_t len, const uint8_t *data)
Adds token of length len to pdu.
Definition coap_pdu.c:356
#define COAP_OPTION_CONTENT_TYPE
Definition coap_pdu.h:129
size_t coap_add_option(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.
Definition coap_pdu.c:766
#define COAP_OPTION_Q_BLOCK2
Definition coap_pdu.h:140
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
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.
Definition coap_pdu.c:99
int coap_get_data_large(const coap_pdu_t *pdu, size_t *len, const uint8_t **data, size_t *offset, size_t *total)
Retrieves the data from a PDU, with support for large bodies of data that spans multiple PDUs.
Definition coap_pdu.c:880
#define COAP_INVALID_MID
Indicates an invalid message id.
Definition coap_pdu.h:266
#define COAP_OPTION_MAXAGE
Definition coap_pdu.h:131
#define COAP_OPTION_ETAG
Definition coap_pdu.h:121
#define COAP_OPTION_OBSERVE
Definition coap_pdu.h:123
#define COAP_OPTION_ECHO
Definition coap_pdu.h:144
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.
Definition coap_pdu.c:841
@ COAP_REQUEST_CODE_GET
Definition coap_pdu.h:329
@ COAP_REQUEST_CODE_FETCH
Definition coap_pdu.h:333
@ COAP_MESSAGE_NON
Definition coap_pdu.h:70
@ COAP_MESSAGE_ACK
Definition coap_pdu.h:71
@ COAP_MESSAGE_CON
Definition coap_pdu.h:69
#define COAP_NON_RECEIVE_TIMEOUT_TICKS(s)
The NON_RECEIVE_TIMEOUT definition for the session (s).
#define COAP_NON_TIMEOUT_TICKS(s)
void coap_handle_nack(coap_session_t *session, coap_pdu_t *sent, const coap_nack_reason_t reason, const coap_mid_t mid)
#define COAP_MAX_TRANSMIT_WAIT_TICKS(s)
#define COAP_NON_PARTIAL_TIMEOUT_TICKS(s)
The NON_PARTIAL_TIMEOUT definition for the session (s).
coap_tick_t coap_get_non_timeout_random_ticks(coap_session_t *session)
#define COAP_NSTART(s)
#define COAP_MAX_PAYLOADS(s)
size_t coap_session_max_pdu_size_lkd(const coap_session_t *session)
Get maximum acceptable PDU size.
#define COAP_NON_MAX_RETRANSMIT(s)
#define COAP_PROTO_NOT_RELIABLE(p)
#define COAP_PROTO_RELIABLE(p)
void coap_session_new_token(coap_session_t *session, size_t *len, uint8_t *data)
Creates a new token for use.
@ COAP_SESSION_TYPE_CLIENT
client-side
void coap_delete_bin_const(coap_bin_const_t *s)
Deletes the given const binary data and releases any memory allocated.
Definition coap_str.c:120
void coap_delete_str_const(coap_str_const_t *s)
Deletes the given const string and releases any memory allocated.
Definition coap_str.c:61
coap_binary_t * coap_new_binary(size_t size)
Returns a new binary object with at least size bytes storage allocated.
Definition coap_str.c:77
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...
Definition coap_str.c:110
coap_binary_t * coap_resize_binary(coap_binary_t *s, size_t size)
Resizes the given coap_binary_t object.
Definition coap_str.c:82
void coap_delete_binary(coap_binary_t *s)
Deletes the given coap_binary_t object and releases any memory allocated.
Definition coap_str.c:105
#define coap_binary_equal(binary1, binary2)
Compares the two binary data for equality.
Definition coap_str.h:211
#define coap_string_equal(string1, string2)
Compares the two strings for equality.
Definition coap_str.h:197
coap_string_t * coap_new_string(size_t size)
Returns a new string object with at least size+1 bytes storage allocated.
Definition coap_str.c:21
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...
Definition coap_str.c:51
void coap_delete_string(coap_string_t *s)
Deletes the given string and releases any memory allocated.
Definition coap_str.c:46
int coap_cancel_observe_lkd(coap_session_t *session, coap_binary_t *token, coap_pdu_type_t message_type)
Cancel an observe that is being tracked by the client large receive logic.
int coap_q_block_is_supported(void)
Check whether Q-BlockX is available.
Definition coap_block.c:37
#define COAP_STATIC_INLINE
Definition libcoap.h:53
CoAP binary data definition with const data.
Definition coap_str.h:64
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
CoAP binary data definition.
Definition coap_str.h:56
size_t length
length of binary data
Definition coap_str.h:57
uint8_t * s
binary data
Definition coap_str.h:58
Structure of Block options with BERT support.
Definition coap_block.h:51
unsigned int num
block number
Definition coap_block.h:52
uint32_t chunk_size
‍1024 if BERT
Definition coap_block.h:58
unsigned int bert
Operating as BERT.
Definition coap_block.h:57
unsigned int aszx
block size (0-7 including BERT
Definition coap_block.h:55
unsigned int defined
Set if block found.
Definition coap_block.h:56
unsigned int m
1 if more blocks follow, 0 otherwise
Definition coap_block.h:53
unsigned int szx
block size (0-6)
Definition coap_block.h:54
Structure of Block options.
Definition coap_block.h:42
unsigned int num
block number
Definition coap_block.h:43
unsigned int szx
block size
Definition coap_block.h:45
unsigned int m
1 if more blocks follow, 0 otherwise
Definition coap_block.h:44
The CoAP stack's global state is stored in a coap_context_t object.
coap_response_handler_t response_handler
Called when a response is received.
uint32_t block_mode
Zero or more COAP_BLOCK_ or'd options.
coap_resource_t * proxy_uri_resource
can be used for handling proxy URI resources
coap_resource_t * unknown_resource
can be used for handling unknown resources
uint64_t state_token
state token
size_t bert_size
size of last BERT block
uint32_t count
the number of packets sent for payload
coap_binary_t * app_token
original PDU token
coap_pdu_code_t request_method
Method used to request this data.
uint8_t rtag_length
RTag length.
coap_string_t * query
Associated query for the resource.
uint64_t etag
ETag value.
coap_resource_t * resource
associated resource
coap_time_t maxage_expire
When this entry expires.
uint8_t rtag_set
Set if RTag is in receive PDU.
uint8_t rtag[8]
RTag for block checking.
Structure to hold large body (many blocks) client receive information.
uint16_t block_option
Block option in use.
uint8_t etag[8]
ETag for block checking.
uint8_t etag_length
ETag length.
uint8_t last_type
Last request type (CON/NON)
uint8_t observe_length
Length of observe data.
uint8_t observe[3]
Observe data (if observe_set) (only 24 bits)
uint8_t etag_set
Set if ETag is in receive PDU.
coap_rblock_t rec_blocks
uint8_t initial
If set, has not been used yet.
uint8_t szx
size of individual blocks
uint16_t o_block_option
Block CoAP option used when initiating Observe.
uint16_t content_format
Content format for the set of blocks.
coap_pdu_t pdu
skeletal PDU
uint8_t o_blk_size
Block size used when initiating Observe.
coap_tick_t last_used
< list of received blocks
coap_bin_const_t ** obs_token
Tokens used in setting up Observe (to handle large FETCH)
uint64_t state_token
state token
coap_binary_t * app_token
app requesting PDU token
uint16_t retry_counter
Retry counter (part of state token)
coap_binary_t * body_data
Used for re-assembling entire body.
size_t obs_token_cnt
number of tokens used to set up Observe
uint8_t observe_set
Set if this is an observe receive PDU.
size_t total_len
Length as indicated by SIZE2 option.
Structure to hold large body (many blocks) server receive information.
uint8_t rtag[8]
RTag for block checking.
coap_mid_t last_mid
Last received mid for this set of packets.
uint8_t rtag_set
Set if RTag is in receive PDU.
uint16_t block_option
Block option in use.
size_t total_len
Length as indicated by SIZE1 option.
uint8_t observe_length
Length of observe data.
coap_rblock_t rec_blocks
set to uri_path if unknown resource
uint8_t no_more_seen
Set if block with more not set seen.
coap_binary_t * body_data
Used for re-assembling entire body.
coap_resource_t * resource
associated resource
uint8_t observe_set
Set if this is an observe receive PDU.
uint8_t rtag_length
RTag length.
uint8_t last_type
Last request type (CON/NON)
uint8_t szx
size of individual blocks
coap_tick_t last_used
Last time data sent or 0.
uint8_t observe[3]
Observe data (if set) (only 24 bits)
uint16_t content_format
Content format for the set of blocks.
coap_bin_const_t * last_token
< list of received blocks
coap_str_const_t * uri_path
Structure to hold large body (many blocks) transmission information.
coap_tick_t last_all_sent
Last time all data sent or 0.
coap_release_large_data_t release_func
large data de-alloc function
uint8_t blk_size
large block transmission size
coap_tick_t last_sent
Last time any data sent.
coap_lg_crcv_t * lg_crcv
The lg_crcv associated with this blocked xmit.
union coap_lg_xmit_t::@1 b
const uint8_t * data
large data ptr
int last_block
last acknowledged block number Block1 last transmitted Q-Block2
coap_tick_t last_payload
Last time MAX_PAYLOAD was sent or 0.
size_t offset
large data next offset to transmit
coap_pdu_t pdu
skeletal PDU
size_t length
large data length
coap_l_block1_t b1
coap_l_block2_t b2
uint16_t option
large block transmisson CoAP option
void * app_ptr
applicaton provided ptr for de-alloc function
coap_tick_t last_obs
Last time used (Observe tracking) or 0.
Iterator to run through PDU options.
coap_option_num_t number
decoded option number
structure for CoAP PDUs
uint8_t max_hdr_size
space reserved for protocol-specific header
uint8_t * token
first byte of token (or extended length bytes prefix), if any, or options
coap_lg_xmit_t * lg_xmit
Holds ptr to lg_xmit if sending a set of blocks.
size_t body_length
Holds body data length.
size_t max_size
maximum size for token, options and payload, or zero for variable size pdu
const uint8_t * body_data
Holds ptr to re-assembled data or NULL.
size_t body_offset
Holds body data offset.
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.
uint8_t * data
first byte of payload, if any
coap_mid_t mid
message id, if any, in regular host byte order
uint32_t e_token_length
length of Token space (includes leading extended bytes
size_t used_size
used bytes of storage for token, options and payload
coap_binary_t * data_free
Data to be freed off by coap_delete_pdu()
size_t body_total
Holds body data total size.
coap_pdu_type_t type
message type
Queue entry.
Structure to keep track of received blocks.
uint32_t total_blocks
Set to block no + 1 when More bit unset.
uint32_t used
Number of range blocks in use.
struct coap_lg_range range[COAP_RBLOCK_CNT]
Abstraction of resource that can be attached to coap_context_t.
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
uint32_t block_mode
Zero or more COAP_BLOCK_ or'd options.
uint8_t csm_bert_rem_support
CSM TCP BERT blocks supported (remote)
uint64_t tx_token
Next token number to use.
coap_mid_t remote_test_mid
mid used for checking remote support
uint8_t csm_bert_loc_support
CSM TCP BERT blocks supported (local)
coap_proto_t proto
protocol used
coap_response_t last_con_handler_res
The result of calling the response handler of the last CON.
uint8_t doing_send_recv
Set if coap_send_recv() active.
uint8_t con_active
Active CON request sent.
coap_queue_t * delayqueue
list of delayed messages waiting to be sent
uint32_t tx_rtag
Next Request-Tag number to use.
coap_lg_srcv_t * lg_srcv
Server list of expected large receives.
coap_bin_const_t * req_token
Token in request pdu of coap_send_recv()
coap_pdu_t * resp_pdu
PDU returned in coap_send_recv() call.
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
coap_bin_const_t * echo
last token used to make a request
CoAP string data definition.
Definition coap_str.h:38
uint8_t * s
string data
Definition coap_str.h:40
size_t length
length of string
Definition coap_str.h:39