2017-11-22 12:28:11 +01:00
|
|
|
/*
|
2019-05-02 11:22:56 +02:00
|
|
|
* Copyright (C) 2016-18 Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
* 2018 Inria
|
|
|
|
* 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 sock helpers
|
|
|
|
*
|
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
2022-03-23 17:18:10 +01:00
|
|
|
* @author Benjamin Valentin <benjamin.valentin@ml-pa.com>
|
2017-11-22 12:28:11 +01:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
2022-04-14 15:40:29 +02:00
|
|
|
#include "atomic_utils.h"
|
2018-12-20 05:16:30 +01:00
|
|
|
#include "net/nanocoap_sock.h"
|
2022-01-06 13:08:59 +01:00
|
|
|
#include "net/sock/util.h"
|
2017-11-22 12:28:11 +01:00
|
|
|
#include "net/sock/udp.h"
|
2022-04-14 15:18:16 +02:00
|
|
|
#include "random.h"
|
2022-03-23 17:18:10 +01:00
|
|
|
#include "sys/uio.h"
|
2020-10-21 21:26:18 +02:00
|
|
|
#include "timex.h"
|
2017-11-22 12:28:11 +01:00
|
|
|
|
2020-10-22 11:35:22 +02:00
|
|
|
#define ENABLE_DEBUG 0
|
2017-11-22 12:28:11 +01:00
|
|
|
#include "debug.h"
|
|
|
|
|
2022-03-23 17:18:10 +01:00
|
|
|
enum {
|
|
|
|
STATE_SEND_REQUEST,
|
|
|
|
STATE_AWAIT_RESPONSE,
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
coap_blockwise_cb_t callback;
|
|
|
|
void *arg;
|
|
|
|
bool more;
|
|
|
|
} _block_ctx_t;
|
|
|
|
|
2022-04-14 15:40:29 +02:00
|
|
|
static uint16_t _get_id(void)
|
|
|
|
{
|
|
|
|
__attribute__((section(".noinit")))
|
|
|
|
static uint16_t id;
|
|
|
|
return atomic_fetch_add_u16(&id, 1);
|
|
|
|
}
|
|
|
|
|
2022-02-10 18:47:05 +01:00
|
|
|
int nanocoap_sock_connect(nanocoap_sock_t *sock, sock_udp_ep_t *local, sock_udp_ep_t *remote)
|
2017-11-22 12:28:11 +01:00
|
|
|
{
|
|
|
|
if (!remote->port) {
|
|
|
|
remote->port = COAP_PORT;
|
|
|
|
}
|
|
|
|
|
2022-01-05 22:30:07 +01:00
|
|
|
return sock_udp_create(sock, local, remote, 0);
|
|
|
|
}
|
|
|
|
|
2022-03-23 17:18:10 +01:00
|
|
|
static int _get_error(const coap_pkt_t *pkt)
|
|
|
|
{
|
|
|
|
switch (coap_get_code_class(pkt)) {
|
|
|
|
case COAP_CLASS_CLIENT_FAILURE:
|
|
|
|
return -ENXIO;
|
|
|
|
case COAP_CLASS_SERVER_FAILURE:
|
|
|
|
return -ENETRESET;
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _send_ack(nanocoap_sock_t *sock, coap_pkt_t *pkt)
|
|
|
|
{
|
|
|
|
coap_hdr_t ack;
|
|
|
|
unsigned tkl = coap_get_token_len(pkt);
|
|
|
|
|
|
|
|
coap_build_hdr(&ack, COAP_TYPE_ACK, pkt->token, tkl,
|
|
|
|
COAP_CODE_VALID, ntohs(pkt->hdr->id));
|
|
|
|
|
|
|
|
return sock_udp_send(sock, &ack, sizeof(ack), NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t nanocoap_sock_request_cb(nanocoap_sock_t *sock, coap_pkt_t *pkt,
|
|
|
|
coap_request_cb_t cb, void *arg)
|
2022-01-05 22:30:07 +01:00
|
|
|
{
|
2022-03-23 17:18:10 +01:00
|
|
|
ssize_t tmp, res = 0;
|
2022-01-05 22:30:07 +01:00
|
|
|
size_t pdu_len = (pkt->payload - (uint8_t *)pkt->hdr) + pkt->payload_len;
|
|
|
|
uint8_t *buf = (uint8_t*)pkt->hdr;
|
2022-04-20 11:45:39 +02:00
|
|
|
unsigned id = coap_get_id(pkt);
|
2022-03-23 17:18:10 +01:00
|
|
|
void *payload, *ctx = NULL;
|
|
|
|
|
|
|
|
unsigned state = STATE_SEND_REQUEST;
|
2017-11-22 12:28:11 +01:00
|
|
|
|
2022-04-14 15:18:16 +02:00
|
|
|
uint32_t timeout = random_uint32_range(CONFIG_COAP_ACK_TIMEOUT_MS * US_PER_MS,
|
|
|
|
CONFIG_COAP_ACK_TIMEOUT_MS * CONFIG_COAP_RANDOM_FACTOR_1000);
|
2022-01-05 22:30:07 +01:00
|
|
|
/* add 1 for initial transmit */
|
|
|
|
unsigned tries_left = CONFIG_COAP_MAX_RETRANSMIT + 1;
|
|
|
|
|
|
|
|
/* check if we expect a reply */
|
2022-03-23 17:18:10 +01:00
|
|
|
const bool confirmable = coap_get_type(pkt) == COAP_TYPE_CON;
|
|
|
|
|
2022-04-20 11:45:39 +02:00
|
|
|
/* try receiving another packet without re-request */
|
|
|
|
bool retry_rx = false;
|
|
|
|
|
2022-03-23 17:18:10 +01:00
|
|
|
while (1) {
|
|
|
|
switch (state) {
|
|
|
|
case STATE_SEND_REQUEST:
|
2022-04-20 11:45:39 +02:00
|
|
|
DEBUG("nanocoap: send %u bytes (%u tries left)\n", pdu_len, tries_left);
|
2022-03-23 17:18:10 +01:00
|
|
|
if (--tries_left == 0) {
|
|
|
|
DEBUG("nanocoap: maximum retries reached\n");
|
|
|
|
return -ETIMEDOUT;
|
|
|
|
}
|
2022-01-05 22:30:07 +01:00
|
|
|
|
|
|
|
res = sock_udp_send(sock, buf, pdu_len, NULL);
|
|
|
|
if (res <= 0) {
|
|
|
|
DEBUG("nanocoap: error sending coap request, %d\n", (int)res);
|
2022-03-23 17:18:10 +01:00
|
|
|
return res;
|
2022-01-05 22:30:07 +01:00
|
|
|
}
|
2017-11-22 12:28:11 +01:00
|
|
|
|
2022-04-20 11:45:39 +02:00
|
|
|
/* no response needed and no response handler given */
|
|
|
|
if (!confirmable && !cb) {
|
2022-03-23 17:18:10 +01:00
|
|
|
return 0;
|
|
|
|
}
|
2022-04-20 11:45:39 +02:00
|
|
|
|
|
|
|
/* ctx must have been released at this point */
|
|
|
|
assert(ctx == NULL);
|
|
|
|
state = STATE_AWAIT_RESPONSE;
|
2022-03-23 17:18:10 +01:00
|
|
|
/* fall-through */
|
|
|
|
case STATE_AWAIT_RESPONSE:
|
2022-04-20 11:45:39 +02:00
|
|
|
DEBUG("nanocoap: waiting for response (timeout: %lu µs)\n", timeout);
|
2022-03-23 17:18:10 +01:00
|
|
|
tmp = sock_udp_recv_buf(sock, &payload, &ctx, timeout, NULL);
|
|
|
|
if (tmp == 0) {
|
|
|
|
/* no more data */
|
2022-04-20 11:45:39 +02:00
|
|
|
/* sock_udp_recv_buf() needs to be called in a loop until ctx is NULL again
|
|
|
|
* to release the buffer */
|
|
|
|
if (retry_rx) {
|
|
|
|
retry_rx = false;
|
|
|
|
continue;
|
|
|
|
}
|
2022-03-23 17:18:10 +01:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
res = tmp;
|
2017-11-22 12:28:11 +01:00
|
|
|
if (res == -ETIMEDOUT) {
|
|
|
|
DEBUG("nanocoap: timeout\n");
|
2022-03-23 17:18:10 +01:00
|
|
|
timeout *= 2;
|
|
|
|
state = STATE_SEND_REQUEST;
|
|
|
|
continue;
|
2017-11-22 12:28:11 +01:00
|
|
|
}
|
2022-03-23 17:18:10 +01:00
|
|
|
if (res < 0) {
|
|
|
|
DEBUG("nanocoap: error receiving coap response, %d\n", (int)res);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* parse response */
|
|
|
|
if (coap_parse(pkt, payload, res) < 0) {
|
2018-05-06 14:40:37 +02:00
|
|
|
DEBUG("nanocoap: error parsing packet\n");
|
2022-04-20 11:45:39 +02:00
|
|
|
retry_rx = true;
|
|
|
|
continue;
|
2017-11-22 12:28:11 +01:00
|
|
|
}
|
2022-01-05 22:30:07 +01:00
|
|
|
else if (coap_get_id(pkt) != id) {
|
2022-04-20 11:45:39 +02:00
|
|
|
DEBUG("nanocoap: ID mismatch %u != %u\n", coap_get_id(pkt), id);
|
|
|
|
retry_rx = true;
|
2022-01-05 22:30:07 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2022-03-23 17:18:10 +01:00
|
|
|
switch (coap_get_type(pkt)) {
|
|
|
|
case COAP_TYPE_RST:
|
|
|
|
/* TODO: handle different? */
|
|
|
|
return -EBADMSG;
|
|
|
|
case COAP_TYPE_CON:
|
|
|
|
_send_ack(sock, pkt);
|
|
|
|
/* fall-through */
|
|
|
|
case COAP_TYPE_NON:
|
|
|
|
case COAP_TYPE_ACK:
|
|
|
|
/* call user callback */
|
|
|
|
if (cb) {
|
|
|
|
res = cb(arg, pkt);
|
|
|
|
} else {
|
|
|
|
res = 0;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2017-11-22 12:28:11 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2022-03-23 17:18:10 +01:00
|
|
|
static int _request_cb(void *arg, coap_pkt_t *pkt)
|
|
|
|
{
|
|
|
|
struct iovec *buf = arg;
|
|
|
|
size_t pkt_len = coap_get_total_len(pkt);
|
|
|
|
|
|
|
|
int res = _get_error(pkt);
|
|
|
|
if (res) {
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pkt_len > buf->iov_len) {
|
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(buf->iov_base, pkt->hdr, pkt_len);
|
|
|
|
|
|
|
|
pkt->hdr = buf->iov_base;
|
|
|
|
pkt->token = (uint8_t*)pkt->hdr + sizeof(coap_hdr_t);
|
|
|
|
pkt->payload = (uint8_t*)pkt->hdr + (pkt_len - pkt->payload_len);
|
|
|
|
|
|
|
|
return pkt_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t nanocoap_sock_request(sock_udp_t *sock, coap_pkt_t *pkt, size_t len)
|
|
|
|
{
|
|
|
|
struct iovec buf = {
|
|
|
|
.iov_base = pkt->hdr,
|
|
|
|
.iov_len = len,
|
|
|
|
};
|
|
|
|
return nanocoap_sock_request_cb(sock, pkt, _request_cb, &buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _get_cb(void *arg, coap_pkt_t *pkt)
|
|
|
|
{
|
|
|
|
struct iovec *buf = arg;
|
|
|
|
|
|
|
|
int res = _get_error(pkt);
|
|
|
|
if (res) {
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pkt->payload_len > buf->iov_len) {
|
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(buf->iov_base, pkt->payload, pkt->payload_len);
|
|
|
|
|
|
|
|
return pkt->payload_len;
|
|
|
|
}
|
|
|
|
|
2022-02-10 18:47:05 +01:00
|
|
|
ssize_t nanocoap_sock_get(nanocoap_sock_t *sock, const char *path, void *buf, size_t len)
|
2018-05-06 14:40:37 +02:00
|
|
|
{
|
|
|
|
ssize_t res;
|
|
|
|
coap_pkt_t pkt;
|
|
|
|
uint8_t *pktpos = buf;
|
|
|
|
|
2022-03-23 17:18:10 +01:00
|
|
|
struct iovec ctx = {
|
|
|
|
.iov_base = buf,
|
|
|
|
.iov_len = len,
|
|
|
|
};
|
|
|
|
|
2022-01-05 22:30:07 +01:00
|
|
|
pkt.hdr = buf;
|
2022-04-14 15:40:29 +02:00
|
|
|
pktpos += coap_build_hdr(pkt.hdr, COAP_TYPE_CON, NULL, 0, COAP_METHOD_GET, _get_id());
|
2018-08-23 11:50:46 +02:00
|
|
|
pktpos += coap_opt_put_uri_path(pktpos, 0, path);
|
2018-05-06 14:40:37 +02:00
|
|
|
pkt.payload = pktpos;
|
|
|
|
pkt.payload_len = 0;
|
|
|
|
|
2022-03-23 17:18:10 +01:00
|
|
|
res = nanocoap_sock_request_cb(sock, &pkt, _get_cb, &ctx);
|
2018-05-06 14:40:37 +02:00
|
|
|
if (res < 0) {
|
|
|
|
return res;
|
|
|
|
}
|
2022-03-23 17:18:10 +01:00
|
|
|
|
|
|
|
if (coap_get_code(&pkt) != 205) {
|
|
|
|
return -ENOENT;
|
2018-05-06 14:40:37 +02:00
|
|
|
}
|
2022-03-23 17:18:10 +01:00
|
|
|
|
2018-05-06 14:40:37 +02:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2022-02-10 18:47:05 +01:00
|
|
|
ssize_t nanocoap_request(coap_pkt_t *pkt, sock_udp_ep_t *local,
|
|
|
|
sock_udp_ep_t *remote, size_t len)
|
|
|
|
{
|
|
|
|
int res;
|
|
|
|
nanocoap_sock_t sock;
|
|
|
|
|
|
|
|
res = nanocoap_sock_connect(&sock, local, remote);
|
|
|
|
if (res) {
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
res = nanocoap_sock_request(&sock, pkt, len);
|
|
|
|
nanocoap_sock_close(&sock);
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t nanocoap_get(sock_udp_ep_t *remote, const char *path, void *buf, size_t len)
|
|
|
|
{
|
|
|
|
int res;
|
|
|
|
nanocoap_sock_t sock;
|
|
|
|
|
|
|
|
res = nanocoap_sock_connect(&sock, NULL, remote);
|
|
|
|
if (res) {
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
res = nanocoap_sock_get(&sock, path, buf, len);
|
|
|
|
nanocoap_sock_close(&sock);
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2022-03-23 17:18:10 +01:00
|
|
|
static int _block_cb(void *arg, coap_pkt_t *pkt)
|
2022-01-06 13:08:59 +01:00
|
|
|
{
|
2022-03-23 17:18:10 +01:00
|
|
|
_block_ctx_t *ctx = arg;
|
|
|
|
coap_block1_t block2;
|
|
|
|
|
|
|
|
int res = _get_error(pkt);
|
|
|
|
if (res) {
|
|
|
|
return res;
|
|
|
|
}
|
2022-01-06 13:08:59 +01:00
|
|
|
|
2022-03-23 17:18:10 +01:00
|
|
|
/* response was not block-wise */
|
|
|
|
if (!coap_get_block2(pkt, &block2)) {
|
|
|
|
block2.offset = 0;
|
2022-04-15 15:24:42 +02:00
|
|
|
block2.more = false;
|
2022-03-23 17:18:10 +01:00
|
|
|
}
|
|
|
|
|
2022-04-15 15:24:42 +02:00
|
|
|
ctx->more = block2.more;
|
2022-03-23 17:18:10 +01:00
|
|
|
return ctx->callback(ctx->arg, block2.offset, pkt->payload, pkt->payload_len, block2.more);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _fetch_block(coap_pkt_t *pkt, sock_udp_t *sock,
|
|
|
|
const char *path, coap_blksize_t blksize, unsigned num,
|
|
|
|
_block_ctx_t *ctx)
|
|
|
|
{
|
|
|
|
uint8_t *pktpos = (void *)pkt->hdr;
|
|
|
|
uint16_t lastonum = 0;
|
2022-01-06 13:08:59 +01:00
|
|
|
|
2022-04-14 15:40:29 +02:00
|
|
|
pktpos += coap_build_hdr(pkt->hdr, COAP_TYPE_CON, NULL, 0, COAP_METHOD_GET, _get_id());
|
2022-01-06 13:08:59 +01:00
|
|
|
pktpos += coap_opt_put_uri_pathquery(pktpos, &lastonum, path);
|
|
|
|
pktpos += coap_opt_put_uint(pktpos, lastonum, COAP_OPT_BLOCK2,
|
|
|
|
(num << 4) | blksize);
|
|
|
|
|
|
|
|
pkt->payload = pktpos;
|
|
|
|
pkt->payload_len = 0;
|
|
|
|
|
2022-03-23 17:18:10 +01:00
|
|
|
int res = nanocoap_sock_request_cb(sock, pkt, _block_cb, ctx);
|
2022-01-06 13:08:59 +01:00
|
|
|
if (res < 0) {
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2022-03-23 17:18:10 +01:00
|
|
|
DEBUG("code=%i\n", coap_get_code(pkt));
|
2022-01-06 13:08:59 +01:00
|
|
|
|
2022-03-23 17:18:10 +01:00
|
|
|
return _get_error(pkt);
|
2022-01-06 13:08:59 +01:00
|
|
|
}
|
|
|
|
|
2022-02-10 18:47:05 +01:00
|
|
|
int nanocoap_sock_get_blockwise(nanocoap_sock_t *sock, const char *path,
|
2022-03-23 17:18:10 +01:00
|
|
|
coap_blksize_t blksize,
|
|
|
|
coap_blockwise_cb_t callback, void *arg)
|
2022-01-06 13:08:59 +01:00
|
|
|
{
|
2022-03-23 17:18:10 +01:00
|
|
|
uint8_t buf[CONFIG_NANOCOAP_BLOCK_HEADER_MAX];
|
2022-01-06 13:08:59 +01:00
|
|
|
|
2022-03-23 17:18:10 +01:00
|
|
|
_block_ctx_t ctx = {
|
|
|
|
.callback = callback,
|
|
|
|
.arg = arg,
|
|
|
|
.more = true,
|
|
|
|
};
|
2022-01-06 13:08:59 +01:00
|
|
|
|
2022-03-23 17:18:10 +01:00
|
|
|
unsigned num = 0;
|
|
|
|
while (ctx.more) {
|
|
|
|
coap_pkt_t pkt = {
|
|
|
|
.hdr = (void *)buf,
|
|
|
|
};
|
2022-01-06 13:08:59 +01:00
|
|
|
|
2022-03-23 17:18:10 +01:00
|
|
|
DEBUG("fetching block %u\n", num);
|
|
|
|
int res = _fetch_block(&pkt, sock, path, blksize, num, &ctx);
|
2022-01-25 17:35:42 +01:00
|
|
|
|
2022-01-06 13:08:59 +01:00
|
|
|
if (res) {
|
2022-03-23 17:18:10 +01:00
|
|
|
DEBUG("error fetching block %u: %d\n", num, res);
|
2022-04-15 15:23:40 +02:00
|
|
|
return res;
|
2022-01-06 13:08:59 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
num += 1;
|
2022-03-23 17:18:10 +01:00
|
|
|
}
|
2022-01-06 13:08:59 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2022-04-18 01:32:28 +02:00
|
|
|
int nanocoap_sock_url_connect(const char *url, nanocoap_sock_t *sock)
|
2022-01-06 13:08:59 +01:00
|
|
|
{
|
|
|
|
char hostport[CONFIG_SOCK_HOSTPORT_MAXLEN];
|
|
|
|
sock_udp_ep_t remote;
|
|
|
|
|
|
|
|
if (strncmp(url, "coap://", 7)) {
|
|
|
|
DEBUG("nanocoap: URL doesn't start with \"coap://\"\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2022-04-18 01:32:28 +02:00
|
|
|
if (sock_urlsplit(url, hostport, NULL) < 0) {
|
2022-01-06 13:08:59 +01:00
|
|
|
DEBUG("nanocoap: invalid URL\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2022-03-24 17:09:16 +01:00
|
|
|
if (sock_udp_name2ep(&remote, hostport) < 0) {
|
2022-01-06 13:08:59 +01:00
|
|
|
DEBUG("nanocoap: invalid URL\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2022-04-18 01:32:28 +02:00
|
|
|
return nanocoap_sock_connect(sock, NULL, &remote);
|
|
|
|
}
|
|
|
|
|
|
|
|
int nanocoap_get_blockwise_url(const char *url,
|
|
|
|
coap_blksize_t blksize,
|
|
|
|
coap_blockwise_cb_t callback, void *arg)
|
|
|
|
{
|
|
|
|
nanocoap_sock_t sock;
|
|
|
|
int res = nanocoap_sock_url_connect(url, &sock);
|
2022-01-06 13:08:59 +01:00
|
|
|
if (res) {
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2022-04-18 01:32:28 +02:00
|
|
|
res = nanocoap_sock_get_blockwise(&sock, sock_urlpath(url), blksize, callback, arg);
|
2022-02-10 18:47:05 +01:00
|
|
|
nanocoap_sock_close(&sock);
|
2022-01-06 13:08:59 +01:00
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2022-03-21 18:26:11 +01:00
|
|
|
typedef struct {
|
|
|
|
uint8_t *ptr;
|
|
|
|
size_t len;
|
|
|
|
} _buf_t;
|
|
|
|
|
|
|
|
static int _2buf(void *arg, size_t offset, uint8_t *buf, size_t len, int more)
|
|
|
|
{
|
|
|
|
_buf_t *dst = arg;
|
|
|
|
|
|
|
|
if (offset + len > dst->len) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(dst->ptr + offset, buf, len);
|
|
|
|
|
|
|
|
if (!more) {
|
|
|
|
dst->len = offset + len;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t nanocoap_get_blockwise_url_to_buf(const char *url,
|
2022-03-23 17:18:10 +01:00
|
|
|
coap_blksize_t blksize,
|
2022-03-21 18:26:11 +01:00
|
|
|
void *buf, size_t len)
|
|
|
|
{
|
|
|
|
_buf_t _buf = { .ptr = buf, .len = len };
|
2022-03-23 17:18:10 +01:00
|
|
|
int res = nanocoap_get_blockwise_url(url, blksize, _2buf, &_buf);
|
2022-03-21 18:26:11 +01:00
|
|
|
|
|
|
|
return (res < 0) ? (ssize_t)res : (ssize_t)_buf.len;
|
|
|
|
}
|
|
|
|
|
2017-11-22 12:28:11 +01:00
|
|
|
int nanocoap_server(sock_udp_ep_t *local, uint8_t *buf, size_t bufsize)
|
|
|
|
{
|
2022-02-10 18:47:05 +01:00
|
|
|
nanocoap_sock_t sock;
|
2017-11-22 12:28:11 +01:00
|
|
|
sock_udp_ep_t remote;
|
|
|
|
|
|
|
|
if (!local->port) {
|
|
|
|
local->port = COAP_PORT;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t res = sock_udp_create(&sock, local, NULL, 0);
|
2019-01-07 17:59:40 +01:00
|
|
|
if (res != 0) {
|
2017-11-22 12:28:11 +01:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
res = sock_udp_recv(&sock, buf, bufsize, -1, &remote);
|
2019-01-07 17:59:40 +01:00
|
|
|
if (res < 0) {
|
2019-02-26 18:35:49 +01:00
|
|
|
DEBUG("error receiving UDP packet %d\n", (int)res);
|
2017-11-22 12:28:11 +01:00
|
|
|
}
|
2019-01-07 17:59:40 +01:00
|
|
|
else if (res > 0) {
|
2017-11-22 12:28:11 +01:00
|
|
|
coap_pkt_t pkt;
|
|
|
|
if (coap_parse(&pkt, (uint8_t *)buf, res) < 0) {
|
|
|
|
DEBUG("error parsing packet\n");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if ((res = coap_handle_req(&pkt, buf, bufsize)) > 0) {
|
2019-10-27 10:49:41 +01:00
|
|
|
sock_udp_send(&sock, buf, res, &remote);
|
2017-11-22 12:28:11 +01:00
|
|
|
}
|
2019-01-22 14:42:16 +01:00
|
|
|
else {
|
|
|
|
DEBUG("error handling request %d\n", (int)res);
|
|
|
|
}
|
2017-11-22 12:28:11 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|