libcoap 4.3.5-develop-4c3f4af
Loading...
Searching...
No Matches
coap_block_internal.h
Go to the documentation of this file.
1/*
2 * coap_block_internal.h -- Structures, Enums & Functions that are not
3 * exposed to application programming
4 *
5 * Copyright (C) 2010-2026 Olaf Bergmann <bergmann@tzi.org>
6 * Copyright (C) 2021-2026 Jon Shallow <supjps-libcoap@jpshallow.com>
7 *
8 * SPDX-License-Identifier: BSD-2-Clause
9 *
10 * This file is part of the CoAP library libcoap. Please see README for terms
11 * of use.
12 */
13
18
19#ifndef COAP_BLOCK_INTERNAL_H_
20#define COAP_BLOCK_INTERNAL_H_
21
22#ifdef __cplusplus
23extern "C" {
24#endif
25
32
33/*
34 * Use the top 20 bits of a 64 bit token for tracking current block in large transfer.
35 * However, if smaller tokens are required. this vould be set up to 48, leaving 16 bits
36 * for unique tokens.
37 */
38#ifndef STATE_MAX_BLK_CNT_BITS
39#define STATE_MAX_BLK_CNT_BITS 20
40#endif /* STATE_MAX_BLK_CNT_BITS */
41#define STATE_TOKEN_BASE(t) ((t) & (0xffffffffffffffffULL >> STATE_MAX_BLK_CNT_BITS))
42#define STATE_TOKEN_RETRY(t) ((uint64_t)(t) >> (64 - STATE_MAX_BLK_CNT_BITS))
43#define STATE_TOKEN_FULL(t,r) (STATE_TOKEN_BASE(t) + ((uint64_t)(r) << (64 - STATE_MAX_BLK_CNT_BITS)))
44
45#if COAP_Q_BLOCK_SUPPORT
46#define COAP_BLOCK_SET_MASK (COAP_BLOCK_USE_LIBCOAP | \
47 COAP_BLOCK_SINGLE_BODY | \
48 COAP_BLOCK_TRY_Q_BLOCK | \
49 COAP_BLOCK_USE_M_Q_BLOCK | \
50 COAP_BLOCK_NO_PREEMPTIVE_RTAG | \
51 COAP_BLOCK_STLESS_FETCH | \
52 COAP_BLOCK_STLESS_BLOCK2 | \
53 COAP_BLOCK_NOT_RANDOM_BLOCK1 | \
54 COAP_BLOCK_CACHE_RESPONSE | \
55 COAP_BLOCK_FORCE_Q_BLOCK)
56#else /* ! COAP_Q_BLOCK_SUPPORT */
57#define COAP_BLOCK_SET_MASK (COAP_BLOCK_USE_LIBCOAP | \
58 COAP_BLOCK_SINGLE_BODY | \
59 COAP_BLOCK_NO_PREEMPTIVE_RTAG | \
60 COAP_BLOCK_STLESS_FETCH | \
61 COAP_BLOCK_STLESS_BLOCK2 | \
62 COAP_BLOCK_NOT_RANDOM_BLOCK1 | \
63 COAP_BLOCK_CACHE_RESPONSE)
64#endif /* ! COAP_Q_BLOCK_SUPPORT */
65
66#define COAP_BLOCK_MAX_SIZE_MASK 0x7000000 /* (svr)Mask to get the max supported block size */
67#define COAP_BLOCK_MAX_SIZE_SHIFT 24 /* (svr)Mask shift to get the max supported block size */
68#define COAP_BLOCK_MAX_SIZE_GET(a) (((a) & COAP_BLOCK_MAX_SIZE_MASK) >> COAP_BLOCK_MAX_SIZE_SHIFT)
69#define COAP_BLOCK_MAX_SIZE_SET(a) (((a) << COAP_BLOCK_MAX_SIZE_SHIFT) & COAP_BLOCK_MAX_SIZE_MASK)
70
71#if COAP_Q_BLOCK_SUPPORT
72/* Internal use only and are dropped when setting block_mode */
73#define COAP_BLOCK_HAS_Q_BLOCK 0x40000000 /* Set when Q_BLOCK supported */
74#define COAP_BLOCK_PROBE_Q_BLOCK 0x80000000 /* Set when Q_BLOCK probing */
75
76#define set_block_mode_probe_q(block_mode) \
77 do { \
78 block_mode |= COAP_BLOCK_PROBE_Q_BLOCK; \
79 block_mode &= ~(COAP_BLOCK_TRY_Q_BLOCK | COAP_BLOCK_HAS_Q_BLOCK); \
80 } while (0)
81
82#define set_block_mode_has_q(block_mode) \
83 do { \
84 block_mode |= COAP_BLOCK_HAS_Q_BLOCK; \
85 block_mode &= ~(COAP_BLOCK_TRY_Q_BLOCK | COAP_BLOCK_PROBE_Q_BLOCK); \
86 } while (0)
87
88#define set_block_mode_drop_q(block_mode) \
89 do { \
90 block_mode &= ~(COAP_BLOCK_TRY_Q_BLOCK |\
91 COAP_BLOCK_PROBE_Q_BLOCK |\
92 COAP_BLOCK_HAS_Q_BLOCK | \
93 COAP_BLOCK_USE_M_Q_BLOCK | \
94 COAP_BLOCK_FORCE_Q_BLOCK); \
95 } while (0)
96
97#define COAP_SINGLE_BLOCK_OR_Q (COAP_BLOCK_SINGLE_BODY|COAP_BLOCK_HAS_Q_BLOCK)
98#else /* ! COAP_Q_BLOCK_SUPPORT */
99#define COAP_SINGLE_BLOCK_OR_Q (COAP_BLOCK_SINGLE_BODY)
100#endif /* ! COAP_Q_BLOCK_SUPPORT */
101
106
108 uint32_t begin;
109 uint32_t end;
110};
111
112#ifndef COAP_RBLOCK_CNT
113#define COAP_RBLOCK_CNT 4
114#endif
115#if COAP_RBLOCK_CNT > COAP_DEFAULT_MAX_PAYLOADS -1
116#error COAP_RBLOCK_CNT too large
117#endif
118#if COAP_RBLOCK_CNT < 2
119#error COAP_RBLOCK_CNT too small
120#endif
121
125typedef struct coap_rblock_t {
126 uint32_t used;
127 uint32_t retry;
128#if COAP_Q_BLOCK_SUPPORT
129 uint32_t processing_payload_set;
130 uint32_t latest_payload_set;
131#endif /* COAP_Q_BLOCK_SUPPORT */
134 uint32_t total_blocks;
136
141typedef struct coap_l_block1_t {
143 uint64_t state_token;
144 size_t bert_size;
145 uint32_t count;
146 coap_address_t upstream; /* Unicast IP of server if mcast client session */
148
163
172
178 uint8_t blk_size;
179 uint16_t option;
183 union {
186 } b;
192#if COAP_Q_BLOCK_SUPPORT
193 coap_tick_t non_timeout_random_ticks;
194 coap_rblock_t send_blocks;
195#endif /* COAP_Q_BLOCK_SUPPORT */
196 uint32_t ref;
197};
198
199#if COAP_CLIENT_SUPPORT
203struct coap_lg_crcv_t {
204 struct coap_lg_crcv_t *next;
205 uint8_t observe[3];
206 uint8_t observe_length;
207 uint8_t observe_set;
208 uint8_t szx;
209 uint8_t etag_set;
210 uint8_t etag_length;
211 uint8_t etag[8];
212 uint16_t content_format;
213 uint8_t last_type;
214 uint8_t initial;
215 uint16_t block_option;
216 uint16_t retry_counter;
217 size_t total_len;
218 coap_binary_t *body_data;
219 coap_binary_t *app_token;
220 coap_bin_const_t **obs_token;
222 size_t obs_token_cnt;
223 uint16_t o_block_option;
224 uint8_t o_blk_size;
225 uint32_t ref;
226 uint64_t state_token;
227 coap_pdu_t *sent_pdu;
228 coap_rblock_t rec_blocks;
229 coap_tick_t last_used;
230 coap_address_t upstream;
231 coap_lg_xmit_data_t *obs_data;
232};
233#endif /* COAP_CLIENT_SUPPORT */
234
235#if COAP_SERVER_SUPPORT
239struct coap_lg_srcv_t {
240 struct coap_lg_srcv_t *next;
241 uint8_t observe[3];
242 uint8_t observe_length;
243 uint8_t observe_set;
244 uint8_t no_more_seen;
245 uint8_t rtag_set;
246 uint8_t rtag_length;
247 uint8_t rtag[8];
248 uint16_t content_format;
249 uint8_t last_type;
250 uint8_t szx;
251 uint16_t block_option;
252 uint8_t dont_timeout;
253 coap_mid_t last_mid;
254 coap_tick_t last_used;
255 size_t total_len;
256 coap_binary_t *body_data;
257 coap_resource_t *resource;
258 coap_str_const_t *uri_path;
259 coap_rblock_t rec_blocks;
260 coap_bin_const_t *last_token;
261 uint32_t ref;
262};
263#endif /* COAP_SERVER_SUPPORT */
264
265#if COAP_Q_BLOCK_SUPPORT
266typedef enum {
267 COAP_SEND_SKIP_PDU,
268 COAP_SEND_INC_PDU
269} coap_send_pdu_t;
270#endif /* COAP_Q_BLOCK_SUPPORT */
271
272#if COAP_CLIENT_SUPPORT
273coap_lg_crcv_t *coap_block_new_lg_crcv(coap_session_t *session,
274 coap_pdu_t *pdu,
275 coap_lg_xmit_t *lg_xmit);
276
288void coap_block_delete_lg_crcv(coap_session_t *session,
289 coap_lg_crcv_t *lg_crcv);
290
302coap_lg_crcv_release_lkd(coap_session_t *session,coap_lg_crcv_t *lg_crcv) {
303 coap_block_delete_lg_crcv(session, lg_crcv);
304 return;
305}
306
315coap_lg_crcv_reference_lkd(coap_lg_crcv_t *lg_crcv) {
316 lg_crcv->ref++;
317 return;
318}
319
320int coap_block_check_lg_crcv_timeouts(coap_session_t *session,
321 coap_tick_t now,
322 coap_tick_t *tim_rem);
323
324#if COAP_Q_BLOCK_SUPPORT
325coap_mid_t coap_send_q_block1(coap_session_t *session,
326 coap_block_b_t block,
327 coap_pdu_t *request,
328 coap_send_pdu_t send_request);
329
339int coap_block_check_q_block1_xmit(coap_session_t *session,
340 coap_tick_t now, coap_tick_t *tim_rem);
341
342coap_mid_t coap_block_test_q_block(coap_session_t *session, coap_pdu_t *actual);
343#endif /* COAP_Q_BLOCK_SUPPORT */
344
345#endif /* COAP_CLIENT_SUPPORT */
346
347#if COAP_Q_BLOCK_SUPPORT
363coap_mid_t coap_send_q_blocks(coap_session_t *session,
364 coap_lg_xmit_t *lg_xmit,
365 coap_block_b_t block,
366 coap_pdu_t *pdu,
367 coap_send_pdu_t send_pdu);
368#endif /* COAP_Q_BLOCK_SUPPORT */
369
370#if COAP_SERVER_SUPPORT
382void coap_block_delete_lg_srcv(coap_session_t *session,
383 coap_lg_srcv_t *lg_srcv);
384
396coap_lg_srcv_release_lkd(coap_session_t *session,coap_lg_srcv_t *lg_srcv) {
397 coap_block_delete_lg_srcv(session, lg_srcv);
398 return;
399}
400
409coap_lg_srcv_reference_lkd(coap_lg_srcv_t *lg_srcv) {
410 lg_srcv->ref++;
411 return;
412}
413
414int coap_block_check_lg_srcv_timeouts(coap_session_t *session,
415 coap_tick_t now,
416 coap_tick_t *tim_rem);
417
418#if COAP_Q_BLOCK_SUPPORT
428int coap_block_check_q_block2_xmit(coap_session_t *session,
429 coap_tick_t now, coap_tick_t *tim_rem);
430
431coap_mid_t coap_send_q_block2(coap_session_t *session,
432 coap_resource_t *resource,
433 const coap_string_t *query,
434 coap_pdu_code_t request_method,
435 coap_block_b_t block,
436 coap_pdu_t *response,
437 coap_send_pdu_t send_response);
438#endif /* COAP_Q_BLOCK_SUPPORT */
439
440int coap_handle_request_send_block(coap_session_t *session,
441 coap_pdu_t *pdu,
442 coap_pdu_t *response,
443 coap_resource_t *resource,
444 coap_string_t *query);
445
446int coap_handle_request_put_block(coap_context_t *context,
447 coap_session_t *session,
448 coap_pdu_t *pdu,
449 coap_pdu_t *response,
450 coap_resource_t *resource,
451 coap_string_t *uri_path,
452 coap_opt_t *observe,
453 int *added_block,
454 coap_lg_srcv_t **free_lg_srcv);
455
456coap_lg_xmit_t *coap_find_lg_xmit_response(const coap_session_t *session,
457 const coap_pdu_t *request,
458 const coap_resource_t *resource,
459 const coap_string_t *query);
460
513int coap_add_data_large_response_lkd(coap_resource_t *resource,
514 coap_session_t *session,
515 const coap_pdu_t *request,
516 coap_pdu_t *response,
517 const coap_string_t *query,
518 uint16_t media_type,
519 int maxage,
520 uint64_t etag,
521 size_t length,
522 const uint8_t *data,
523 coap_release_large_data_t release_func,
524 void *app_ptr);
525
526#endif /* COAP_SERVER_SUPPORT */
527
528#if COAP_CLIENT_SUPPORT
529int coap_handle_response_send_block(coap_session_t *session, coap_pdu_t *sent,
530 coap_pdu_t *rcvd);
531
532int coap_handle_response_get_block(coap_context_t *context,
533 coap_session_t *session,
534 coap_pdu_t *sent,
535 coap_pdu_t *rcvd,
536 coap_recurse_t recursive);
537
538coap_mid_t coap_retransmit_oscore_pdu(coap_session_t *session,
539 coap_pdu_t *pdu,
540 coap_opt_t *echo);
541
551coap_lg_crcv_t *coap_find_lg_crcv(coap_session_t *session, coap_pdu_t *pdu);
552
553#endif /* COAP_CLIENT_SUPPORT */
554
565
578 coap_lg_xmit_t *lg_xmit);
579
592 coap_block_delete_lg_xmit(session, lg_xmit);
593 return;
594}
595
605 lg_xmit->ref++;
606 return;
607}
608
610 coap_tick_t now,
611 coap_tick_t *tim_rem);
612
613#if COAP_Q_BLOCK_SUPPORT
614int coap_block_drop_resp_q_block_xmit(coap_session_t *session,
615 coap_lg_xmit_t *lg_xmit);
616
617int coap_block_drop_resp_q_block2_crcv(coap_session_t *session,
618 coap_lg_crcv_t *lg_crcv,
619 coap_pdu_t *sent);
620#endif /* COAP_Q_BLOCK_SUPPORT */
621
633 const coap_pdu_t *request,
634 coap_pdu_t *response,
635 const coap_resource_t *resource,
636 const coap_string_t *query);
637
657 uint32_t block_mode);
658
673int coap_context_set_max_block_size_lkd(coap_context_t *context, size_t max_block_size);
674
690coap_binary_t *coap_block_build_body_lkd(coap_binary_t *body_data, size_t length,
691 const uint8_t *data, size_t offset, size_t total);
692
693#if COAP_CLIENT_SUPPORT
738int coap_add_data_large_request_lkd(coap_session_t *session,
739 coap_pdu_t *pdu,
740 size_t length,
741 const uint8_t *data,
742 coap_release_large_data_t release_func,
743 void *app_ptr);
744
784int coap_add_data_large_request_app_lkd(coap_session_t *session,
785 coap_pdu_t *pdu,
786 size_t length,
787 coap_release_large_data_t release_func,
788 coap_get_large_data_t get_func,
789 void *app_ptr);
790
799#else /* ! COAP_CLIENT_SUPPORT */
800#define coap_check_update_token(a,b)
801#endif /* ! COAP_CLIENT_SUPPORT */
802
804
805#ifdef __cplusplus
806}
807#endif
808
809#endif /* COAP_BLOCK_INTERNAL_H_ */
struct coap_lg_crcv_t coap_lg_crcv_t
struct coap_resource_t coap_resource_t
struct coap_lg_srcv_t coap_lg_srcv_t
uint8_t coap_opt_t
Use byte-oriented access methods here because sliding a complex struct coap_opt_t over the data buffe...
struct coap_rblock_t coap_rblock_t
Structure to keep track of received blocks.
#define coap_check_update_token(a, b)
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:454
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:429
#define COAP_RBLOCK_CNT
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...
COAP_STATIC_INLINE void coap_lg_xmit_reference_lkd(coap_lg_xmit_t *lg_xmit)
Increment reference counter on a lg_xmit.
void coap_block_delete_lg_xmit(coap_session_t *session, coap_lg_xmit_t *lg_xmit)
Remove a lg_xmit.
coap_binary_t * coap_block_build_body_lkd(coap_binary_t *body_data, size_t length, const uint8_t *data, size_t offset, size_t total)
Re-assemble payloads into a body.
COAP_STATIC_INLINE void coap_lg_xmit_release_lkd(coap_session_t *session, coap_lg_xmit_t *lg_xmit)
Decrement reference counter on a lg_xmit.
coap_lg_xmit_t * coap_find_lg_xmit(coap_session_t *session, coap_pdu_t *pdu)
Find the current lg_xmit for the session that matches the pdu.
Definition coap_block.c:484
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
int(* coap_get_large_data_t)(coap_session_t *session, size_t max, size_t offset, uint8_t *data, size_t *length, void *app_ptr)
Callback handler for getting the data based on app_ptr provided to coap_add_data_large_request_app() ...
Definition coap_block.h:361
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:292
time_t coap_time_t
CoAP time in seconds since epoch.
Definition coap_time.h:154
uint64_t coap_tick_t
This data type represents internal timer ticks with COAP_TICKS_PER_SECOND resolution.
Definition coap_time.h:149
int coap_mid_t
coap_mid_t is used to store the CoAP Message ID of a CoAP PDU.
Definition coap_pdu.h:184
coap_pdu_code_t
Set of codes available for a PDU.
Definition coap_pdu.h:248
struct coap_str_const_t coap_str_const_t
CoAP string data definition with const data.
struct coap_binary_t coap_binary_t
CoAP binary data definition.
struct coap_bin_const_t coap_bin_const_t
CoAP binary data definition with const data.
#define COAP_STATIC_INLINE
Definition libcoap.h:57
Multi-purpose address abstraction.
CoAP binary data definition with const data.
Definition coap_str.h:65
CoAP binary data definition.
Definition coap_str.h:57
Structure of Block options with BERT support.
Definition coap_block.h:55
The CoAP stack's global state is stored in a coap_context_t object.
Structure to keep track of block1 specific information (Requests).
uint64_t state_token
state token
size_t bert_size
size of last BERT block
coap_address_t upstream
uint32_t count
the number of packets sent for payload
coap_binary_t * app_token
original PDU token
Structure to keep track of block2 specific information (Responses).
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.
coap_get_large_data_t get_func
Where to get data id needed.
void * app_ptr
applicaton provided ptr for de-alloc function
uint32_t ref
Reference count.
const uint8_t * data
large data ptr
size_t length
large data length
coap_release_large_data_t release_func
large data de-alloc function
Structure to hold large body (many blocks) transmission information.
coap_tick_t last_all_sent
Last time all data sent or 0.
uint8_t blk_size
large block transmission size
coap_tick_t last_sent
Last time any data sent.
coap_lg_xmit_data_t * data_info
Pointer to large data information.
union coap_lg_xmit_t::@025301310216224123377060072066277301055061031376 b
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.
coap_pdu_t * sent_pdu
The sent pdu with all the data.
coap_l_block1_t b1
coap_l_block2_t b2
uint32_t ref
Reference count.
uint16_t option
large block transmisson CoAP option
struct coap_lg_xmit_t * next
coap_tick_t last_obs
Last time used (Observe tracking) or 0.
structure for CoAP PDUs
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 virtual session that can be attached to coap_context_t (client) or coap_endpoint_t (se...
CoAP string data definition.
Definition coap_str.h:39