2017-07-07 11:04:03 -07:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2017 Linaro Limited
|
2019-01-24 12:22:42 -08:00
|
|
|
* Copyright (c) 2018-2019 Foundries.io
|
2017-07-07 11:04:03 -07:00
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Uses some original concepts by:
|
|
|
|
* Joakim Eriksson <joakime@sics.se>
|
|
|
|
* Niclas Finne <nfi@sics.se>
|
|
|
|
* Joel Hoglund <joel@sics.se>
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* TODO:
|
|
|
|
* - Handle Resource ObjLink type
|
|
|
|
*/
|
|
|
|
|
2018-09-19 11:22:19 +03:00
|
|
|
#define LOG_MODULE_NAME net_lwm2m_engine
|
|
|
|
#define LOG_LEVEL CONFIG_LWM2M_LOG_LEVEL
|
|
|
|
|
|
|
|
#include <logging/log.h>
|
|
|
|
LOG_MODULE_REGISTER(LOG_MODULE_NAME);
|
2017-07-07 11:04:03 -07:00
|
|
|
|
|
|
|
#include <zephyr/types.h>
|
|
|
|
#include <stddef.h>
|
2017-11-24 18:47:00 +08:00
|
|
|
#include <stdlib.h>
|
2017-07-07 11:04:03 -07:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <init.h>
|
|
|
|
#include <misc/printk.h>
|
|
|
|
#include <net/net_ip.h>
|
2019-01-28 11:55:23 -08:00
|
|
|
#include <net/http_parser_url.h>
|
|
|
|
#include <net/socket.h>
|
|
|
|
#if defined(CONFIG_LWM2M_DTLS_SUPPORT)
|
|
|
|
#include <net/tls_credentials.h>
|
|
|
|
#endif
|
2019-01-31 09:21:51 -08:00
|
|
|
#if defined(CONFIG_DNS_RESOLVER)
|
|
|
|
#include <net/dns_resolve.h>
|
|
|
|
#endif
|
2017-07-07 11:04:03 -07:00
|
|
|
|
|
|
|
#include "lwm2m_object.h"
|
|
|
|
#include "lwm2m_engine.h"
|
|
|
|
#include "lwm2m_rw_plain_text.h"
|
|
|
|
#include "lwm2m_rw_oma_tlv.h"
|
|
|
|
#ifdef CONFIG_LWM2M_RW_JSON_SUPPORT
|
|
|
|
#include "lwm2m_rw_json.h"
|
|
|
|
#endif
|
|
|
|
#ifdef CONFIG_LWM2M_RD_CLIENT_SUPPORT
|
|
|
|
#include "lwm2m_rd_client.h"
|
|
|
|
#endif
|
|
|
|
|
2017-10-10 20:55:05 -07:00
|
|
|
#define ENGINE_UPDATE_INTERVAL K_MSEC(500)
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2017-11-15 09:46:13 +08:00
|
|
|
#define WELL_KNOWN_CORE_PATH "</.well-known/core>"
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2017-08-14 17:06:03 +08:00
|
|
|
/*
|
|
|
|
* TODO: to implement a way for clients to specify alternate path
|
|
|
|
* via Kconfig (LwM2M specification 8.2.2 Alternate Path)
|
|
|
|
*
|
|
|
|
* For now, in order to inform server we support JSON format, we have to
|
|
|
|
* report 'ct=11543' to the server. '</>' is required in order to append
|
|
|
|
* content attribute. And resource type attribute is appended because of
|
|
|
|
* Eclipse wakaama will reject the registration when 'rt="oma.lwm2m"' is
|
|
|
|
* missing.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define RESOURCE_TYPE ";rt=\"oma.lwm2m\""
|
|
|
|
|
|
|
|
#if defined(CONFIG_LWM2M_RW_JSON_SUPPORT)
|
|
|
|
#define REG_PREFACE "</>" RESOURCE_TYPE \
|
|
|
|
";ct=" STRINGIFY(LWM2M_FORMAT_OMA_JSON)
|
|
|
|
#else
|
|
|
|
#define REG_PREFACE ""
|
|
|
|
#endif
|
|
|
|
|
2017-11-24 18:47:00 +08:00
|
|
|
#if defined(CONFIG_COAP_EXTENDED_OPTIONS_LEN)
|
|
|
|
#define COAP_OPTION_BUF_LEN (CONFIG_COAP_EXTENDED_OPTIONS_LEN_VALUE + 1)
|
|
|
|
#else
|
|
|
|
#define COAP_OPTION_BUF_LEN 13
|
|
|
|
#endif
|
|
|
|
|
2017-11-28 16:29:21 +08:00
|
|
|
/* TODO: grab this from server obj */
|
|
|
|
#define DEFAULT_SERVER_PMIN 10
|
|
|
|
#define DEFAULT_SERVER_PMAX 60
|
|
|
|
|
2017-10-03 09:34:59 -07:00
|
|
|
#define MAX_TOKEN_LEN 8
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
struct observe_node {
|
|
|
|
sys_snode_t node;
|
2017-09-01 00:42:05 -07:00
|
|
|
struct lwm2m_ctx *ctx;
|
2017-07-07 11:04:03 -07:00
|
|
|
struct lwm2m_obj_path path;
|
2017-10-03 09:34:59 -07:00
|
|
|
u8_t token[MAX_TOKEN_LEN];
|
2017-07-07 11:04:03 -07:00
|
|
|
s64_t event_timestamp;
|
|
|
|
s64_t last_timestamp;
|
|
|
|
u32_t min_period_sec;
|
|
|
|
u32_t max_period_sec;
|
|
|
|
u32_t counter;
|
2017-07-26 14:02:29 +08:00
|
|
|
u16_t format;
|
2017-07-07 11:04:03 -07:00
|
|
|
u8_t tkl;
|
|
|
|
};
|
|
|
|
|
2018-05-07 14:06:46 -07:00
|
|
|
struct notification_attrs {
|
|
|
|
/* use to determine which value is set */
|
|
|
|
float32_value_t gt;
|
|
|
|
float32_value_t lt;
|
|
|
|
float32_value_t st;
|
|
|
|
s32_t pmin;
|
|
|
|
s32_t pmax;
|
|
|
|
u8_t flags;
|
|
|
|
};
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
static struct observe_node observe_node_data[CONFIG_LWM2M_ENGINE_MAX_OBSERVER];
|
|
|
|
|
2017-10-10 20:55:05 -07:00
|
|
|
#define MAX_PERIODIC_SERVICE 10
|
|
|
|
|
|
|
|
struct service_node {
|
|
|
|
sys_snode_t node;
|
2019-04-05 16:14:50 -07:00
|
|
|
k_work_handler_t service_work;
|
2017-10-10 20:55:05 -07:00
|
|
|
u32_t min_call_period;
|
|
|
|
u64_t last_timestamp;
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct service_node service_node_data[MAX_PERIODIC_SERVICE];
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
static sys_slist_t engine_obj_list;
|
|
|
|
static sys_slist_t engine_obj_inst_list;
|
|
|
|
static sys_slist_t engine_observer_list;
|
2017-10-10 20:55:05 -07:00
|
|
|
static sys_slist_t engine_service_list;
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
static K_THREAD_STACK_DEFINE(engine_thread_stack,
|
|
|
|
CONFIG_LWM2M_ENGINE_STACK_SIZE);
|
|
|
|
static struct k_thread engine_thread_data;
|
|
|
|
|
|
|
|
#define MAX_POLL_FD CONFIG_NET_SOCKETS_POLL_MAX
|
|
|
|
|
|
|
|
static struct lwm2m_ctx *sock_ctx[MAX_POLL_FD];
|
|
|
|
static struct pollfd sock_fds[MAX_POLL_FD];
|
|
|
|
static int sock_nfds;
|
|
|
|
|
2017-07-25 16:54:25 +08:00
|
|
|
#define NUM_BLOCK1_CONTEXT CONFIG_LWM2M_NUM_BLOCK1_CONTEXT
|
|
|
|
|
|
|
|
/* TODO: figure out what's correct value */
|
|
|
|
#define TIMEOUT_BLOCKWISE_TRANSFER K_SECONDS(30)
|
|
|
|
|
|
|
|
#define GET_BLOCK_NUM(v) ((v) >> 4)
|
|
|
|
#define GET_BLOCK_SIZE(v) (((v) & 0x7))
|
|
|
|
#define GET_MORE(v) (!!((v) & 0x08))
|
|
|
|
|
|
|
|
struct block_context {
|
2017-09-19 16:04:32 -07:00
|
|
|
struct coap_block_context ctx;
|
2017-07-25 16:54:25 +08:00
|
|
|
s64_t timestamp;
|
|
|
|
u8_t token[8];
|
|
|
|
u8_t tkl;
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct block_context block1_contexts[NUM_BLOCK1_CONTEXT];
|
|
|
|
|
2017-11-24 18:47:00 +08:00
|
|
|
/* write-attribute related definitons */
|
|
|
|
static const char * const LWM2M_ATTR_STR[] = { "pmin", "pmax",
|
|
|
|
"gt", "lt", "st" };
|
|
|
|
static const u8_t LWM2M_ATTR_LEN[] = { 4, 4, 2, 2, 2 };
|
|
|
|
|
|
|
|
static struct lwm2m_attr write_attr_pool[CONFIG_LWM2M_NUM_ATTR];
|
|
|
|
|
2017-09-05 14:29:41 +08:00
|
|
|
static struct lwm2m_engine_obj *get_engine_obj(int obj_id);
|
|
|
|
static struct lwm2m_engine_obj_inst *get_engine_obj_inst(int obj_id,
|
|
|
|
int obj_inst_id);
|
|
|
|
|
2017-09-01 01:11:43 -07:00
|
|
|
/* Shared set of in-flight LwM2M messages */
|
|
|
|
static struct lwm2m_message messages[CONFIG_LWM2M_ENGINE_MAX_MESSAGES];
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
/* for debugging: to print IP addresses */
|
|
|
|
char *lwm2m_sprint_ip_addr(const struct sockaddr *addr)
|
|
|
|
{
|
|
|
|
static char buf[NET_IPV6_ADDR_LEN];
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV6)
|
2017-08-18 10:03:46 +03:00
|
|
|
if (addr->sa_family == AF_INET6) {
|
2017-07-07 11:04:03 -07:00
|
|
|
return net_addr_ntop(AF_INET6, &net_sin6(addr)->sin6_addr,
|
|
|
|
buf, sizeof(buf));
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#if defined(CONFIG_NET_IPV4)
|
2017-08-18 10:03:46 +03:00
|
|
|
if (addr->sa_family == AF_INET) {
|
2017-07-07 11:04:03 -07:00
|
|
|
return net_addr_ntop(AF_INET, &net_sin(addr)->sin_addr,
|
|
|
|
buf, sizeof(buf));
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("Unknown IP address family:%d", addr->sa_family);
|
2017-07-07 11:04:03 -07:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2018-05-09 14:01:29 -07:00
|
|
|
static u8_t to_hex_digit(u8_t digit)
|
|
|
|
{
|
2019-03-26 19:57:45 -06:00
|
|
|
if (digit >= 10U) {
|
|
|
|
return digit - 10U + 'a';
|
2018-05-09 14:01:29 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return digit + '0';
|
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
static char *sprint_token(const u8_t *token, u8_t tkl)
|
|
|
|
{
|
|
|
|
static char buf[32];
|
2018-05-09 14:01:29 -07:00
|
|
|
char *ptr = buf;
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2017-10-03 09:34:59 -07:00
|
|
|
if (token && tkl != LWM2M_MSG_TOKEN_LEN_SKIP) {
|
|
|
|
int i;
|
|
|
|
|
2019-02-11 17:14:19 +00:00
|
|
|
tkl = MIN(tkl, sizeof(buf) / 2 - 1);
|
2018-05-09 14:01:29 -07:00
|
|
|
|
2017-10-03 09:34:59 -07:00
|
|
|
for (i = 0; i < tkl; i++) {
|
2018-05-09 14:01:29 -07:00
|
|
|
*ptr++ = to_hex_digit(token[i] >> 4);
|
|
|
|
*ptr++ = to_hex_digit(token[i] & 0x0F);
|
2017-10-03 09:34:59 -07:00
|
|
|
}
|
|
|
|
|
2018-05-09 14:01:29 -07:00
|
|
|
*ptr = '\0';
|
2017-10-03 09:34:59 -07:00
|
|
|
} else if (tkl == LWM2M_MSG_TOKEN_LEN_SKIP) {
|
|
|
|
strcpy(buf, "[skip-token]");
|
|
|
|
} else {
|
|
|
|
strcpy(buf, "[no-token]");
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
2017-10-03 09:34:59 -07:00
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
2017-07-25 16:54:25 +08:00
|
|
|
/* block-wise transfer functions */
|
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
enum coap_block_size lwm2m_default_block_size(void)
|
2017-07-25 16:54:25 +08:00
|
|
|
{
|
|
|
|
switch (CONFIG_LWM2M_COAP_BLOCK_SIZE) {
|
|
|
|
case 16:
|
2017-09-19 16:04:32 -07:00
|
|
|
return COAP_BLOCK_16;
|
2017-07-25 16:54:25 +08:00
|
|
|
case 32:
|
2017-09-19 16:04:32 -07:00
|
|
|
return COAP_BLOCK_32;
|
2017-07-25 16:54:25 +08:00
|
|
|
case 64:
|
2017-09-19 16:04:32 -07:00
|
|
|
return COAP_BLOCK_64;
|
2017-07-25 16:54:25 +08:00
|
|
|
case 128:
|
2017-09-19 16:04:32 -07:00
|
|
|
return COAP_BLOCK_128;
|
2017-07-25 16:54:25 +08:00
|
|
|
case 256:
|
2017-09-19 16:04:32 -07:00
|
|
|
return COAP_BLOCK_256;
|
2017-07-25 16:54:25 +08:00
|
|
|
case 512:
|
2017-09-19 16:04:32 -07:00
|
|
|
return COAP_BLOCK_512;
|
2017-07-25 16:54:25 +08:00
|
|
|
case 1024:
|
2017-09-19 16:04:32 -07:00
|
|
|
return COAP_BLOCK_1024;
|
2017-07-25 16:54:25 +08:00
|
|
|
}
|
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
return COAP_BLOCK_256;
|
2017-07-25 16:54:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
init_block_ctx(const u8_t *token, u8_t tkl, struct block_context **ctx)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
s64_t timestamp;
|
|
|
|
|
|
|
|
*ctx = NULL;
|
|
|
|
timestamp = k_uptime_get();
|
|
|
|
for (i = 0; i < NUM_BLOCK1_CONTEXT; i++) {
|
2019-03-26 19:57:45 -06:00
|
|
|
if (block1_contexts[i].tkl == 0U) {
|
2017-07-25 16:54:25 +08:00
|
|
|
*ctx = &block1_contexts[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (timestamp - block1_contexts[i].timestamp >
|
|
|
|
TIMEOUT_BLOCKWISE_TRANSFER) {
|
|
|
|
*ctx = &block1_contexts[i];
|
|
|
|
/* TODO: notify application for block
|
|
|
|
* transfer timeout
|
|
|
|
*/
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*ctx == NULL) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("Cannot find free block context");
|
2017-07-25 16:54:25 +08:00
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
(*ctx)->tkl = tkl;
|
|
|
|
memcpy((*ctx)->token, token, tkl);
|
2017-09-19 16:04:32 -07:00
|
|
|
coap_block_transfer_init(&(*ctx)->ctx, lwm2m_default_block_size(), 0);
|
2017-07-25 16:54:25 +08:00
|
|
|
(*ctx)->timestamp = timestamp;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
get_block_ctx(const u8_t *token, u8_t tkl, struct block_context **ctx)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
*ctx = NULL;
|
|
|
|
|
|
|
|
for (i = 0; i < NUM_BLOCK1_CONTEXT; i++) {
|
|
|
|
if (block1_contexts[i].tkl == tkl &&
|
|
|
|
memcmp(token, block1_contexts[i].token, tkl) == 0) {
|
|
|
|
*ctx = &block1_contexts[i];
|
2019-06-18 14:45:40 -04:00
|
|
|
/* refresh timestamp */
|
2017-07-25 16:54:25 +08:00
|
|
|
(*ctx)->timestamp = k_uptime_get();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*ctx == NULL) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("Cannot find block context");
|
2017-07-25 16:54:25 +08:00
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void free_block_ctx(struct block_context *ctx)
|
|
|
|
{
|
|
|
|
if (ctx == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-11-29 11:23:03 -08:00
|
|
|
ctx->tkl = 0U;
|
2017-07-25 16:54:25 +08:00
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
/* observer functions */
|
|
|
|
|
2018-05-07 15:05:04 -07:00
|
|
|
static int update_attrs(void *ref, struct notification_attrs *out)
|
2017-11-24 18:47:00 +08:00
|
|
|
{
|
2018-05-07 15:05:04 -07:00
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < CONFIG_LWM2M_NUM_ATTR; i++) {
|
2018-06-06 12:35:13 -07:00
|
|
|
if (ref != write_attr_pool[i].ref) {
|
2018-05-07 15:05:04 -07:00
|
|
|
continue;
|
|
|
|
}
|
2017-11-24 18:47:00 +08:00
|
|
|
|
2018-05-07 15:05:04 -07:00
|
|
|
switch (write_attr_pool[i].type) {
|
2017-11-24 18:47:00 +08:00
|
|
|
case LWM2M_ATTR_PMIN:
|
2018-05-07 15:05:04 -07:00
|
|
|
out->pmin = write_attr_pool[i].int_val;
|
2017-11-24 18:47:00 +08:00
|
|
|
break;
|
|
|
|
case LWM2M_ATTR_PMAX:
|
2018-05-07 15:05:04 -07:00
|
|
|
out->pmax = write_attr_pool[i].int_val;
|
2017-11-24 18:47:00 +08:00
|
|
|
break;
|
|
|
|
case LWM2M_ATTR_LT:
|
2018-05-07 15:05:04 -07:00
|
|
|
out->lt = write_attr_pool[i].float_val;
|
2017-11-24 18:47:00 +08:00
|
|
|
break;
|
|
|
|
case LWM2M_ATTR_GT:
|
2018-05-07 15:05:04 -07:00
|
|
|
out->gt = write_attr_pool[i].float_val;
|
2017-11-24 18:47:00 +08:00
|
|
|
break;
|
|
|
|
case LWM2M_ATTR_STEP:
|
2018-05-07 15:05:04 -07:00
|
|
|
out->st = write_attr_pool[i].float_val;
|
2017-11-24 18:47:00 +08:00
|
|
|
break;
|
|
|
|
default:
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("Unrecognize attr: %d",
|
|
|
|
write_attr_pool[i].type);
|
2017-11-24 18:47:00 +08:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* mark as set */
|
2018-05-07 15:05:04 -07:00
|
|
|
out->flags |= BIT(write_attr_pool[i].type);
|
2017-11-24 18:47:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-05-07 15:05:04 -07:00
|
|
|
static void clear_attrs(void *ref)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < CONFIG_LWM2M_NUM_ATTR; i++) {
|
|
|
|
if (ref == write_attr_pool[i].ref) {
|
2018-09-11 19:09:03 -07:00
|
|
|
(void)memset(&write_attr_pool[i], 0,
|
|
|
|
sizeof(write_attr_pool[i]));
|
2018-05-07 15:05:04 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
int lwm2m_notify_observer(u16_t obj_id, u16_t obj_inst_id, u16_t res_id)
|
|
|
|
{
|
|
|
|
struct observe_node *obs;
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
/* look for observers which match our resource */
|
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER(&engine_observer_list, obs, node) {
|
|
|
|
if (obs->path.obj_id == obj_id &&
|
|
|
|
obs->path.obj_inst_id == obj_inst_id &&
|
|
|
|
(obs->path.level < 3 ||
|
|
|
|
obs->path.res_id == res_id)) {
|
|
|
|
/* update the event time for this observer */
|
|
|
|
obs->event_timestamp = k_uptime_get();
|
|
|
|
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_DBG("NOTIFY EVENT %u/%u/%u",
|
|
|
|
obj_id, obj_inst_id, res_id);
|
2017-07-07 11:04:03 -07:00
|
|
|
|
|
|
|
ret++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_notify_observer_path(struct lwm2m_obj_path *path)
|
|
|
|
{
|
|
|
|
return lwm2m_notify_observer(path->obj_id, path->obj_inst_id,
|
|
|
|
path->res_id);
|
|
|
|
}
|
|
|
|
|
2017-09-01 01:11:43 -07:00
|
|
|
static int engine_add_observer(struct lwm2m_message *msg,
|
2017-07-07 11:04:03 -07:00
|
|
|
const u8_t *token, u8_t tkl,
|
2017-07-26 14:02:29 +08:00
|
|
|
u16_t format)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
2017-11-28 16:29:21 +08:00
|
|
|
struct lwm2m_engine_obj *obj = NULL;
|
2018-08-27 13:57:23 -07:00
|
|
|
struct lwm2m_engine_obj_field *obj_field = NULL;
|
2017-09-05 14:29:41 +08:00
|
|
|
struct lwm2m_engine_obj_inst *obj_inst = NULL;
|
2017-07-07 11:04:03 -07:00
|
|
|
struct observe_node *obs;
|
2017-11-28 16:29:21 +08:00
|
|
|
struct notification_attrs attrs = {
|
2019-03-13 11:55:29 -05:00
|
|
|
.flags = BIT(LWM2M_ATTR_PMIN) | BIT(LWM2M_ATTR_PMAX),
|
2017-11-28 16:29:21 +08:00
|
|
|
.pmin = DEFAULT_SERVER_PMIN,
|
|
|
|
.pmax = DEFAULT_SERVER_PMAX,
|
|
|
|
};
|
2018-05-07 14:18:37 -07:00
|
|
|
int i, ret;
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2017-09-01 01:11:43 -07:00
|
|
|
if (!msg || !msg->ctx) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("valid lwm2m message is required");
|
2017-09-01 01:11:43 -07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2019-03-26 19:57:45 -06:00
|
|
|
if (!token || (tkl == 0U || tkl > MAX_TOKEN_LEN)) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("token(%p) and token length(%u) must be valid.",
|
|
|
|
token, tkl);
|
2017-10-03 09:34:59 -07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2017-11-28 16:29:21 +08:00
|
|
|
/* TODO: get server object for default pmin/pmax
|
|
|
|
* and observe dup checking
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* make sure this observer doesn't exist already */
|
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER(&engine_observer_list, obs, node) {
|
|
|
|
/* TODO: distinguish server object */
|
|
|
|
if (obs->ctx == msg->ctx &&
|
2019-01-24 14:40:23 -08:00
|
|
|
memcmp(&obs->path, &msg->path, sizeof(msg->path)) == 0) {
|
2017-11-28 16:29:21 +08:00
|
|
|
/* quietly update the token information */
|
|
|
|
memcpy(obs->token, token, tkl);
|
|
|
|
obs->tkl = tkl;
|
|
|
|
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_DBG("OBSERVER DUPLICATE %u/%u/%u(%u) [%s]",
|
2019-01-24 14:40:23 -08:00
|
|
|
msg->path.obj_id, msg->path.obj_inst_id,
|
|
|
|
msg->path.res_id, msg->path.level,
|
2019-05-29 16:49:18 -07:00
|
|
|
log_strdup(
|
|
|
|
lwm2m_sprint_ip_addr(&msg->ctx->remote_addr)));
|
2017-11-28 16:29:21 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-05 14:29:41 +08:00
|
|
|
/* check if object exists */
|
2019-01-24 14:40:23 -08:00
|
|
|
obj = get_engine_obj(msg->path.obj_id);
|
2017-11-28 16:29:21 +08:00
|
|
|
if (!obj) {
|
2019-01-24 14:40:23 -08:00
|
|
|
LOG_ERR("unable to find obj: %u", msg->path.obj_id);
|
2017-09-05 14:29:41 +08:00
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
2018-05-07 15:05:04 -07:00
|
|
|
ret = update_attrs(obj, &attrs);
|
2018-05-07 14:18:37 -07:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
2017-11-28 16:29:21 +08:00
|
|
|
}
|
|
|
|
|
2017-09-05 14:29:41 +08:00
|
|
|
/* check if object instance exists */
|
2019-03-26 19:57:45 -06:00
|
|
|
if (msg->path.level >= 2U) {
|
2019-01-24 14:40:23 -08:00
|
|
|
obj_inst = get_engine_obj_inst(msg->path.obj_id,
|
|
|
|
msg->path.obj_inst_id);
|
2017-09-05 14:29:41 +08:00
|
|
|
if (!obj_inst) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("unable to find obj_inst: %u/%u",
|
2019-01-24 14:40:23 -08:00
|
|
|
msg->path.obj_id, msg->path.obj_inst_id);
|
2017-09-05 14:29:41 +08:00
|
|
|
return -ENOENT;
|
|
|
|
}
|
2017-11-28 16:29:21 +08:00
|
|
|
|
2018-05-07 15:05:04 -07:00
|
|
|
ret = update_attrs(obj_inst, &attrs);
|
2018-05-07 14:18:37 -07:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
2017-11-28 16:29:21 +08:00
|
|
|
}
|
2017-09-05 14:29:41 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* check if resource exists */
|
2019-03-26 19:57:45 -06:00
|
|
|
if (msg->path.level >= 3U) {
|
2017-09-05 14:29:41 +08:00
|
|
|
for (i = 0; i < obj_inst->resource_count; i++) {
|
2019-01-24 14:40:23 -08:00
|
|
|
if (obj_inst->resources[i].res_id == msg->path.res_id) {
|
2017-09-05 14:29:41 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i == obj_inst->resource_count) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("unable to find res_id: %u/%u/%u",
|
2019-01-24 14:40:23 -08:00
|
|
|
msg->path.obj_id, msg->path.obj_inst_id,
|
|
|
|
msg->path.res_id);
|
2017-09-05 14:29:41 +08:00
|
|
|
return -ENOENT;
|
|
|
|
}
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2018-08-27 13:57:23 -07:00
|
|
|
/* load object field data */
|
|
|
|
obj_field = lwm2m_get_engine_obj_field(obj,
|
|
|
|
obj_inst->resources[i].res_id);
|
|
|
|
if (!obj_field) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("unable to find obj_field: %u/%u/%u",
|
2019-01-24 14:40:23 -08:00
|
|
|
msg->path.obj_id, msg->path.obj_inst_id,
|
|
|
|
msg->path.res_id);
|
2018-08-27 13:57:23 -07:00
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check for READ permission on matching resource */
|
|
|
|
if (!LWM2M_HAS_PERM(obj_field, LWM2M_PERM_R)) {
|
|
|
|
return -EPERM;
|
|
|
|
}
|
|
|
|
|
2018-05-07 15:05:04 -07:00
|
|
|
ret = update_attrs(&obj_inst->resources[i], &attrs);
|
2018-05-07 14:18:37 -07:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* find an unused observer index node */
|
|
|
|
for (i = 0; i < CONFIG_LWM2M_ENGINE_MAX_OBSERVER; i++) {
|
2018-05-07 16:03:27 -07:00
|
|
|
if (!observe_node_data[i].ctx) {
|
2017-07-07 11:04:03 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* couldn't find an index */
|
|
|
|
if (i == CONFIG_LWM2M_ENGINE_MAX_OBSERVER) {
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* copy the values and add it to the list */
|
2017-09-01 01:11:43 -07:00
|
|
|
observe_node_data[i].ctx = msg->ctx;
|
2019-01-24 14:40:23 -08:00
|
|
|
memcpy(&observe_node_data[i].path, &msg->path, sizeof(msg->path));
|
2017-07-07 11:04:03 -07:00
|
|
|
memcpy(observe_node_data[i].token, token, tkl);
|
|
|
|
observe_node_data[i].tkl = tkl;
|
|
|
|
observe_node_data[i].last_timestamp = k_uptime_get();
|
|
|
|
observe_node_data[i].event_timestamp =
|
|
|
|
observe_node_data[i].last_timestamp;
|
2017-11-28 16:29:21 +08:00
|
|
|
observe_node_data[i].min_period_sec = attrs.pmin;
|
2019-02-11 17:14:19 +00:00
|
|
|
observe_node_data[i].max_period_sec = MAX(attrs.pmax, attrs.pmin);
|
2017-07-26 14:02:29 +08:00
|
|
|
observe_node_data[i].format = format;
|
2018-11-29 11:23:03 -08:00
|
|
|
observe_node_data[i].counter = 1U;
|
2017-07-07 11:04:03 -07:00
|
|
|
sys_slist_append(&engine_observer_list,
|
|
|
|
&observe_node_data[i].node);
|
|
|
|
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_DBG("OBSERVER ADDED %u/%u/%u(%u) token:'%s' addr:%s",
|
2019-01-24 14:40:23 -08:00
|
|
|
msg->path.obj_id, msg->path.obj_inst_id,
|
|
|
|
msg->path.res_id, msg->path.level,
|
2019-05-29 16:49:18 -07:00
|
|
|
log_strdup(sprint_token(token, tkl)),
|
|
|
|
log_strdup(lwm2m_sprint_ip_addr(&msg->ctx->remote_addr)));
|
2017-07-07 11:04:03 -07:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int engine_remove_observer(const u8_t *token, u8_t tkl)
|
|
|
|
{
|
|
|
|
struct observe_node *obs, *found_obj = NULL;
|
2017-09-05 11:03:47 +08:00
|
|
|
sys_snode_t *prev_node = NULL;
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2019-03-26 19:57:45 -06:00
|
|
|
if (!token || (tkl == 0U || tkl > MAX_TOKEN_LEN)) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("token(%p) and token length(%u) must be valid.",
|
|
|
|
token, tkl);
|
2017-07-07 11:04:03 -07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* find the node index */
|
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER(&engine_observer_list, obs, node) {
|
|
|
|
if (memcmp(obs->token, token, tkl) == 0) {
|
|
|
|
found_obj = obs;
|
|
|
|
break;
|
|
|
|
}
|
2017-09-05 11:03:47 +08:00
|
|
|
|
|
|
|
prev_node = &obs->node;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!found_obj) {
|
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
2017-09-05 11:03:47 +08:00
|
|
|
sys_slist_remove(&engine_observer_list, prev_node, &found_obj->node);
|
2018-09-11 19:09:03 -07:00
|
|
|
(void)memset(found_obj, 0, sizeof(*found_obj));
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2019-05-29 16:49:18 -07:00
|
|
|
LOG_DBG("observer '%s' removed", log_strdup(sprint_token(token, tkl)));
|
2017-07-07 11:04:03 -07:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-09-05 13:49:21 +08:00
|
|
|
static void engine_remove_observer_by_id(u16_t obj_id, s32_t obj_inst_id)
|
|
|
|
{
|
|
|
|
struct observe_node *obs, *tmp;
|
|
|
|
sys_snode_t *prev_node = NULL;
|
|
|
|
|
|
|
|
/* remove observer instances accordingly */
|
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER_SAFE(
|
|
|
|
&engine_observer_list, obs, tmp, node) {
|
|
|
|
if (!(obj_id == obs->path.obj_id &&
|
2017-11-28 16:29:21 +08:00
|
|
|
obj_inst_id == obs->path.obj_inst_id)) {
|
2017-09-05 13:49:21 +08:00
|
|
|
prev_node = &obs->node;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
sys_slist_remove(&engine_observer_list, prev_node, &obs->node);
|
2018-09-11 19:09:03 -07:00
|
|
|
(void)memset(obs, 0, sizeof(*obs));
|
2017-09-05 13:49:21 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
/* engine object */
|
|
|
|
|
|
|
|
void lwm2m_register_obj(struct lwm2m_engine_obj *obj)
|
|
|
|
{
|
|
|
|
sys_slist_append(&engine_obj_list, &obj->node);
|
|
|
|
}
|
|
|
|
|
|
|
|
void lwm2m_unregister_obj(struct lwm2m_engine_obj *obj)
|
|
|
|
{
|
2017-09-05 13:49:21 +08:00
|
|
|
engine_remove_observer_by_id(obj->obj_id, -1);
|
2017-09-05 11:03:47 +08:00
|
|
|
sys_slist_find_and_remove(&engine_obj_list, &obj->node);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static struct lwm2m_engine_obj *get_engine_obj(int obj_id)
|
|
|
|
{
|
|
|
|
struct lwm2m_engine_obj *obj;
|
|
|
|
|
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER(&engine_obj_list, obj, node) {
|
|
|
|
if (obj->obj_id == obj_id) {
|
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct lwm2m_engine_obj_field *
|
|
|
|
lwm2m_get_engine_obj_field(struct lwm2m_engine_obj *obj, int res_id)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (obj && obj->fields && obj->field_count > 0) {
|
|
|
|
for (i = 0; i < obj->field_count; i++) {
|
|
|
|
if (obj->fields[i].res_id == res_id) {
|
|
|
|
return &obj->fields[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* engine object instance */
|
|
|
|
|
|
|
|
static void engine_register_obj_inst(struct lwm2m_engine_obj_inst *obj_inst)
|
|
|
|
{
|
|
|
|
sys_slist_append(&engine_obj_inst_list, &obj_inst->node);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void engine_unregister_obj_inst(struct lwm2m_engine_obj_inst *obj_inst)
|
|
|
|
{
|
2017-09-05 13:49:21 +08:00
|
|
|
engine_remove_observer_by_id(
|
|
|
|
obj_inst->obj->obj_id, obj_inst->obj_inst_id);
|
2017-09-05 11:03:47 +08:00
|
|
|
sys_slist_find_and_remove(&engine_obj_inst_list, &obj_inst->node);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static struct lwm2m_engine_obj_inst *get_engine_obj_inst(int obj_id,
|
|
|
|
int obj_inst_id)
|
|
|
|
{
|
|
|
|
struct lwm2m_engine_obj_inst *obj_inst;
|
|
|
|
|
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER(&engine_obj_inst_list, obj_inst,
|
|
|
|
node) {
|
|
|
|
if (obj_inst->obj->obj_id == obj_id &&
|
|
|
|
obj_inst->obj_inst_id == obj_inst_id) {
|
|
|
|
return obj_inst;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct lwm2m_engine_obj_inst *
|
2018-08-27 17:00:17 -07:00
|
|
|
next_engine_obj_inst(int obj_id, int obj_inst_id)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
2018-08-27 17:00:17 -07:00
|
|
|
struct lwm2m_engine_obj_inst *obj_inst, *next = NULL;
|
|
|
|
|
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER(&engine_obj_inst_list, obj_inst,
|
|
|
|
node) {
|
|
|
|
if (obj_inst->obj->obj_id == obj_id &&
|
|
|
|
obj_inst->obj_inst_id > obj_inst_id &&
|
|
|
|
(!next || next->obj_inst_id > obj_inst->obj_inst_id)) {
|
|
|
|
next = obj_inst;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-27 17:00:17 -07:00
|
|
|
return next;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_create_obj_inst(u16_t obj_id, u16_t obj_inst_id,
|
|
|
|
struct lwm2m_engine_obj_inst **obj_inst)
|
|
|
|
{
|
|
|
|
struct lwm2m_engine_obj *obj;
|
2018-07-12 10:00:01 -07:00
|
|
|
int ret;
|
2017-07-07 11:04:03 -07:00
|
|
|
|
|
|
|
*obj_inst = NULL;
|
|
|
|
obj = get_engine_obj(obj_id);
|
|
|
|
if (!obj) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("unable to find obj: %u", obj_id);
|
2017-07-07 11:04:03 -07:00
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!obj->create_cb) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("obj %u has no create_cb", obj_id);
|
2017-07-07 11:04:03 -07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (obj->instance_count + 1 > obj->max_instance_count) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("no more instances available for obj %u", obj_id);
|
2017-07-07 11:04:03 -07:00
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
*obj_inst = obj->create_cb(obj_inst_id);
|
|
|
|
if (!*obj_inst) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("unable to create obj %u instance %u",
|
|
|
|
obj_id, obj_inst_id);
|
2017-08-17 13:10:59 -07:00
|
|
|
/*
|
|
|
|
* Already checked for instance count total.
|
|
|
|
* This can only be an error if the object instance exists.
|
|
|
|
*/
|
|
|
|
return -EEXIST;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
obj->instance_count++;
|
|
|
|
(*obj_inst)->obj = obj;
|
|
|
|
(*obj_inst)->obj_inst_id = obj_inst_id;
|
|
|
|
engine_register_obj_inst(*obj_inst);
|
2018-07-12 10:00:01 -07:00
|
|
|
|
|
|
|
if (obj->user_create_cb) {
|
|
|
|
ret = obj->user_create_cb(obj_inst_id);
|
|
|
|
if (ret < 0) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("Error in user obj create %u/%u: %d",
|
|
|
|
obj_id, obj_inst_id, ret);
|
2018-07-12 10:00:01 -07:00
|
|
|
lwm2m_delete_obj_inst(obj_id, obj_inst_id);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_delete_obj_inst(u16_t obj_id, u16_t obj_inst_id)
|
|
|
|
{
|
2017-09-04 16:59:37 +08:00
|
|
|
int i, ret = 0;
|
2017-07-07 11:04:03 -07:00
|
|
|
struct lwm2m_engine_obj *obj;
|
|
|
|
struct lwm2m_engine_obj_inst *obj_inst;
|
|
|
|
|
|
|
|
obj = get_engine_obj(obj_id);
|
|
|
|
if (!obj) {
|
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
obj_inst = get_engine_obj_inst(obj_id, obj_inst_id);
|
|
|
|
if (!obj_inst) {
|
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
2018-07-12 10:00:01 -07:00
|
|
|
if (obj->user_delete_cb) {
|
|
|
|
ret = obj->user_delete_cb(obj_inst_id);
|
|
|
|
if (ret < 0) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("Error in user obj delete %u/%u: %d",
|
|
|
|
obj_id, obj_inst_id, ret);
|
2018-07-12 10:00:01 -07:00
|
|
|
/* don't return error */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
engine_unregister_obj_inst(obj_inst);
|
|
|
|
obj->instance_count--;
|
2017-09-04 16:59:37 +08:00
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
if (obj->delete_cb) {
|
2017-09-04 16:59:37 +08:00
|
|
|
ret = obj->delete_cb(obj_inst_id);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2017-09-04 16:59:37 +08:00
|
|
|
/* reset obj_inst and res_inst data structure */
|
|
|
|
for (i = 0; i < obj_inst->resource_count; i++) {
|
2018-05-07 15:05:04 -07:00
|
|
|
clear_attrs(&obj_inst->resources[i]);
|
2018-09-11 19:09:03 -07:00
|
|
|
(void)memset(obj_inst->resources + i, 0,
|
|
|
|
sizeof(struct lwm2m_engine_res_inst));
|
2017-09-04 16:59:37 +08:00
|
|
|
}
|
|
|
|
|
2018-05-07 15:05:04 -07:00
|
|
|
clear_attrs(obj_inst);
|
2018-09-11 19:09:03 -07:00
|
|
|
(void)memset(obj_inst, 0, sizeof(struct lwm2m_engine_obj_inst));
|
2017-09-04 16:59:37 +08:00
|
|
|
return ret;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* utility functions */
|
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
static int get_option_int(const struct coap_packet *cpkt, u8_t opt)
|
2017-07-25 16:54:25 +08:00
|
|
|
{
|
2017-09-19 16:04:32 -07:00
|
|
|
struct coap_option option = {};
|
2018-11-29 11:23:03 -08:00
|
|
|
u16_t count = 1U;
|
2017-07-25 16:54:25 +08:00
|
|
|
int r;
|
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
r = coap_find_options(cpkt, opt, &option, count);
|
2017-07-25 16:54:25 +08:00
|
|
|
if (r <= 0) {
|
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
return coap_option_value_to_int(&option);
|
2017-07-25 16:54:25 +08:00
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
static u16_t atou16(u8_t *buf, u16_t buflen, u16_t *len)
|
|
|
|
{
|
2018-11-29 11:23:03 -08:00
|
|
|
u16_t val = 0U;
|
|
|
|
u16_t pos = 0U;
|
2017-07-07 11:04:03 -07:00
|
|
|
|
|
|
|
/* we should get a value first - consume all numbers */
|
|
|
|
while (pos < buflen && isdigit(buf[pos])) {
|
2019-03-26 19:57:45 -06:00
|
|
|
val = val * 10U + (buf[pos] - '0');
|
2017-07-07 11:04:03 -07:00
|
|
|
pos++;
|
|
|
|
}
|
|
|
|
|
|
|
|
*len = pos;
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
2017-11-24 18:47:00 +08:00
|
|
|
static int atof32(const char *input, float32_value_t *out)
|
|
|
|
{
|
|
|
|
char *pos, *end, buf[24];
|
|
|
|
long int val;
|
|
|
|
s32_t base = 1000000, sign = 1;
|
|
|
|
|
|
|
|
if (!input || !out) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
strncpy(buf, input, sizeof(buf) - 1);
|
|
|
|
buf[sizeof(buf) - 1] = '\0';
|
|
|
|
|
|
|
|
if (strchr(buf, '-')) {
|
|
|
|
sign = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
pos = strchr(buf, '.');
|
|
|
|
if (pos) {
|
|
|
|
*pos = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
errno = 0;
|
|
|
|
val = strtol(buf, &end, 10);
|
2019-02-13 13:37:25 -08:00
|
|
|
if (errno || *end || val < INT_MIN) {
|
2017-11-24 18:47:00 +08:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
out->val1 = (s32_t) val;
|
|
|
|
out->val2 = 0;
|
|
|
|
|
|
|
|
if (!pos) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-07-11 15:31:59 -05:00
|
|
|
while (*(++pos) && base > 1 && isdigit((unsigned char)*pos)) {
|
2017-11-24 18:47:00 +08:00
|
|
|
out->val2 = out->val2 * 10 + (*pos - '0');
|
|
|
|
base /= 10;
|
|
|
|
}
|
|
|
|
|
|
|
|
out->val2 *= sign * base;
|
|
|
|
return !*pos || base == 1 ? 0 : -EINVAL;
|
|
|
|
}
|
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
static int coap_options_to_path(struct coap_option *opt, int options_count,
|
|
|
|
struct lwm2m_obj_path *path)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
2017-11-21 18:00:10 +08:00
|
|
|
u16_t len, *id[4] = { &path->obj_id, &path->obj_inst_id,
|
|
|
|
&path->res_id, &path->res_inst_id };
|
2017-07-07 11:04:03 -07:00
|
|
|
|
|
|
|
path->level = options_count;
|
|
|
|
|
2017-11-21 18:00:10 +08:00
|
|
|
for (int i = 0; i < options_count; i++) {
|
|
|
|
*id[i] = atou16(opt[i].value, opt[i].len, &len);
|
2019-03-26 19:57:45 -06:00
|
|
|
if (len == 0U || opt[i].len != len) {
|
2017-11-21 18:00:10 +08:00
|
|
|
path->level = i;
|
|
|
|
break;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
}
|
2017-08-28 17:06:48 +08:00
|
|
|
|
|
|
|
return options_count == path->level ? 0 : -EINVAL;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
static struct lwm2m_message *find_msg(struct coap_pending *pending,
|
|
|
|
struct coap_reply *reply)
|
2017-09-01 01:11:43 -07:00
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
2017-09-05 21:03:49 -07:00
|
|
|
if (!pending && !reply) {
|
2017-09-01 01:11:43 -07:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < CONFIG_LWM2M_ENGINE_MAX_MESSAGES; i++) {
|
|
|
|
if (messages[i].ctx && messages[i].pending == pending) {
|
|
|
|
return &messages[i];
|
|
|
|
}
|
2017-09-05 21:03:49 -07:00
|
|
|
|
|
|
|
if (messages[i].ctx && messages[i].reply == reply) {
|
|
|
|
return &messages[i];
|
|
|
|
}
|
2017-09-01 01:11:43 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct lwm2m_message *lwm2m_get_message(struct lwm2m_ctx *client_ctx)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < CONFIG_LWM2M_ENGINE_MAX_MESSAGES; i++) {
|
|
|
|
if (!messages[i].ctx) {
|
|
|
|
messages[i].ctx = client_ctx;
|
|
|
|
return &messages[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2017-10-11 11:29:54 -07:00
|
|
|
void lwm2m_reset_message(struct lwm2m_message *msg, bool release)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
2017-09-01 01:11:43 -07:00
|
|
|
if (!msg) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (msg->pending) {
|
2017-09-19 16:04:32 -07:00
|
|
|
coap_pending_clear(msg->pending);
|
2017-09-01 01:11:43 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (msg->reply) {
|
|
|
|
/* make sure we want to clear the reply */
|
2017-09-19 16:04:32 -07:00
|
|
|
coap_reply_clear(msg->reply);
|
2017-09-01 01:11:43 -07:00
|
|
|
}
|
|
|
|
|
2017-10-11 11:29:54 -07:00
|
|
|
if (release) {
|
2018-09-11 19:09:03 -07:00
|
|
|
(void)memset(msg, 0, sizeof(*msg));
|
2017-10-11 11:29:54 -07:00
|
|
|
} else {
|
|
|
|
msg->message_timeout_cb = NULL;
|
2018-09-11 19:09:03 -07:00
|
|
|
(void)memset(&msg->cpkt, 0, sizeof(msg->cpkt));
|
2017-10-11 11:29:54 -07:00
|
|
|
}
|
2017-09-01 01:11:43 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_init_message(struct lwm2m_message *msg)
|
|
|
|
{
|
2018-11-29 11:23:03 -08:00
|
|
|
u8_t tokenlen = 0U;
|
2017-09-19 16:04:32 -07:00
|
|
|
u8_t *token = NULL;
|
|
|
|
int r = 0;
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2017-09-01 01:11:43 -07:00
|
|
|
if (!msg || !msg->ctx) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("LwM2M message is invalid.");
|
2017-09-01 01:11:43 -07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2017-10-03 09:34:59 -07:00
|
|
|
/*
|
2017-09-19 16:04:32 -07:00
|
|
|
* msg->tkl == 0 is for a new TOKEN
|
|
|
|
* msg->tkl == LWM2M_MSG_TOKEN_LEN_SKIP means dont set
|
2017-10-03 09:34:59 -07:00
|
|
|
*/
|
2019-03-26 19:57:45 -06:00
|
|
|
if (msg->tkl == 0U) {
|
2018-11-29 11:23:03 -08:00
|
|
|
tokenlen = 0U;
|
2017-09-19 16:04:32 -07:00
|
|
|
token = coap_next_token();
|
2017-09-01 01:11:43 -07:00
|
|
|
} else if (msg->token && msg->tkl != LWM2M_MSG_TOKEN_LEN_SKIP) {
|
2017-09-19 16:04:32 -07:00
|
|
|
tokenlen = msg->tkl;
|
|
|
|
token = msg->token;
|
|
|
|
}
|
|
|
|
|
2019-01-25 14:13:07 -08:00
|
|
|
r = coap_packet_init(&msg->cpkt, msg->msg_data, sizeof(msg->msg_data),
|
|
|
|
1, msg->type, tokenlen, token, msg->code,
|
2017-09-19 16:04:32 -07:00
|
|
|
(msg->mid > 0 ? msg->mid : coap_next_id()));
|
|
|
|
if (r < 0) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("coap packet init error (err:%d)", r);
|
2017-09-19 16:04:32 -07:00
|
|
|
goto cleanup;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2017-09-01 01:11:43 -07:00
|
|
|
/* only TYPE_CON messages need pending tracking / reply handling */
|
2017-09-19 16:04:32 -07:00
|
|
|
if (msg->type != COAP_TYPE_CON) {
|
2017-09-01 01:11:43 -07:00
|
|
|
return 0;
|
|
|
|
}
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
msg->pending = coap_pending_next_unused(
|
2017-09-01 01:11:43 -07:00
|
|
|
msg->ctx->pendings,
|
|
|
|
CONFIG_LWM2M_ENGINE_MAX_PENDING);
|
|
|
|
if (!msg->pending) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("Unable to find a free pending to track "
|
|
|
|
"retransmissions.");
|
2017-09-01 01:11:43 -07:00
|
|
|
r = -ENOMEM;
|
|
|
|
goto cleanup;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2019-01-25 14:39:41 -08:00
|
|
|
r = coap_pending_init(msg->pending, &msg->cpkt, &msg->ctx->remote_addr);
|
2017-09-01 01:11:43 -07:00
|
|
|
if (r < 0) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("Unable to initialize a pending "
|
|
|
|
"retransmission (err:%d).", r);
|
2017-09-01 01:11:43 -07:00
|
|
|
goto cleanup;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2017-09-01 01:11:43 -07:00
|
|
|
if (msg->reply_cb) {
|
2017-09-19 16:04:32 -07:00
|
|
|
msg->reply = coap_reply_next_unused(
|
2017-09-01 01:11:43 -07:00
|
|
|
msg->ctx->replies,
|
|
|
|
CONFIG_LWM2M_ENGINE_MAX_REPLIES);
|
|
|
|
if (!msg->reply) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("No resources for waiting for replies.");
|
2017-09-01 01:11:43 -07:00
|
|
|
r = -ENOMEM;
|
|
|
|
goto cleanup;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2018-03-23 10:49:44 -07:00
|
|
|
coap_reply_clear(msg->reply);
|
2017-09-19 16:04:32 -07:00
|
|
|
coap_reply_init(msg->reply, &msg->cpkt);
|
2017-09-01 01:11:43 -07:00
|
|
|
msg->reply->reply = msg->reply_cb;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2017-09-01 01:11:43 -07:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
cleanup:
|
2017-10-11 11:29:54 -07:00
|
|
|
lwm2m_reset_message(msg, true);
|
2017-09-01 01:11:43 -07:00
|
|
|
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_send_message(struct lwm2m_message *msg)
|
|
|
|
{
|
|
|
|
if (!msg || !msg->ctx) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("LwM2M message is invalid.");
|
2017-09-01 01:11:43 -07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2018-01-30 13:39:34 -08:00
|
|
|
if (msg->type == COAP_TYPE_CON) {
|
|
|
|
coap_pending_cycle(msg->pending);
|
|
|
|
}
|
2017-12-08 11:55:01 +08:00
|
|
|
|
2017-09-01 01:11:43 -07:00
|
|
|
msg->send_attempts++;
|
2019-01-25 14:13:07 -08:00
|
|
|
|
2019-02-12 17:10:41 -08:00
|
|
|
if (send(msg->ctx->sock_fd, msg->cpkt.data, msg->cpkt.offset, 0) < 0) {
|
2018-01-30 13:39:34 -08:00
|
|
|
if (msg->type == COAP_TYPE_CON) {
|
|
|
|
coap_pending_clear(msg->pending);
|
|
|
|
}
|
|
|
|
|
2019-02-12 17:10:41 -08:00
|
|
|
return -errno;
|
2017-09-01 01:11:43 -07:00
|
|
|
}
|
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
if (msg->type == COAP_TYPE_CON) {
|
2018-01-30 13:39:34 -08:00
|
|
|
/* don't re-queue the retransmit work on retransmits */
|
2017-09-01 01:11:43 -07:00
|
|
|
if (msg->send_attempts > 1) {
|
2018-01-30 13:39:34 -08:00
|
|
|
return 0;
|
2017-09-01 01:11:43 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
k_delayed_work_submit(&msg->ctx->retransmit_work,
|
|
|
|
msg->pending->timeout);
|
|
|
|
} else {
|
2017-10-11 11:29:54 -07:00
|
|
|
lwm2m_reset_message(msg, true);
|
2017-09-01 01:11:43 -07:00
|
|
|
}
|
|
|
|
|
2019-02-12 17:10:41 -08:00
|
|
|
return 0;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
u16_t lwm2m_get_rd_data(u8_t *client_data, u16_t size)
|
|
|
|
{
|
|
|
|
struct lwm2m_engine_obj *obj;
|
|
|
|
struct lwm2m_engine_obj_inst *obj_inst;
|
|
|
|
u8_t temp[32];
|
2018-11-29 11:23:03 -08:00
|
|
|
u16_t pos = 0U;
|
2017-07-07 11:04:03 -07:00
|
|
|
int len;
|
|
|
|
|
2017-08-14 17:06:03 +08:00
|
|
|
/* Add resource-type/content-type to the registration message */
|
|
|
|
memcpy(client_data, REG_PREFACE, sizeof(REG_PREFACE) - 1);
|
|
|
|
pos += sizeof(REG_PREFACE) - 1;
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER(&engine_obj_list, obj, node) {
|
2017-07-10 18:19:35 +08:00
|
|
|
/* Security obj MUST NOT be part of registration message */
|
|
|
|
if (obj->obj_id == LWM2M_OBJECT_SECURITY_ID) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-08-14 16:16:11 +08:00
|
|
|
/* Only report <OBJ_ID> when no instance available */
|
2019-03-26 19:57:45 -06:00
|
|
|
if (obj->instance_count == 0U) {
|
2017-10-11 13:49:46 -07:00
|
|
|
len = snprintk(temp, sizeof(temp), "%s</%u>",
|
2017-08-14 16:16:11 +08:00
|
|
|
(pos > 0) ? "," : "", obj->obj_id);
|
|
|
|
if (pos + len >= size) {
|
|
|
|
/* full buffer -- exit loop */
|
|
|
|
break;
|
|
|
|
}
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2017-08-14 16:16:11 +08:00
|
|
|
memcpy(&client_data[pos], temp, len);
|
|
|
|
pos += len;
|
|
|
|
continue;
|
|
|
|
}
|
2017-07-07 11:04:03 -07:00
|
|
|
|
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER(&engine_obj_inst_list,
|
|
|
|
obj_inst, node) {
|
|
|
|
if (obj_inst->obj->obj_id == obj->obj_id) {
|
2017-10-11 13:49:46 -07:00
|
|
|
len = snprintk(temp, sizeof(temp),
|
2018-05-07 13:59:11 -07:00
|
|
|
"%s</%u/%u>",
|
2017-07-07 11:04:03 -07:00
|
|
|
(pos > 0) ? "," : "",
|
2018-05-07 13:59:11 -07:00
|
|
|
obj_inst->obj->obj_id,
|
|
|
|
obj_inst->obj_inst_id);
|
2017-07-07 11:04:03 -07:00
|
|
|
/*
|
|
|
|
* TODO: iterate through resources once block
|
|
|
|
* transfer is handled correctly
|
|
|
|
*/
|
|
|
|
if (pos + len >= size) {
|
|
|
|
/* full buffer -- exit loop */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(&client_data[pos], temp, len);
|
|
|
|
pos += len;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
client_data[pos] = '\0';
|
|
|
|
return pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* input / output selection */
|
|
|
|
|
2018-02-05 16:05:51 +08:00
|
|
|
static int select_writer(struct lwm2m_output_context *out, u16_t accept)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
|
|
|
switch (accept) {
|
|
|
|
|
2017-10-15 13:42:45 -07:00
|
|
|
case LWM2M_FORMAT_APP_LINK_FORMAT:
|
|
|
|
/* TODO: rewrite do_discover as content formatter */
|
|
|
|
break;
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
case LWM2M_FORMAT_PLAIN_TEXT:
|
|
|
|
case LWM2M_FORMAT_OMA_PLAIN_TEXT:
|
|
|
|
out->writer = &plain_text_writer;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_FORMAT_OMA_TLV:
|
|
|
|
case LWM2M_FORMAT_OMA_OLD_TLV:
|
|
|
|
out->writer = &oma_tlv_writer;
|
|
|
|
break;
|
|
|
|
|
|
|
|
#ifdef CONFIG_LWM2M_RW_JSON_SUPPORT
|
|
|
|
case LWM2M_FORMAT_OMA_JSON:
|
|
|
|
case LWM2M_FORMAT_OMA_OLD_JSON:
|
|
|
|
out->writer = &json_writer;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
default:
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_WRN("Unknown content type %u", accept);
|
2018-02-05 16:05:51 +08:00
|
|
|
return -ENOMSG;
|
2017-07-07 11:04:03 -07:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2018-02-05 16:05:51 +08:00
|
|
|
return 0;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2018-01-30 15:18:57 +08:00
|
|
|
static int select_reader(struct lwm2m_input_context *in, u16_t format)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
|
|
|
switch (format) {
|
|
|
|
|
2017-10-30 15:18:35 +08:00
|
|
|
case LWM2M_FORMAT_APP_OCTET_STREAM:
|
2017-07-05 10:54:31 +08:00
|
|
|
case LWM2M_FORMAT_PLAIN_TEXT:
|
2017-07-07 11:04:03 -07:00
|
|
|
case LWM2M_FORMAT_OMA_PLAIN_TEXT:
|
|
|
|
in->reader = &plain_text_reader;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_FORMAT_OMA_TLV:
|
|
|
|
case LWM2M_FORMAT_OMA_OLD_TLV:
|
|
|
|
in->reader = &oma_tlv_reader;
|
|
|
|
break;
|
|
|
|
|
2019-01-25 15:56:23 -08:00
|
|
|
#ifdef CONFIG_LWM2M_RW_JSON_SUPPORT
|
|
|
|
case LWM2M_FORMAT_OMA_JSON:
|
|
|
|
case LWM2M_FORMAT_OMA_OLD_JSON:
|
|
|
|
in->reader = &json_reader;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
default:
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_WRN("Unknown content type %u", format);
|
2018-01-30 15:18:57 +08:00
|
|
|
return -ENOMSG;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2018-02-05 16:05:51 +08:00
|
|
|
return 0;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* user data setter functions */
|
|
|
|
|
|
|
|
static int string_to_path(char *pathstr, struct lwm2m_obj_path *path,
|
|
|
|
char delim)
|
|
|
|
{
|
|
|
|
u16_t value, len;
|
|
|
|
int i, tokstart = -1, toklen;
|
|
|
|
int end_index = strlen(pathstr) - 1;
|
|
|
|
|
2018-09-11 19:09:03 -07:00
|
|
|
(void)memset(path, 0, sizeof(*path));
|
2017-07-07 11:04:03 -07:00
|
|
|
for (i = 0; i <= end_index; i++) {
|
|
|
|
/* search for first numeric */
|
|
|
|
if (tokstart == -1) {
|
2018-07-11 15:31:59 -05:00
|
|
|
if (!isdigit((unsigned char)pathstr[i])) {
|
2017-07-07 11:04:03 -07:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
tokstart = i;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* find delimiter char or end of string */
|
|
|
|
if (pathstr[i] == delim || i == end_index) {
|
|
|
|
toklen = i - tokstart + 1;
|
|
|
|
|
|
|
|
/* don't process delimiter char */
|
|
|
|
if (pathstr[i] == delim) {
|
|
|
|
toklen--;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (toklen <= 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
value = atou16(&pathstr[tokstart], toklen, &len);
|
|
|
|
switch (path->level) {
|
|
|
|
|
|
|
|
case 0:
|
|
|
|
path->obj_id = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 1:
|
|
|
|
path->obj_inst_id = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 2:
|
|
|
|
path->res_id = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 3:
|
|
|
|
path->res_inst_id = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("invalid level (%d)", path->level);
|
2017-07-07 11:04:03 -07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/* increase the path level for each token found */
|
|
|
|
path->level++;
|
|
|
|
tokstart = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-04-30 15:02:04 -07:00
|
|
|
static int path_to_objs(const struct lwm2m_obj_path *path,
|
|
|
|
struct lwm2m_engine_obj_inst **obj_inst,
|
|
|
|
struct lwm2m_engine_obj_field **obj_field,
|
|
|
|
struct lwm2m_engine_res_inst **res)
|
|
|
|
{
|
|
|
|
struct lwm2m_engine_obj_inst *oi;
|
|
|
|
struct lwm2m_engine_obj_field *of;
|
|
|
|
struct lwm2m_engine_res_inst *r = NULL;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (!path) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
oi = get_engine_obj_inst(path->obj_id, path->obj_inst_id);
|
|
|
|
if (!oi) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("obj instance %d/%d not found",
|
|
|
|
path->obj_id, path->obj_inst_id);
|
2018-04-30 15:02:04 -07:00
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
2019-03-26 19:57:45 -06:00
|
|
|
if (!oi->resources || oi->resource_count == 0U) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("obj instance has no resources");
|
2018-04-30 15:02:04 -07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
of = lwm2m_get_engine_obj_field(oi->obj, path->res_id);
|
|
|
|
if (!of) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("obj field %d not found", path->res_id);
|
2018-04-30 15:02:04 -07:00
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < oi->resource_count; i++) {
|
|
|
|
if (oi->resources[i].res_id == path->res_id) {
|
|
|
|
r = &oi->resources[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!r) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("res instance %d not found", path->res_id);
|
2018-04-30 15:02:04 -07:00
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (obj_inst) {
|
|
|
|
*obj_inst = oi;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (obj_field) {
|
|
|
|
*obj_field = of;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (res) {
|
|
|
|
*res = r;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
int lwm2m_engine_create_obj_inst(char *pathstr)
|
|
|
|
{
|
|
|
|
struct lwm2m_obj_path path;
|
|
|
|
struct lwm2m_engine_obj_inst *obj_inst;
|
|
|
|
int ret = 0;
|
|
|
|
|
2019-05-29 16:49:18 -07:00
|
|
|
LOG_DBG("path:%s", log_strdup(pathstr));
|
2017-07-07 11:04:03 -07:00
|
|
|
|
|
|
|
/* translate path -> path_obj */
|
|
|
|
ret = string_to_path(pathstr, &path, '/');
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2019-03-26 19:57:45 -06:00
|
|
|
if (path.level != 2U) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("path must have 2 parts");
|
2017-07-07 11:04:03 -07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return lwm2m_create_obj_inst(path.obj_id, path.obj_inst_id, &obj_inst);
|
|
|
|
}
|
|
|
|
|
2018-04-30 16:43:25 -07:00
|
|
|
int lwm2m_engine_set_res_data(char *pathstr, void *data_ptr, u16_t data_len,
|
|
|
|
u8_t data_flags)
|
|
|
|
{
|
|
|
|
struct lwm2m_obj_path path;
|
|
|
|
struct lwm2m_engine_res_inst *res = NULL;
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
/* translate path -> path_obj */
|
|
|
|
ret = string_to_path(pathstr, &path, '/');
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (path.level < 3) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("path must have 3 parts");
|
2018-04-30 16:43:25 -07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* look up resource obj */
|
|
|
|
ret = path_to_objs(&path, NULL, NULL, &res);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* assign data elements */
|
|
|
|
res->data_ptr = data_ptr;
|
|
|
|
res->data_len = data_len;
|
|
|
|
res->data_flags = data_flags;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
static int lwm2m_engine_set(char *pathstr, void *value, u16_t len)
|
|
|
|
{
|
|
|
|
struct lwm2m_obj_path path;
|
|
|
|
struct lwm2m_engine_obj_inst *obj_inst;
|
|
|
|
struct lwm2m_engine_obj_field *obj_field;
|
|
|
|
struct lwm2m_engine_res_inst *res = NULL;
|
|
|
|
void *data_ptr = NULL;
|
|
|
|
size_t data_len = 0;
|
2018-04-30 15:02:04 -07:00
|
|
|
int ret = 0;
|
|
|
|
bool changed = false;
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2019-05-29 16:49:18 -07:00
|
|
|
LOG_DBG("path:%s, value:%p, len:%d", log_strdup(pathstr), value, len);
|
2017-07-07 11:04:03 -07:00
|
|
|
|
|
|
|
/* translate path -> path_obj */
|
|
|
|
ret = string_to_path(pathstr, &path, '/');
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (path.level < 3) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("path must have 3 parts");
|
2017-07-07 11:04:03 -07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2018-04-30 15:02:04 -07:00
|
|
|
/* look up resource obj */
|
|
|
|
ret = path_to_objs(&path, &obj_inst, &obj_field, &res);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!res) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("res instance %d not found", path.res_id);
|
2017-07-07 11:04:03 -07:00
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
2018-04-30 16:43:25 -07:00
|
|
|
if (LWM2M_HAS_RES_FLAG(res, LWM2M_RES_DATA_FLAG_RO)) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("res data pointer is read-only");
|
2018-04-30 16:43:25 -07:00
|
|
|
return -EACCES;
|
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
/* setup initial data elements */
|
|
|
|
data_ptr = res->data_ptr;
|
|
|
|
data_len = res->data_len;
|
|
|
|
|
|
|
|
/* allow user to override data elements via callback */
|
|
|
|
if (res->pre_write_cb) {
|
|
|
|
data_ptr = res->pre_write_cb(obj_inst->obj_inst_id, &data_len);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!data_ptr) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("res data pointer is NULL");
|
2017-07-07 11:04:03 -07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check length (note: we add 1 to string length for NULL pad) */
|
|
|
|
if (len > res->data_len -
|
|
|
|
(obj_field->data_type == LWM2M_RES_TYPE_STRING ? 1 : 0)) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("length %u is too long for resource %d data",
|
|
|
|
len, path.res_id);
|
2017-07-07 11:04:03 -07:00
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (memcmp(data_ptr, value, len) != 0) {
|
|
|
|
changed = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (obj_field->data_type) {
|
|
|
|
|
2017-10-26 23:45:59 -07:00
|
|
|
case LWM2M_RES_TYPE_OPAQUE:
|
|
|
|
memcpy((u8_t *)data_ptr, value, len);
|
|
|
|
break;
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
case LWM2M_RES_TYPE_STRING:
|
|
|
|
memcpy((u8_t *)data_ptr, value, len);
|
|
|
|
((u8_t *)data_ptr)[len] = '\0';
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_U64:
|
|
|
|
*((u64_t *)data_ptr) = *(u64_t *)value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_U32:
|
|
|
|
case LWM2M_RES_TYPE_TIME:
|
|
|
|
*((u32_t *)data_ptr) = *(u32_t *)value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_U16:
|
|
|
|
*((u16_t *)data_ptr) = *(u16_t *)value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_U8:
|
|
|
|
*((u8_t *)data_ptr) = *(u8_t *)value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_S64:
|
|
|
|
*((s64_t *)data_ptr) = *(s64_t *)value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_S32:
|
|
|
|
*((s32_t *)data_ptr) = *(s32_t *)value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_S16:
|
|
|
|
*((s16_t *)data_ptr) = *(s16_t *)value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_S8:
|
|
|
|
*((s8_t *)data_ptr) = *(s8_t *)value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_BOOL:
|
|
|
|
*((bool *)data_ptr) = *(bool *)value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_FLOAT32:
|
|
|
|
((float32_value_t *)data_ptr)->val1 =
|
|
|
|
((float32_value_t *)value)->val1;
|
|
|
|
((float32_value_t *)data_ptr)->val2 =
|
|
|
|
((float32_value_t *)value)->val2;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_FLOAT64:
|
|
|
|
((float64_value_t *)data_ptr)->val1 =
|
|
|
|
((float64_value_t *)value)->val1;
|
|
|
|
((float64_value_t *)data_ptr)->val2 =
|
|
|
|
((float64_value_t *)value)->val2;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("unknown obj data_type %d", obj_field->data_type);
|
2017-07-07 11:04:03 -07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (res->post_write_cb) {
|
2017-10-27 15:14:42 -07:00
|
|
|
ret = res->post_write_cb(obj_inst->obj_inst_id, data_ptr, len,
|
|
|
|
false, 0);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (changed) {
|
|
|
|
NOTIFY_OBSERVER_PATH(&path);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2017-10-26 23:45:59 -07:00
|
|
|
int lwm2m_engine_set_opaque(char *pathstr, char *data_ptr, u16_t data_len)
|
|
|
|
{
|
|
|
|
return lwm2m_engine_set(pathstr, data_ptr, data_len);
|
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
int lwm2m_engine_set_string(char *pathstr, char *data_ptr)
|
|
|
|
{
|
|
|
|
return lwm2m_engine_set(pathstr, data_ptr, strlen(data_ptr));
|
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_engine_set_u8(char *pathstr, u8_t value)
|
|
|
|
{
|
|
|
|
return lwm2m_engine_set(pathstr, &value, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_engine_set_u16(char *pathstr, u16_t value)
|
|
|
|
{
|
|
|
|
return lwm2m_engine_set(pathstr, &value, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_engine_set_u32(char *pathstr, u32_t value)
|
|
|
|
{
|
|
|
|
return lwm2m_engine_set(pathstr, &value, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_engine_set_u64(char *pathstr, u64_t value)
|
|
|
|
{
|
|
|
|
return lwm2m_engine_set(pathstr, &value, 8);
|
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_engine_set_s8(char *pathstr, s8_t value)
|
|
|
|
{
|
|
|
|
return lwm2m_engine_set(pathstr, &value, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_engine_set_s16(char *pathstr, s16_t value)
|
|
|
|
{
|
|
|
|
return lwm2m_engine_set(pathstr, &value, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_engine_set_s32(char *pathstr, s32_t value)
|
|
|
|
{
|
|
|
|
return lwm2m_engine_set(pathstr, &value, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_engine_set_s64(char *pathstr, s64_t value)
|
|
|
|
{
|
|
|
|
return lwm2m_engine_set(pathstr, &value, 8);
|
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_engine_set_bool(char *pathstr, bool value)
|
|
|
|
{
|
|
|
|
u8_t temp = (value != 0 ? 1 : 0);
|
|
|
|
|
|
|
|
return lwm2m_engine_set(pathstr, &temp, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_engine_set_float32(char *pathstr, float32_value_t *value)
|
|
|
|
{
|
|
|
|
return lwm2m_engine_set(pathstr, value, sizeof(float32_value_t));
|
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_engine_set_float64(char *pathstr, float64_value_t *value)
|
|
|
|
{
|
|
|
|
return lwm2m_engine_set(pathstr, value, sizeof(float64_value_t));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* user data getter functions */
|
|
|
|
|
2018-04-30 16:43:25 -07:00
|
|
|
int lwm2m_engine_get_res_data(char *pathstr, void **data_ptr, u16_t *data_len,
|
|
|
|
u8_t *data_flags)
|
|
|
|
{
|
|
|
|
struct lwm2m_obj_path path;
|
|
|
|
struct lwm2m_engine_res_inst *res = NULL;
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
/* translate path -> path_obj */
|
|
|
|
ret = string_to_path(pathstr, &path, '/');
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (path.level < 3) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("path must have 3 parts");
|
2018-04-30 16:43:25 -07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* look up resource obj */
|
|
|
|
ret = path_to_objs(&path, NULL, NULL, &res);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
*data_ptr = res->data_ptr;
|
|
|
|
*data_len = res->data_len;
|
|
|
|
*data_flags = res->data_flags;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
static int lwm2m_engine_get(char *pathstr, void *buf, u16_t buflen)
|
|
|
|
{
|
2018-04-30 15:02:04 -07:00
|
|
|
int ret = 0;
|
2017-07-07 11:04:03 -07:00
|
|
|
struct lwm2m_obj_path path;
|
|
|
|
struct lwm2m_engine_obj_inst *obj_inst;
|
|
|
|
struct lwm2m_engine_obj_field *obj_field;
|
|
|
|
struct lwm2m_engine_res_inst *res = NULL;
|
|
|
|
void *data_ptr = NULL;
|
|
|
|
size_t data_len = 0;
|
|
|
|
|
2019-05-29 16:49:18 -07:00
|
|
|
LOG_DBG("path:%s, buf:%p, buflen:%d", log_strdup(pathstr), buf, buflen);
|
2017-07-07 11:04:03 -07:00
|
|
|
|
|
|
|
/* translate path -> path_obj */
|
|
|
|
ret = string_to_path(pathstr, &path, '/');
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (path.level < 3) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("path must have 3 parts");
|
2017-07-07 11:04:03 -07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2018-04-30 15:02:04 -07:00
|
|
|
/* look up resource obj */
|
|
|
|
ret = path_to_objs(&path, &obj_inst, &obj_field, &res);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!res) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("res instance %d not found", path.res_id);
|
2017-07-07 11:04:03 -07:00
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* setup initial data elements */
|
|
|
|
data_ptr = res->data_ptr;
|
|
|
|
data_len = res->data_len;
|
|
|
|
|
|
|
|
/* allow user to override data elements via callback */
|
|
|
|
if (res->read_cb) {
|
|
|
|
data_ptr = res->read_cb(obj_inst->obj_inst_id, &data_len);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO: handle data_len > buflen case */
|
|
|
|
|
|
|
|
if (data_ptr && data_len > 0) {
|
|
|
|
switch (obj_field->data_type) {
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_OPAQUE:
|
2017-10-26 23:45:59 -07:00
|
|
|
if (data_len > buflen) {
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(buf, data_ptr, data_len);
|
2017-07-07 11:04:03 -07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_STRING:
|
|
|
|
strncpy((u8_t *)buf, (u8_t *)data_ptr, buflen);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_U64:
|
|
|
|
*(u64_t *)buf = *(u64_t *)data_ptr;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_U32:
|
|
|
|
case LWM2M_RES_TYPE_TIME:
|
|
|
|
*(u32_t *)buf = *(u32_t *)data_ptr;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_U16:
|
|
|
|
*(u16_t *)buf = *(u16_t *)data_ptr;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_U8:
|
|
|
|
*(u8_t *)buf = *(u8_t *)data_ptr;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_S64:
|
|
|
|
*(s64_t *)buf = *(s64_t *)data_ptr;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_S32:
|
|
|
|
*(s32_t *)buf = *(s32_t *)data_ptr;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_S16:
|
|
|
|
*(s16_t *)buf = *(s16_t *)data_ptr;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_S8:
|
|
|
|
*(s8_t *)buf = *(s8_t *)data_ptr;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_BOOL:
|
|
|
|
*(bool *)buf = *(bool *)data_ptr;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_FLOAT32:
|
|
|
|
((float32_value_t *)buf)->val1 =
|
|
|
|
((float32_value_t *)data_ptr)->val1;
|
|
|
|
((float32_value_t *)buf)->val2 =
|
|
|
|
((float32_value_t *)data_ptr)->val2;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_FLOAT64:
|
|
|
|
((float64_value_t *)buf)->val1 =
|
|
|
|
((float64_value_t *)data_ptr)->val1;
|
|
|
|
((float64_value_t *)buf)->val2 =
|
|
|
|
((float64_value_t *)data_ptr)->val2;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("unknown obj data_type %d",
|
|
|
|
obj_field->data_type);
|
2017-07-07 11:04:03 -07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-10-26 23:45:59 -07:00
|
|
|
int lwm2m_engine_get_opaque(char *pathstr, void *buf, u16_t buflen)
|
|
|
|
{
|
|
|
|
return lwm2m_engine_get(pathstr, buf, buflen);
|
|
|
|
}
|
|
|
|
|
2017-10-11 12:48:06 -07:00
|
|
|
int lwm2m_engine_get_string(char *pathstr, void *buf, u16_t buflen)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
2017-10-11 12:48:06 -07:00
|
|
|
return lwm2m_engine_get(pathstr, buf, buflen);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2018-04-30 16:38:36 -07:00
|
|
|
int lwm2m_engine_get_u8(char *pathstr, u8_t *value)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
2018-04-30 16:38:36 -07:00
|
|
|
return lwm2m_engine_get(pathstr, value, 1);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2018-04-30 16:38:36 -07:00
|
|
|
int lwm2m_engine_get_u16(char *pathstr, u16_t *value)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
2018-04-30 16:38:36 -07:00
|
|
|
return lwm2m_engine_get(pathstr, value, 2);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2018-04-30 16:38:36 -07:00
|
|
|
int lwm2m_engine_get_u32(char *pathstr, u32_t *value)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
2018-04-30 16:38:36 -07:00
|
|
|
return lwm2m_engine_get(pathstr, value, 4);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2018-04-30 16:38:36 -07:00
|
|
|
int lwm2m_engine_get_u64(char *pathstr, u64_t *value)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
2018-04-30 16:38:36 -07:00
|
|
|
return lwm2m_engine_get(pathstr, value, 8);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2018-04-30 16:38:36 -07:00
|
|
|
int lwm2m_engine_get_s8(char *pathstr, s8_t *value)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
2018-04-30 16:38:36 -07:00
|
|
|
return lwm2m_engine_get(pathstr, value, 1);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2018-04-30 16:38:36 -07:00
|
|
|
int lwm2m_engine_get_s16(char *pathstr, s16_t *value)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
2018-04-30 16:38:36 -07:00
|
|
|
return lwm2m_engine_get(pathstr, value, 2);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2018-04-30 16:38:36 -07:00
|
|
|
int lwm2m_engine_get_s32(char *pathstr, s32_t *value)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
2018-04-30 16:38:36 -07:00
|
|
|
return lwm2m_engine_get(pathstr, value, 4);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2018-04-30 16:38:36 -07:00
|
|
|
int lwm2m_engine_get_s64(char *pathstr, s64_t *value)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
2018-04-30 16:38:36 -07:00
|
|
|
return lwm2m_engine_get(pathstr, value, 8);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2018-04-30 16:38:36 -07:00
|
|
|
int lwm2m_engine_get_bool(char *pathstr, bool *value)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
2018-04-30 16:38:36 -07:00
|
|
|
int ret = 0;
|
|
|
|
s8_t temp = 0;
|
|
|
|
|
|
|
|
ret = lwm2m_engine_get_s8(pathstr, &temp);
|
|
|
|
if (!ret) {
|
|
|
|
*value = temp != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2018-04-30 16:38:36 -07:00
|
|
|
int lwm2m_engine_get_float32(char *pathstr, float32_value_t *buf)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
|
|
|
return lwm2m_engine_get(pathstr, buf, sizeof(float32_value_t));
|
|
|
|
}
|
|
|
|
|
2018-04-30 16:38:36 -07:00
|
|
|
int lwm2m_engine_get_float64(char *pathstr, float64_value_t *buf)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
|
|
|
return lwm2m_engine_get(pathstr, buf, sizeof(float64_value_t));
|
|
|
|
}
|
|
|
|
|
2017-10-26 23:41:17 -07:00
|
|
|
int lwm2m_engine_get_resource(char *pathstr, struct lwm2m_engine_res_inst **res)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
struct lwm2m_obj_path path;
|
|
|
|
|
|
|
|
ret = string_to_path(pathstr, &path, '/');
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (path.level < 3) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("path must have 3 parts");
|
2017-07-07 11:04:03 -07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2018-04-30 15:02:04 -07:00
|
|
|
return path_to_objs(&path, NULL, NULL, res);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_engine_register_read_callback(char *pathstr,
|
|
|
|
lwm2m_engine_get_data_cb_t cb)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
struct lwm2m_engine_res_inst *res = NULL;
|
|
|
|
|
2017-10-26 23:41:17 -07:00
|
|
|
ret = lwm2m_engine_get_resource(pathstr, &res);
|
2017-07-07 11:04:03 -07:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
res->read_cb = cb;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_engine_register_pre_write_callback(char *pathstr,
|
|
|
|
lwm2m_engine_get_data_cb_t cb)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
struct lwm2m_engine_res_inst *res = NULL;
|
|
|
|
|
2017-10-26 23:41:17 -07:00
|
|
|
ret = lwm2m_engine_get_resource(pathstr, &res);
|
2017-07-07 11:04:03 -07:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
res->pre_write_cb = cb;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_engine_register_post_write_callback(char *pathstr,
|
|
|
|
lwm2m_engine_set_data_cb_t cb)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
struct lwm2m_engine_res_inst *res = NULL;
|
|
|
|
|
2017-10-26 23:41:17 -07:00
|
|
|
ret = lwm2m_engine_get_resource(pathstr, &res);
|
2017-07-07 11:04:03 -07:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
res->post_write_cb = cb;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_engine_register_exec_callback(char *pathstr,
|
2018-07-12 10:00:00 -07:00
|
|
|
lwm2m_engine_user_cb_t cb)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
struct lwm2m_engine_res_inst *res = NULL;
|
|
|
|
|
2017-10-26 23:41:17 -07:00
|
|
|
ret = lwm2m_engine_get_resource(pathstr, &res);
|
2017-07-07 11:04:03 -07:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
res->execute_cb = cb;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-07-12 10:00:01 -07:00
|
|
|
int lwm2m_engine_register_create_callback(u16_t obj_id,
|
|
|
|
lwm2m_engine_user_cb_t cb)
|
|
|
|
{
|
|
|
|
struct lwm2m_engine_obj *obj = NULL;
|
|
|
|
|
|
|
|
obj = get_engine_obj(obj_id);
|
|
|
|
if (!obj) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("unable to find obj: %u", obj_id);
|
2018-07-12 10:00:01 -07:00
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
obj->user_create_cb = cb;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_engine_register_delete_callback(u16_t obj_id,
|
|
|
|
lwm2m_engine_user_cb_t cb)
|
|
|
|
{
|
|
|
|
struct lwm2m_engine_obj *obj = NULL;
|
|
|
|
|
|
|
|
obj = get_engine_obj(obj_id);
|
|
|
|
if (!obj) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("unable to find obj: %u", obj_id);
|
2018-07-12 10:00:01 -07:00
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
obj->user_delete_cb = cb;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
/* generic data handlers */
|
|
|
|
|
|
|
|
static int lwm2m_read_handler(struct lwm2m_engine_obj_inst *obj_inst,
|
|
|
|
struct lwm2m_engine_res_inst *res,
|
|
|
|
struct lwm2m_engine_obj_field *obj_field,
|
2019-01-24 14:40:23 -08:00
|
|
|
struct lwm2m_message *msg)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
|
|
|
int i, loop_max = 1;
|
2018-11-29 11:23:03 -08:00
|
|
|
u16_t res_inst_id_tmp = 0U;
|
2017-07-07 11:04:03 -07:00
|
|
|
void *data_ptr = NULL;
|
|
|
|
size_t data_len = 0;
|
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
if (!obj_inst || !res || !obj_field || !msg) {
|
2017-07-07 11:04:03 -07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* setup initial data elements */
|
|
|
|
data_ptr = res->data_ptr;
|
|
|
|
data_len = res->data_len;
|
|
|
|
|
|
|
|
/* allow user to override data elements via callback */
|
|
|
|
if (res->read_cb) {
|
|
|
|
data_ptr = res->read_cb(obj_inst->obj_inst_id, &data_len);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!data_ptr || data_len == 0) {
|
2018-04-30 15:20:19 -07:00
|
|
|
return -ENOENT;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (res->multi_count_var != NULL) {
|
2018-02-05 15:29:24 -08:00
|
|
|
/* if multi_count_var is 0 (none assigned) return NOT_FOUND */
|
2019-03-26 19:57:45 -06:00
|
|
|
if (*res->multi_count_var == 0U) {
|
2018-02-05 15:29:24 -08:00
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_put_begin_ri(&msg->out, &msg->path);
|
2017-07-07 11:04:03 -07:00
|
|
|
loop_max = *res->multi_count_var;
|
2019-01-24 14:40:23 -08:00
|
|
|
res_inst_id_tmp = msg->path.res_inst_id;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < loop_max; i++) {
|
|
|
|
if (res->multi_count_var != NULL) {
|
2019-01-24 14:40:23 -08:00
|
|
|
msg->path.res_inst_id = (u16_t) i;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
switch (obj_field->data_type) {
|
|
|
|
|
|
|
|
/* do nothing for OPAQUE (probably has a callback) */
|
|
|
|
case LWM2M_RES_TYPE_OPAQUE:
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* TODO: handle multi count for string? */
|
|
|
|
case LWM2M_RES_TYPE_STRING:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_put_string(&msg->out, &msg->path,
|
|
|
|
(u8_t *)data_ptr,
|
2017-07-07 11:04:03 -07:00
|
|
|
strlen((u8_t *)data_ptr));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_U64:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_put_s64(&msg->out, &msg->path,
|
2017-07-07 11:04:03 -07:00
|
|
|
(s64_t)((u64_t *)data_ptr)[i]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_U32:
|
|
|
|
case LWM2M_RES_TYPE_TIME:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_put_s32(&msg->out, &msg->path,
|
2017-07-07 11:04:03 -07:00
|
|
|
(s32_t)((u32_t *)data_ptr)[i]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_U16:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_put_s16(&msg->out, &msg->path,
|
2017-07-07 11:04:03 -07:00
|
|
|
(s16_t)((u16_t *)data_ptr)[i]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_U8:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_put_s8(&msg->out, &msg->path,
|
2017-07-07 11:04:03 -07:00
|
|
|
(s8_t)((u8_t *)data_ptr)[i]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_S64:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_put_s64(&msg->out, &msg->path,
|
2017-07-07 11:04:03 -07:00
|
|
|
((s64_t *)data_ptr)[i]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_S32:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_put_s32(&msg->out, &msg->path,
|
2017-07-07 11:04:03 -07:00
|
|
|
((s32_t *)data_ptr)[i]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_S16:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_put_s16(&msg->out, &msg->path,
|
2017-07-07 11:04:03 -07:00
|
|
|
((s16_t *)data_ptr)[i]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_S8:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_put_s8(&msg->out, &msg->path,
|
2017-07-07 11:04:03 -07:00
|
|
|
((s8_t *)data_ptr)[i]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_BOOL:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_put_bool(&msg->out, &msg->path,
|
2017-07-07 11:04:03 -07:00
|
|
|
((bool *)data_ptr)[i]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_FLOAT32:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_put_float32fix(&msg->out, &msg->path,
|
2017-07-07 11:04:03 -07:00
|
|
|
&((float32_value_t *)data_ptr)[i]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_FLOAT64:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_put_float64fix(&msg->out, &msg->path,
|
2017-07-07 11:04:03 -07:00
|
|
|
&((float64_value_t *)data_ptr)[i]);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("unknown obj data_type %d",
|
|
|
|
obj_field->data_type);
|
2017-07-07 11:04:03 -07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (res->multi_count_var != NULL) {
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_put_end_ri(&msg->out, &msg->path);
|
|
|
|
msg->path.res_inst_id = res_inst_id_tmp;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-10-26 23:32:47 -07:00
|
|
|
size_t lwm2m_engine_get_opaque_more(struct lwm2m_input_context *in,
|
|
|
|
u8_t *buf, size_t buflen, bool *last_block)
|
|
|
|
{
|
|
|
|
u16_t in_len = in->opaque_len;
|
|
|
|
|
|
|
|
if (in_len > buflen) {
|
|
|
|
in_len = buflen;
|
|
|
|
}
|
|
|
|
|
|
|
|
in->opaque_len -= in_len;
|
2019-03-26 19:57:45 -06:00
|
|
|
if (in->opaque_len == 0U) {
|
2017-10-26 23:32:47 -07:00
|
|
|
*last_block = true;
|
|
|
|
}
|
|
|
|
|
2019-01-25 14:13:07 -08:00
|
|
|
if (buf_read(buf, in_len, CPKT_BUF_READ(in->in_cpkt),
|
|
|
|
&in->offset) < 0) {
|
2017-10-26 23:32:47 -07:00
|
|
|
*last_block = true;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (size_t)in_len;
|
|
|
|
}
|
|
|
|
|
2017-10-26 23:45:59 -07:00
|
|
|
static int lwm2m_write_handler_opaque(struct lwm2m_engine_obj_inst *obj_inst,
|
|
|
|
struct lwm2m_engine_res_inst *res,
|
|
|
|
struct lwm2m_input_context *in,
|
|
|
|
void *data_ptr, size_t data_len,
|
|
|
|
bool last_block, size_t total_size)
|
|
|
|
{
|
|
|
|
size_t len = 1;
|
|
|
|
bool last_pkt_block = false, first_read = true;
|
2017-10-27 15:14:42 -07:00
|
|
|
int ret = 0;
|
2017-10-26 23:45:59 -07:00
|
|
|
|
|
|
|
while (!last_pkt_block && len > 0) {
|
|
|
|
if (first_read) {
|
|
|
|
len = engine_get_opaque(in, (u8_t *)data_ptr,
|
|
|
|
data_len, &last_pkt_block);
|
2019-01-25 22:36:47 -08:00
|
|
|
if (len == 0) {
|
|
|
|
/* ignore empty content and continue */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-10-26 23:45:59 -07:00
|
|
|
first_read = false;
|
|
|
|
} else {
|
|
|
|
len = lwm2m_engine_get_opaque_more(in, (u8_t *)data_ptr,
|
|
|
|
data_len,
|
|
|
|
&last_pkt_block);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (len == 0) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (res->post_write_cb) {
|
2017-10-27 15:14:42 -07:00
|
|
|
ret = res->post_write_cb(obj_inst->obj_inst_id,
|
|
|
|
data_ptr, len,
|
|
|
|
last_pkt_block && last_block,
|
|
|
|
total_size);
|
2017-11-01 14:36:50 +08:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
2017-10-26 23:45:59 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-27 15:14:42 -07:00
|
|
|
return ret;
|
2017-10-26 23:45:59 -07:00
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
/* This function is exposed for the content format writers */
|
|
|
|
int lwm2m_write_handler(struct lwm2m_engine_obj_inst *obj_inst,
|
|
|
|
struct lwm2m_engine_res_inst *res,
|
|
|
|
struct lwm2m_engine_obj_field *obj_field,
|
2019-01-24 14:40:23 -08:00
|
|
|
struct lwm2m_message *msg)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
2019-01-25 14:13:07 -08:00
|
|
|
struct block_context *block_ctx = NULL;
|
2017-07-07 11:04:03 -07:00
|
|
|
void *data_ptr = NULL;
|
|
|
|
size_t data_len = 0;
|
|
|
|
size_t len = 0;
|
2017-07-25 16:54:25 +08:00
|
|
|
size_t total_size = 0;
|
2019-01-25 14:13:07 -08:00
|
|
|
s64_t temp64 = 0;
|
|
|
|
s32_t temp32 = 0;
|
2017-07-25 16:54:25 +08:00
|
|
|
int ret = 0;
|
2017-09-19 16:04:32 -07:00
|
|
|
u8_t token[8];
|
2019-01-25 14:13:07 -08:00
|
|
|
u8_t tkl = 0U;
|
2017-07-25 16:54:25 +08:00
|
|
|
bool last_block = true;
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
if (!obj_inst || !res || !obj_field || !msg) {
|
2017-07-07 11:04:03 -07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2018-04-30 16:43:25 -07:00
|
|
|
if (LWM2M_HAS_RES_FLAG(res, LWM2M_RES_DATA_FLAG_RO)) {
|
|
|
|
return -EACCES;
|
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
/* setup initial data elements */
|
|
|
|
data_ptr = res->data_ptr;
|
|
|
|
data_len = res->data_len;
|
|
|
|
|
|
|
|
/* allow user to override data elements via callback */
|
|
|
|
if (res->pre_write_cb) {
|
|
|
|
data_ptr = res->pre_write_cb(obj_inst->obj_inst_id, &data_len);
|
|
|
|
}
|
|
|
|
|
2017-10-26 23:45:59 -07:00
|
|
|
if (res->post_write_cb) {
|
|
|
|
/* Get block1 option for checking MORE block flag */
|
2019-01-24 14:40:23 -08:00
|
|
|
ret = get_option_int(msg->in.in_cpkt, COAP_OPTION_BLOCK1);
|
2017-10-26 23:45:59 -07:00
|
|
|
if (ret >= 0) {
|
|
|
|
last_block = !GET_MORE(ret);
|
|
|
|
|
|
|
|
/* Get block_ctx for total_size (might be zero) */
|
2019-01-24 14:40:23 -08:00
|
|
|
tkl = coap_header_get_token(msg->in.in_cpkt, token);
|
2017-10-26 23:45:59 -07:00
|
|
|
if (tkl && !get_block_ctx(token, tkl, &block_ctx)) {
|
|
|
|
total_size = block_ctx->ctx.total_size;
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_DBG("BLOCK1: total:%zu current:%zu"
|
|
|
|
" last:%u",
|
|
|
|
block_ctx->ctx.total_size,
|
|
|
|
block_ctx->ctx.current,
|
|
|
|
last_block);
|
2017-10-26 23:45:59 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
if (data_ptr && data_len > 0) {
|
|
|
|
switch (obj_field->data_type) {
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_OPAQUE:
|
2019-01-24 14:40:23 -08:00
|
|
|
ret = lwm2m_write_handler_opaque(obj_inst, res,
|
|
|
|
&msg->in,
|
2017-10-26 23:45:59 -07:00
|
|
|
data_ptr, data_len,
|
|
|
|
last_block,
|
|
|
|
total_size);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_STRING:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_get_string(&msg->in, (u8_t *)data_ptr, data_len);
|
2017-07-07 11:04:03 -07:00
|
|
|
len = strlen((char *)data_ptr);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_U64:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_get_s64(&msg->in, &temp64);
|
2017-07-07 11:04:03 -07:00
|
|
|
*(u64_t *)data_ptr = temp64;
|
|
|
|
len = 8;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_U32:
|
|
|
|
case LWM2M_RES_TYPE_TIME:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_get_s32(&msg->in, &temp32);
|
2017-07-07 11:04:03 -07:00
|
|
|
*(u32_t *)data_ptr = temp32;
|
|
|
|
len = 4;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_U16:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_get_s32(&msg->in, &temp32);
|
2017-07-07 11:04:03 -07:00
|
|
|
*(u16_t *)data_ptr = temp32;
|
|
|
|
len = 2;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_U8:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_get_s32(&msg->in, &temp32);
|
2017-07-07 11:04:03 -07:00
|
|
|
*(u8_t *)data_ptr = temp32;
|
|
|
|
len = 1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_S64:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_get_s64(&msg->in, (s64_t *)data_ptr);
|
2017-07-07 11:04:03 -07:00
|
|
|
len = 8;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_S32:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_get_s32(&msg->in, (s32_t *)data_ptr);
|
2017-07-07 11:04:03 -07:00
|
|
|
len = 4;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_S16:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_get_s32(&msg->in, &temp32);
|
2017-07-07 11:04:03 -07:00
|
|
|
*(s16_t *)data_ptr = temp32;
|
|
|
|
len = 2;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_S8:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_get_s32(&msg->in, &temp32);
|
2017-07-07 11:04:03 -07:00
|
|
|
*(s8_t *)data_ptr = temp32;
|
|
|
|
len = 1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_BOOL:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_get_bool(&msg->in, (bool *)data_ptr);
|
2017-07-07 11:04:03 -07:00
|
|
|
len = 1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_FLOAT32:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_get_float32fix(&msg->in,
|
2017-07-07 11:04:03 -07:00
|
|
|
(float32_value_t *)data_ptr);
|
2019-02-21 00:52:00 -08:00
|
|
|
len = sizeof(float32_value_t);
|
2017-07-07 11:04:03 -07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_RES_TYPE_FLOAT64:
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_get_float64fix(&msg->in,
|
2017-07-07 11:04:03 -07:00
|
|
|
(float64_value_t *)data_ptr);
|
2019-02-21 00:52:00 -08:00
|
|
|
len = sizeof(float64_value_t);
|
2017-07-07 11:04:03 -07:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("unknown obj data_type %d",
|
|
|
|
obj_field->data_type);
|
2017-07-07 11:04:03 -07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
}
|
2018-04-30 15:20:19 -07:00
|
|
|
} else {
|
|
|
|
return -ENOENT;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2017-10-26 23:45:59 -07:00
|
|
|
if (res->post_write_cb &&
|
|
|
|
obj_field->data_type != LWM2M_RES_TYPE_OPAQUE) {
|
2017-07-25 16:54:25 +08:00
|
|
|
ret = res->post_write_cb(obj_inst->obj_inst_id, data_ptr, len,
|
|
|
|
last_block, total_size);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
NOTIFY_OBSERVER_PATH(&msg->path);
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2017-07-25 16:54:25 +08:00
|
|
|
return ret;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static int lwm2m_write_attr_handler(struct lwm2m_engine_obj *obj,
|
2019-01-24 14:40:23 -08:00
|
|
|
struct lwm2m_message *msg)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
2017-12-12 16:24:03 +08:00
|
|
|
bool update_observe_node = false;
|
2017-11-24 18:47:00 +08:00
|
|
|
char opt_buf[COAP_OPTION_BUF_LEN];
|
2018-05-07 15:05:04 -07:00
|
|
|
int nr_opt, i, ret = 0;
|
2017-11-24 18:47:00 +08:00
|
|
|
struct coap_option options[NR_LWM2M_ATTR];
|
|
|
|
struct lwm2m_engine_obj_inst *obj_inst = NULL;
|
|
|
|
struct lwm2m_engine_res_inst *res = NULL;
|
2018-05-07 15:05:04 -07:00
|
|
|
struct lwm2m_attr *attr;
|
2017-11-24 18:47:00 +08:00
|
|
|
struct notification_attrs nattrs = { 0 };
|
2017-12-12 16:24:03 +08:00
|
|
|
struct observe_node *obs;
|
2018-11-29 11:23:03 -08:00
|
|
|
u8_t type = 0U;
|
2017-11-24 18:47:00 +08:00
|
|
|
void *nattr_ptrs[NR_LWM2M_ATTR] = {
|
|
|
|
&nattrs.pmin, &nattrs.pmax, &nattrs.gt, &nattrs.lt, &nattrs.st
|
|
|
|
};
|
2018-05-07 15:05:04 -07:00
|
|
|
void *ref;
|
2017-11-24 18:47:00 +08:00
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
if (!obj || !msg) {
|
2017-07-07 11:04:03 -07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2017-11-24 18:47:00 +08:00
|
|
|
/* do not expose security obj */
|
|
|
|
if (obj->obj_id == LWM2M_OBJECT_SECURITY_ID) {
|
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
nr_opt = coap_find_options(msg->in.in_cpkt, COAP_OPTION_URI_QUERY,
|
2017-11-24 18:47:00 +08:00
|
|
|
options, NR_LWM2M_ATTR);
|
|
|
|
if (nr_opt <= 0) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("No attribute found!");
|
2017-11-24 18:47:00 +08:00
|
|
|
/* translate as bad request */
|
|
|
|
return -EEXIST;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* get lwm2m_attr slist */
|
2019-03-26 19:57:45 -06:00
|
|
|
if (msg->path.level == 3U) {
|
2019-01-24 14:40:23 -08:00
|
|
|
ret = path_to_objs(&msg->path, NULL, NULL, &res);
|
2017-11-24 18:47:00 +08:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2018-05-07 15:05:04 -07:00
|
|
|
ref = res;
|
2019-03-26 19:57:45 -06:00
|
|
|
} else if (msg->path.level == 1U) {
|
2018-05-07 15:05:04 -07:00
|
|
|
ref = obj;
|
2019-03-26 19:57:45 -06:00
|
|
|
} else if (msg->path.level == 2U) {
|
2019-01-24 14:40:23 -08:00
|
|
|
obj_inst = get_engine_obj_inst(msg->path.obj_id,
|
|
|
|
msg->path.obj_inst_id);
|
2017-11-24 18:47:00 +08:00
|
|
|
if (!obj_inst) {
|
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
2018-05-07 15:05:04 -07:00
|
|
|
ref = obj_inst;
|
2017-11-24 18:47:00 +08:00
|
|
|
} else {
|
|
|
|
/* bad request */
|
|
|
|
return -EEXIST;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* retrieve existing attributes */
|
2018-05-07 15:05:04 -07:00
|
|
|
ret = update_attrs(ref, &nattrs);
|
2018-05-07 14:18:37 -07:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
2017-11-24 18:47:00 +08:00
|
|
|
|
|
|
|
/* loop through options to parse attribute */
|
2018-05-07 15:05:04 -07:00
|
|
|
for (i = 0; i < nr_opt; i++) {
|
2019-02-11 17:14:19 +00:00
|
|
|
int limit = MIN(options[i].len, 5), plen = 0, vlen;
|
2017-11-24 18:47:00 +08:00
|
|
|
float32_value_t val = { 0 };
|
2018-11-29 11:23:03 -08:00
|
|
|
type = 0U;
|
2017-11-24 18:47:00 +08:00
|
|
|
|
|
|
|
/* search for '=' */
|
|
|
|
while (plen < limit && options[i].value[plen] != '=') {
|
|
|
|
plen += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* either length = 2(gt/lt/st) or = 4(pmin/pmax) */
|
|
|
|
if (plen != 2 && plen != 4) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* matching attribute name */
|
2018-11-29 11:23:03 -08:00
|
|
|
for (type = 0U; type < NR_LWM2M_ATTR; type++) {
|
2017-11-24 18:47:00 +08:00
|
|
|
if (LWM2M_ATTR_LEN[type] == plen &&
|
|
|
|
!memcmp(options[i].value, LWM2M_ATTR_STR[type],
|
|
|
|
LWM2M_ATTR_LEN[type])) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* unrecognized attribute */
|
|
|
|
if (type == NR_LWM2M_ATTR) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* unset attribute when no value's given */
|
|
|
|
if (options[i].len == plen) {
|
|
|
|
nattrs.flags &= ~BIT(type);
|
|
|
|
|
2018-09-11 19:09:03 -07:00
|
|
|
(void)memset(nattr_ptrs[type], 0,
|
|
|
|
type <= LWM2M_ATTR_PMAX ? sizeof(s32_t) :
|
|
|
|
sizeof(float32_value_t));
|
2017-11-24 18:47:00 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* gt/lt/st cannot be assigned to obj/obj_inst unless unset */
|
2019-03-26 19:57:45 -06:00
|
|
|
if (plen == 2 && msg->path.level <= 2U) {
|
2017-11-24 18:47:00 +08:00
|
|
|
return -EEXIST;
|
|
|
|
}
|
|
|
|
|
|
|
|
vlen = options[i].len - plen - 1;
|
|
|
|
memcpy(opt_buf, options[i].value + plen + 1, vlen);
|
|
|
|
opt_buf[vlen] = '\0';
|
|
|
|
|
|
|
|
/* convert value to integer or float */
|
|
|
|
if (plen == 4) {
|
|
|
|
char *end;
|
|
|
|
long int v;
|
|
|
|
|
|
|
|
/* pmin/pmax: integer (sec 5.1.2)
|
|
|
|
* however, negative is non-sense
|
|
|
|
*/
|
|
|
|
errno = 0;
|
|
|
|
v = strtol(opt_buf, &end, 10);
|
2019-02-13 13:37:25 -08:00
|
|
|
if (errno || *end || v < 0) {
|
2017-11-24 18:47:00 +08:00
|
|
|
ret = -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
val.val1 = v;
|
|
|
|
} else {
|
|
|
|
/* gt/lt/st: type float */
|
|
|
|
ret = atof32(opt_buf, &val);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ret < 0) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("invalid attr[%s] value",
|
2019-05-29 16:49:18 -07:00
|
|
|
log_strdup(LWM2M_ATTR_STR[type]));
|
2017-11-24 18:47:00 +08:00
|
|
|
/* bad request */
|
|
|
|
return -EEXIST;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (type <= LWM2M_ATTR_PMAX) {
|
|
|
|
*(s32_t *)nattr_ptrs[type] = val.val1;
|
|
|
|
} else {
|
|
|
|
memcpy(nattr_ptrs[type], &val, sizeof(float32_value_t));
|
|
|
|
}
|
|
|
|
|
|
|
|
nattrs.flags |= BIT(type);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((nattrs.flags & (BIT(LWM2M_ATTR_PMIN) | BIT(LWM2M_ATTR_PMAX))) &&
|
|
|
|
nattrs.pmin > nattrs.pmax) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_DBG("pmin (%d) > pmax (%d)", nattrs.pmin, nattrs.pmax);
|
2017-11-24 18:47:00 +08:00
|
|
|
return -EEXIST;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nattrs.flags & (BIT(LWM2M_ATTR_LT) | BIT(LWM2M_ATTR_GT))) {
|
|
|
|
if (!((nattrs.lt.val1 < nattrs.gt.val1) ||
|
|
|
|
(nattrs.lt.val2 < nattrs.gt.val2))) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_DBG("lt > gt");
|
2017-11-24 18:47:00 +08:00
|
|
|
return -EEXIST;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nattrs.flags & BIT(LWM2M_ATTR_STEP)) {
|
|
|
|
s32_t st1 = nattrs.st.val1 * 2 +
|
|
|
|
nattrs.st.val2 * 2 / 1000000;
|
|
|
|
s32_t st2 = nattrs.st.val2 * 2 % 1000000;
|
|
|
|
if (!(((nattrs.lt.val1 + st1) < nattrs.gt.val1) ||
|
|
|
|
((nattrs.lt.val2 + st2) < nattrs.gt.val2))) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_DBG("lt + 2*st > gt");
|
2017-11-24 18:47:00 +08:00
|
|
|
return -EEXIST;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-07 15:05:04 -07:00
|
|
|
/* find matching attributes */
|
|
|
|
for (i = 0; i < CONFIG_LWM2M_NUM_ATTR; i++) {
|
|
|
|
if (ref != write_attr_pool[i].ref) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
attr = write_attr_pool + i;
|
2017-11-24 18:47:00 +08:00
|
|
|
type = attr->type;
|
|
|
|
|
|
|
|
if (!(BIT(type) & nattrs.flags)) {
|
2019-05-29 16:49:18 -07:00
|
|
|
LOG_DBG("Unset attr %s",
|
|
|
|
log_strdup(LWM2M_ATTR_STR[type]));
|
2018-09-11 19:09:03 -07:00
|
|
|
(void)memset(attr, 0, sizeof(*attr));
|
2017-12-12 16:24:03 +08:00
|
|
|
|
|
|
|
if (type <= LWM2M_ATTR_PMAX) {
|
|
|
|
update_observe_node = true;
|
|
|
|
}
|
|
|
|
|
2017-11-24 18:47:00 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
nattrs.flags &= ~BIT(type);
|
|
|
|
|
|
|
|
if (type <= LWM2M_ATTR_PMAX) {
|
2017-12-12 16:24:03 +08:00
|
|
|
if (attr->int_val == *(s32_t *)nattr_ptrs[type]) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-11-24 18:47:00 +08:00
|
|
|
attr->int_val = *(s32_t *)nattr_ptrs[type];
|
2017-12-12 16:24:03 +08:00
|
|
|
update_observe_node = true;
|
2017-11-24 18:47:00 +08:00
|
|
|
} else {
|
2017-12-12 16:24:03 +08:00
|
|
|
if (!memcmp(&attr->float_val, nattr_ptrs[type],
|
|
|
|
sizeof(float32_value_t))) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-11-24 18:47:00 +08:00
|
|
|
memcpy(&attr->float_val, nattr_ptrs[type],
|
|
|
|
sizeof(float32_value_t));
|
|
|
|
}
|
|
|
|
|
2019-05-29 16:49:18 -07:00
|
|
|
LOG_DBG("Update %s to %d.%06d",
|
|
|
|
log_strdup(LWM2M_ATTR_STR[type]),
|
2018-09-19 11:22:19 +03:00
|
|
|
attr->float_val.val1, attr->float_val.val2);
|
2017-11-24 18:47:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* add attribute to obj/obj_inst/res */
|
2018-11-29 11:23:03 -08:00
|
|
|
for (type = 0U; nattrs.flags && type < NR_LWM2M_ATTR; type++) {
|
2017-11-24 18:47:00 +08:00
|
|
|
if (!(BIT(type) & nattrs.flags)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* grab an entry for newly added attribute */
|
|
|
|
for (i = 0; i < CONFIG_LWM2M_NUM_ATTR; i++) {
|
2018-05-07 15:05:04 -07:00
|
|
|
if (!write_attr_pool[i].ref) {
|
2017-11-24 18:47:00 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i == CONFIG_LWM2M_NUM_ATTR) {
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
attr = write_attr_pool + i;
|
|
|
|
attr->type = type;
|
2018-05-07 15:05:04 -07:00
|
|
|
attr->ref = ref;
|
|
|
|
|
2017-11-24 18:47:00 +08:00
|
|
|
if (type <= LWM2M_ATTR_PMAX) {
|
|
|
|
attr->int_val = *(s32_t *)nattr_ptrs[type];
|
2017-12-12 16:24:03 +08:00
|
|
|
update_observe_node = true;
|
2017-11-24 18:47:00 +08:00
|
|
|
} else {
|
|
|
|
memcpy(&attr->float_val, nattr_ptrs[type],
|
|
|
|
sizeof(float32_value_t));
|
|
|
|
}
|
|
|
|
|
|
|
|
nattrs.flags &= ~BIT(type);
|
2019-05-29 16:49:18 -07:00
|
|
|
LOG_DBG("Add %s to %d.%06d", log_strdup(LWM2M_ATTR_STR[type]),
|
2018-09-19 11:22:19 +03:00
|
|
|
attr->float_val.val1, attr->float_val.val2);
|
2017-11-24 18:47:00 +08:00
|
|
|
}
|
|
|
|
|
2017-12-12 16:24:03 +08:00
|
|
|
/* check only pmin/pmax */
|
|
|
|
if (!update_observe_node) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* update observe_node accordingly */
|
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER(&engine_observer_list, obs, node) {
|
|
|
|
/* updated path is deeper than obs node, skip */
|
2019-01-24 14:40:23 -08:00
|
|
|
if (msg->path.level > obs->path.level) {
|
2017-12-12 16:24:03 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check obj id matched or not */
|
2019-01-24 14:40:23 -08:00
|
|
|
if (msg->path.obj_id != obs->path.obj_id) {
|
2017-12-12 16:24:03 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO: grab default from server obj */
|
|
|
|
nattrs.pmin = DEFAULT_SERVER_PMIN;
|
|
|
|
nattrs.pmax = DEFAULT_SERVER_PMAX;
|
|
|
|
|
2018-05-07 15:05:04 -07:00
|
|
|
ret = update_attrs(obj, &nattrs);
|
2018-05-07 14:18:37 -07:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
2017-12-12 16:24:03 +08:00
|
|
|
|
|
|
|
if (obs->path.level > 1) {
|
2019-01-24 14:40:23 -08:00
|
|
|
if (msg->path.level > 1 &&
|
|
|
|
msg->path.obj_inst_id != obs->path.obj_inst_id) {
|
2017-12-12 16:24:03 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* get obj_inst */
|
|
|
|
if (!obj_inst || obj_inst->obj_inst_id !=
|
|
|
|
obs->path.obj_inst_id) {
|
|
|
|
obj_inst = get_engine_obj_inst(
|
|
|
|
obs->path.obj_id,
|
|
|
|
obs->path.obj_inst_id);
|
|
|
|
if (!obj_inst) {
|
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-07 15:05:04 -07:00
|
|
|
ret = update_attrs(obj_inst, &nattrs);
|
2018-05-07 14:18:37 -07:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
2017-12-12 16:24:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (obs->path.level > 2) {
|
2019-01-24 14:40:23 -08:00
|
|
|
if (msg->path.level > 2 &&
|
|
|
|
msg->path.res_id != obs->path.res_id) {
|
2017-12-12 16:24:03 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!res || res->res_id != obs->path.res_id) {
|
2018-04-30 15:02:04 -07:00
|
|
|
ret = path_to_objs(&obs->path, NULL, NULL,
|
|
|
|
&res);
|
2017-12-12 16:24:03 +08:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-07 15:05:04 -07:00
|
|
|
ret = update_attrs(res, &nattrs);
|
2018-05-07 14:18:37 -07:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
2017-12-12 16:24:03 +08:00
|
|
|
}
|
|
|
|
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_DBG("%d/%d/%d(%d) updated from %d/%d to %u/%u",
|
|
|
|
obs->path.obj_id, obs->path.obj_inst_id,
|
|
|
|
obs->path.res_id, obs->path.level,
|
|
|
|
obs->min_period_sec, obs->max_period_sec,
|
2019-02-11 17:14:19 +00:00
|
|
|
nattrs.pmin, MAX(nattrs.pmin, nattrs.pmax));
|
2017-12-12 16:24:03 +08:00
|
|
|
obs->min_period_sec = (u32_t)nattrs.pmin;
|
2019-02-11 17:14:19 +00:00
|
|
|
obs->max_period_sec = (u32_t)MAX(nattrs.pmin, nattrs.pmax);
|
2018-09-11 19:09:03 -07:00
|
|
|
(void)memset(&nattrs, 0, sizeof(nattrs));
|
2017-12-12 16:24:03 +08:00
|
|
|
}
|
|
|
|
|
2017-11-24 18:47:00 +08:00
|
|
|
return 0;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static int lwm2m_exec_handler(struct lwm2m_engine_obj *obj,
|
2019-01-24 14:40:23 -08:00
|
|
|
struct lwm2m_message *msg)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
|
|
|
struct lwm2m_engine_obj_inst *obj_inst;
|
|
|
|
struct lwm2m_engine_res_inst *res = NULL;
|
|
|
|
int ret;
|
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
if (!obj || !msg) {
|
2017-07-07 11:04:03 -07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
ret = path_to_objs(&msg->path, &obj_inst, NULL, &res);
|
2017-07-07 11:04:03 -07:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (res->execute_cb) {
|
|
|
|
return res->execute_cb(obj_inst->obj_inst_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO: something else to handle for execute? */
|
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int lwm2m_delete_handler(struct lwm2m_engine_obj *obj,
|
2019-01-24 14:40:23 -08:00
|
|
|
struct lwm2m_message *msg)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
2019-01-25 22:36:47 -08:00
|
|
|
int ret;
|
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
if (!msg) {
|
2017-07-07 11:04:03 -07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2019-01-25 22:36:47 -08:00
|
|
|
ret = lwm2m_delete_obj_inst(msg->path.obj_id, msg->path.obj_inst_id);
|
|
|
|
#if defined(CONFIG_LWM2M_RD_CLIENT_SUPPORT)
|
|
|
|
if (!ret && !msg->ctx->bootstrap_mode) {
|
|
|
|
engine_trigger_update();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return ret;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static int do_read_op(struct lwm2m_engine_obj *obj,
|
2019-01-24 14:40:23 -08:00
|
|
|
struct lwm2m_message *msg, u16_t content_format)
|
2018-08-29 13:35:50 -07:00
|
|
|
{
|
|
|
|
switch (content_format) {
|
|
|
|
|
|
|
|
case LWM2M_FORMAT_APP_OCTET_STREAM:
|
|
|
|
case LWM2M_FORMAT_PLAIN_TEXT:
|
|
|
|
case LWM2M_FORMAT_OMA_PLAIN_TEXT:
|
2019-01-24 14:40:23 -08:00
|
|
|
return do_read_op_plain_text(obj, msg, content_format);
|
2018-08-29 13:35:50 -07:00
|
|
|
|
|
|
|
case LWM2M_FORMAT_OMA_TLV:
|
|
|
|
case LWM2M_FORMAT_OMA_OLD_TLV:
|
2019-01-24 14:40:23 -08:00
|
|
|
return do_read_op_tlv(obj, msg, content_format);
|
2018-08-29 13:35:50 -07:00
|
|
|
|
|
|
|
#if defined(CONFIG_LWM2M_RW_JSON_SUPPORT)
|
|
|
|
case LWM2M_FORMAT_OMA_JSON:
|
|
|
|
case LWM2M_FORMAT_OMA_OLD_JSON:
|
2019-01-24 14:40:23 -08:00
|
|
|
return do_read_op_json(obj, msg, content_format);
|
2018-08-29 13:35:50 -07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
default:
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("Unsupported content-format: %u", content_format);
|
2018-08-29 13:35:50 -07:00
|
|
|
return -ENOMSG;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_perform_read_op(struct lwm2m_engine_obj *obj,
|
2019-01-24 14:40:23 -08:00
|
|
|
struct lwm2m_message *msg, u16_t content_format)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
2018-08-27 17:09:32 -07:00
|
|
|
struct lwm2m_engine_obj_inst *obj_inst = NULL;
|
2018-08-29 14:22:09 -07:00
|
|
|
struct lwm2m_engine_res_inst *res = NULL;
|
2017-07-07 11:04:03 -07:00
|
|
|
struct lwm2m_engine_obj_field *obj_field;
|
2019-01-24 14:40:23 -08:00
|
|
|
struct lwm2m_obj_path temp_path;
|
2018-08-29 14:11:32 -07:00
|
|
|
int ret = 0, index;
|
2018-11-29 11:23:03 -08:00
|
|
|
u8_t num_read = 0U;
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2019-03-26 19:57:45 -06:00
|
|
|
if (msg->path.level >= 2U) {
|
2019-01-24 14:40:23 -08:00
|
|
|
obj_inst = get_engine_obj_inst(msg->path.obj_id,
|
|
|
|
msg->path.obj_inst_id);
|
2019-03-26 19:57:45 -06:00
|
|
|
} else if (msg->path.level == 1U) {
|
2018-08-27 17:09:32 -07:00
|
|
|
/* find first obj_inst with path's obj_id */
|
2019-01-24 14:40:23 -08:00
|
|
|
obj_inst = next_engine_obj_inst(msg->path.obj_id, -1);
|
2018-08-27 17:09:32 -07:00
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
if (!obj_inst) {
|
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
2017-10-19 16:27:32 -07:00
|
|
|
/* set output content-format */
|
2019-01-24 14:40:23 -08:00
|
|
|
ret = coap_append_option_int(msg->out.out_cpkt,
|
|
|
|
COAP_OPTION_CONTENT_FORMAT,
|
2017-10-19 16:27:32 -07:00
|
|
|
content_format);
|
|
|
|
if (ret < 0) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("Error setting response content-format: %d", ret);
|
2017-10-19 16:27:32 -07:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
ret = coap_packet_append_payload_marker(msg->out.out_cpkt);
|
2017-10-19 16:27:32 -07:00
|
|
|
if (ret < 0) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("Error appending payload marker: %d", ret);
|
2017-10-19 16:27:32 -07:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2018-08-29 14:32:51 -07:00
|
|
|
/* store original path values so we can change them during processing */
|
2019-01-24 14:40:23 -08:00
|
|
|
memcpy(&temp_path, &msg->path, sizeof(temp_path));
|
|
|
|
engine_put_begin(&msg->out, &msg->path);
|
2017-10-19 16:27:32 -07:00
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
while (obj_inst) {
|
2019-03-26 19:57:45 -06:00
|
|
|
if (!obj_inst->resources || obj_inst->resource_count == 0U) {
|
2018-08-27 17:00:17 -07:00
|
|
|
goto move_forward;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2018-08-29 14:44:23 -07:00
|
|
|
/* update the obj_inst_id as we move through the instances */
|
2019-01-24 14:40:23 -08:00
|
|
|
msg->path.obj_inst_id = obj_inst->obj_inst_id;
|
2018-08-29 14:44:23 -07:00
|
|
|
|
2019-03-26 19:57:45 -06:00
|
|
|
if (msg->path.level <= 1U) {
|
2018-08-29 14:44:23 -07:00
|
|
|
/* start instance formatting */
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_put_begin_oi(&msg->out, &msg->path);
|
2018-08-29 14:44:23 -07:00
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
for (index = 0; index < obj_inst->resource_count; index++) {
|
2019-01-24 14:40:23 -08:00
|
|
|
if (msg->path.level > 2 &&
|
|
|
|
msg->path.res_id !=
|
|
|
|
obj_inst->resources[index].res_id) {
|
2018-08-29 14:22:09 -07:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
res = &obj_inst->resources[index];
|
|
|
|
|
|
|
|
/*
|
2018-08-27 16:56:44 -07:00
|
|
|
* On an entire object instance, we need to set path's
|
|
|
|
* res_id for lwm2m_read_handler to read this specific
|
2017-07-07 11:04:03 -07:00
|
|
|
* resource.
|
|
|
|
*/
|
2019-01-24 14:40:23 -08:00
|
|
|
msg->path.res_id = res->res_id;
|
2017-07-07 11:04:03 -07:00
|
|
|
obj_field = lwm2m_get_engine_obj_field(obj_inst->obj,
|
|
|
|
res->res_id);
|
|
|
|
if (!obj_field) {
|
|
|
|
ret = -ENOENT;
|
2018-04-30 15:30:12 -07:00
|
|
|
} else if (!LWM2M_HAS_PERM(obj_field, LWM2M_PERM_R)) {
|
2017-07-07 11:04:03 -07:00
|
|
|
ret = -EPERM;
|
|
|
|
} else {
|
2018-08-29 14:44:23 -07:00
|
|
|
/* start resource formatting */
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_put_begin_r(&msg->out, &msg->path);
|
2018-08-29 14:44:23 -07:00
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
/* perform read operation on this resource */
|
|
|
|
ret = lwm2m_read_handler(obj_inst, res,
|
2019-01-24 14:40:23 -08:00
|
|
|
obj_field, msg);
|
2017-07-07 11:04:03 -07:00
|
|
|
if (ret < 0) {
|
2018-08-27 16:56:44 -07:00
|
|
|
/* ignore errors unless single read */
|
2019-01-24 14:40:23 -08:00
|
|
|
if (msg->path.level > 2 &&
|
2018-04-30 16:43:25 -07:00
|
|
|
!LWM2M_HAS_PERM(obj_field,
|
|
|
|
BIT(LWM2M_FLAG_OPTIONAL))) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("READ OP: %d", ret);
|
2018-04-30 16:43:25 -07:00
|
|
|
}
|
2017-07-07 11:04:03 -07:00
|
|
|
} else {
|
2019-03-26 19:57:45 -06:00
|
|
|
num_read += 1U;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
2018-08-29 14:44:23 -07:00
|
|
|
|
|
|
|
/* end resource formatting */
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_put_end_r(&msg->out, &msg->path);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* on single read break if errors */
|
2019-01-24 14:40:23 -08:00
|
|
|
if (ret < 0 && msg->path.level > 2) {
|
2017-07-07 11:04:03 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* when reading multiple resources ignore return code */
|
|
|
|
ret = 0;
|
|
|
|
}
|
|
|
|
|
2018-08-27 17:00:17 -07:00
|
|
|
move_forward:
|
2019-03-26 19:57:45 -06:00
|
|
|
if (msg->path.level <= 1U) {
|
2018-08-29 14:44:23 -07:00
|
|
|
/* end instance formatting */
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_put_end_oi(&msg->out, &msg->path);
|
2018-08-29 14:44:23 -07:00
|
|
|
}
|
|
|
|
|
2019-03-26 19:57:45 -06:00
|
|
|
if (msg->path.level <= 1U) {
|
2018-08-27 17:00:17 -07:00
|
|
|
/* advance to the next object instance */
|
2019-01-24 14:40:23 -08:00
|
|
|
obj_inst = next_engine_obj_inst(msg->path.obj_id,
|
2018-08-27 17:00:17 -07:00
|
|
|
obj_inst->obj_inst_id);
|
|
|
|
} else {
|
|
|
|
obj_inst = NULL;
|
|
|
|
}
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
engine_put_end(&msg->out, &msg->path);
|
2018-08-29 14:15:43 -07:00
|
|
|
|
2018-08-29 14:32:51 -07:00
|
|
|
/* restore original path values */
|
2019-01-24 14:40:23 -08:00
|
|
|
memcpy(&msg->path, &temp_path, sizeof(temp_path));
|
2018-08-29 14:32:51 -07:00
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
/* did not read anything even if we should have - on single item */
|
2019-03-26 19:57:45 -06:00
|
|
|
if (ret == 0 && num_read == 0U && msg->path.level == 3U) {
|
2017-07-07 11:04:03 -07:00
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2019-01-25 14:13:07 -08:00
|
|
|
static int print_attr(struct lwm2m_output_context *out,
|
|
|
|
u8_t *buf, u16_t buflen, void *ref)
|
2017-11-28 11:51:36 +08:00
|
|
|
{
|
|
|
|
struct lwm2m_attr *attr;
|
2019-01-25 14:13:07 -08:00
|
|
|
int i, used, base, ret;
|
2017-11-28 11:51:36 +08:00
|
|
|
u8_t digit;
|
|
|
|
s32_t fraction;
|
|
|
|
|
2018-05-07 15:05:04 -07:00
|
|
|
for (i = 0; i < CONFIG_LWM2M_NUM_ATTR; i++) {
|
|
|
|
if (ref != write_attr_pool[i].ref) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
attr = write_attr_pool + i;
|
|
|
|
|
2017-11-28 11:51:36 +08:00
|
|
|
/* assuming integer will have float_val.val2 set as 0 */
|
net: lwm2m: fix reporting attributes with negative fraction
Fraction could be stored with negative value.
The implementation was only considering the positive value case.
Therefore, we have to modify the code to take care of the case.
To test it
======================================================================
1. launch eclipse/wakaama lwm2m server
2. launch zephyr lwm2m client and wait for registration completed
3. Issue commands from server
* attr 0 /1/0/1 -0.1 0.1
* disc 0 /1/0
Current output
----------------------------------------------------------------------
Client #0 /1/0 : 2.05 (COAP_205_CONTENT)
105 bytes received of type application/link-format:
</1/0>,</1/0/0>,</1/0/1>;gt=0.1;lt=0/00000,</1/0/2>,</1/0/3>,</1/0/4>,
</1/0/5>,</1/0/6>,</1/0/7>,</1/0/8>
Expected output
----------------------------------------------------------------------
Client #0 /1/0 : 2.05 (COAP_205_CONTENT)
102 bytes received of type application/link-format:
</1/0>,</1/0/0>,</1/0/1>;gt=0.1;lt=-0.1,</1/0/2>,</1/0/3>,</1/0/4>,
</1/0/5>,</1/0/6>,</1/0/7>,</1/0/8>
Signed-off-by: Robert Chou <robert.ch.chou@acer.com>
2018-01-15 09:39:34 +08:00
|
|
|
|
|
|
|
used = snprintk(buf, buflen, ";%s=%s%d%s",
|
2017-11-28 11:51:36 +08:00
|
|
|
LWM2M_ATTR_STR[attr->type],
|
net: lwm2m: fix reporting attributes with negative fraction
Fraction could be stored with negative value.
The implementation was only considering the positive value case.
Therefore, we have to modify the code to take care of the case.
To test it
======================================================================
1. launch eclipse/wakaama lwm2m server
2. launch zephyr lwm2m client and wait for registration completed
3. Issue commands from server
* attr 0 /1/0/1 -0.1 0.1
* disc 0 /1/0
Current output
----------------------------------------------------------------------
Client #0 /1/0 : 2.05 (COAP_205_CONTENT)
105 bytes received of type application/link-format:
</1/0>,</1/0/0>,</1/0/1>;gt=0.1;lt=0/00000,</1/0/2>,</1/0/3>,</1/0/4>,
</1/0/5>,</1/0/6>,</1/0/7>,</1/0/8>
Expected output
----------------------------------------------------------------------
Client #0 /1/0 : 2.05 (COAP_205_CONTENT)
102 bytes received of type application/link-format:
</1/0>,</1/0/0>,</1/0/1>;gt=0.1;lt=-0.1,</1/0/2>,</1/0/3>,</1/0/4>,
</1/0/5>,</1/0/6>,</1/0/7>,</1/0/8>
Signed-off-by: Robert Chou <robert.ch.chou@acer.com>
2018-01-15 09:39:34 +08:00
|
|
|
attr->float_val.val1 == 0 &&
|
|
|
|
attr->float_val.val2 < 0 ? "-" : "",
|
2017-11-28 11:51:36 +08:00
|
|
|
attr->float_val.val1,
|
net: lwm2m: fix reporting attributes with negative fraction
Fraction could be stored with negative value.
The implementation was only considering the positive value case.
Therefore, we have to modify the code to take care of the case.
To test it
======================================================================
1. launch eclipse/wakaama lwm2m server
2. launch zephyr lwm2m client and wait for registration completed
3. Issue commands from server
* attr 0 /1/0/1 -0.1 0.1
* disc 0 /1/0
Current output
----------------------------------------------------------------------
Client #0 /1/0 : 2.05 (COAP_205_CONTENT)
105 bytes received of type application/link-format:
</1/0>,</1/0/0>,</1/0/1>;gt=0.1;lt=0/00000,</1/0/2>,</1/0/3>,</1/0/4>,
</1/0/5>,</1/0/6>,</1/0/7>,</1/0/8>
Expected output
----------------------------------------------------------------------
Client #0 /1/0 : 2.05 (COAP_205_CONTENT)
102 bytes received of type application/link-format:
</1/0>,</1/0/0>,</1/0/1>;gt=0.1;lt=-0.1,</1/0/2>,</1/0/3>,</1/0/4>,
</1/0/5>,</1/0/6>,</1/0/7>,</1/0/8>
Signed-off-by: Robert Chou <robert.ch.chou@acer.com>
2018-01-15 09:39:34 +08:00
|
|
|
attr->float_val.val2 != 0 ? "." : "");
|
2017-11-28 11:51:36 +08:00
|
|
|
|
|
|
|
base = 100000;
|
net: lwm2m: fix reporting attributes with negative fraction
Fraction could be stored with negative value.
The implementation was only considering the positive value case.
Therefore, we have to modify the code to take care of the case.
To test it
======================================================================
1. launch eclipse/wakaama lwm2m server
2. launch zephyr lwm2m client and wait for registration completed
3. Issue commands from server
* attr 0 /1/0/1 -0.1 0.1
* disc 0 /1/0
Current output
----------------------------------------------------------------------
Client #0 /1/0 : 2.05 (COAP_205_CONTENT)
105 bytes received of type application/link-format:
</1/0>,</1/0/0>,</1/0/1>;gt=0.1;lt=0/00000,</1/0/2>,</1/0/3>,</1/0/4>,
</1/0/5>,</1/0/6>,</1/0/7>,</1/0/8>
Expected output
----------------------------------------------------------------------
Client #0 /1/0 : 2.05 (COAP_205_CONTENT)
102 bytes received of type application/link-format:
</1/0>,</1/0/0>,</1/0/1>;gt=0.1;lt=-0.1,</1/0/2>,</1/0/3>,</1/0/4>,
</1/0/5>,</1/0/6>,</1/0/7>,</1/0/8>
Signed-off-by: Robert Chou <robert.ch.chou@acer.com>
2018-01-15 09:39:34 +08:00
|
|
|
fraction = attr->float_val.val2 < 0 ?
|
|
|
|
-attr->float_val.val2 : attr->float_val.val2;
|
2017-11-28 11:51:36 +08:00
|
|
|
while (fraction && used < buflen && base > 0) {
|
|
|
|
digit = fraction / base;
|
|
|
|
buf[used++] = '0' + digit;
|
|
|
|
fraction -= digit * base;
|
|
|
|
base /= 10;
|
|
|
|
}
|
|
|
|
|
2019-01-25 14:13:07 -08:00
|
|
|
ret = buf_append(CPKT_BUF_WRITE(out->out_cpkt), buf, used);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
2017-11-28 11:51:36 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
static int do_discover_op(struct lwm2m_message *msg, bool well_known)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
2017-10-19 16:27:32 -07:00
|
|
|
static char disc_buf[24];
|
2017-11-15 09:46:13 +08:00
|
|
|
struct lwm2m_engine_obj *obj;
|
2017-07-07 11:04:03 -07:00
|
|
|
struct lwm2m_engine_obj_inst *obj_inst;
|
2017-11-15 09:46:13 +08:00
|
|
|
int ret;
|
|
|
|
bool reported = false;
|
|
|
|
|
2019-01-25 22:36:47 -08:00
|
|
|
/* object ID is required unless it's bootstrap discover or it's
|
2017-11-15 09:46:13 +08:00
|
|
|
* a ".well-known/core" discovery
|
|
|
|
* ref: lwm2m spec 20170208-A table 11
|
|
|
|
*/
|
2019-01-25 22:36:47 -08:00
|
|
|
if (!msg->ctx->bootstrap_mode && !well_known &&
|
2019-03-26 19:57:45 -06:00
|
|
|
(msg->path.level == 0U ||
|
2019-01-24 14:40:23 -08:00
|
|
|
(msg->path.level > 0 &&
|
|
|
|
msg->path.obj_id == LWM2M_OBJECT_SECURITY_ID))) {
|
2017-11-15 09:46:13 +08:00
|
|
|
return -EPERM;
|
|
|
|
}
|
2017-07-07 11:04:03 -07:00
|
|
|
|
|
|
|
/* set output content-format */
|
2019-01-24 14:40:23 -08:00
|
|
|
ret = coap_append_option_int(msg->out.out_cpkt,
|
2017-11-15 09:46:13 +08:00
|
|
|
COAP_OPTION_CONTENT_FORMAT,
|
|
|
|
LWM2M_FORMAT_APP_LINK_FORMAT);
|
2017-09-19 16:04:32 -07:00
|
|
|
if (ret < 0) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("Error setting response content-format: %d", ret);
|
2017-09-19 16:04:32 -07:00
|
|
|
return ret;
|
|
|
|
}
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
ret = coap_packet_append_payload_marker(msg->out.out_cpkt);
|
2017-10-19 16:27:32 -07:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2017-11-15 09:46:13 +08:00
|
|
|
/* Handle CoAP .well-known/core discover */
|
|
|
|
if (well_known) {
|
|
|
|
/* </.well-known/core> */
|
2019-01-25 14:13:07 -08:00
|
|
|
ret = buf_append(CPKT_BUF_WRITE(msg->out.out_cpkt),
|
|
|
|
WELL_KNOWN_CORE_PATH,
|
|
|
|
strlen(WELL_KNOWN_CORE_PATH));
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
2017-11-15 09:46:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER(&engine_obj_list, obj, node) {
|
|
|
|
snprintk(disc_buf, sizeof(disc_buf), ",</%u>",
|
|
|
|
obj->obj_id);
|
2019-01-25 14:13:07 -08:00
|
|
|
|
|
|
|
ret = buf_append(CPKT_BUF_WRITE(msg->out.out_cpkt),
|
|
|
|
disc_buf, strlen(disc_buf));
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
2017-11-15 09:46:13 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2017-10-19 16:27:32 -07:00
|
|
|
}
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2019-01-25 22:36:47 -08:00
|
|
|
/*
|
|
|
|
* lwm2m spec 20170208-A sec 5.2.7.3 bootstrap discover on "/"
|
|
|
|
* - (TODO) prefixed w/ lwm2m enabler version. e.g. lwm2m="1.0"
|
2017-11-15 09:46:13 +08:00
|
|
|
* - returns object and object instances only
|
|
|
|
*/
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER(&engine_obj_inst_list, obj_inst, node) {
|
2019-01-25 22:36:47 -08:00
|
|
|
/*
|
|
|
|
* - Avoid discovery for security object (5.2.7.3) unless
|
|
|
|
* Bootstrap discover
|
|
|
|
* - Skip reporting unrelated object
|
2017-07-10 18:19:35 +08:00
|
|
|
*/
|
2019-01-25 22:36:47 -08:00
|
|
|
if ((!msg->ctx->bootstrap_mode &&
|
|
|
|
obj_inst->obj->obj_id == LWM2M_OBJECT_SECURITY_ID) ||
|
2019-01-24 14:40:23 -08:00
|
|
|
obj_inst->obj->obj_id != msg->path.obj_id) {
|
2017-11-15 09:46:13 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-03-26 19:57:45 -06:00
|
|
|
if (msg->path.level == 1U) {
|
2017-11-15 09:46:13 +08:00
|
|
|
snprintk(disc_buf, sizeof(disc_buf), "%s</%u>",
|
|
|
|
reported ? "," : "",
|
|
|
|
obj_inst->obj->obj_id);
|
2019-01-25 14:13:07 -08:00
|
|
|
|
|
|
|
ret = buf_append(CPKT_BUF_WRITE(msg->out.out_cpkt),
|
|
|
|
disc_buf, strlen(disc_buf));
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
2017-11-15 09:46:13 +08:00
|
|
|
}
|
|
|
|
|
2017-11-28 11:51:36 +08:00
|
|
|
/* report object attrs (5.4.2) */
|
2019-01-25 14:13:07 -08:00
|
|
|
ret = print_attr(&msg->out, disc_buf, sizeof(disc_buf),
|
2018-05-07 15:05:04 -07:00
|
|
|
obj_inst->obj);
|
2017-11-28 11:51:36 +08:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2017-11-15 09:46:13 +08:00
|
|
|
reported = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* skip unrelated object instance */
|
2019-01-24 14:40:23 -08:00
|
|
|
if (msg->path.level > 1 &&
|
|
|
|
msg->path.obj_inst_id != obj_inst->obj_inst_id) {
|
2017-07-07 11:04:03 -07:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-03-26 19:57:45 -06:00
|
|
|
if (msg->path.level == 2U) {
|
2017-11-15 09:46:13 +08:00
|
|
|
snprintk(disc_buf, sizeof(disc_buf), "%s</%u/%u>",
|
|
|
|
reported ? "," : "",
|
|
|
|
obj_inst->obj->obj_id, obj_inst->obj_inst_id);
|
2019-01-25 14:13:07 -08:00
|
|
|
|
|
|
|
ret = buf_append(CPKT_BUF_WRITE(msg->out.out_cpkt),
|
|
|
|
disc_buf, strlen(disc_buf));
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
2017-11-15 09:46:13 +08:00
|
|
|
}
|
2017-10-19 16:27:32 -07:00
|
|
|
|
2017-11-28 11:51:36 +08:00
|
|
|
/* report object instance attrs (5.4.2) */
|
2019-01-25 14:13:07 -08:00
|
|
|
ret = print_attr(&msg->out, disc_buf, sizeof(disc_buf),
|
2018-05-07 15:05:04 -07:00
|
|
|
obj_inst);
|
2017-11-28 11:51:36 +08:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2017-11-15 09:46:13 +08:00
|
|
|
reported = true;
|
2017-10-19 16:27:32 -07:00
|
|
|
}
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2019-01-25 22:36:47 -08:00
|
|
|
/* don't return resource info for bootstrap discovery */
|
|
|
|
if (msg->ctx->bootstrap_mode) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-11-15 09:46:13 +08:00
|
|
|
for (int i = 0; i < obj_inst->resource_count; i++) {
|
|
|
|
/* skip unrelated resources */
|
2019-03-26 19:57:45 -06:00
|
|
|
if (msg->path.level == 3U &&
|
2019-01-24 14:40:23 -08:00
|
|
|
msg->path.res_id != obj_inst->resources[i].res_id) {
|
2017-11-15 09:46:13 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-10-19 16:27:32 -07:00
|
|
|
snprintk(disc_buf, sizeof(disc_buf),
|
2017-11-15 09:46:13 +08:00
|
|
|
"%s</%u/%u/%u>",
|
|
|
|
reported ? "," : "",
|
|
|
|
obj_inst->obj->obj_id,
|
|
|
|
obj_inst->obj_inst_id,
|
|
|
|
obj_inst->resources[i].res_id);
|
2019-01-25 14:13:07 -08:00
|
|
|
|
|
|
|
ret = buf_append(CPKT_BUF_WRITE(msg->out.out_cpkt),
|
|
|
|
disc_buf, strlen(disc_buf));
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
2017-10-19 16:27:32 -07:00
|
|
|
}
|
2017-11-15 09:46:13 +08:00
|
|
|
|
2017-11-28 11:51:36 +08:00
|
|
|
/* report resource attrs when path > 1 (5.4.2) */
|
2019-01-24 14:40:23 -08:00
|
|
|
if (msg->path.level > 1) {
|
2019-01-25 14:13:07 -08:00
|
|
|
ret = print_attr(&msg->out,
|
2018-05-07 15:05:04 -07:00
|
|
|
disc_buf, sizeof(disc_buf),
|
|
|
|
&obj_inst->resources[i]);
|
2017-11-28 11:51:36 +08:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-15 09:46:13 +08:00
|
|
|
reported = true;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-15 09:46:13 +08:00
|
|
|
return reported ? 0 : -ENOENT;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
int lwm2m_get_or_create_engine_obj(struct lwm2m_message *msg,
|
2017-07-07 11:04:03 -07:00
|
|
|
struct lwm2m_engine_obj_inst **obj_inst,
|
|
|
|
u8_t *created)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
if (created) {
|
2018-11-29 11:23:03 -08:00
|
|
|
*created = 0U;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
*obj_inst = get_engine_obj_inst(msg->path.obj_id,
|
|
|
|
msg->path.obj_inst_id);
|
2017-07-07 11:04:03 -07:00
|
|
|
if (!*obj_inst) {
|
2019-01-24 14:40:23 -08:00
|
|
|
ret = lwm2m_create_obj_inst(msg->path.obj_id,
|
|
|
|
msg->path.obj_inst_id,
|
2017-07-07 11:04:03 -07:00
|
|
|
obj_inst);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* set created flag to one */
|
|
|
|
if (created) {
|
2018-11-29 11:23:03 -08:00
|
|
|
*created = 1U;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
2019-01-25 22:36:47 -08:00
|
|
|
|
|
|
|
#if defined(CONFIG_LWM2M_RD_CLIENT_SUPPORT)
|
|
|
|
if (!msg->ctx->bootstrap_mode) {
|
|
|
|
engine_trigger_update();
|
|
|
|
}
|
|
|
|
#endif
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int do_write_op(struct lwm2m_engine_obj *obj,
|
2019-01-24 14:40:23 -08:00
|
|
|
struct lwm2m_message *msg,
|
2017-07-07 11:04:03 -07:00
|
|
|
u16_t format)
|
|
|
|
{
|
|
|
|
switch (format) {
|
|
|
|
|
2017-10-30 15:18:35 +08:00
|
|
|
case LWM2M_FORMAT_APP_OCTET_STREAM:
|
2017-07-07 11:04:03 -07:00
|
|
|
case LWM2M_FORMAT_PLAIN_TEXT:
|
|
|
|
case LWM2M_FORMAT_OMA_PLAIN_TEXT:
|
2019-01-24 14:40:23 -08:00
|
|
|
return do_write_op_plain_text(obj, msg);
|
2017-07-07 11:04:03 -07:00
|
|
|
|
|
|
|
case LWM2M_FORMAT_OMA_TLV:
|
|
|
|
case LWM2M_FORMAT_OMA_OLD_TLV:
|
2019-01-24 14:40:23 -08:00
|
|
|
return do_write_op_tlv(obj, msg);
|
2017-07-07 11:04:03 -07:00
|
|
|
|
|
|
|
#ifdef CONFIG_LWM2M_RW_JSON_SUPPORT
|
|
|
|
case LWM2M_FORMAT_OMA_JSON:
|
|
|
|
case LWM2M_FORMAT_OMA_OLD_JSON:
|
2019-01-24 14:40:23 -08:00
|
|
|
return do_write_op_json(obj, msg);
|
2017-07-07 11:04:03 -07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
default:
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("Unsupported format: %u", format);
|
2018-01-30 15:18:57 +08:00
|
|
|
return -ENOMSG;
|
2017-07-07 11:04:03 -07:00
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-25 22:36:47 -08:00
|
|
|
#if defined(CONFIG_LWM2M_RD_CLIENT_SUPPORT_BOOTSTRAP)
|
|
|
|
static int bootstrap_delete(void)
|
|
|
|
{
|
|
|
|
struct lwm2m_engine_obj_inst *obj_inst, *tmp;
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
/* delete SECURITY instances > 0 */
|
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER_SAFE(&engine_obj_inst_list,
|
|
|
|
obj_inst, tmp, node) {
|
|
|
|
if (obj_inst->obj->obj_id == LWM2M_OBJECT_SECURITY_ID &&
|
|
|
|
obj_inst->obj_inst_id > 0) {
|
|
|
|
ret = lwm2m_delete_obj_inst(obj_inst->obj->obj_id,
|
|
|
|
obj_inst->obj_inst_id);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
static int handle_request(struct coap_packet *request,
|
2017-09-01 01:11:43 -07:00
|
|
|
struct lwm2m_message *msg)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
|
|
|
int r;
|
|
|
|
u8_t code;
|
2017-09-19 16:04:32 -07:00
|
|
|
struct coap_option options[4];
|
2017-11-15 09:46:13 +08:00
|
|
|
struct lwm2m_engine_obj *obj = NULL;
|
2017-09-19 16:04:32 -07:00
|
|
|
u8_t token[8];
|
2018-11-29 11:23:03 -08:00
|
|
|
u8_t tkl = 0U;
|
2017-11-21 18:16:52 +08:00
|
|
|
u16_t format = LWM2M_FORMAT_NONE, accept;
|
2017-07-07 11:04:03 -07:00
|
|
|
int observe = -1; /* default to -1, 0 = ENABLE, 1 = DISABLE */
|
2017-11-15 09:46:13 +08:00
|
|
|
bool well_known = false;
|
2017-07-25 16:54:25 +08:00
|
|
|
struct block_context *block_ctx = NULL;
|
2017-09-19 16:04:32 -07:00
|
|
|
enum coap_block_size block_size;
|
2019-01-25 14:13:07 -08:00
|
|
|
u16_t payload_len = 0U;
|
2017-10-23 10:10:08 +08:00
|
|
|
bool last_block = false;
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
/* set CoAP request / message */
|
2019-01-24 14:40:23 -08:00
|
|
|
msg->in.in_cpkt = request;
|
|
|
|
msg->out.out_cpkt = &msg->cpkt;
|
2017-07-07 11:04:03 -07:00
|
|
|
|
|
|
|
/* set default reader/writer */
|
2019-01-24 14:40:23 -08:00
|
|
|
msg->in.reader = &plain_text_reader;
|
|
|
|
msg->out.writer = &plain_text_writer;
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
code = coap_header_get_code(msg->in.in_cpkt);
|
2017-08-28 18:11:02 +08:00
|
|
|
|
2017-11-15 16:20:05 +08:00
|
|
|
/* setup response token */
|
2019-01-24 14:40:23 -08:00
|
|
|
tkl = coap_header_get_token(msg->in.in_cpkt, token);
|
2017-11-15 16:20:05 +08:00
|
|
|
if (tkl) {
|
|
|
|
msg->tkl = tkl;
|
|
|
|
msg->token = token;
|
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
/* parse the URL path into components */
|
2019-01-24 14:40:23 -08:00
|
|
|
r = coap_find_options(msg->in.in_cpkt, COAP_OPTION_URI_PATH, options,
|
2018-07-12 10:00:02 -07:00
|
|
|
ARRAY_SIZE(options));
|
2017-08-28 14:54:40 +08:00
|
|
|
if (r <= 0) {
|
2019-01-25 22:36:47 -08:00
|
|
|
switch (code & COAP_REQUEST_MASK) {
|
|
|
|
#if defined(CONFIG_LWM2M_RD_CLIENT_SUPPORT_BOOTSTRAP)
|
|
|
|
case COAP_METHOD_DELETE:
|
|
|
|
if (msg->ctx->bootstrap_mode) {
|
|
|
|
r = bootstrap_delete();
|
|
|
|
if (r < 0) {
|
|
|
|
goto error;
|
|
|
|
}
|
2017-08-28 14:54:40 +08:00
|
|
|
|
2019-01-25 22:36:47 -08:00
|
|
|
msg->code = COAP_RESPONSE_CODE_DELETED;
|
|
|
|
r = lwm2m_init_message(msg);
|
|
|
|
} else {
|
|
|
|
r = -EPERM;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (r < 0) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
#endif
|
|
|
|
default:
|
|
|
|
r = -EPERM;
|
|
|
|
goto error;
|
|
|
|
}
|
2017-08-28 14:54:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* check for .well-known/core URI query (DISCOVER) */
|
|
|
|
if (r == 2 &&
|
2019-03-26 19:57:45 -06:00
|
|
|
(options[0].len == 11U &&
|
2017-08-28 14:54:40 +08:00
|
|
|
strncmp(options[0].value, ".well-known", 11) == 0) &&
|
2019-03-26 19:57:45 -06:00
|
|
|
(options[1].len == 4U &&
|
2017-08-28 14:54:40 +08:00
|
|
|
strncmp(options[1].value, "core", 4) == 0)) {
|
2017-09-19 16:04:32 -07:00
|
|
|
if ((code & COAP_REQUEST_MASK) != COAP_METHOD_GET) {
|
2017-08-28 18:11:02 +08:00
|
|
|
r = -EPERM;
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2017-11-15 09:46:13 +08:00
|
|
|
well_known = true;
|
2017-08-28 14:54:40 +08:00
|
|
|
} else {
|
2019-01-24 14:40:23 -08:00
|
|
|
r = coap_options_to_path(options, r, &msg->path);
|
2017-08-28 17:06:48 +08:00
|
|
|
if (r < 0) {
|
|
|
|
r = -ENOENT;
|
|
|
|
goto error;
|
|
|
|
}
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2018-02-05 16:05:51 +08:00
|
|
|
/* read Content Format / setup in.reader */
|
2019-01-24 14:40:23 -08:00
|
|
|
r = coap_find_options(msg->in.in_cpkt, COAP_OPTION_CONTENT_FORMAT,
|
2017-07-07 11:04:03 -07:00
|
|
|
options, 1);
|
|
|
|
if (r > 0) {
|
2018-01-30 15:18:57 +08:00
|
|
|
format = coap_option_value_to_int(&options[0]);
|
2019-01-24 14:40:23 -08:00
|
|
|
r = select_reader(&msg->in, format);
|
2018-01-30 15:18:57 +08:00
|
|
|
if (r < 0) {
|
|
|
|
goto error;
|
|
|
|
}
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2018-02-05 16:05:51 +08:00
|
|
|
/* read Accept / setup out.writer */
|
2019-01-24 14:40:23 -08:00
|
|
|
r = coap_find_options(msg->in.in_cpkt, COAP_OPTION_ACCEPT, options, 1);
|
2017-07-07 11:04:03 -07:00
|
|
|
if (r > 0) {
|
2017-09-19 16:04:32 -07:00
|
|
|
accept = coap_option_value_to_int(&options[0]);
|
2017-07-07 11:04:03 -07:00
|
|
|
} else {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_DBG("No accept option given. Assume OMA TLV.");
|
2017-07-05 10:54:31 +08:00
|
|
|
accept = LWM2M_FORMAT_OMA_TLV;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
r = select_writer(&msg->out, accept);
|
2018-02-05 16:05:51 +08:00
|
|
|
if (r < 0) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2017-11-15 09:46:13 +08:00
|
|
|
if (!well_known) {
|
|
|
|
/* find registered obj */
|
2019-01-24 14:40:23 -08:00
|
|
|
obj = get_engine_obj(msg->path.obj_id);
|
2017-11-15 09:46:13 +08:00
|
|
|
if (!obj) {
|
|
|
|
/* No matching object found - ignore request */
|
|
|
|
r = -ENOENT;
|
|
|
|
goto error;
|
|
|
|
}
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* set the operation */
|
2017-09-19 16:04:32 -07:00
|
|
|
switch (code & COAP_REQUEST_MASK) {
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
case COAP_METHOD_GET:
|
2017-10-15 13:42:45 -07:00
|
|
|
/*
|
2017-11-20 15:33:32 +08:00
|
|
|
* LwM2M V1_0_1-20170704-A, table 25,
|
|
|
|
* Discover: CoAP GET + accept=LWM2M_FORMAT_APP_LINK_FORMAT
|
2017-10-15 13:42:45 -07:00
|
|
|
*/
|
2017-11-20 15:33:32 +08:00
|
|
|
if (well_known || accept == LWM2M_FORMAT_APP_LINK_FORMAT) {
|
2019-01-24 14:40:23 -08:00
|
|
|
msg->operation = LWM2M_OP_DISCOVER;
|
2017-07-07 11:04:03 -07:00
|
|
|
accept = LWM2M_FORMAT_APP_LINK_FORMAT;
|
|
|
|
} else {
|
2019-01-24 14:40:23 -08:00
|
|
|
msg->operation = LWM2M_OP_READ;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
2017-11-15 09:46:13 +08:00
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
/* check for observe */
|
2019-01-24 14:40:23 -08:00
|
|
|
observe = get_option_int(msg->in.in_cpkt, COAP_OPTION_OBSERVE);
|
2017-09-19 16:04:32 -07:00
|
|
|
msg->code = COAP_RESPONSE_CODE_CONTENT;
|
2017-07-07 11:04:03 -07:00
|
|
|
break;
|
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
case COAP_METHOD_POST:
|
2019-03-26 19:57:45 -06:00
|
|
|
if (msg->path.level == 1U) {
|
2017-11-20 17:36:23 +08:00
|
|
|
/* create an object instance */
|
2019-01-24 14:40:23 -08:00
|
|
|
msg->operation = LWM2M_OP_CREATE;
|
2017-10-11 13:44:35 -07:00
|
|
|
msg->code = COAP_RESPONSE_CODE_CREATED;
|
2019-03-26 19:57:45 -06:00
|
|
|
} else if (msg->path.level == 2U) {
|
2017-11-20 17:36:23 +08:00
|
|
|
/* write values to an object instance */
|
2019-01-24 14:40:23 -08:00
|
|
|
msg->operation = LWM2M_OP_WRITE;
|
2017-11-20 17:36:23 +08:00
|
|
|
msg->code = COAP_RESPONSE_CODE_CHANGED;
|
2017-07-07 11:04:03 -07:00
|
|
|
} else {
|
2019-01-24 14:40:23 -08:00
|
|
|
msg->operation = LWM2M_OP_EXECUTE;
|
2017-10-11 13:44:35 -07:00
|
|
|
msg->code = COAP_RESPONSE_CODE_CHANGED;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
2017-11-20 17:36:23 +08:00
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
break;
|
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
case COAP_METHOD_PUT:
|
2017-11-21 18:16:52 +08:00
|
|
|
/* write attributes if content-format is absent */
|
|
|
|
if (format == LWM2M_FORMAT_NONE) {
|
2019-01-24 14:40:23 -08:00
|
|
|
msg->operation = LWM2M_OP_WRITE_ATTR;
|
2017-11-21 18:16:52 +08:00
|
|
|
} else {
|
2019-01-24 14:40:23 -08:00
|
|
|
msg->operation = LWM2M_OP_WRITE;
|
2017-11-21 18:16:52 +08:00
|
|
|
}
|
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
msg->code = COAP_RESPONSE_CODE_CHANGED;
|
2017-07-07 11:04:03 -07:00
|
|
|
break;
|
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
case COAP_METHOD_DELETE:
|
2019-01-24 14:40:23 -08:00
|
|
|
msg->operation = LWM2M_OP_DELETE;
|
2017-09-19 16:04:32 -07:00
|
|
|
msg->code = COAP_RESPONSE_CODE_DELETED;
|
2017-07-07 11:04:03 -07:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2017-10-19 16:27:32 -07:00
|
|
|
/* setup incoming data */
|
2019-01-25 14:13:07 -08:00
|
|
|
msg->in.offset = msg->in.in_cpkt->hdr_len + msg->in.in_cpkt->opt_len;
|
|
|
|
coap_packet_get_payload(msg->in.in_cpkt, &payload_len);
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2017-07-25 16:54:25 +08:00
|
|
|
/* Check for block transfer */
|
2019-01-24 14:40:23 -08:00
|
|
|
r = get_option_int(msg->in.in_cpkt, COAP_OPTION_BLOCK1);
|
2017-07-25 16:54:25 +08:00
|
|
|
if (r > 0) {
|
2017-10-23 10:10:08 +08:00
|
|
|
last_block = !GET_MORE(r);
|
|
|
|
|
2017-07-25 16:54:25 +08:00
|
|
|
/* RFC7252: 4.6. Message Size */
|
|
|
|
block_size = GET_BLOCK_SIZE(r);
|
2017-10-23 10:10:08 +08:00
|
|
|
if (!last_block &&
|
2019-01-25 14:13:07 -08:00
|
|
|
coap_block_size_to_bytes(block_size) > payload_len) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_DBG("Trailing payload is discarded!");
|
2017-07-25 16:54:25 +08:00
|
|
|
r = -EFBIG;
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (GET_BLOCK_NUM(r) == 0) {
|
|
|
|
r = init_block_ctx(token, tkl, &block_ctx);
|
|
|
|
} else {
|
|
|
|
r = get_block_ctx(token, tkl, &block_ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (r < 0) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
r = coap_update_from_block(msg->in.in_cpkt, &block_ctx->ctx);
|
2017-10-23 10:10:08 +08:00
|
|
|
if (r < 0) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("Error from block update: %d", r);
|
2017-10-23 10:10:08 +08:00
|
|
|
goto error;
|
|
|
|
}
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2017-10-23 10:10:08 +08:00
|
|
|
/* Handle blockwise 1 (Part 1): Set response code */
|
|
|
|
if (!last_block) {
|
2017-09-19 16:04:32 -07:00
|
|
|
msg->code = COAP_RESPONSE_CODE_CONTINUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* render CoAP packet header */
|
2017-10-11 14:04:39 -07:00
|
|
|
r = lwm2m_init_message(msg);
|
|
|
|
if (r < 0) {
|
|
|
|
goto error;
|
|
|
|
}
|
2017-09-19 16:04:32 -07:00
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
switch (msg->operation) {
|
2017-07-07 11:04:03 -07:00
|
|
|
|
|
|
|
case LWM2M_OP_READ:
|
|
|
|
if (observe == 0) {
|
|
|
|
/* add new observer */
|
2017-09-19 16:04:32 -07:00
|
|
|
if (msg->token) {
|
2019-01-24 14:40:23 -08:00
|
|
|
r = coap_append_option_int(msg->out.out_cpkt,
|
2017-09-19 16:04:32 -07:00
|
|
|
COAP_OPTION_OBSERVE,
|
|
|
|
1);
|
2017-10-11 14:04:39 -07:00
|
|
|
if (r < 0) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("OBSERVE option error: %d", r);
|
2017-10-11 14:04:39 -07:00
|
|
|
goto error;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
r = engine_add_observer(msg, token, tkl,
|
2017-08-24 09:39:33 -07:00
|
|
|
accept);
|
2017-07-07 11:04:03 -07:00
|
|
|
if (r < 0) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("add OBSERVE error: %d", r);
|
2018-08-27 13:59:18 -07:00
|
|
|
goto error;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
} else {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("OBSERVE request missing token");
|
2018-08-27 13:59:18 -07:00
|
|
|
r = -EINVAL;
|
|
|
|
goto error;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
} else if (observe == 1) {
|
|
|
|
/* remove observer */
|
|
|
|
r = engine_remove_observer(token, tkl);
|
|
|
|
if (r < 0) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("remove observe error: %d", r);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
r = do_read_op(obj, msg, accept);
|
2017-07-07 11:04:03 -07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_OP_DISCOVER:
|
2019-01-24 14:40:23 -08:00
|
|
|
r = do_discover_op(msg, well_known);
|
2017-07-07 11:04:03 -07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_OP_WRITE:
|
2017-08-17 12:21:28 -07:00
|
|
|
case LWM2M_OP_CREATE:
|
2019-01-24 14:40:23 -08:00
|
|
|
r = do_write_op(obj, msg, format);
|
2017-07-07 11:04:03 -07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_OP_WRITE_ATTR:
|
2019-01-24 14:40:23 -08:00
|
|
|
r = lwm2m_write_attr_handler(obj, msg);
|
2017-07-07 11:04:03 -07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_OP_EXECUTE:
|
2019-01-24 14:40:23 -08:00
|
|
|
r = lwm2m_exec_handler(obj, msg);
|
2017-07-07 11:04:03 -07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case LWM2M_OP_DELETE:
|
2019-01-24 14:40:23 -08:00
|
|
|
r = lwm2m_delete_handler(obj, msg);
|
2017-07-07 11:04:03 -07:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2019-01-24 14:40:23 -08:00
|
|
|
LOG_ERR("Unknown operation: %u", msg->operation);
|
2017-08-28 17:20:27 +08:00
|
|
|
r = -EINVAL;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2017-10-11 14:04:39 -07:00
|
|
|
if (r < 0) {
|
2017-07-25 16:54:25 +08:00
|
|
|
goto error;
|
|
|
|
}
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2017-10-23 10:10:08 +08:00
|
|
|
/* Handle blockwise 1 (Part 2): Append BLOCK1 option / free context */
|
2017-07-25 16:54:25 +08:00
|
|
|
if (block_ctx) {
|
2017-10-23 10:10:08 +08:00
|
|
|
if (!last_block) {
|
2017-07-25 16:54:25 +08:00
|
|
|
/* More to come, ack with correspond block # */
|
2019-01-24 14:40:23 -08:00
|
|
|
r = coap_append_block1_option(msg->out.out_cpkt,
|
2017-09-19 16:04:32 -07:00
|
|
|
&block_ctx->ctx);
|
2017-10-19 16:27:32 -07:00
|
|
|
if (r < 0) {
|
2017-07-25 16:54:25 +08:00
|
|
|
/* report as internal server error */
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("Fail adding block1 option: %d", r);
|
2017-07-25 16:54:25 +08:00
|
|
|
r = -EINVAL;
|
|
|
|
goto error;
|
|
|
|
}
|
2017-10-26 23:45:59 -07:00
|
|
|
} else {
|
|
|
|
/* Free context when finished */
|
|
|
|
free_block_ctx(block_ctx);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
2017-07-25 16:54:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
error:
|
2017-10-11 14:04:39 -07:00
|
|
|
lwm2m_reset_message(msg, false);
|
2017-07-25 16:54:25 +08:00
|
|
|
if (r == -ENOENT) {
|
2017-09-19 16:04:32 -07:00
|
|
|
msg->code = COAP_RESPONSE_CODE_NOT_FOUND;
|
2017-07-25 16:54:25 +08:00
|
|
|
} else if (r == -EPERM) {
|
2017-09-19 16:04:32 -07:00
|
|
|
msg->code = COAP_RESPONSE_CODE_NOT_ALLOWED;
|
2017-07-25 16:54:25 +08:00
|
|
|
} else if (r == -EEXIST) {
|
2017-09-19 16:04:32 -07:00
|
|
|
msg->code = COAP_RESPONSE_CODE_BAD_REQUEST;
|
2017-11-02 18:36:24 +08:00
|
|
|
} else if (r == -EFAULT) {
|
|
|
|
msg->code = COAP_RESPONSE_CODE_INCOMPLETE;
|
2017-07-25 16:54:25 +08:00
|
|
|
} else if (r == -EFBIG) {
|
2017-09-19 16:04:32 -07:00
|
|
|
msg->code = COAP_RESPONSE_CODE_REQUEST_TOO_LARGE;
|
2017-11-21 18:16:52 +08:00
|
|
|
} else if (r == -ENOTSUP) {
|
|
|
|
msg->code = COAP_RESPONSE_CODE_NOT_IMPLEMENTED;
|
2018-01-30 15:18:57 +08:00
|
|
|
} else if (r == -ENOMSG) {
|
|
|
|
msg->code = COAP_RESPONSE_CODE_UNSUPPORTED_CONTENT_FORMAT;
|
2017-07-25 16:54:25 +08:00
|
|
|
} else {
|
|
|
|
/* Failed to handle the request */
|
2017-09-19 16:04:32 -07:00
|
|
|
msg->code = COAP_RESPONSE_CODE_INTERNAL_ERROR;
|
2017-07-25 16:54:25 +08:00
|
|
|
}
|
|
|
|
|
2017-10-11 14:04:39 -07:00
|
|
|
r = lwm2m_init_message(msg);
|
|
|
|
if (r < 0) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("Error recreating message: %d", r);
|
2017-10-11 14:04:39 -07:00
|
|
|
}
|
|
|
|
|
2017-07-25 16:54:25 +08:00
|
|
|
/* Free block context when error happened */
|
|
|
|
free_block_ctx(block_ctx);
|
|
|
|
|
|
|
|
return 0;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
static void lwm2m_udp_receive(struct lwm2m_ctx *client_ctx,
|
|
|
|
u8_t *buf, u16_t buf_len,
|
|
|
|
struct sockaddr *from_addr,
|
2019-01-25 21:43:52 -08:00
|
|
|
udp_request_handler_cb_t udp_request_handler)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
2017-09-01 01:11:43 -07:00
|
|
|
struct lwm2m_message *msg = NULL;
|
2017-09-19 16:04:32 -07:00
|
|
|
struct coap_pending *pending;
|
|
|
|
struct coap_reply *reply;
|
|
|
|
struct coap_packet response;
|
|
|
|
int r;
|
|
|
|
u8_t token[8];
|
2017-07-07 11:04:03 -07:00
|
|
|
u8_t tkl;
|
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
r = coap_packet_parse(&response, buf, buf_len, NULL, 0);
|
2017-07-07 11:04:03 -07:00
|
|
|
if (r < 0) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("Invalid data received (err:%d)", r);
|
2019-01-25 14:13:07 -08:00
|
|
|
return;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
tkl = coap_header_get_token(&response, token);
|
|
|
|
pending = coap_pending_received(&response, client_ctx->pendings,
|
2017-08-24 08:51:25 -07:00
|
|
|
CONFIG_LWM2M_ENGINE_MAX_PENDING);
|
2017-09-01 01:11:43 -07:00
|
|
|
/*
|
2017-09-19 16:04:32 -07:00
|
|
|
* Clear pending pointer because coap_pending_received() calls
|
2017-10-11 11:29:54 -07:00
|
|
|
* coap_pending_clear, and later when we call lwm2m_reset_message()
|
2017-09-19 16:04:32 -07:00
|
|
|
* it will try and call coap_pending_clear() again if msg->pending
|
2017-09-01 01:11:43 -07:00
|
|
|
* is != NULL.
|
|
|
|
*/
|
2017-07-07 11:04:03 -07:00
|
|
|
if (pending) {
|
2017-09-05 21:03:49 -07:00
|
|
|
msg = find_msg(pending, NULL);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_DBG("checking for reply from [%s]",
|
2019-05-29 16:49:18 -07:00
|
|
|
log_strdup(lwm2m_sprint_ip_addr(from_addr)));
|
2019-01-28 11:55:23 -08:00
|
|
|
reply = coap_response_received(&response, from_addr,
|
2017-08-24 08:51:25 -07:00
|
|
|
client_ctx->replies,
|
|
|
|
CONFIG_LWM2M_ENGINE_MAX_REPLIES);
|
2017-09-01 01:11:43 -07:00
|
|
|
if (reply) {
|
2017-07-28 19:21:58 -03:00
|
|
|
/*
|
|
|
|
* Separate response is composed of 2 messages, empty ACK with
|
|
|
|
* no token and an additional message with a matching token id
|
|
|
|
* (based on the token used by the CON request).
|
|
|
|
*
|
|
|
|
* Since the ACK received by the notify CON messages are also
|
|
|
|
* empty with no token (consequence of always using the same
|
|
|
|
* token id for all notifications), we have to use an
|
|
|
|
* additional flag to decide when to clear the reply callback.
|
|
|
|
*/
|
2019-01-25 21:43:52 -08:00
|
|
|
if (client_ctx->handle_separate_response && !tkl &&
|
2017-09-19 16:04:32 -07:00
|
|
|
coap_header_get_type(&response) == COAP_TYPE_ACK) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_DBG("separated response, not removing reply");
|
2019-01-25 14:13:07 -08:00
|
|
|
return;
|
2017-09-01 01:11:43 -07:00
|
|
|
}
|
2017-09-05 21:03:49 -07:00
|
|
|
|
|
|
|
if (!msg) {
|
|
|
|
msg = find_msg(pending, reply);
|
|
|
|
}
|
2017-09-01 01:11:43 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (reply || pending) {
|
2018-03-23 10:49:44 -07:00
|
|
|
/* skip release if reply->user_data has error condition */
|
|
|
|
if (reply && reply->user_data != COAP_REPLY_STATUS_NONE) {
|
|
|
|
/* reset reply->user_data for next time */
|
|
|
|
reply->user_data = (void *)COAP_REPLY_STATUS_NONE;
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_DBG("reply %p NOT removed", reply);
|
2019-01-25 14:13:07 -08:00
|
|
|
return;
|
2018-03-23 10:49:44 -07:00
|
|
|
}
|
|
|
|
|
2017-09-01 01:11:43 -07:00
|
|
|
/* free up msg resources */
|
|
|
|
if (msg) {
|
2017-10-11 11:29:54 -07:00
|
|
|
lwm2m_reset_message(msg, true);
|
2017-09-01 01:11:43 -07:00
|
|
|
}
|
|
|
|
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_DBG("reply %p handled and removed", reply);
|
2019-01-25 14:13:07 -08:00
|
|
|
return;
|
2017-09-01 01:11:43 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If no normal response handler is found, then this is
|
|
|
|
* a new request coming from the server. Let's look
|
|
|
|
* at registered objects to find a handler.
|
|
|
|
*/
|
|
|
|
if (udp_request_handler &&
|
2017-09-19 16:04:32 -07:00
|
|
|
coap_header_get_type(&response) == COAP_TYPE_CON) {
|
2017-09-01 01:11:43 -07:00
|
|
|
msg = lwm2m_get_message(client_ctx);
|
|
|
|
if (!msg) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("Unable to get a lwm2m message!");
|
2019-01-25 14:13:07 -08:00
|
|
|
return;
|
2017-09-01 01:11:43 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Create a response message if we reach this point */
|
2017-09-19 16:04:32 -07:00
|
|
|
msg->type = COAP_TYPE_ACK;
|
|
|
|
msg->code = coap_header_get_code(&response);
|
|
|
|
msg->mid = coap_header_get_id(&response);
|
2017-09-01 01:11:43 -07:00
|
|
|
/* skip token generation by default */
|
|
|
|
msg->tkl = LWM2M_MSG_TOKEN_LEN_SKIP;
|
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
/* process the response to this request */
|
|
|
|
r = udp_request_handler(&response, msg);
|
2017-09-01 01:11:43 -07:00
|
|
|
if (r < 0) {
|
2019-01-25 14:13:07 -08:00
|
|
|
return;
|
2017-09-01 01:11:43 -07:00
|
|
|
}
|
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
r = lwm2m_send_message(msg);
|
2017-09-01 01:11:43 -07:00
|
|
|
if (r < 0) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("Err sending response: %d", r);
|
2017-10-11 11:29:54 -07:00
|
|
|
lwm2m_reset_message(msg, true);
|
2017-07-28 19:21:58 -03:00
|
|
|
}
|
2017-09-01 01:11:43 -07:00
|
|
|
} else {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("No handler for response");
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void retransmit_request(struct k_work *work)
|
|
|
|
{
|
2017-08-23 23:14:06 -07:00
|
|
|
struct lwm2m_ctx *client_ctx;
|
2017-09-01 01:11:43 -07:00
|
|
|
struct lwm2m_message *msg;
|
2017-09-19 16:04:32 -07:00
|
|
|
struct coap_pending *pending;
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2017-08-23 23:14:06 -07:00
|
|
|
client_ctx = CONTAINER_OF(work, struct lwm2m_ctx, retransmit_work);
|
2017-09-19 16:04:32 -07:00
|
|
|
pending = coap_pending_next_to_expire(client_ctx->pendings,
|
2017-08-24 08:51:25 -07:00
|
|
|
CONFIG_LWM2M_ENGINE_MAX_PENDING);
|
2017-07-07 11:04:03 -07:00
|
|
|
if (!pending) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-09-05 21:03:49 -07:00
|
|
|
msg = find_msg(pending, NULL);
|
2017-09-01 01:11:43 -07:00
|
|
|
if (!msg) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("pending has no valid LwM2M message!");
|
2017-07-07 11:04:03 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-09-13 08:52:42 -07:00
|
|
|
if (!coap_pending_cycle(pending)) {
|
|
|
|
/* pending request has expired */
|
|
|
|
if (msg->message_timeout_cb) {
|
|
|
|
msg->message_timeout_cb(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* coap_pending_clear() is called in lwm2m_reset_message()
|
|
|
|
* which balances the ref we made in coap_pending_cycle()
|
|
|
|
*/
|
|
|
|
lwm2m_reset_message(msg, true);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-02-18 21:54:12 -08:00
|
|
|
LOG_INF("Resending message: %p", msg);
|
2018-01-30 13:50:59 -08:00
|
|
|
msg->send_attempts++;
|
2019-02-12 17:10:41 -08:00
|
|
|
if (send(msg->ctx->sock_fd, msg->cpkt.data, msg->cpkt.offset, 0) < 0) {
|
|
|
|
LOG_ERR("Error sending lwm2m message: %d", -errno);
|
2018-01-30 13:50:59 -08:00
|
|
|
/* don't error here, retry until timeout */
|
|
|
|
}
|
|
|
|
|
2017-08-23 23:14:06 -07:00
|
|
|
k_delayed_work_submit(&client_ctx->retransmit_work, pending->timeout);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
static int notify_message_reply_cb(const struct coap_packet *response,
|
|
|
|
struct coap_reply *reply,
|
2017-07-07 11:04:03 -07:00
|
|
|
const struct sockaddr *from)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
u8_t type, code;
|
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
type = coap_header_get_type(response);
|
|
|
|
code = coap_header_get_code(response);
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_DBG("NOTIFY ACK type:%u code:%d.%d reply_token:'%s'",
|
2017-07-07 11:04:03 -07:00
|
|
|
type,
|
2017-09-19 16:04:32 -07:00
|
|
|
COAP_RESPONSE_CODE_CLASS(code),
|
|
|
|
COAP_RESPONSE_CODE_DETAIL(code),
|
2019-05-29 16:49:18 -07:00
|
|
|
log_strdup(sprint_token(reply->token, reply->tkl)));
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
/* remove observer on COAP_TYPE_RESET */
|
|
|
|
if (type == COAP_TYPE_RESET) {
|
2017-07-07 11:04:03 -07:00
|
|
|
if (reply->tkl > 0) {
|
|
|
|
ret = engine_remove_observer(reply->token, reply->tkl);
|
|
|
|
if (ret) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("remove observe error: %d", ret);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
} else {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("notify reply missing token -- ignored.");
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int generate_notify_message(struct observe_node *obs,
|
|
|
|
bool manual_trigger)
|
|
|
|
{
|
2017-09-01 01:11:43 -07:00
|
|
|
struct lwm2m_message *msg;
|
2017-07-07 11:04:03 -07:00
|
|
|
struct lwm2m_engine_obj_inst *obj_inst;
|
|
|
|
int ret = 0;
|
|
|
|
|
2017-09-01 00:42:05 -07:00
|
|
|
if (!obs->ctx) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("observer has no valid LwM2M ctx!");
|
2017-08-23 23:14:06 -07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
msg = lwm2m_get_message(obs->ctx);
|
|
|
|
if (!msg) {
|
|
|
|
LOG_ERR("Unable to get a lwm2m message!");
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* copy path */
|
|
|
|
memcpy(&msg->path, &obs->path, sizeof(struct lwm2m_obj_path));
|
|
|
|
msg->operation = LWM2M_OP_READ;
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_DBG("[%s] NOTIFY MSG START: %u/%u/%u(%u) token:'%s' [%s] %lld",
|
|
|
|
manual_trigger ? "MANUAL" : "AUTO",
|
|
|
|
obs->path.obj_id,
|
|
|
|
obs->path.obj_inst_id,
|
|
|
|
obs->path.res_id,
|
|
|
|
obs->path.level,
|
2019-05-29 16:49:18 -07:00
|
|
|
log_strdup(sprint_token(obs->token, obs->tkl)),
|
|
|
|
log_strdup(lwm2m_sprint_ip_addr(&obs->ctx->remote_addr)),
|
2018-09-19 11:22:19 +03:00
|
|
|
k_uptime_get());
|
2017-07-07 11:04:03 -07:00
|
|
|
|
|
|
|
obj_inst = get_engine_obj_inst(obs->path.obj_id,
|
|
|
|
obs->path.obj_inst_id);
|
|
|
|
if (!obj_inst) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("unable to get engine obj for %u/%u",
|
|
|
|
obs->path.obj_id,
|
|
|
|
obs->path.obj_inst_id);
|
2019-01-24 14:40:23 -08:00
|
|
|
ret = -EINVAL;
|
|
|
|
goto cleanup;
|
2017-09-01 01:11:43 -07:00
|
|
|
}
|
|
|
|
|
2017-09-19 16:04:32 -07:00
|
|
|
msg->type = COAP_TYPE_CON;
|
|
|
|
msg->code = COAP_RESPONSE_CODE_CONTENT;
|
2018-11-29 11:23:03 -08:00
|
|
|
msg->mid = 0U;
|
2017-09-01 01:11:43 -07:00
|
|
|
msg->token = obs->token;
|
|
|
|
msg->tkl = obs->tkl;
|
|
|
|
msg->reply_cb = notify_message_reply_cb;
|
2019-01-24 14:40:23 -08:00
|
|
|
msg->out.out_cpkt = &msg->cpkt;
|
2017-09-01 01:11:43 -07:00
|
|
|
|
|
|
|
ret = lwm2m_init_message(msg);
|
2017-09-19 16:04:32 -07:00
|
|
|
if (ret < 0) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("Unable to init lwm2m message! (err: %d)", ret);
|
2017-09-19 16:04:32 -07:00
|
|
|
goto cleanup;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* each notification should increment the obs counter */
|
|
|
|
obs->counter++;
|
2017-09-19 16:04:32 -07:00
|
|
|
ret = coap_append_option_int(&msg->cpkt, COAP_OPTION_OBSERVE,
|
|
|
|
obs->counter);
|
|
|
|
if (ret < 0) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("OBSERVE option error: %d", ret);
|
2017-07-07 11:04:03 -07:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* set the output writer */
|
2019-01-24 14:40:23 -08:00
|
|
|
select_writer(&msg->out, obs->format);
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2019-01-24 14:40:23 -08:00
|
|
|
ret = do_read_op(obj_inst->obj, msg, obs->format);
|
2017-09-19 16:04:32 -07:00
|
|
|
if (ret < 0) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("error in multi-format read (err:%d)", ret);
|
2017-07-07 11:04:03 -07:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2017-09-01 01:11:43 -07:00
|
|
|
ret = lwm2m_send_message(msg);
|
2017-07-07 11:04:03 -07:00
|
|
|
if (ret < 0) {
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_ERR("Error sending LWM2M packet (err:%d).", ret);
|
2017-07-07 11:04:03 -07:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2018-09-19 11:22:19 +03:00
|
|
|
LOG_DBG("NOTIFY MSG: SENT");
|
2017-09-01 01:11:43 -07:00
|
|
|
return 0;
|
2017-07-07 11:04:03 -07:00
|
|
|
|
|
|
|
cleanup:
|
2017-10-11 11:29:54 -07:00
|
|
|
lwm2m_reset_message(msg, true);
|
2017-07-07 11:04:03 -07:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2017-10-10 20:55:05 -07:00
|
|
|
s32_t engine_next_service_timeout_ms(u32_t max_timeout)
|
|
|
|
{
|
|
|
|
struct service_node *srv;
|
|
|
|
u64_t time_left_ms, timestamp = k_uptime_get();
|
|
|
|
u32_t timeout = max_timeout;
|
|
|
|
|
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER(&engine_service_list, srv, node) {
|
|
|
|
time_left_ms = srv->last_timestamp +
|
|
|
|
K_MSEC(srv->min_call_period);
|
|
|
|
|
|
|
|
/* service is due */
|
|
|
|
if (time_left_ms < timestamp) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* service timeout is less than the current timeout */
|
|
|
|
time_left_ms -= timestamp;
|
|
|
|
if (time_left_ms < timeout) {
|
|
|
|
timeout = time_left_ms;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return timeout;
|
|
|
|
}
|
|
|
|
|
2019-01-25 22:47:58 -08:00
|
|
|
int lwm2m_engine_add_service(k_work_handler_t service, u32_t period_ms)
|
2017-10-10 20:55:05 -07:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* find an unused service index node */
|
|
|
|
for (i = 0; i < MAX_PERIODIC_SERVICE; i++) {
|
2019-04-05 16:14:50 -07:00
|
|
|
if (!service_node_data[i].service_work) {
|
2017-10-10 20:55:05 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i == MAX_PERIODIC_SERVICE) {
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
2019-04-05 16:14:50 -07:00
|
|
|
service_node_data[i].service_work = service;
|
2017-10-10 20:55:05 -07:00
|
|
|
service_node_data[i].min_call_period = period_ms;
|
2019-03-26 19:57:45 -06:00
|
|
|
service_node_data[i].last_timestamp = 0U;
|
2017-10-10 20:55:05 -07:00
|
|
|
|
|
|
|
sys_slist_append(&engine_service_list,
|
|
|
|
&service_node_data[i].node);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-04-05 16:14:50 -07:00
|
|
|
static int lwm2m_engine_service(void)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
|
|
|
struct observe_node *obs;
|
2017-10-10 20:55:05 -07:00
|
|
|
struct service_node *srv;
|
|
|
|
s64_t timestamp, service_due_timestamp;
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2019-01-25 22:47:58 -08:00
|
|
|
/*
|
|
|
|
* 1. scan the observer list
|
|
|
|
* 2. For each notify event found, scan the observer list
|
|
|
|
* 3. For each observer match, generate a NOTIFY message,
|
|
|
|
* attaching the notify response handler
|
|
|
|
*/
|
|
|
|
timestamp = k_uptime_get();
|
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER(&engine_observer_list, obs, node) {
|
2017-07-07 11:04:03 -07:00
|
|
|
/*
|
2019-01-25 22:47:58 -08:00
|
|
|
* manual notify requirements:
|
|
|
|
* - event_timestamp > last_timestamp
|
|
|
|
* - current timestamp > last_timestamp + min_period_sec
|
2017-07-07 11:04:03 -07:00
|
|
|
*/
|
2019-01-25 22:47:58 -08:00
|
|
|
if (obs->event_timestamp > obs->last_timestamp &&
|
|
|
|
timestamp > obs->last_timestamp +
|
|
|
|
K_SECONDS(obs->min_period_sec)) {
|
|
|
|
obs->last_timestamp = k_uptime_get();
|
|
|
|
generate_notify_message(obs, true);
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2019-01-25 22:47:58 -08:00
|
|
|
/*
|
|
|
|
* automatic time-based notify requirements:
|
|
|
|
* - current timestamp > last_timestamp + max_period_sec
|
|
|
|
*/
|
|
|
|
} else if (timestamp > obs->last_timestamp +
|
|
|
|
K_SECONDS(obs->min_period_sec)) {
|
|
|
|
obs->last_timestamp = k_uptime_get();
|
|
|
|
generate_notify_message(obs, false);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2019-01-25 22:47:58 -08:00
|
|
|
}
|
2017-10-10 20:55:05 -07:00
|
|
|
|
2019-01-25 22:47:58 -08:00
|
|
|
timestamp = k_uptime_get();
|
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER(&engine_service_list, srv, node) {
|
|
|
|
service_due_timestamp = srv->last_timestamp +
|
|
|
|
K_MSEC(srv->min_call_period);
|
|
|
|
/* service is due */
|
2019-04-08 11:31:46 -07:00
|
|
|
if (timestamp >= service_due_timestamp) {
|
2019-01-25 22:47:58 -08:00
|
|
|
srv->last_timestamp = k_uptime_get();
|
2019-04-05 16:14:50 -07:00
|
|
|
srv->service_work(NULL);
|
2017-10-10 20:55:05 -07:00
|
|
|
}
|
2019-01-25 22:47:58 -08:00
|
|
|
}
|
2017-10-10 20:55:05 -07:00
|
|
|
|
2019-01-25 22:47:58 -08:00
|
|
|
/* calculate how long to sleep till the next service */
|
2019-04-05 16:14:50 -07:00
|
|
|
return engine_next_service_timeout_ms(ENGINE_UPDATE_INTERVAL);
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
2019-01-25 22:36:47 -08:00
|
|
|
int lwm2m_engine_context_close(struct lwm2m_ctx *client_ctx)
|
|
|
|
{
|
2019-02-18 21:51:01 -08:00
|
|
|
struct observe_node *obs, *tmp;
|
|
|
|
sys_snode_t *prev_node = NULL;
|
2019-01-28 11:55:23 -08:00
|
|
|
int sock_fd = client_ctx->sock_fd;
|
|
|
|
|
2019-02-18 21:51:01 -08:00
|
|
|
/* Remove observes for this context */
|
|
|
|
SYS_SLIST_FOR_EACH_CONTAINER_SAFE(&engine_observer_list,
|
|
|
|
obs, tmp, node) {
|
|
|
|
if (obs->ctx == client_ctx) {
|
|
|
|
sys_slist_remove(&engine_observer_list, prev_node,
|
|
|
|
&obs->node);
|
|
|
|
(void)memset(obs, 0, sizeof(*obs));
|
|
|
|
} else {
|
|
|
|
prev_node = &obs->node;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
lwm2m_socket_del(client_ctx);
|
|
|
|
client_ctx->sock_fd = -1;
|
|
|
|
if (sock_fd >= 0) {
|
|
|
|
return close(sock_fd);
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
2019-01-25 22:36:47 -08:00
|
|
|
}
|
|
|
|
|
2017-09-08 08:22:59 -07:00
|
|
|
void lwm2m_engine_context_init(struct lwm2m_ctx *client_ctx)
|
|
|
|
{
|
|
|
|
k_delayed_work_init(&client_ctx->retransmit_work, retransmit_request);
|
|
|
|
}
|
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
/* LwM2M Socket Integration */
|
|
|
|
|
|
|
|
int lwm2m_socket_add(struct lwm2m_ctx *ctx)
|
2017-12-04 21:54:16 -08:00
|
|
|
{
|
2019-01-28 11:55:23 -08:00
|
|
|
int i;
|
|
|
|
|
|
|
|
if (sock_nfds < MAX_POLL_FD) {
|
|
|
|
i = sock_nfds++;
|
|
|
|
} else {
|
|
|
|
for (i = 0; i < MAX_POLL_FD; i++) {
|
|
|
|
if (sock_ctx[i] == NULL) {
|
|
|
|
goto found;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
found:
|
|
|
|
sock_ctx[i] = ctx;
|
|
|
|
sock_fds[i].fd = ctx->sock_fd;
|
|
|
|
sock_fds[i].events = POLLIN;
|
2017-12-04 21:54:16 -08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
void lwm2m_socket_del(struct lwm2m_ctx *ctx)
|
2017-07-07 11:04:03 -07:00
|
|
|
{
|
2019-01-28 11:55:23 -08:00
|
|
|
for (int i = 0; i < sock_nfds; i++) {
|
|
|
|
if (sock_ctx[i] == ctx) {
|
|
|
|
sock_ctx[i] = NULL;
|
|
|
|
sock_fds[i].fd = -1;
|
|
|
|
sock_nfds--;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-05-16 15:50:32 -07:00
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
/* LwM2M main work loop */
|
2018-05-16 15:50:32 -07:00
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
static void socket_receive_loop(void)
|
|
|
|
{
|
|
|
|
static u8_t in_buf[NET_IPV6_MTU];
|
|
|
|
static struct sockaddr from_addr;
|
|
|
|
socklen_t from_addr_len;
|
|
|
|
ssize_t len;
|
|
|
|
int i;
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
from_addr_len = sizeof(from_addr);
|
|
|
|
while (1) {
|
|
|
|
/* wait for sockets */
|
|
|
|
if (sock_nfds < 1) {
|
2019-04-05 16:14:50 -07:00
|
|
|
k_sleep(lwm2m_engine_service());
|
2019-01-28 11:55:23 -08:00
|
|
|
continue;
|
|
|
|
}
|
2017-08-28 14:39:50 -07:00
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
/*
|
|
|
|
* FIXME: Currently we timeout and restart poll in case fds
|
|
|
|
* were modified.
|
|
|
|
*/
|
2019-04-05 16:14:50 -07:00
|
|
|
if (poll(sock_fds, sock_nfds, lwm2m_engine_service()) < 0) {
|
2019-01-28 11:55:23 -08:00
|
|
|
LOG_ERR("Error in poll:%d", errno);
|
|
|
|
errno = 0;
|
|
|
|
k_sleep(ENGINE_UPDATE_INTERVAL);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < sock_nfds; i++) {
|
|
|
|
if (sock_fds[i].revents & POLLERR) {
|
|
|
|
LOG_ERR("Error in poll.. waiting a moment.");
|
|
|
|
k_sleep(ENGINE_UPDATE_INTERVAL);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(sock_fds[i].revents & POLLIN) ||
|
|
|
|
sock_ctx[i] == NULL) {
|
|
|
|
sock_fds[i].revents = 0;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
sock_fds[i].revents = 0;
|
|
|
|
len = recvfrom(sock_ctx[i]->sock_fd, in_buf,
|
|
|
|
sizeof(in_buf) - 1, 0,
|
|
|
|
&from_addr, &from_addr_len);
|
|
|
|
|
|
|
|
if (len < 0) {
|
|
|
|
LOG_ERR("Error reading response: %d", errno);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (len == 0) {
|
|
|
|
LOG_ERR("Zero length recv");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-03-26 19:57:45 -06:00
|
|
|
in_buf[len] = 0U;
|
2019-01-28 11:55:23 -08:00
|
|
|
|
|
|
|
lwm2m_udp_receive(sock_ctx[i], in_buf, len, &from_addr,
|
|
|
|
handle_request);
|
2019-01-25 21:43:52 -08:00
|
|
|
}
|
2017-12-04 21:54:16 -08:00
|
|
|
}
|
2019-01-28 11:55:23 -08:00
|
|
|
}
|
2017-12-04 21:54:16 -08:00
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
#if defined(CONFIG_LWM2M_DTLS_SUPPORT)
|
|
|
|
static int load_tls_credential(struct lwm2m_ctx *client_ctx, u16_t res_id,
|
|
|
|
enum tls_credential_type type)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
void *cred = NULL;
|
|
|
|
u16_t cred_len;
|
|
|
|
u8_t cred_flags;
|
|
|
|
char pathstr[MAX_RESOURCE_LEN];
|
|
|
|
|
|
|
|
/* ignore error value */
|
|
|
|
tls_credential_delete(client_ctx->tls_tag, type);
|
|
|
|
|
|
|
|
snprintk(pathstr, sizeof(pathstr), "0/%d/%u", client_ctx->sec_obj_inst,
|
|
|
|
res_id);
|
|
|
|
|
|
|
|
ret = lwm2m_engine_get_res_data(pathstr, &cred, &cred_len, &cred_flags);
|
2017-08-28 14:39:50 -07:00
|
|
|
if (ret < 0) {
|
2019-05-29 16:49:18 -07:00
|
|
|
LOG_ERR("Unable to get resource data for '%s'",
|
|
|
|
log_strdup(pathstr));
|
2019-01-28 11:55:23 -08:00
|
|
|
return ret;
|
2017-08-28 14:39:50 -07:00
|
|
|
}
|
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
/* Set correct PSK_ID length */
|
|
|
|
if (type == TLS_CREDENTIAL_PSK_ID) {
|
|
|
|
cred_len = strlen(cred);
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = tls_credential_add(client_ctx->tls_tag, type, cred, cred_len);
|
|
|
|
if (ret < 0) {
|
2019-05-29 16:49:18 -07:00
|
|
|
LOG_ERR("Unable to get resource data for '%s'",
|
|
|
|
log_strdup(pathstr));
|
2019-01-25 14:39:41 -08:00
|
|
|
}
|
2017-08-28 14:39:50 -07:00
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
return ret;
|
|
|
|
}
|
2019-01-28 11:55:23 -08:00
|
|
|
#endif /* CONFIG_LWM2M_DTLS_SUPPORT */
|
2017-07-07 11:04:03 -07:00
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
int lwm2m_socket_start(struct lwm2m_ctx *client_ctx)
|
2019-01-25 21:43:52 -08:00
|
|
|
{
|
2019-01-28 11:55:23 -08:00
|
|
|
#if defined(CONFIG_LWM2M_DTLS_SUPPORT)
|
|
|
|
int ret;
|
2019-01-25 21:43:52 -08:00
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
ret = load_tls_credential(client_ctx, 3, TLS_CREDENTIAL_PSK_ID);
|
2019-01-25 21:43:52 -08:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
ret = load_tls_credential(client_ctx, 5, TLS_CREDENTIAL_PSK);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
2019-01-25 21:43:52 -08:00
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
if (client_ctx->use_dtls) {
|
|
|
|
client_ctx->sock_fd = socket(client_ctx->remote_addr.sa_family,
|
|
|
|
SOCK_DGRAM, IPPROTO_DTLS_1_2);
|
|
|
|
} else
|
|
|
|
#endif /* CONFIG_LWM2M_DTLS_SUPPORT */
|
|
|
|
{
|
|
|
|
client_ctx->sock_fd = socket(client_ctx->remote_addr.sa_family,
|
|
|
|
SOCK_DGRAM, IPPROTO_UDP);
|
2019-01-25 21:43:52 -08:00
|
|
|
}
|
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
if (client_ctx->sock_fd < 0) {
|
|
|
|
LOG_ERR("Failed to create socket: %d", errno);
|
|
|
|
return -errno;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined(CONFIG_LWM2M_DTLS_SUPPORT)
|
|
|
|
if (client_ctx->use_dtls) {
|
|
|
|
sec_tag_t tls_tag_list[] = {
|
|
|
|
client_ctx->tls_tag,
|
|
|
|
};
|
|
|
|
|
|
|
|
ret = setsockopt(client_ctx->sock_fd, SOL_TLS, TLS_SEC_TAG_LIST,
|
|
|
|
tls_tag_list, sizeof(tls_tag_list));
|
|
|
|
if (ret < 0) {
|
|
|
|
LOG_ERR("Failed to set TLS_SEC_TAG_LIST option: %d",
|
|
|
|
errno);
|
|
|
|
lwm2m_engine_context_close(client_ctx);
|
|
|
|
return -errno;
|
|
|
|
}
|
2019-01-25 21:43:52 -08:00
|
|
|
}
|
2019-01-28 11:55:23 -08:00
|
|
|
#endif /* CONFIG_LWM2M_DTLS_SUPPORT */
|
2019-01-25 21:43:52 -08:00
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
if (connect(client_ctx->sock_fd, &client_ctx->remote_addr,
|
|
|
|
NET_SOCKADDR_MAX_SIZE) < 0) {
|
|
|
|
LOG_ERR("Cannot connect UDP (-%d)", errno);
|
|
|
|
lwm2m_engine_context_close(client_ctx);
|
|
|
|
return -errno;
|
|
|
|
}
|
|
|
|
|
|
|
|
lwm2m_socket_add(client_ctx);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_parse_peerinfo(char *url, struct sockaddr *addr, bool *use_dtls)
|
|
|
|
{
|
|
|
|
struct http_parser_url parser;
|
2019-01-31 09:21:51 -08:00
|
|
|
#if defined(CONFIG_DNS_RESOLVER)
|
|
|
|
struct addrinfo hints, *res;
|
|
|
|
#endif
|
2019-01-28 11:55:23 -08:00
|
|
|
int ret;
|
|
|
|
u16_t off, len;
|
|
|
|
u8_t tmp;
|
|
|
|
|
|
|
|
http_parser_url_init(&parser);
|
|
|
|
ret = http_parser_parse_url(url, strlen(url), 0, &parser);
|
|
|
|
if (ret < 0) {
|
2019-05-29 16:49:18 -07:00
|
|
|
LOG_ERR("Invalid url: %s", log_strdup(url));
|
2019-01-28 11:55:23 -08:00
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
|
|
|
|
|
|
|
off = parser.field_data[UF_SCHEMA].off;
|
|
|
|
len = parser.field_data[UF_SCHEMA].len;
|
|
|
|
|
|
|
|
/* check for supported protocol */
|
|
|
|
if (strncmp(url + off, "coaps", len) != 0) {
|
2019-01-25 21:43:52 -08:00
|
|
|
return -EPROTONOSUPPORT;
|
|
|
|
}
|
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
/* check for DTLS requirement */
|
|
|
|
*use_dtls = false;
|
2019-03-26 19:57:45 -06:00
|
|
|
if (len == 5U && strncmp(url + off, "coaps", len) == 0) {
|
2019-01-25 21:43:52 -08:00
|
|
|
#if defined(CONFIG_LWM2M_DTLS_SUPPORT)
|
2019-01-28 11:55:23 -08:00
|
|
|
*use_dtls = true;
|
2019-01-25 21:43:52 -08:00
|
|
|
#else
|
|
|
|
return -EPROTONOSUPPORT;
|
|
|
|
#endif /* CONFIG_LWM2M_DTLS_SUPPORT */
|
|
|
|
}
|
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
if (!(parser.field_set & (1 << UF_PORT))) {
|
|
|
|
/* Set to default port of CoAP */
|
|
|
|
parser.port = CONFIG_LWM2M_PEER_PORT;
|
2019-01-25 21:43:52 -08:00
|
|
|
}
|
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
off = parser.field_data[UF_HOST].off;
|
|
|
|
len = parser.field_data[UF_HOST].len;
|
2019-01-25 21:43:52 -08:00
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
/* truncate host portion */
|
|
|
|
tmp = url[off + len];
|
|
|
|
url[off + len] = '\0';
|
|
|
|
|
|
|
|
/* initialize addr */
|
|
|
|
(void)memset(addr, 0, sizeof(*addr));
|
2019-01-25 21:43:52 -08:00
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
/* try and set IP address directly */
|
|
|
|
ret = -EINVAL;
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV6)
|
|
|
|
addr->sa_family = AF_INET6;
|
|
|
|
ret = net_addr_pton(AF_INET6, url + off,
|
|
|
|
&((struct sockaddr_in6 *)addr)->sin6_addr);
|
|
|
|
#endif /* CONFIG_NET_IPV6 */
|
|
|
|
|
|
|
|
#if defined(CONFIG_NET_IPV4)
|
|
|
|
if (ret < 0) {
|
|
|
|
addr->sa_family = AF_INET;
|
|
|
|
ret = net_addr_pton(AF_INET, url + off,
|
|
|
|
&((struct sockaddr_in *)addr)->sin_addr);
|
|
|
|
}
|
|
|
|
#endif /* CONFIG_NET_IPV4 */
|
|
|
|
|
|
|
|
if (ret < 0) {
|
2019-01-31 09:21:51 -08:00
|
|
|
#if defined(CONFIG_DNS_RESOLVER)
|
|
|
|
#if defined(CONFIG_NET_IPV6) && defined(CONFIG_NET_IPV4)
|
|
|
|
hints.ai_family = AF_UNSPEC;
|
|
|
|
#elif defined(CONFIG_NET_IPV6)
|
|
|
|
hints.ai_family = AF_INET6;
|
|
|
|
#else
|
|
|
|
hints.ai_family = AF_INET;
|
|
|
|
#endif /* defined(CONFIG_NET_IPV6) && defined(CONFIG_NET_IPV4) */
|
|
|
|
hints.ai_socktype = SOCK_DGRAM;
|
|
|
|
ret = getaddrinfo(url + off, NULL, &hints, &res);
|
|
|
|
if (ret != 0) {
|
|
|
|
LOG_ERR("Unable to resolve address");
|
|
|
|
/* DNS error codes don't align with normal errors */
|
|
|
|
ret = -ENOENT;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(addr, res->ai_addr, sizeof(*addr));
|
|
|
|
addr->sa_family = res->ai_family;
|
|
|
|
free(res);
|
|
|
|
#else
|
2019-01-28 11:55:23 -08:00
|
|
|
goto cleanup;
|
2019-01-31 09:21:51 -08:00
|
|
|
#endif /* CONFIG_DNS_RESOLVER */
|
2019-01-28 11:55:23 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* set port */
|
|
|
|
if (addr->sa_family == AF_INET6) {
|
|
|
|
net_sin6(addr)->sin6_port = htons(parser.port);
|
|
|
|
} else if (addr->sa_family == AF_INET) {
|
|
|
|
net_sin(addr)->sin_port = htons(parser.port);
|
|
|
|
} else {
|
|
|
|
ret = -EPROTONOSUPPORT;
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
/* restore host separator */
|
|
|
|
url[off + len] = tmp;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int lwm2m_engine_start(struct lwm2m_ctx *client_ctx)
|
|
|
|
{
|
|
|
|
char pathstr[MAX_RESOURCE_LEN];
|
|
|
|
char *url;
|
|
|
|
u16_t url_len;
|
|
|
|
u8_t url_data_flags;
|
|
|
|
int ret = 0U;
|
|
|
|
|
|
|
|
/* get the server URL */
|
|
|
|
snprintk(pathstr, sizeof(pathstr), "0/%d/0", client_ctx->sec_obj_inst);
|
|
|
|
ret = lwm2m_engine_get_res_data(pathstr, (void **)&url, &url_len,
|
|
|
|
&url_data_flags);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
url[url_len] = '\0';
|
|
|
|
ret = lwm2m_parse_peerinfo(url, &client_ctx->remote_addr,
|
|
|
|
&client_ctx->use_dtls);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
lwm2m_engine_context_init(client_ctx);
|
|
|
|
return lwm2m_socket_start(client_ctx);
|
2019-01-25 21:43:52 -08:00
|
|
|
}
|
|
|
|
|
2017-07-07 11:04:03 -07:00
|
|
|
static int lwm2m_engine_init(struct device *dev)
|
|
|
|
{
|
2019-04-05 16:14:50 -07:00
|
|
|
int ret = 0;
|
2019-02-21 00:56:55 -08:00
|
|
|
|
2018-09-11 19:09:03 -07:00
|
|
|
(void)memset(block1_contexts, 0,
|
|
|
|
sizeof(struct block_context) * NUM_BLOCK1_CONTEXT);
|
2017-07-25 16:54:25 +08:00
|
|
|
|
2019-01-28 11:55:23 -08:00
|
|
|
/* start sock receive thread */
|
|
|
|
k_thread_create(&engine_thread_data,
|
|
|
|
&engine_thread_stack[0],
|
|
|
|
K_THREAD_STACK_SIZEOF(engine_thread_stack),
|
|
|
|
(k_thread_entry_t) socket_receive_loop,
|
|
|
|
NULL, NULL, NULL,
|
|
|
|
/* Lowest priority cooperative thread */
|
|
|
|
K_PRIO_COOP(CONFIG_NUM_COOP_PRIORITIES - 1),
|
|
|
|
0, K_NO_WAIT);
|
|
|
|
k_thread_name_set(&engine_thread_data, "lwm2m-sock-recv");
|
|
|
|
LOG_DBG("LWM2M engine socket receive thread started");
|
|
|
|
|
2019-04-05 16:14:50 -07:00
|
|
|
return ret;
|
2017-07-07 11:04:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
SYS_INIT(lwm2m_engine_init, APPLICATION, CONFIG_KERNEL_INIT_PRIORITY_DEFAULT);
|