2017-11-22 12:28:11 +01:00
|
|
|
/*
|
2018-03-16 09:35:51 +01:00
|
|
|
* Copyright (C) 2016-18 Kaspar Schleiser <kaspar@schleiser.de>
|
2018-08-23 11:50:46 +02:00
|
|
|
* 2018 Freie Universität Berlin
|
2017-11-22 12:28:11 +01:00
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2018-06-01 13:19:39 +02:00
|
|
|
* @ingroup net_nanocoap
|
2017-11-22 12:28:11 +01:00
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Nanocoap implementation
|
|
|
|
*
|
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
2018-08-23 11:50:46 +02:00
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
2017-11-22 12:28:11 +01:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "net/nanocoap.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
static int _decode_value(unsigned val, uint8_t **pkt_pos_ptr, uint8_t *pkt_end);
|
2018-03-16 09:35:51 +01:00
|
|
|
int coap_get_option_uint(coap_pkt_t *pkt, unsigned opt_num, uint32_t *target);
|
2018-03-16 09:36:05 +01:00
|
|
|
static uint32_t _decode_uint(uint8_t *pkt_pos, unsigned nbytes);
|
|
|
|
static size_t _encode_uint(uint32_t *val);
|
2017-11-22 12:28:11 +01:00
|
|
|
|
|
|
|
/* http://tools.ietf.org/html/rfc7252#section-3
|
|
|
|
* 0 1 2 3
|
|
|
|
* 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
|
|
|
|
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
|
|
* |Ver| T | TKL | Code | Message ID |
|
|
|
|
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
|
|
* | Token (if any, TKL bytes) ...
|
|
|
|
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
|
|
* | Options (if any) ...
|
|
|
|
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
|
|
* |1 1 1 1 1 1 1 1| Payload (if any) ...
|
|
|
|
* +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
|
|
*/
|
|
|
|
int coap_parse(coap_pkt_t *pkt, uint8_t *buf, size_t len)
|
|
|
|
{
|
|
|
|
coap_hdr_t *hdr = (coap_hdr_t *)buf;
|
|
|
|
pkt->hdr = hdr;
|
|
|
|
|
|
|
|
uint8_t *pkt_pos = hdr->data;
|
|
|
|
uint8_t *pkt_end = buf + len;
|
|
|
|
|
2018-03-16 09:35:51 +01:00
|
|
|
pkt->payload = NULL;
|
2017-11-22 12:28:11 +01:00
|
|
|
pkt->payload_len = 0;
|
|
|
|
|
|
|
|
/* token value (tkl bytes) */
|
|
|
|
if (coap_get_token_len(pkt)) {
|
|
|
|
pkt->token = pkt_pos;
|
|
|
|
pkt_pos += coap_get_token_len(pkt);
|
2017-11-23 11:28:28 +01:00
|
|
|
}
|
|
|
|
else {
|
2017-11-22 12:28:11 +01:00
|
|
|
pkt->token = NULL;
|
|
|
|
}
|
|
|
|
|
2018-03-16 09:35:51 +01:00
|
|
|
coap_optpos_t *optpos = pkt->options;
|
|
|
|
unsigned option_count = 0;
|
|
|
|
unsigned option_nr = 0;
|
|
|
|
|
2017-11-22 12:28:11 +01:00
|
|
|
/* parse options */
|
|
|
|
while (pkt_pos != pkt_end) {
|
2018-03-16 09:35:51 +01:00
|
|
|
uint8_t *option_start = pkt_pos;
|
2017-11-22 12:28:11 +01:00
|
|
|
uint8_t option_byte = *pkt_pos++;
|
|
|
|
if (option_byte == 0xff) {
|
|
|
|
pkt->payload = pkt_pos;
|
|
|
|
pkt->payload_len = buf + len - pkt_pos;
|
|
|
|
DEBUG("payload len = %u\n", pkt->payload_len);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
int option_delta = _decode_value(option_byte >> 4, &pkt_pos, pkt_end);
|
|
|
|
if (option_delta < 0) {
|
|
|
|
DEBUG("bad op delta\n");
|
|
|
|
return -EBADMSG;
|
|
|
|
}
|
|
|
|
int option_len = _decode_value(option_byte & 0xf, &pkt_pos, pkt_end);
|
|
|
|
if (option_len < 0) {
|
|
|
|
DEBUG("bad op len\n");
|
|
|
|
return -EBADMSG;
|
|
|
|
}
|
|
|
|
option_nr += option_delta;
|
2018-03-16 09:35:51 +01:00
|
|
|
DEBUG("option count=%u nr=%u len=%i\n", option_count, option_nr, option_len);
|
|
|
|
|
|
|
|
if (option_delta) {
|
|
|
|
optpos->opt_num = option_nr;
|
|
|
|
optpos->offset = (uintptr_t)option_start - (uintptr_t)hdr;
|
|
|
|
DEBUG("optpos option_nr=%u %u\n", (unsigned)option_nr, (unsigned)optpos->offset);
|
|
|
|
optpos++;
|
|
|
|
option_count++;
|
2017-11-22 12:28:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
pkt_pos += option_len;
|
2018-03-16 09:35:51 +01:00
|
|
|
|
|
|
|
if (pkt_pos > (buf + len)) {
|
|
|
|
DEBUG("nanocoap: bad pkt\n");
|
|
|
|
return -EBADMSG;
|
|
|
|
}
|
2017-11-22 12:28:11 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-16 09:35:51 +01:00
|
|
|
pkt->options_len = option_count;
|
|
|
|
if (!pkt->payload) {
|
|
|
|
pkt->payload = pkt_pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef MODULE_GCOAP
|
2018-08-23 11:50:46 +02:00
|
|
|
coap_get_uri_path(pkt, pkt->url);
|
2018-03-16 09:35:51 +01:00
|
|
|
pkt->content_type = coap_get_content_type(pkt);
|
|
|
|
|
|
|
|
if (coap_get_option_uint(pkt, COAP_OPT_OBSERVE, &pkt->observe_value) != 0) {
|
|
|
|
pkt->observe_value = UINT32_MAX;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
DEBUG("coap pkt parsed. code=%u detail=%u payload_len=%u, nopts=%u, 0x%02x\n",
|
2017-11-23 11:28:28 +01:00
|
|
|
coap_get_code_class(pkt),
|
|
|
|
coap_get_code_detail(pkt),
|
2018-03-16 09:35:51 +01:00
|
|
|
pkt->payload_len, option_count, hdr->code);
|
2017-11-22 12:28:11 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-08-23 11:50:46 +02:00
|
|
|
uint8_t *coap_find_option(const coap_pkt_t *pkt, unsigned opt_num)
|
2018-03-16 09:35:51 +01:00
|
|
|
{
|
2018-08-23 11:50:46 +02:00
|
|
|
const coap_optpos_t *optpos = pkt->options;
|
2018-03-16 09:35:51 +01:00
|
|
|
unsigned opt_count = pkt->options_len;
|
|
|
|
|
|
|
|
while (opt_count--) {
|
|
|
|
if (optpos->opt_num == opt_num) {
|
|
|
|
return (uint8_t*)pkt->hdr + optpos->offset;
|
|
|
|
}
|
|
|
|
optpos++;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2018-08-23 11:50:46 +02:00
|
|
|
static uint8_t *_parse_option(const coap_pkt_t *pkt,
|
|
|
|
uint8_t *pkt_pos, uint16_t *delta, int *opt_len)
|
2018-03-16 09:35:51 +01:00
|
|
|
{
|
|
|
|
uint8_t *hdr_end = pkt->payload;
|
|
|
|
|
|
|
|
if (pkt_pos == hdr_end) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t option_byte = *pkt_pos++;
|
|
|
|
|
|
|
|
*delta = _decode_value(option_byte >> 4, &pkt_pos, hdr_end);
|
|
|
|
*opt_len = _decode_value(option_byte & 0xf, &pkt_pos, hdr_end);
|
|
|
|
|
|
|
|
return pkt_pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
int coap_get_option_uint(coap_pkt_t *pkt, unsigned opt_num, uint32_t *target)
|
|
|
|
{
|
|
|
|
assert(target);
|
|
|
|
|
|
|
|
uint8_t *opt_pos = coap_find_option(pkt, opt_num);
|
|
|
|
if (opt_pos) {
|
|
|
|
uint16_t delta;
|
|
|
|
int option_len = 0;
|
|
|
|
uint8_t *pkt_pos = _parse_option(pkt, opt_pos, &delta, &option_len);
|
|
|
|
if (option_len >= 0) {
|
|
|
|
if (option_len > 4) {
|
|
|
|
DEBUG("nanocoap: uint option with len > 4 (unsupported).\n");
|
|
|
|
return -ENOSPC;
|
|
|
|
}
|
|
|
|
*target = _decode_uint(pkt_pos, option_len);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
DEBUG("nanocoap: discarding packet with invalid option length.\n");
|
|
|
|
return -EBADMSG;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2018-08-23 11:50:46 +02:00
|
|
|
uint8_t *coap_iterate_option(const coap_pkt_t *pkt, uint8_t **optpos,
|
|
|
|
int *opt_len, int first)
|
2018-03-16 09:35:51 +01:00
|
|
|
{
|
|
|
|
uint8_t *data_start;
|
|
|
|
|
|
|
|
uint16_t delta = 0;
|
|
|
|
data_start = _parse_option(pkt, *optpos, &delta, opt_len);
|
|
|
|
if (data_start && (first || !delta)) {
|
|
|
|
*optpos = data_start + *opt_len;
|
|
|
|
return data_start;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*optpos = NULL;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned coap_get_content_type(coap_pkt_t *pkt)
|
|
|
|
{
|
|
|
|
uint8_t *opt_pos = coap_find_option(pkt, COAP_OPT_CONTENT_FORMAT);
|
|
|
|
unsigned content_type = COAP_FORMAT_NONE;
|
|
|
|
if (opt_pos) {
|
|
|
|
uint16_t delta;
|
|
|
|
int option_len = 0;
|
|
|
|
uint8_t *pkt_pos = _parse_option(pkt, opt_pos, &delta, &option_len);
|
|
|
|
|
|
|
|
if (option_len == 0) {
|
|
|
|
content_type = 0;
|
|
|
|
} else if (option_len == 1) {
|
|
|
|
content_type = *pkt_pos;
|
|
|
|
} else if (option_len == 2) {
|
|
|
|
memcpy(&content_type, pkt_pos, 2);
|
|
|
|
content_type = ntohs(content_type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return content_type;
|
|
|
|
}
|
|
|
|
|
2018-08-23 11:50:46 +02:00
|
|
|
ssize_t coap_opt_get_string(const coap_pkt_t *pkt, uint16_t optnum,
|
|
|
|
uint8_t *target, size_t max_len, char separator)
|
2018-03-16 09:35:51 +01:00
|
|
|
{
|
2018-08-23 11:50:46 +02:00
|
|
|
assert(pkt && target && (max_len > 1));
|
|
|
|
|
|
|
|
uint8_t *opt_pos = coap_find_option(pkt, optnum);
|
2018-03-16 09:35:51 +01:00
|
|
|
if (!opt_pos) {
|
2018-08-23 11:50:46 +02:00
|
|
|
*target++ = (uint8_t)separator;
|
2018-03-16 09:35:51 +01:00
|
|
|
*target = '\0';
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
|
2018-08-23 11:50:46 +02:00
|
|
|
unsigned left = max_len - 1;
|
2018-03-16 09:35:51 +01:00
|
|
|
uint8_t *part_start = NULL;
|
|
|
|
do {
|
|
|
|
int opt_len;
|
2018-08-23 11:50:46 +02:00
|
|
|
part_start = coap_iterate_option(pkt, &opt_pos, &opt_len,
|
|
|
|
(part_start == NULL));
|
2018-03-16 09:35:51 +01:00
|
|
|
if (part_start) {
|
|
|
|
if (left < (unsigned)(opt_len + 1)) {
|
|
|
|
return -ENOSPC;
|
|
|
|
}
|
2018-08-23 11:50:46 +02:00
|
|
|
*target++ = (uint8_t)separator;
|
2018-03-16 09:35:51 +01:00
|
|
|
memcpy(target, part_start, opt_len);
|
|
|
|
target += opt_len;
|
|
|
|
left -= (opt_len + 1);
|
|
|
|
}
|
|
|
|
} while(opt_pos);
|
|
|
|
|
|
|
|
*target = '\0';
|
|
|
|
|
2018-08-23 11:50:46 +02:00
|
|
|
return (int)(max_len - left);
|
2018-03-16 09:35:51 +01:00
|
|
|
}
|
|
|
|
|
2018-03-16 09:36:05 +01:00
|
|
|
int coap_get_blockopt(coap_pkt_t *pkt, uint16_t option, uint32_t *blknum, unsigned *szx)
|
|
|
|
{
|
|
|
|
uint8_t *optpos = coap_find_option(pkt, option);
|
|
|
|
if (!optpos) {
|
|
|
|
*blknum = 0;
|
|
|
|
*szx = 0;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int option_len;
|
|
|
|
uint16_t delta;
|
|
|
|
|
|
|
|
uint8_t *data_start = _parse_option(pkt, optpos, &delta, &option_len);
|
|
|
|
uint32_t blkopt = _decode_uint(data_start, option_len);
|
|
|
|
|
|
|
|
DEBUG("nanocoap: blkopt len: %i\n", option_len);
|
|
|
|
DEBUG("nanocoap: blkopt: 0x%08x\n", (unsigned)blkopt);
|
|
|
|
*blknum = blkopt >> COAP_BLOCKWISE_NUM_OFF;
|
|
|
|
*szx = blkopt & COAP_BLOCKWISE_SZX_MASK;
|
|
|
|
|
|
|
|
return (blkopt & 0x8) ? 1 : 0;
|
|
|
|
}
|
|
|
|
|
2017-11-22 12:28:11 +01:00
|
|
|
ssize_t coap_handle_req(coap_pkt_t *pkt, uint8_t *resp_buf, unsigned resp_buf_len)
|
|
|
|
{
|
|
|
|
if (coap_get_code_class(pkt) != COAP_REQ) {
|
|
|
|
DEBUG("coap_handle_req(): not a request.\n");
|
|
|
|
return -EBADMSG;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pkt->hdr->code == 0) {
|
|
|
|
return coap_build_reply(pkt, COAP_CODE_EMPTY, resp_buf, resp_buf_len, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned method_flag = coap_method2flag(coap_get_code_detail(pkt));
|
|
|
|
|
2018-03-16 09:35:51 +01:00
|
|
|
#ifdef MODULE_GCOAP
|
|
|
|
uint8_t *uri = pkt->url;
|
|
|
|
#else
|
|
|
|
uint8_t uri[NANOCOAP_URI_MAX];
|
2018-08-23 11:50:46 +02:00
|
|
|
if (coap_get_uri_path(pkt, uri) <= 0) {
|
2018-03-16 09:35:51 +01:00
|
|
|
return -EBADMSG;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
DEBUG("nanocoap: URI path: \"%s\"\n", uri);
|
|
|
|
|
2017-11-22 12:28:11 +01:00
|
|
|
for (unsigned i = 0; i < coap_resources_numof; i++) {
|
2018-02-02 18:54:29 +01:00
|
|
|
const coap_resource_t *resource = &coap_resources[i];
|
|
|
|
if (!(resource->methods & method_flag)) {
|
2017-11-22 12:28:11 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-03-16 09:35:51 +01:00
|
|
|
int res = strcmp((char *)uri, resource->path);
|
2017-11-22 12:28:11 +01:00
|
|
|
if (res > 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else if (res < 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else {
|
2018-02-02 18:54:29 +01:00
|
|
|
return resource->handler(pkt, resp_buf, resp_buf_len, resource->context);
|
2017-11-22 12:28:11 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return coap_build_reply(pkt, COAP_CODE_404, resp_buf, resp_buf_len, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t coap_reply_simple(coap_pkt_t *pkt,
|
2017-11-23 11:28:28 +01:00
|
|
|
unsigned code,
|
|
|
|
uint8_t *buf, size_t len,
|
|
|
|
unsigned ct,
|
|
|
|
const uint8_t *payload, uint8_t payload_len)
|
2017-11-22 12:28:11 +01:00
|
|
|
{
|
|
|
|
uint8_t *payload_start = buf + coap_get_total_hdr_len(pkt);
|
|
|
|
uint8_t *bufpos = payload_start;
|
|
|
|
|
|
|
|
if (payload_len) {
|
|
|
|
bufpos += coap_put_option_ct(bufpos, 0, ct);
|
|
|
|
*bufpos++ = 0xff;
|
|
|
|
|
|
|
|
memcpy(bufpos, payload, payload_len);
|
|
|
|
bufpos += payload_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
return coap_build_reply(pkt, code, buf, len, bufpos - payload_start);
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t coap_build_reply(coap_pkt_t *pkt, unsigned code,
|
2017-11-23 11:28:28 +01:00
|
|
|
uint8_t *rbuf, unsigned rlen, unsigned payload_len)
|
2017-11-22 12:28:11 +01:00
|
|
|
{
|
|
|
|
unsigned tkl = coap_get_token_len(pkt);
|
|
|
|
unsigned len = sizeof(coap_hdr_t) + tkl;
|
|
|
|
|
|
|
|
if ((len + payload_len + 1) > rlen) {
|
|
|
|
return -ENOSPC;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if code is COAP_CODE_EMPTY (zero), use RST as type, else RESP */
|
|
|
|
unsigned type = code ? COAP_RESP : COAP_RST;
|
|
|
|
|
2018-01-12 11:40:03 +01:00
|
|
|
coap_build_hdr((coap_hdr_t *)rbuf, type, pkt->token, tkl, code,
|
|
|
|
ntohs(pkt->hdr->id));
|
2017-11-23 11:28:28 +01:00
|
|
|
coap_hdr_set_type((coap_hdr_t *)rbuf, type);
|
|
|
|
coap_hdr_set_code((coap_hdr_t *)rbuf, code);
|
2017-11-22 12:28:11 +01:00
|
|
|
|
|
|
|
len += payload_len;
|
|
|
|
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t coap_build_hdr(coap_hdr_t *hdr, unsigned type, uint8_t *token, size_t token_len, unsigned code, uint16_t id)
|
|
|
|
{
|
|
|
|
assert(!(type & ~0x3));
|
|
|
|
assert(!(token_len & ~0x1f));
|
|
|
|
|
|
|
|
memset(hdr, 0, sizeof(coap_hdr_t));
|
|
|
|
hdr->ver_t_tkl = (0x1 << 6) | (type << 4) | token_len;
|
|
|
|
hdr->code = code;
|
2018-01-12 11:40:03 +01:00
|
|
|
hdr->id = htons(id);
|
2017-11-22 12:28:11 +01:00
|
|
|
|
|
|
|
if (token_len) {
|
|
|
|
memcpy(hdr->data, token, token_len);
|
|
|
|
}
|
|
|
|
|
|
|
|
return sizeof(coap_hdr_t) + token_len;
|
|
|
|
}
|
|
|
|
|
2018-05-01 19:15:54 +02:00
|
|
|
void coap_pkt_init(coap_pkt_t *pkt, uint8_t *buf, size_t len, size_t header_len)
|
|
|
|
{
|
|
|
|
memset(pkt, 0, sizeof(coap_pkt_t));
|
|
|
|
pkt->hdr = (coap_hdr_t *)buf;
|
|
|
|
pkt->token = buf + sizeof(coap_hdr_t);
|
|
|
|
pkt->payload = buf + header_len;
|
|
|
|
pkt->payload_len = len - header_len;
|
|
|
|
}
|
|
|
|
|
2017-11-22 12:28:11 +01:00
|
|
|
static int _decode_value(unsigned val, uint8_t **pkt_pos_ptr, uint8_t *pkt_end)
|
|
|
|
{
|
|
|
|
uint8_t *pkt_pos = *pkt_pos_ptr;
|
|
|
|
size_t left = pkt_end - pkt_pos;
|
|
|
|
int res;
|
2017-11-23 11:28:28 +01:00
|
|
|
|
2017-11-22 12:28:11 +01:00
|
|
|
switch (val) {
|
|
|
|
case 13:
|
2017-11-23 11:28:28 +01:00
|
|
|
{
|
2017-11-22 12:28:11 +01:00
|
|
|
/* An 8-bit unsigned integer follows the initial byte and
|
|
|
|
indicates the Option Delta minus 13. */
|
|
|
|
if (left < 1) {
|
|
|
|
return -ENOSPC;
|
|
|
|
}
|
|
|
|
uint8_t delta = *pkt_pos++;
|
|
|
|
res = delta + 13;
|
|
|
|
break;
|
2017-11-23 11:28:28 +01:00
|
|
|
}
|
2017-11-22 12:28:11 +01:00
|
|
|
case 14:
|
2017-11-23 11:28:28 +01:00
|
|
|
{
|
2017-11-22 12:28:11 +01:00
|
|
|
/* A 16-bit unsigned integer in network byte order follows
|
|
|
|
* the initial byte and indicates the Option Delta minus
|
|
|
|
* 269. */
|
|
|
|
if (left < 2) {
|
|
|
|
return -ENOSPC;
|
|
|
|
}
|
|
|
|
uint16_t delta;
|
2017-11-23 11:28:28 +01:00
|
|
|
uint8_t *_tmp = (uint8_t *)δ
|
|
|
|
*_tmp++ = *pkt_pos++;
|
|
|
|
*_tmp++ = *pkt_pos++;
|
2017-11-22 12:28:11 +01:00
|
|
|
res = ntohs(delta) + 269;
|
|
|
|
break;
|
2017-11-23 11:28:28 +01:00
|
|
|
}
|
2017-11-22 12:28:11 +01:00
|
|
|
case 15:
|
|
|
|
/* Reserved for the Payload Marker. If the field is set to
|
|
|
|
* this value but the entire byte is not the payload
|
|
|
|
* marker, this MUST be processed as a message format
|
|
|
|
* error. */
|
|
|
|
return -EBADMSG;
|
|
|
|
default:
|
|
|
|
res = val;
|
|
|
|
}
|
|
|
|
|
|
|
|
*pkt_pos_ptr = pkt_pos;
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2018-03-16 09:36:05 +01:00
|
|
|
static uint32_t _decode_uint(uint8_t *pkt_pos, unsigned nbytes)
|
|
|
|
{
|
|
|
|
assert(nbytes <= 4);
|
|
|
|
|
|
|
|
uint32_t res = 0;
|
|
|
|
if (nbytes) {
|
|
|
|
memcpy(((uint8_t *)&res) + (4 - nbytes), pkt_pos, nbytes);
|
|
|
|
}
|
|
|
|
return ntohl(res);
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t _encode_uint(uint32_t *val)
|
|
|
|
{
|
|
|
|
uint8_t *tgt = (uint8_t *)val;
|
|
|
|
size_t size = 0;
|
|
|
|
|
|
|
|
/* count number of used bytes */
|
|
|
|
uint32_t tmp = *val;
|
|
|
|
while(tmp) {
|
|
|
|
size++;
|
|
|
|
tmp >>= 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* convert to network byte order */
|
|
|
|
tmp = htonl(*val);
|
|
|
|
|
|
|
|
/* copy bytewise, starting with first actually used byte */
|
|
|
|
*val = 0;
|
|
|
|
uint8_t *tmp_u8 = (uint8_t *)&tmp;
|
|
|
|
tmp_u8 += (4 - size);
|
|
|
|
for (unsigned n = 0; n < size; n++) {
|
|
|
|
*tgt++ = *tmp_u8++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
2017-11-22 12:28:11 +01:00
|
|
|
static unsigned _put_delta_optlen(uint8_t *buf, unsigned offset, unsigned shift, unsigned val)
|
|
|
|
{
|
|
|
|
if (val < 13) {
|
|
|
|
*buf |= (val << shift);
|
|
|
|
}
|
|
|
|
else if (val < (256 + 13)) {
|
|
|
|
*buf |= (13 << shift);
|
|
|
|
buf[offset++] = (val - 13);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*buf |= (14 << shift);
|
|
|
|
uint16_t tmp = (val - 269);
|
|
|
|
tmp = htons(tmp);
|
|
|
|
memcpy(buf + offset, &tmp, 2);
|
|
|
|
offset += 2;
|
|
|
|
}
|
|
|
|
return offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t coap_put_option(uint8_t *buf, uint16_t lastonum, uint16_t onum, uint8_t *odata, size_t olen)
|
|
|
|
{
|
|
|
|
assert(lastonum <= onum);
|
|
|
|
|
|
|
|
unsigned delta = (onum - lastonum);
|
|
|
|
*buf = 0;
|
|
|
|
|
|
|
|
/* write delta value to option header: 4 upper bits of header (shift 4) +
|
|
|
|
* 1 or 2 optional bytes depending on delta value) */
|
|
|
|
unsigned n = _put_delta_optlen(buf, 1, 4, delta);
|
|
|
|
/* write option length to option header: 4 lower bits of header (shift 0) +
|
|
|
|
* 1 or 2 optional bytes depending of the length of the option */
|
|
|
|
n = _put_delta_optlen(buf, n, 0, olen);
|
2017-11-23 11:28:28 +01:00
|
|
|
if (olen) {
|
2017-11-22 12:28:11 +01:00
|
|
|
memcpy(buf + n, odata, olen);
|
|
|
|
n += olen;
|
|
|
|
}
|
|
|
|
return (size_t)n;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t coap_put_option_ct(uint8_t *buf, uint16_t lastonum, uint16_t content_type)
|
|
|
|
{
|
|
|
|
if (content_type == 0) {
|
|
|
|
return coap_put_option(buf, lastonum, COAP_OPT_CONTENT_FORMAT, NULL, 0);
|
|
|
|
}
|
|
|
|
else if (content_type <= 255) {
|
|
|
|
uint8_t tmp = content_type;
|
|
|
|
return coap_put_option(buf, lastonum, COAP_OPT_CONTENT_FORMAT, &tmp, sizeof(tmp));
|
|
|
|
}
|
|
|
|
else {
|
2017-11-23 11:28:28 +01:00
|
|
|
return coap_put_option(buf, lastonum, COAP_OPT_CONTENT_FORMAT, (uint8_t *)&content_type, sizeof(content_type));
|
2017-11-22 12:28:11 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-16 09:36:05 +01:00
|
|
|
static size_t coap_put_option_block(uint8_t *buf, uint16_t lastonum, unsigned blknum, unsigned szx, int more, uint16_t option)
|
|
|
|
{
|
|
|
|
uint32_t blkopt = (blknum << 4) | szx | (more ? 0x8 : 0);
|
|
|
|
size_t olen = _encode_uint(&blkopt);
|
|
|
|
return coap_put_option(buf, lastonum, option, (uint8_t*)&blkopt, olen);
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t coap_put_option_block1(uint8_t *buf, uint16_t lastonum, unsigned blknum, unsigned szx, int more)
|
|
|
|
{
|
|
|
|
return coap_put_option_block(buf, lastonum, blknum, szx, more, COAP_OPT_BLOCK1);
|
|
|
|
}
|
|
|
|
|
|
|
|
int coap_get_block1(coap_pkt_t *pkt, coap_block1_t *block1)
|
|
|
|
{
|
|
|
|
uint32_t blknum;
|
|
|
|
unsigned szx;
|
|
|
|
block1->more = coap_get_blockopt(pkt, COAP_OPT_BLOCK1, &blknum, &szx);
|
|
|
|
if (block1->more >= 0) {
|
|
|
|
block1->offset = blknum << (szx + 4);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
block1->offset = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
block1->blknum = blknum;
|
|
|
|
block1->szx = szx;
|
|
|
|
|
|
|
|
return (block1->more >= 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t coap_put_block1_ok(uint8_t *pkt_pos, coap_block1_t *block1, uint16_t lastonum)
|
|
|
|
{
|
|
|
|
if (block1->more >= 1) {
|
|
|
|
return coap_put_option_block1(pkt_pos, lastonum, block1->blknum, block1->szx, block1->more);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-23 11:50:46 +02:00
|
|
|
size_t coap_opt_put_string(uint8_t *buf, uint16_t lastonum, uint16_t optnum,
|
|
|
|
const char *string, char separator)
|
2017-11-22 12:28:11 +01:00
|
|
|
{
|
2018-08-23 11:50:46 +02:00
|
|
|
size_t len = strlen(string);
|
2017-11-23 11:28:28 +01:00
|
|
|
|
2018-08-23 11:50:46 +02:00
|
|
|
if (len == 0) {
|
2017-11-22 12:28:11 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t *bufpos = buf;
|
2018-08-23 11:50:46 +02:00
|
|
|
char *uripos = (char *)string;
|
2017-11-22 12:28:11 +01:00
|
|
|
|
2018-08-23 11:50:46 +02:00
|
|
|
while (len) {
|
2017-11-22 12:28:11 +01:00
|
|
|
size_t part_len;
|
|
|
|
uripos++;
|
2017-11-23 11:28:28 +01:00
|
|
|
uint8_t *part_start = (uint8_t *)uripos;
|
2017-11-22 12:28:11 +01:00
|
|
|
|
2018-08-23 11:50:46 +02:00
|
|
|
while (len--) {
|
2017-11-22 12:28:11 +01:00
|
|
|
if ((*uripos == separator) || (*uripos == '\0')) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
uripos++;
|
|
|
|
}
|
|
|
|
|
2017-11-23 11:28:28 +01:00
|
|
|
part_len = (uint8_t *)uripos - part_start;
|
2017-11-22 12:28:11 +01:00
|
|
|
|
2018-08-31 18:07:25 +02:00
|
|
|
/* Creates empty option if part for Uri-Path or Uri-Location contains only *
|
|
|
|
* a trailing slash, except for root path ("/"). */
|
|
|
|
if (part_len || ((separator == '/') && (lastonum == optnum))) {
|
2017-11-22 12:28:11 +01:00
|
|
|
bufpos += coap_put_option(bufpos, lastonum, optnum, part_start, part_len);
|
|
|
|
lastonum = optnum;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return bufpos - buf;
|
|
|
|
}
|
|
|
|
|
2018-05-01 19:15:54 +02:00
|
|
|
/* Common functionality for addition of an option */
|
|
|
|
static ssize_t _add_opt_pkt(coap_pkt_t *pkt, uint16_t optnum, uint8_t *val,
|
|
|
|
size_t val_len)
|
|
|
|
{
|
|
|
|
assert(pkt->options_len < NANOCOAP_NOPTS_MAX);
|
|
|
|
|
|
|
|
uint16_t lastonum = (pkt->options_len)
|
|
|
|
? pkt->options[pkt->options_len - 1].opt_num : 0;
|
|
|
|
assert(optnum >= lastonum);
|
|
|
|
|
|
|
|
size_t optlen = coap_put_option(pkt->payload, lastonum, optnum, val, val_len);
|
|
|
|
assert(pkt->payload_len > optlen);
|
|
|
|
|
|
|
|
pkt->options[pkt->options_len].opt_num = optnum;
|
|
|
|
pkt->options[pkt->options_len].offset = pkt->payload - (uint8_t *)pkt->hdr;
|
|
|
|
pkt->options_len++;
|
|
|
|
pkt->payload += optlen;
|
|
|
|
pkt->payload_len -= optlen;
|
|
|
|
|
|
|
|
return optlen;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t coap_opt_add_string(coap_pkt_t *pkt, uint16_t optnum, const char *string,
|
|
|
|
char separator)
|
|
|
|
{
|
|
|
|
size_t unread_len = strlen(string);
|
|
|
|
if (!unread_len) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
char *uripos = (char *)string;
|
|
|
|
size_t write_len = 0;
|
|
|
|
|
|
|
|
while (unread_len) {
|
|
|
|
size_t part_len;
|
|
|
|
uripos++;
|
|
|
|
uint8_t *part_start = (uint8_t *)uripos;
|
|
|
|
|
|
|
|
while (unread_len--) {
|
|
|
|
if ((*uripos == separator) || (*uripos == '\0')) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
uripos++;
|
|
|
|
}
|
|
|
|
|
|
|
|
part_len = (uint8_t *)uripos - part_start;
|
|
|
|
|
2018-08-31 18:07:25 +02:00
|
|
|
/* Creates empty option if part for Uri-Path or Uri-Location contains
|
|
|
|
* only a trailing slash, except for root path ("/"). */
|
|
|
|
if (part_len || ((separator == '/') && write_len)) {
|
2018-05-01 19:15:54 +02:00
|
|
|
if (pkt->options_len == NANOCOAP_NOPTS_MAX) {
|
|
|
|
return -ENOSPC;
|
|
|
|
}
|
|
|
|
write_len += _add_opt_pkt(pkt, optnum, part_start, part_len);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return write_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t coap_opt_add_uint(coap_pkt_t *pkt, uint16_t optnum, uint32_t value)
|
|
|
|
{
|
|
|
|
uint32_t tmp = value;
|
|
|
|
unsigned tmp_len = _encode_uint(&tmp);
|
|
|
|
return _add_opt_pkt(pkt, optnum, (uint8_t *)&tmp, tmp_len);
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t coap_opt_finish(coap_pkt_t *pkt, uint16_t flags)
|
|
|
|
{
|
|
|
|
if (flags & COAP_OPT_FINISH_PAYLOAD) {
|
|
|
|
assert(pkt->payload_len > 1);
|
|
|
|
|
|
|
|
*pkt->payload++ = 0xFF;
|
|
|
|
pkt->payload_len--;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
pkt->payload_len = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return pkt->payload - (uint8_t *)pkt->hdr;
|
|
|
|
}
|
|
|
|
|
2017-11-23 11:28:28 +01:00
|
|
|
ssize_t coap_well_known_core_default_handler(coap_pkt_t *pkt, uint8_t *buf, \
|
2018-02-02 18:54:29 +01:00
|
|
|
size_t len, void *context)
|
2017-11-22 12:28:11 +01:00
|
|
|
{
|
2018-02-02 18:54:29 +01:00
|
|
|
(void)context;
|
|
|
|
|
2017-11-22 12:28:11 +01:00
|
|
|
uint8_t *payload = buf + coap_get_total_hdr_len(pkt);
|
|
|
|
|
|
|
|
uint8_t *bufpos = payload;
|
|
|
|
|
|
|
|
bufpos += coap_put_option_ct(bufpos, 0, COAP_CT_LINK_FORMAT);
|
|
|
|
*bufpos++ = 0xff;
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < coap_resources_numof; i++) {
|
|
|
|
if (i) {
|
|
|
|
*bufpos++ = ',';
|
|
|
|
}
|
|
|
|
*bufpos++ = '<';
|
|
|
|
unsigned url_len = strlen(coap_resources[i].path);
|
|
|
|
memcpy(bufpos, coap_resources[i].path, url_len);
|
|
|
|
bufpos += url_len;
|
|
|
|
*bufpos++ = '>';
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned payload_len = bufpos - payload;
|
|
|
|
|
|
|
|
return coap_build_reply(pkt, COAP_CODE_205, buf, len, payload_len);
|
|
|
|
}
|
2018-03-16 09:35:51 +01:00
|
|
|
|
|
|
|
unsigned coap_get_len(coap_pkt_t *pkt)
|
|
|
|
{
|
|
|
|
unsigned pktlen = sizeof(coap_hdr_t) + coap_get_token_len(pkt);
|
|
|
|
if (pkt->payload) {
|
|
|
|
pktlen += pkt->payload_len + 1;
|
|
|
|
}
|
|
|
|
return pktlen;
|
|
|
|
}
|