19#define min(a,b) ((a) < (b) ? (a) : (b))
23#ifndef COAP_ETAG_MAX_BYTES
24#define COAP_ETAG_MAX_BYTES 4
26#if COAP_ETAG_MAX_BYTES < 1 || COAP_ETAG_MAX_BYTES > 8
27#error COAP_ETAG_MAX_BYTES byte size invalid
30#if COAP_Q_BLOCK_SUPPORT
76 if (block->
szx == 7) {
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;
124 block->
m = block_b.
m;
134 unsigned int blk_size,
size_t total) {
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;
140 assert(start <= total);
141 memset(block, 0,
sizeof(*block));
143 block->
szx = block->
aszx = blk_size;
144 if (can_use_bert && blk_size == 6 && avail >= 1024 && session != NULL &&
149 block->
chunk_size = (uint32_t)((avail / 1024) * 1024);
151 block->
chunk_size = (size_t)1 << (blk_size + 4);
152 if (avail < block->chunk_size && (total - start) >= avail) {
158 coap_log_debug(
"not enough space, even the smallest block does not fit (1)\n");
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);
164 block->
szx = new_blk_size;
165 block->
num <<= szx - block->
szx;
166 block->
chunk_size = (size_t)1 << (new_blk_size + 4);
177 unsigned char buf[4];
182 start = block->
num << (block->
szx + 4);
183 if (block->
num != 0 && data_length <= start) {
193 block->
szx, data_length))
198 ((block_b.
num << 4) |
211 unsigned char buf[4];
215 start = block->
num << (block->
szx + 4);
216 if (block->
num != 0 && data_length <= start) {
224 block->
szx, data_length))
239 unsigned int block_num,
unsigned char block_szx) {
241 start = block_num << (block_szx + 4);
247 min(len - start, ((
size_t)1 << (block_szx + 4))),
254 unsigned int start = block->
num << (block->
szx + 4);
262 max_size = ((pdu->
max_size - token_options) / 1024) * 1024;
264 max_size = (size_t)1 << (block->
szx + 4);
269 min(len - start, max_size),
284 unsigned char buf[4];
286 int block2_requested = 0;
287#if COAP_SERVER_SUPPORT
293 memset(&block2, 0,
sizeof(block2));
300 block2_requested = 1;
301 if (block2.
num != 0 && length <= (block2.
num << (block2.
szx + 4))) {
304 length >> (block2.
szx + 4));
312#if COAP_SERVER_SUPPORT
323 memcpy(&etag, digest.
key,
sizeof(etag));
324#if COAP_ETAG_MAX_BYTES != 8
347 if (block2_requested) {
400#if COAP_SERVER_SUPPORT
410 uint32_t block_mode) {
418 uint32_t block_mode) {
424#if ! COAP_Q_BLOCK_SUPPORT
432 size_t max_block_size) {
443 switch (max_block_size) {
454 coap_log_info(
"coap_context_set_max_block_size: Invalid max block size (%zu)\n",
459 max_block_size = (
coap_fls((uint32_t)max_block_size >> 4) - 1) & 0x07;
460 context->
block_mode &= ~COAP_BLOCK_MAX_SIZE_MASK;
467 const uint8_t *b,
size_t blen) {
468 return alen == blen && (alen == 0 || memcmp(a, b, alen) == 0);
471#if COAP_CLIENT_SUPPORT
495 coap_log_debug(
"** %s: coap_cancel_observe: COAP_BLOCK_USE_LIBCOAP not enabled\n",
500 LL_FOREACH_SAFE(session->
lg_crcv, lg_crcv, q) {
508#if COAP_Q_BLOCK_SUPPORT
550#if COAP_Q_BLOCK_SUPPORT
552 if (using_q_block1) {
553 mid = coap_send_q_block1(session, block, pdu, COAP_SEND_INC_PDU);
569#if COAP_OSCORE_SUPPORT
575 uint64_t token_match =
586 LL_FOREACH(session->
lg_crcv, lg_crcv) {
625#if COAP_SERVER_SUPPORT
642 LL_FOREACH(session->
lg_xmit, lg_xmit) {
649 lg_xmit->b.b2.query ?
650 lg_xmit->b.b2.query : &empty)) {
659 memcmp(lg_xmit->
b.
b2.
rtag, rtag, rtag_length) != 0)
684#if COAP_Q_BLOCK_SUPPORT
690 int have_block_defined = 0;
692 uint8_t max_blk_size;
694 size_t token_options;
697#if COAP_Q_BLOCK_SUPPORT
701#if !COAP_SERVER_SUPPORT
707 coap_log_warn(
"coap_add_data_large: PDU already contains data\n");
715 coap_log_debug(
"** %s: coap_add_data_large: COAP_BLOCK_USE_LIBCOAP not enabled\n",
729#define MAX_BLK_LEN (((1UL << 20) - 1) * (1 << (6 + 4)))
731#if UINT_MAX > MAX_BLK_LEN
738#if COAP_SERVER_SUPPORT
755 memcpy(&etag, digest.
key,
sizeof(etag));
756#if COAP_ETAG_MAX_BYTES != 8
782 if (etag == etag_r) {
797#if COAP_Q_BLOCK_SUPPORT
813 LL_FOREACH_SAFE(session->
lg_xmit, lg_xmit, q) {
816 LL_DELETE(session->
lg_xmit, lg_xmit);
826#if COAP_Q_BLOCK_SUPPORT
827 if (session->
block_mode & COAP_BLOCK_HAS_Q_BLOCK) {
840#if COAP_SERVER_SUPPORT
848 LL_DELETE(session->
lg_xmit, lg_xmit);
855#if COAP_OSCORE_SUPPORT
856 if (session->oscore_encryption) {
864 avail = pdu->
max_size - token_options;
867#if COAP_OSCORE_SUPPORT
872 blk_size =
coap_flsll((
long long)avail) - 4 - 1;
877 if (max_blk_size && blk_size > max_blk_size)
878 blk_size = max_blk_size;
882 if (block.
szx < blk_size)
883 blk_size = block.
szx;
884 have_block_defined = 1;
886#if COAP_Q_BLOCK_SUPPORT
889 if (have_block_defined) {
891 coap_log_warn(
"Both BlockX and Q-BlockX cannot be set at the same time\n");
895 if (block.
szx < blk_size)
896 blk_size = block.
szx;
897 have_block_defined = 1;
903 if (avail < 16 && ((ssize_t)length > avail || have_block_defined)) {
905 coap_log_debug(
"not enough space, even the smallest block does not fit (2)\n");
909 chunk = (size_t)1 << (blk_size + 4);
910 if ((have_block_defined && block.
num != 0) || single_request ||
915 if (length >= block.
num * chunk) {
916#if COAP_SERVER_SUPPORT
922 (
unsigned int)length),
936 (block.
num << 4) | (block.
m << 3) | block.
aszx),
941 if (chunk > length - block.
num * chunk)
942 rem = length - block.
num * chunk;
949 }
else if ((have_block_defined && length > chunk) || (ssize_t)length > avail) {
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 =
977 lg_xmit->release_func = release_func;
978 lg_xmit->app_ptr = app_ptr;
985 if (!lg_xmit->b.b1.app_token)
993 lg_xmit->b.b1.count = 1;
995 lg_xmit->b.b1.count);
1003 (
unsigned int)length),
1016 lg_xmit->b.b2.resource = resource;
1019 if (lg_xmit->b.b2.query) {
1020 memcpy(lg_xmit->b.b2.query->s, query->
s, query->
length);
1023 lg_xmit->b.b2.query = NULL;
1028 sizeof(lg_xmit->b.b2.rtag));
1030 lg_xmit->b.b2.rtag_set = 1;
1032 lg_xmit->b.b2.rtag_set = 0;
1034 lg_xmit->b.b2.etag = etag;
1035 lg_xmit->b.b2.request_method = request_method;
1042 lg_xmit->b.b2.maxage_expire = 0;
1047 (
unsigned int)length),
1052 blk_size, lg_xmit->length))
1059 (block.
num << 4) | (block.
m << 3) | block.
aszx),
1063 memcpy(&lg_xmit->pdu, pdu,
sizeof(lg_xmit->pdu));
1065 lg_xmit->pdu.used_size + lg_xmit->pdu.max_hdr_size);
1066 if (!lg_xmit->pdu.token)
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);
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 -
1080 avail = pdu->
max_size - token_options;
1085#if COAP_OSCORE_SUPPORT
1088 if (avail < (ssize_t)chunk) {
1091 coap_log_warn(
"not enough space, even the smallest block does not fit (3)\n");
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);
1103 (block.
num << 4) | (block.
m << 3) | lg_xmit->blk_size),
1108 if (rem > lg_xmit->length - block.
num * chunk)
1109 rem = lg_xmit->length - block.
num * chunk;
1114 lg_xmit->b.b1.bert_size = rem;
1116 lg_xmit->last_block = -1;
1119 LL_PREPEND(session->
lg_xmit,lg_xmit);
1122 if (have_block_defined) {
1126 (0 << 4) | (0 << 3) | blk_size), buf);
1141 }
else if (release_func) {
1147#if COAP_CLIENT_SUPPORT
1152 const uint8_t *data,
1160 release_func, app_ptr);
1169 const uint8_t *data,
1183 length, data, release_func, app_ptr, 0, 0);
1187#if COAP_SERVER_SUPPORT
1194 uint16_t media_type,
1198 const uint8_t *data,
1206 response, query, media_type, maxage, etag,
1207 length, data, release_func, app_ptr);
1218 uint16_t media_type,
1222 const uint8_t *data,
1226 unsigned char buf[4];
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) ?
1237 memset(&block, 0,
sizeof(block));
1244 block_requested = 1;
1245 if (block.
num != 0 && length <= (block.
num << (block.
szx + 4))) {
1248 length >> (block.
szx + 4));
1253#if COAP_Q_BLOCK_SUPPORT
1255 block_requested = 1;
1256 if (block.
num != 0 && length <= (block.
num << (block.
szx + 4))) {
1259 length >> (block.
szx + 4));
1263 if (!(session->
block_mode & COAP_BLOCK_HAS_Q_BLOCK)) {
1284 if (block_requested) {
1308 query, maxage, etag, length, data,
1309 release_func, app_ptr, single_request,
1312 goto error_released;
1322#if COAP_ERROR_PHRASE_LENGTH > 0
1340#if COAP_Q_BLOCK_SUPPORT
1350 LL_FOREACH_SAFE(session->
lg_xmit, lg_xmit, q) {
1354 LL_DELETE(session->
lg_xmit, lg_xmit);
1358 if (*tim_rem > lg_xmit->
last_all_sent + idle_timeout - now) {
1364 if (lg_xmit->
last_sent + partial_timeout <= now) {
1366 LL_DELETE(session->
lg_xmit, lg_xmit);
1371 if (*tim_rem > lg_xmit->
last_sent + partial_timeout - now) {
1372 *tim_rem = lg_xmit->
last_sent + partial_timeout - now;
1381#if COAP_CLIENT_SUPPORT
1382#if COAP_Q_BLOCK_SUPPORT
1410 int block_payload_set = -1;
1426 block_size = (size_t)1 << (lg_crcv->
szx + 4);
1427 sofar = block * block_size;
1428 if (sofar < lg_crcv->total_len) {
1431 pdu = coap_build_missing_pdu(session, lg_crcv);
1437 (block << 4) | (1 << 3) | lg_crcv->
szx),
1450 pdu = coap_build_missing_pdu(session, lg_crcv);
1455 if (block_payload_set == -1)
1461 (block << 4) | (0 << 3) | lg_crcv->
szx),
1469 block_size = (size_t)1 << (lg_crcv->
szx + 4);
1470 sofar = (block + 1) * block_size;
1471 if (sofar < lg_crcv->total_len) {
1474 pdu = coap_build_missing_pdu(session, lg_crcv);
1478 sofar = (lg_crcv->
total_len + block_size - 1)/block_size;
1480 if (block_payload_set == -1)
1482 for (; block < (ssize_t)sofar &&
1486 (block << 4) | (0 << 3) | lg_crcv->
szx),
1494 if (block_payload_set != -1)
1495 lg_crcv->
rec_blocks.processing_payload_set = block_payload_set;
1510#if COAP_Q_BLOCK_SUPPORT
1516#if COAP_Q_BLOCK_SUPPORT
1518 session->
block_mode & COAP_BLOCK_HAS_Q_BLOCK)
1524 LL_FOREACH_SAFE(session->
lg_crcv, lg_crcv, q) {
1528#if COAP_Q_BLOCK_SUPPORT
1530 size_t scaled_timeout = receive_timeout *
1540 coap_request_missing_q_block2(session, lg_crcv);
1552 lg_crcv->
last_used + partial_timeout <= now) {
1553#if COAP_Q_BLOCK_SUPPORT
1557 LL_DELETE(session->
lg_crcv, lg_crcv);
1561 if (*tim_rem > lg_crcv->
last_used + partial_timeout - now) {
1562 *tim_rem = lg_crcv->
last_used + partial_timeout - now;
1571#if COAP_SERVER_SUPPORT
1572#if COAP_Q_BLOCK_SUPPORT
1600 assert(block >= 0 && block < (1 << 20));
1602 if (block < 0 || block >= (1 << 20)) {
1604 }
else if (block < 24) {
1607 }
else if (block < 0x100) {
1611 }
else if (block < 0x10000) {
1614 val[1] = block >> 8;
1615 val[2] = block & 0xff;
1619 val[1] = block >> 16;
1620 val[2] = (block >> 8) & 0xff;
1621 val[3] = block & 0xff;
1637 for (i = 0; i < rec_blocks->
used; i++) {
1638 if (block_num < rec_blocks->range[i].begin)
1640 if (block_num <= rec_blocks->range[i].end)
1646#if COAP_SERVER_SUPPORT
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;
1652 if (rec_blocks->
range[rec_blocks->
used-1].
end + 1 == block_num)
1664 for (i = 0; i < rec_blocks->
used; i++) {
1665 if (block < rec_blocks->range[i].begin)
1667 if (block < rec_blocks->range[i].end)
1671 if (block + 1 < total_blocks)
1677#if COAP_CLIENT_SUPPORT
1678#if COAP_Q_BLOCK_SUPPORT
1682 if (rec_blocks->
used &&
1684 rec_blocks->processing_payload_set)
1692 if (rec_blocks->
used > 1 &&
1694 rec_blocks->processing_payload_set)
1701#if COAP_SERVER_SUPPORT
1712#if COAP_Q_BLOCK_SUPPORT
1718#if COAP_Q_BLOCK_SUPPORT
1720 session->
block_mode & COAP_BLOCK_HAS_Q_BLOCK)
1726 LL_FOREACH_SAFE(session->
lg_srcv, lg_srcv, q) {
1730#if COAP_Q_BLOCK_SUPPORT
1732 size_t scaled_timeout = receive_timeout *
1742 size_t block_size = (size_t)1 << (lg_srcv->
szx + 4);
1743 size_t final_block = (lg_srcv->
total_len + block_size - 1)/block_size - 1;
1745 size_t last_payload_block;
1747 size_t no_blocks = 0;
1760 if (no_blocks == 0 && block == (
int)final_block)
1766 if (final_block > last_payload_block) {
1767 final_block = last_payload_block;
1769 no_blocks += final_block - block;
1770 if (no_blocks == 0) {
1772 final_block = (lg_srcv->
total_len + block_size - 1)/block_size - 1;
1774 if (final_block > last_payload_block) {
1775 final_block = last_payload_block;
1785 pdu = pdu_408_build(session, lg_srcv);
1791 if (!add_408_block(pdu, block)) {
1801 for (; block <= (int)final_block; block++) {
1803 pdu = pdu_408_build(session, lg_srcv);
1807 if (!add_408_block(pdu, block)) {
1827#if COAP_Q_BLOCK_SUPPORT
1847 (
const uint8_t *)
"Missing interim block");
1851 LL_DELETE(session->
lg_srcv, lg_srcv);
1855 if (*tim_rem > lg_srcv->
last_used + partial_timeout - now) {
1856 *tim_rem = lg_srcv->
last_used + partial_timeout - now;
1865#if COAP_Q_BLOCK_SUPPORT
1874 coap_send_pdu_t send_pdu) {
1879 const uint8_t *ptoken;
1881 size_t ltoken_length;
1882 uint32_t delayqueue_cnt = 0;
1885 if (send_pdu == COAP_SEND_INC_PDU)
1894 (send_pdu == COAP_SEND_INC_PDU ? 1 : 0);
1895 LL_FOREACH(session->
delayqueue, delayqueue) {
1923 ptoken, &drop_options);
1929 if (send_pdu == COAP_SEND_INC_PDU &&
1939 size_t chunk = ((size_t)1 << (lg_xmit->
blk_size + 4));
1965 ltoken_length, ptoken, &drop_options);
1970 ((block.
num) << 4) |
2009#if COAP_CLIENT_SUPPORT
2017 LL_FOREACH_SAFE(session->
lg_xmit, lg_xmit, q) {
2018 coap_tick_t non_timeout = lg_xmit->non_timeout_random_ticks;
2020 if (now < non_timeout)
2021 return non_timeout - now;
2022 timed_out = now - non_timeout;
2028 size_t chunk = (size_t)1 << (lg_xmit->
blk_size + 4);
2030 memset(&block, 0,
sizeof(block));
2031 block.
num = (uint32_t)(lg_xmit->
offset / chunk);
2034 coap_send_q_blocks(session, lg_xmit, block, &lg_xmit->
pdu, COAP_SEND_SKIP_PDU);
2035 if (tim_rem > non_timeout)
2036 tim_rem = non_timeout;
2046 LL_DELETE(session->
lg_xmit, lg_xmit);
2050 if (tim_rem > lg_xmit->
last_all_sent + 4 * non_timeout - now)
2059#if COAP_SERVER_SUPPORT
2067 LL_FOREACH_SAFE(session->
lg_xmit, lg_xmit, q) {
2068 coap_tick_t non_timeout = lg_xmit->non_timeout_random_ticks;
2070 if (now < non_timeout)
2071 return non_timeout - now;
2072 timed_out = now - non_timeout;
2078 size_t chunk = (size_t)1 << (lg_xmit->
blk_size + 4);
2080 memset(&block, 0,
sizeof(block));
2081 block.
num = (uint32_t)(lg_xmit->
offset / chunk);
2085 coap_send_q_blocks(session, lg_xmit, block, &lg_xmit->
pdu, COAP_SEND_SKIP_PDU);
2086 if (tim_rem > non_timeout)
2087 tim_rem = non_timeout;
2097 LL_DELETE(session->
lg_xmit, lg_xmit);
2101 if (tim_rem > lg_xmit->
last_all_sent + 4 * non_timeout - now)
2111#if COAP_CLIENT_SUPPORT
2125 if (opt && lg_crcv) {
2126 int observe_action = -1;
2137 (block_num + 1) *
sizeof(lg_crcv->
obs_token[0]));
2150 if (lg_crcv->
obs_token[block_num] == NULL)
2154 if (block_num < lg_crcv->obs_token_cnt) {
2162#if COAP_Q_BLOCK_SUPPORT
2167 coap_send_pdu_t send_request) {
2170 uint64_t token_match =
2174 LL_FOREACH(session->
lg_xmit, lg_xmit) {
2183 return coap_send_q_blocks(session, lg_xmit, block, request, send_request);
2188#if COAP_SERVER_SUPPORT
2189#if COAP_Q_BLOCK_SUPPORT
2200 coap_send_pdu_t send_response) {
2205 LL_FOREACH(session->
lg_xmit, lg_xmit) {
2210 lg_xmit->b.b2.query ? lg_xmit->b.b2.query : &empty))
2213 return coap_send_q_blocks(session, lg_xmit, block, response, send_response);
2218#if COAP_CLIENT_SUPPORT
2219#if COAP_Q_BLOCK_SUPPORT
2251 11, (
const u_char *)
".well-known");
2253 4, (
const u_char *)
"core");
2266 (0 << 4) | (0 << 3) | 0),
2283 size_t token_options = pdu->
data ? (size_t)(pdu->
data - pdu->
token) :
2285 size_t data_len = lg_xmit ? lg_xmit->
length :
2291 if (lg_crcv == NULL)
2294 coap_log_debug(
"** %s: lg_crcv %p initialized - stateless token xxxx%012llx\n",
2298 lg_crcv->initial = 1;
2301 memcpy(&lg_crcv->pdu, pdu,
sizeof(lg_crcv->pdu));
2303 lg_crcv->pdu.max_size = token_options + data_len + 9;
2304 lg_crcv->pdu.used_size = token_options + data_len;
2306 token_options + data_len + lg_crcv->pdu.max_hdr_size);
2307 if (!lg_crcv->pdu.token) {
2311 lg_crcv->pdu.token += lg_crcv->pdu.max_hdr_size;
2312 memcpy(lg_crcv->pdu.token, pdu->
token, token_options);
2313 if (lg_crcv->pdu.data) {
2314 lg_crcv->pdu.data = lg_crcv->pdu.token + token_options;
2316 memcpy(lg_crcv->pdu.data, lg_xmit ? lg_xmit->
data : pdu->data, data_len);
2321 if (!lg_crcv->app_token) {
2328 lg_crcv->retry_counter = 1;
2329 lg_crcv->state_token = state_token;
2335 new_token = track_fetch_observe(pdu, lg_crcv, 0, &pdu->
actual_token);
2343 lg_crcv->o_blk_size = block.
aszx;
2359#if (COAP_MAX_LOGGING_LEVEL < _COAP_LOG_DEBUG)
2362 if (lg_crcv == NULL)
2365 LL_FOREACH(session->
lg_xmit, lg_xmit) {
2366 if (lg_xmit->
lg_crcv == lg_crcv) {
2386#if COAP_SERVER_SUPPORT
2390#if (COAP_MAX_LOGGING_LEVEL < _COAP_LOG_DEBUG)
2393 if (lg_srcv == NULL)
2408 if (lg_xmit == NULL)
2427#if COAP_SERVER_SUPPORT
2434add_block_send(uint32_t num,
int is_continue, send_track *out_blocks,
2435 uint32_t *count, uint32_t max_count) {
2438 for (i = 0; i < *count && *count < max_count; i++) {
2439 if (num == out_blocks[i].num)
2441 else if (num < out_blocks[i].num) {
2443 memmove(&out_blocks[i], &out_blocks[i+1], *count - i -1);
2444 out_blocks[i].num = num;
2445 out_blocks[i].is_continue = is_continue;
2450 if (*count < max_count) {
2451 out_blocks[i].num = num;
2452 out_blocks[i].is_continue = is_continue;
2485 uint16_t block_opt = 0;
2486 send_track *out_blocks = NULL;
2487 const char *error_phrase;
2492 uint32_t request_cnt, i;
2495#if COAP_Q_BLOCK_SUPPORT
2500 if (!(session->
block_mode & COAP_BLOCK_HAS_Q_BLOCK))
2507 coap_log_warn(
"Block2 and Q-Block2 cannot be in the same request\n");
2508 coap_add_data(response,
sizeof(
"Both Block2 and Q-Block2 invalid")-1,
2509 (
const uint8_t *)
"Both Block2 and Q-Block2 invalid");
2511 goto skip_app_handler;
2518 if (block.
num == 0) {
2523 if (lg_xmit == NULL)
2526#if COAP_Q_BLOCK_SUPPORT
2532 goto internal_issue;
2545 if (etag == lg_xmit->
b.
b2.
etag) {
2559 chunk = (size_t)1 << (lg_xmit->
blk_size + 4);
2562 coap_log_debug(
"found Block option, block is BERT, block nr. %u, M %d\n",
2563 block.
num, block.
m);
2565 coap_log_debug(
"found Block option, block size is %u, block nr. %u, M %d\n",
2566 1 << (block.
szx + 4), block.
num, block.
m);
2569 if (block.
num == 0) {
2570 if ((lg_xmit->
offset + chunk) % ((
size_t)1 << (block.
szx + 4)) == 0) {
2575 block.
num = (uint32_t)(((lg_xmit->
offset + chunk) >> (block.
szx + 4)) - 1);
2577 chunk = (size_t)1 << (lg_xmit->
blk_size + 4);
2579 coap_log_debug(
"new Block size is %u, block number %u completed\n",
2580 1 << (block.
szx + 4), block.
num);
2583 "next block is not aligned on requested block size "
2584 "boundary. (%zu x %u mod %u = %zu (which is not 0)\n",
2586 (1 << (block.
szx + 4)),
2587 (lg_xmit->
offset + chunk) % ((
size_t)1 << (block.
szx + 4)));
2590 coap_log_debug(
"ignoring request to change Block size from %u to %u\n",
2591 (1 << (lg_xmit->
blk_size + 4)), (1 << (block.
szx + 4)));
2603#if COAP_Q_BLOCK_SUPPORT
2604 max_block = (lg_xmit->
length + chunk - 1)/chunk;
2616 sizeof(
"Changing blocksize during request invalid")-1,
2617 (
const uint8_t *)
"Changing blocksize during request invalid");
2619 goto skip_app_handler;
2621#if COAP_Q_BLOCK_SUPPORT
2626 goto call_app_handler;
2630 num + i < max_block; i++) {
2631 add_block_send(num + i, 1, out_blocks, &request_cnt,
2639 num + i < max_block; i++) {
2640 add_block_send(num + i, 0, out_blocks, &request_cnt,
2645 add_block_send(num, 0, out_blocks, &request_cnt,
2648 add_block_send(num, 0, out_blocks, &request_cnt, 1);
2652 if (request_cnt == 0) {
2656 out_blocks[0].num = 0;
2657 out_blocks[0].is_continue = 0;
2661 for (i = 0; i < request_cnt; i++) {
2664 block.
num = out_blocks[i].num;
2667 if (i + 1 < request_cnt) {
2674 if (out_blocks[i].is_continue) {
2682 goto internal_issue;
2685 if (out_blocks[i].is_continue)
2700 goto internal_issue;
2710 ((out_pdu->
max_size - token_options) /1024) * 1024;
2721 goto internal_issue;
2723 if (!(lg_xmit->
offset + chunk < lg_xmit->length)) {
2731 if (!(lg_xmit->
offset + chunk < lg_xmit->length)) {
2749 goto internal_issue;
2757 goto internal_issue;
2759 if (i + 1 < request_cnt) {
2765 goto skip_app_handler;
2766#if COAP_Q_BLOCK_SUPPORT
2776 (
const uint8_t *)error_phrase);
2792 rec_blocks->
retry = 0;
2794 for (i = 0; i < rec_blocks->
used; i++) {
2795 if (block_num >= rec_blocks->
range[i].
begin &&
2796 block_num <= rec_blocks->range[i].end)
2799 if (block_num < rec_blocks->range[i].begin) {
2800 if (block_num + 1 == rec_blocks->
range[i].
begin) {
2807 memmove(&rec_blocks->
range[i+1], &rec_blocks->
range[i],
2808 (rec_blocks->
used - i) *
sizeof(rec_blocks->
range[0]));
2814 if (block_num == rec_blocks->
range[i].
end + 1) {
2815 rec_blocks->
range[i].
end = block_num;
2816 if (i + 1 < rec_blocks->
used) {
2817 if (rec_blocks->
range[i+1].
begin == block_num + 1) {
2820 if (i+2 < rec_blocks->
used) {
2821 memmove(&rec_blocks->
range[i+1], &rec_blocks->
range[i+2],
2822 (rec_blocks->
used - (i+2)) *
sizeof(rec_blocks->
range[0]));
2830 if (i == rec_blocks->
used) {
2841#if COAP_SERVER_SUPPORT
2861 const uint8_t *data = NULL;
2866 uint16_t block_option = 0;
2873 const uint8_t *rtag;
2874 uint32_t max_block_szx;
2877 unsigned int saved_num;
2878 size_t saved_offset;
2881 *pfree_lg_srcv = NULL;
2888#if COAP_Q_BLOCK_SUPPORT
2891 coap_add_data(response,
sizeof(
"Block1 + Q-Block1 together")-1,
2892 (
const uint8_t *)
"Block1 + Q-Block1 together");
2894 goto skip_app_handler;
2898#if COAP_Q_BLOCK_SUPPORT
2904 if (!block_option ||
2907 goto call_app_handler;
2933 goto skip_app_handler;
2937 offset = block.
num << (block.
szx + 4);
2940#
if COAP_Q_BLOCK_SUPPORT
2959 if (total < (length + offset + (block.
m ? 1 : 0)))
2960 total = length + offset + (block.m ? 1 : 0);
2962 *added_block = block.
m;
2964 goto call_app_handler;
2970 LL_FOREACH(session->
lg_srcv, lg_srcv) {
2971 if (rtag_opt || lg_srcv->
rtag_set == 1) {
2972 if (!(rtag_opt && lg_srcv->
rtag_set == 1))
2975 memcmp(lg_srcv->
rtag, rtag, rtag_length) != 0)
2978 if (resource == lg_srcv->
resource) {
2989 (
const uint8_t *)
"Missing block 0");
2991 goto skip_app_handler;
2997 if (lg_srcv == NULL) {
2999 (
const uint8_t *)
"Memory issue");
3001 goto skip_app_handler;
3014 if (!block.
bert && block.
num == 0 && max_block_szx != 0 &&
3015 max_block_szx < block.
szx) {
3016 lg_srcv->
szx = max_block_szx;
3018 lg_srcv->
szx = block.
szx;
3032 LL_PREPEND(session->
lg_srcv, lg_srcv);
3039 (
const uint8_t *)
"Missing interim block");
3041 goto skip_app_handler;
3045 coap_add_data(response,
sizeof(
"Content-Format mismatch")-1,
3046 (
const uint8_t *)
"Content-Format mismatch");
3051#if COAP_Q_BLOCK_SUPPORT
3055 (
const uint8_t *)
"Size1 mismatch");
3068 chunk = (size_t)1 << (block.
szx + 4);
3070 saved_num = block.
num;
3071 saved_offset = offset;
3073 while (offset < saved_offset + length) {
3078 coap_add_data(response,
sizeof(
"Too many missing blocks")-1,
3079 (
const uint8_t *)
"Too many missing blocks");
3086 offset = block.
num << (block.
szx + 4);
3092#if COAP_Q_BLOCK_SUPPORT
3096 if (lg_srcv->
total_len < saved_offset + length) {
3097 lg_srcv->
total_len = saved_offset + length;
3103 (
const uint8_t *)
"Memory issue");
3105 goto skip_app_handler;
3111 (uint32_t)(lg_srcv->
total_len + chunk -1)/chunk)) {
3116#if COAP_Q_BLOCK_SUPPORT
3119 (uint32_t)(lg_srcv->
total_len + chunk -1)/chunk)) {
3130 goto skip_app_handler;
3134 goto skip_app_handler;
3140 if (!block.
bert && saved_num == 0 && max_block_szx != 0 &&
3141 max_block_szx < block.
aszx) {
3142 block.
aszx = max_block_szx;
3152 (uint32_t)(lg_srcv->
total_len + chunk -1)/chunk)) {
3179#if COAP_Q_BLOCK_SUPPORT
3198#if COAP_Q_BLOCK_SUPPORT
3202 goto skip_app_handler;
3228 *pfree_lg_srcv = lg_srcv;
3234 LL_DELETE(session->
lg_srcv, lg_srcv);
3242#if COAP_CLIENT_SUPPORT
3243#if COAP_Q_BLOCK_SUPPORT
3245derive_cbor_value(
const uint8_t **bp,
size_t rem_len) {
3246 uint32_t value = **bp & 0x1f;
3250 }
else if (value == 24) {
3252 return (uint32_t)-1;
3256 }
else if (value == 25) {
3258 return (uint32_t)-1;
3266 return (uint32_t)-1;
3269 value |= **bp << 16;
3287 if (sent || lg_xmit || lg_crcv) {
3291 const uint8_t *data;
3301 }
else if (lg_xmit) {
3302 sent = &lg_xmit->
pdu;
3304 size_t blk_size = (size_t)1 << (lg_xmit->
blk_size + 4);
3305 size_t offset = (lg_xmit->
last_block + 1) * blk_size;
3307 data = &lg_xmit->
data[offset];
3308 data_len = (lg_xmit->
length - offset) > blk_size ? blk_size :
3309 lg_xmit->length - offset;
3312 sent = &lg_crcv->
pdu;
3334 track_fetch_observe(echo_pdu, lg_crcv, 0, &echo_pdu->
actual_token);
3335#if COAP_OSCORE_SUPPORT
3336 if (session->oscore_encryption &&
3393 uint64_t token_match =
3398 LL_FOREACH_SAFE(session->
lg_xmit, lg_xmit, q) {
3406 size_t chunk = (size_t)1 << (lg_xmit->
blk_size + 4);
3417 coap_log_debug(
"found Block option, block is BERT, block nr. %u (%zu)\n",
3420 coap_log_debug(
"found Block option, block size is %u, block nr. %u\n",
3421 1 << (block.
szx + 4), block.
num);
3427 1 << (block.
szx + 4), 1 << (lg_xmit->
blk_size + 4));
3428 }
else if ((lg_xmit->
offset + chunk) % ((
size_t)1 << (block.
szx + 4)) == 0) {
3433 block.
num = (uint32_t)(((lg_xmit->
offset + chunk) >> (block.
szx + 4)) - 1);
3435 chunk = (size_t)1 << (lg_xmit->
blk_size + 4);
3437 coap_log_debug(
"new Block size is %u, block number %u completed\n",
3438 1 << (block.
szx + 4), block.
num);
3443 "next block is not aligned on requested block size boundary. "
3444 "(%zu x %u mod %u = %zu != 0)\n",
3446 (1 << (block.
szx + 4)),
3447 (lg_xmit->
offset + chunk) % ((
size_t)1 << (block.
szx + 4)));
3450 track_echo(session, rcvd);
3468 lg_xmit->
offset = (block.
num + 1) * chunk;
3484 new_token = track_fetch_observe(&lg_xmit->
pdu, lg_crcv, block.
num + 1,
3487 assert(len <=
sizeof(buf));
3489 memcpy(buf, new_token->
s, len);
3500 size_t token_options = pdu->
data ? (size_t)(pdu->
data - pdu->
token) :
3503 ((pdu->
max_size - token_options) /1024) * 1024;
3521#if COAP_Q_BLOCK_SUPPORT
3524 if (coap_send_q_block1(session, block, pdu,
3544 int observe_action = -1;
3547 goto lg_xmit_finished;
3557 goto lg_xmit_finished;
3565 goto lg_xmit_finished;
3567 if (check_freshness(session, rcvd, sent, lg_xmit, NULL))
3569#if COAP_Q_BLOCK_SUPPORT
3578 const uint8_t *data;
3583 uint16_t fmt = fmt_opt ?
3593 coap_log_debug(
"Unexpected 4.08 - protocol violation - ignore\n");
3599 const uint8_t *bp = data;
3605 size_t ltoken_length;
3607 for (i = 0; (bp < data + length) &&
3609 if ((*bp & 0xc0) != 0x00)
3611 block.
num = derive_cbor_value(&bp, data + length - bp);
3613 if (block.
num > (1 << 20) -1)
3615 block.
m = (block.
num + 1) * chunk < lg_xmit->length;
3645 coap_log_info(
"Invalid application/missing-blocks+cbor-seq\n");
3648 goto lg_xmit_finished;
3685 LL_DELETE(session->
lg_xmit, lg_xmit);
3696 const uint8_t *data,
size_t offset,
size_t total) {
3699 if (body_data == NULL && total) {
3702 if (body_data == NULL)
3706 if (offset + length <= total && body_data->length >= total) {
3707 memcpy(&body_data->
s[offset], data, length);
3723 memcpy(&body_data->
s[offset], data, length);
3732#if COAP_CLIENT_SUPPORT
3752#if COAP_Q_BLOCK_SUPPORT
3756 uint16_t block_opt = 0;
3758 int ack_rst_sent = 0;
3759 uint64_t token_match =
3764 memset(&block, 0,
sizeof(block));
3765#if COAP_Q_BLOCK_SUPPORT
3766 memset(&qblock, 0,
sizeof(qblock));
3768 LL_FOREACH(session->
lg_crcv, lg_crcv) {
3783 const uint8_t *data;
3786 size_t size2 = size_opt ?
3798#if COAP_Q_BLOCK_SUPPORT
3801 coap_log_warn(
"Both Block1 and Q-Block1 not supported in a response\n");
3807 if (!(session->
block_mode & COAP_BLOCK_HAS_Q_BLOCK)) {
3812 track_echo(session, rcvd);
3813 if (have_block && (block.
m || length)) {
3817 uint16_t fmt = fmt_opt ?
3824 size_t saved_offset;
3837 goto expire_lg_crcv;
3840 chunk = (size_t)1 << (block.
szx + 4);
3841 offset = block.
num * chunk;
3842 if (size2 < (offset + length)) {
3844 size2 = offset + length + 1;
3846 size2 = offset + length;
3848 saved_offset = offset;
3851#if COAP_Q_BLOCK_SUPPORT
3868 lg_crcv->
szx = block.
szx;
3872#if COAP_Q_BLOCK_SUPPORT
3873 lg_crcv->
rec_blocks.processing_payload_set = 0;
3890#if COAP_Q_BLOCK_SUPPORT
3895 coap_log_warn(
"Data body updated during receipt - new request started\n");
3912 (0 << 4) | (0 << 3) | block.
aszx),
3918 goto skip_app_handler;
3930#if COAP_Q_BLOCK_SUPPORT
3936 if (block.
num == 0) {
3949 while (offset < saved_offset + length) {
3951#if COAP_Q_BLOCK_SUPPORT
3955 coap_log_debug(
"found Block option, block size is %u, block nr. %u\n",
3956 1 << (block.
szx + 4), block.
num);
3957#if COAP_Q_BLOCK_SUPPORT
3959 this_payload_set > lg_crcv->
rec_blocks.processing_payload_set &&
3960 this_payload_set != lg_crcv->
rec_blocks.latest_payload_set) {
3961 coap_request_missing_q_block2(session, lg_crcv);
3963 lg_crcv->
rec_blocks.latest_payload_set = this_payload_set;
3973 offset = block.
num << (block.
szx + 4);
3979 if (updated_block) {
3981 if (size2 < saved_offset + length) {
3982 size2 = saved_offset + length;
3985 saved_offset, size2);
3991 (size2 + chunk -1) / chunk)) {
3998#if COAP_Q_BLOCK_SUPPORT
4002 (size2 + chunk -1) / chunk)) {
4005 if (check_all_blocks_in_for_payload_set(session,
4011 if (check_any_blocks_next_payload_set(session,
4014 coap_request_missing_q_block2(session, lg_crcv);
4015 goto skip_app_handler;
4020 goto skip_app_handler;
4024 goto skip_app_handler;
4045 ((block.
num + 1) << 4) |
4046 (block.
m << 3) | block.
aszx),
4051 coap_add_data_large_internal(session, NULL, pdu, NULL, NULL, -1, 0, length, data, NULL, NULL, 0, 0);
4057 goto skip_app_handler;
4062#if COAP_Q_BLOCK_SUPPORT
4078 goto call_app_handler;
4080#if COAP_Q_BLOCK_SUPPORT
4091#if COAP_Q_BLOCK_SUPPORT
4093 lg_crcv->
total_len : saved_offset + length;
4101#if COAP_Q_BLOCK_SUPPORT
4159 LL_DELETE(session->
lg_crcv, lg_crcv);
4161 goto skip_app_handler;
4171 goto skip_app_handler;
4190 goto expire_lg_crcv;
4195#if COAP_OSCORE_SUPPORT
4196 if (check_freshness(session, rcvd,
4197 (session->oscore_encryption == 0) ? sent : NULL,
4200 if (check_freshness(session, rcvd, sent, NULL, lg_crcv))
4202 goto skip_app_handler;
4203 goto expire_lg_crcv;
4206 goto expire_lg_crcv;
4228#
if COAP_Q_BLOCK_SUPPORT
4235 goto skip_app_handler;
4239#if COAP_Q_BLOCK_SUPPORT
4240 if (session->
block_mode & COAP_BLOCK_PROBE_Q_BLOCK) {
4246 if (block.
num != 0) {
4249 const uint8_t *data;
4250 size_t chunk = (size_t)1 << (block.
szx + 4);
4255 goto call_app_handler;
4258#if COAP_Q_BLOCK_SUPPORT
4262 if (!(session->
block_mode & COAP_BLOCK_HAS_Q_BLOCK)) {
4271 LL_PREPEND(session->
lg_crcv, lg_crcv);
4276 track_echo(session, rcvd);
4281 LL_PREPEND(session->
lg_crcv, lg_crcv);
4305 LL_DELETE(session->
lg_crcv, lg_crcv);
4318#if COAP_SERVER_SUPPORT
4327 if (response->
code == 0)
4330 if (lg_xmit && lg_xmit->
pdu.
code == 0) {
4337#if COAP_CLIENT_SUPPORT
4340 uint64_t token_match =
4347 LL_FOREACH(session->
lg_crcv, lg_crcv) {
4360 LL_FOREACH(session->
lg_xmit, lg_xmit) {
#define COAP_ETAG_MAX_BYTES
COAP_STATIC_INLINE int full_match(const uint8_t *a, size_t alen, const uint8_t *b, size_t blen)
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)
static int check_all_blocks_in(coap_rblock_t *rec_blocks, size_t total_blocks)
static int update_received_blocks(coap_rblock_t *rec_blocks, uint32_t block_num)
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)
static int check_if_received_block(coap_rblock_t *rec_blocks, uint32_t block_num)
int coap_flsll(long long j)
int coap_fls(unsigned int i)
@ COAP_NACK_TOO_MANY_RETRIES
Library specific build wrapper for coap_internal.h.
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
uint8_t coap_opt_t
Use byte-oriented access methods here because sliding a complex struct coap_opt_t over the data buffe...
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.
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.
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...
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.
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)
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_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.
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
#define COAP_OPT_BLOCK_SZX(opt)
Returns the value of the SZX-field of a Block option opt.
#define COAP_BLOCK_STLESS_BLOCK2
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
#define COAP_BLOCK_STLESS_FETCH
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...
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.
#define COAP_BLOCK_SINGLE_BODY
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.
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.
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...
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.
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.
#define COAP_OPT_BLOCK_MORE(opt)
Returns the value of the More-bit of a Block option opt.
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.
int coap_get_block(const coap_pdu_t *pdu, coap_option_num_t number, coap_block_t *block)
Initializes block from pdu.
#define COAP_BLOCK_NOT_RANDOM_BLOCK1
#define COAP_OPT_BLOCK_END_BYTE(opt)
Returns the value of the last byte of opt.
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.
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
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.
coap_digest_ctx_t * coap_digest_setup(void)
Initialize a coap_digest.
time_t coap_time_t
CoAP time in seconds since epoch.
uint64_t coap_tick_t
This data type represents internal timer ticks with COAP_TICKS_PER_SECOND resolution.
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.
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.
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.
coap_mid_t coap_send_internal(coap_session_t *session, coap_pdu_t *pdu)
Sends a CoAP message to given peer.
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.
@ COAP_RESPONSE_OK
Response is fine.
unsigned int coap_encode_var_safe(uint8_t *buf, size_t length, unsigned int val)
Encodes multiple-length byte sequences.
unsigned int coap_decode_var_bytes(const uint8_t *buf, size_t len)
Decodes multiple-length byte sequences.
uint64_t coap_decode_var_bytes8(const uint8_t *buf, size_t len)
Decodes multiple-length byte sequences.
unsigned int coap_encode_var_safe8(uint8_t *buf, size_t length, uint64_t val)
Encodes multiple-length byte sequences.
@ COAP_EVENT_PARTIAL_BLOCK
Triggered when not all of a large body has been received.
@ COAP_EVENT_XMIT_BLOCK_FAIL
Triggered when not all of a large body has been transmitted.
#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(...)
void coap_show_pdu(coap_log_t level, const coap_pdu_t *pdu)
Display the contents of the specified pdu.
const char * coap_session_str(const coap_session_t *session)
Get session description.
#define coap_log_info(...)
#define coap_log_warn(...)
#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)
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.
int coap_remove_option(coap_pdu_t *pdu, coap_option_num_t number)
Removes (first) option of given number from the pdu.
int coap_update_token(coap_pdu_t *pdu, size_t len, const uint8_t *data)
Updates token in pdu with length len and data.
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.
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.
#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.
#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.
#define COAP_OPTION_BLOCK2
const char * coap_response_phrase(unsigned char code)
Returns a human-readable response phrase for the specified CoAP response code.
#define COAP_MEDIATYPE_APPLICATION_MB_CBOR_SEQ
#define COAP_OPTION_CONTENT_FORMAT
#define COAP_OPTION_SIZE2
#define COAP_OPTION_BLOCK1
#define COAP_OPTION_Q_BLOCK1
void coap_delete_pdu(coap_pdu_t *pdu)
Dispose of an CoAP PDU and frees associated storage.
int coap_mid_t
coap_mid_t is used to store the CoAP Message ID of a CoAP PDU.
#define COAP_OPTION_URI_PATH
#define COAP_RESPONSE_CODE(N)
#define COAP_RESPONSE_CLASS(C)
coap_pdu_code_t
Set of codes available for a PDU.
#define COAP_OPTION_SIZE1
coap_pdu_type_t
CoAP PDU message type definitions.
#define COAP_MEDIATYPE_TEXT_PLAIN
int coap_add_token(coap_pdu_t *pdu, size_t len, const uint8_t *data)
Adds token of length len to pdu.
#define COAP_OPTION_CONTENT_TYPE
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.
#define COAP_OPTION_Q_BLOCK2
int coap_get_data(const coap_pdu_t *pdu, size_t *len, const uint8_t **data)
Retrieves the length and data pointer of specified PDU.
coap_pdu_t * coap_pdu_init(coap_pdu_type_t type, coap_pdu_code_t code, coap_mid_t mid, size_t size)
Creates a new CoAP PDU with at least enough storage space for the given size maximum message size.
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.
#define COAP_INVALID_MID
Indicates an invalid message id.
#define COAP_OPTION_MAXAGE
#define COAP_OPTION_OBSERVE
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.
@ COAP_REQUEST_CODE_FETCH
#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_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.
void coap_delete_str_const(coap_str_const_t *s)
Deletes the given const string and releases any memory allocated.
coap_binary_t * coap_new_binary(size_t size)
Returns a new binary object with at least size bytes storage allocated.
coap_bin_const_t * coap_new_bin_const(const uint8_t *data, size_t size)
Take the specified byte array (text) and create a coap_bin_const_t * Returns a new const binary objec...
coap_binary_t * coap_resize_binary(coap_binary_t *s, size_t size)
Resizes the given coap_binary_t object.
void coap_delete_binary(coap_binary_t *s)
Deletes the given coap_binary_t object and releases any memory allocated.
#define coap_binary_equal(binary1, binary2)
Compares the two binary data for equality.
#define coap_string_equal(string1, string2)
Compares the two strings for equality.
coap_string_t * coap_new_string(size_t size)
Returns a new string object with at least size+1 bytes storage allocated.
coap_str_const_t * coap_new_str_const(const uint8_t *data, size_t size)
Returns a new const string object with at least size+1 bytes storage allocated, and the provided data...
void coap_delete_string(coap_string_t *s)
Deletes the given string and releases any memory allocated.
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.
#define COAP_STATIC_INLINE
CoAP binary data definition with const data.
size_t length
length of binary data
const uint8_t * s
read-only binary data
CoAP binary data definition.
size_t length
length of binary data
Structure of Block options with BERT support.
unsigned int num
block number
uint32_t chunk_size
1024 if BERT
unsigned int bert
Operating as BERT.
unsigned int aszx
block size (0-7 including BERT
unsigned int defined
Set if block found.
unsigned int m
1 if more blocks follow, 0 otherwise
unsigned int szx
block size (0-6)
Structure of Block options.
unsigned int num
block number
unsigned int szx
block size
unsigned int m
1 if more blocks follow, 0 otherwise
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.
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.
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
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
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.
unsigned ref
reference count
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
Structure to keep track of received blocks.
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.
size_t length
length of string