1 /*
2  * Copyright (c) 2017 Linaro Limited
3  * Copyright (c) 2018-2019 Foundries.io
4  *
5  * SPDX-License-Identifier: Apache-2.0
6  */
7 
8 /*
9  * Copyright (c) 2015, Yanzi Networks AB.
10  * All rights reserved.
11  *
12  * Redistribution and use in source and binary forms, with or without
13  * modification, are permitted provided that the following conditions
14  * are met:
15  * 1. Redistributions of source code must retain the above copyright
16  *    notice, this list of conditions and the following disclaimer.
17  * 2. Redistributions in binary form must reproduce the above copyright
18  *    notice, this list of conditions and the following disclaimer in the
19  *    documentation and/or other materials provided with the distribution.
20  * 3. Neither the name of the copyright holder nor the names of its
21  *    contributors may be used to endorse or promote products derived
22  *    from this software without specific prior written permission.
23  *
24  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER AND CONTRIBUTORS
25  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
26  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
27  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
28  * COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
29  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
30  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
31  * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
32  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
33  * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
34  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
35  * OF THE POSSIBILITY OF SUCH DAMAGE.
36  */
37 
38 /*
39  * Original Authors:
40  *         Joakim Eriksson <joakime@sics.se>
41  *         Niclas Finne <nfi@sics.se>
42  */
43 
44 /*
45  * Zephyr Contribution by Michael Scott <michael.scott@linaro.org>
46  * - Zephyr code style changes / code cleanup
47  * - Move to Zephyr APIs where possible
48  * - Convert to Zephyr int/uint types
49  * - Remove engine dependency (replace with writer context)
50  * - Add write int64 function
51  */
52 
53 /*
54  * TODO:
55  * - Lots of byte-order API clean up
56  * - Var / parameter type cleanup
57  * - Replace magic #'s with defines
58  */
59 
60 #define LOG_MODULE_NAME net_lwm2m_oma_tlv
61 #define LOG_LEVEL CONFIG_LWM2M_LOG_LEVEL
62 
63 #include <zephyr/logging/log.h>
64 LOG_MODULE_REGISTER(LOG_MODULE_NAME);
65 
66 #include <string.h>
67 #include <stdint.h>
68 #include <zephyr/sys/byteorder.h>
69 
70 #include "lwm2m_rw_oma_tlv.h"
71 #include "lwm2m_engine.h"
72 #include "lwm2m_rd_client.h"
73 #include "lwm2m_util.h"
74 
75 enum {
76 	OMA_TLV_TYPE_OBJECT_INSTANCE   = 0,
77 	OMA_TLV_TYPE_RESOURCE_INSTANCE = 1,
78 	OMA_TLV_TYPE_MULTI_RESOURCE    = 2,
79 	OMA_TLV_TYPE_RESOURCE          = 3
80 };
81 
82 struct oma_tlv {
83 	uint8_t  type;
84 	uint16_t id; /* can be 8-bit or 16-bit when serialized */
85 	uint32_t length;
86 };
87 
get_len_type(const struct oma_tlv * tlv)88 static uint8_t get_len_type(const struct oma_tlv *tlv)
89 {
90 	if (tlv->length < 8) {
91 		return 0;
92 	} else if (tlv->length < 0x100) {
93 		return 1;
94 	} else if (tlv->length < 0x10000) {
95 		return 2;
96 	}
97 
98 	return 3;
99 }
100 
tlv_calc_type(uint8_t flags)101 static uint8_t tlv_calc_type(uint8_t flags)
102 {
103 	return flags & WRITER_RESOURCE_INSTANCE ?
104 			OMA_TLV_TYPE_RESOURCE_INSTANCE : OMA_TLV_TYPE_RESOURCE;
105 }
106 
tlv_calc_id(uint8_t flags,struct lwm2m_obj_path * path)107 static uint16_t tlv_calc_id(uint8_t flags, struct lwm2m_obj_path *path)
108 {
109 	return flags & WRITER_RESOURCE_INSTANCE ?
110 			path->res_inst_id : path->res_id;
111 }
112 
tlv_setup(struct oma_tlv * tlv,uint8_t type,uint16_t id,uint32_t buflen)113 static void tlv_setup(struct oma_tlv *tlv, uint8_t type, uint16_t id,
114 		      uint32_t buflen)
115 {
116 	if (tlv) {
117 		tlv->type = type;
118 		tlv->id = id;
119 		tlv->length = buflen;
120 	}
121 }
122 
oma_tlv_put_u8(struct lwm2m_output_context * out,uint8_t value,bool insert)123 static int oma_tlv_put_u8(struct lwm2m_output_context *out,
124 			  uint8_t value, bool insert)
125 {
126 	struct tlv_out_formatter_data *fd;
127 	int ret;
128 
129 	if (insert) {
130 		fd = engine_get_out_user_data(out);
131 		if (!fd) {
132 			return -EINVAL;
133 		}
134 
135 		ret = buf_insert(CPKT_BUF_WRITE(out->out_cpkt),
136 				 fd->mark_pos, &value, 1);
137 		if (ret < 0) {
138 			return ret;
139 		}
140 
141 		fd->mark_pos++;
142 	} else {
143 		ret = buf_append(CPKT_BUF_WRITE(out->out_cpkt), &value, 1);
144 		if (ret < 0) {
145 			return ret;
146 		}
147 	}
148 
149 	return 0;
150 }
151 
oma_tlv_put(const struct oma_tlv * tlv,struct lwm2m_output_context * out,uint8_t * value,bool insert)152 static int oma_tlv_put(const struct oma_tlv *tlv,
153 		       struct lwm2m_output_context *out, uint8_t *value,
154 		       bool insert)
155 {
156 	size_t pos;
157 	int ret, i;
158 	uint8_t len_type, tmp;
159 
160 	/* len_type is the same as number of bytes required for length */
161 	len_type = get_len_type(tlv);
162 
163 	/* first type byte in TLV header */
164 	tmp = (tlv->type << 6) |
165 	      (tlv->id > 255 ? (1 << 5) : 0) |
166 	      (len_type << 3) |
167 	      (len_type == 0U ? tlv->length : 0);
168 
169 	ret = oma_tlv_put_u8(out, tmp, insert);
170 	if (ret < 0) {
171 		return ret;
172 	}
173 
174 	pos = 1;
175 
176 	/* The ID */
177 	if (tlv->id > 255) {
178 		ret = oma_tlv_put_u8(out, (tlv->id >> 8) & 0xff, insert);
179 		if (ret < 0) {
180 			return ret;
181 		}
182 
183 		pos++;
184 	}
185 
186 	ret = oma_tlv_put_u8(out, tlv->id & 0xff, insert);
187 	if (ret < 0) {
188 		return ret;
189 	}
190 
191 	pos++;
192 
193 	for (i = 2; i >= 0; i--) {
194 		if (len_type > i) {
195 			ret = oma_tlv_put_u8(out,
196 					     (tlv->length >> (i * 8)) & 0xff,
197 					     insert);
198 			if (ret < 0) {
199 				return ret;
200 			}
201 
202 			pos++;
203 		}
204 	}
205 
206 	/* finally add the value */
207 	if (value != NULL && tlv->length > 0 && !insert) {
208 		ret = buf_append(CPKT_BUF_WRITE(out->out_cpkt),
209 				 value, tlv->length);
210 		if (ret < 0) {
211 			return ret;
212 		}
213 	}
214 
215 	return pos + tlv->length;
216 }
217 
oma_tlv_get(struct oma_tlv * tlv,struct lwm2m_input_context * in,bool dont_advance)218 static int oma_tlv_get(struct oma_tlv *tlv, struct lwm2m_input_context *in,
219 		       bool dont_advance)
220 {
221 	uint8_t len_type;
222 	uint8_t len_pos = 1U;
223 	size_t tlv_len;
224 	uint16_t tmp_offset;
225 	uint8_t buf[2];
226 	int ret;
227 
228 	tmp_offset = in->offset;
229 	ret = buf_read_u8(&buf[0], CPKT_BUF_READ(in->in_cpkt), &tmp_offset);
230 	if (ret < 0) {
231 		goto error;
232 	}
233 
234 	tlv->type = (buf[0] >> 6) & 3;
235 	len_type = (buf[0] >> 3) & 3;
236 	len_pos = 1 + (((buf[0] & (1 << 5)) != 0U) ? 2 : 1);
237 
238 	ret = buf_read_u8(&buf[1], CPKT_BUF_READ(in->in_cpkt), &tmp_offset);
239 	if (ret < 0) {
240 		goto error;
241 	}
242 
243 	tlv->id = buf[1];
244 
245 	/* if len_pos > 2 it means that there are more ID to read */
246 	if (len_pos > 2) {
247 		ret = buf_read_u8(&buf[1], CPKT_BUF_READ(in->in_cpkt),
248 				  &tmp_offset);
249 		if (ret < 0) {
250 			goto error;
251 		}
252 
253 		tlv->id = (tlv->id << 8) + buf[1];
254 	}
255 
256 	if (len_type == 0U) {
257 		tlv_len = buf[0] & 7;
258 	} else {
259 		/* read the length */
260 		tlv_len = 0;
261 		while (len_type > 0) {
262 			ret = buf_read_u8(&buf[1], CPKT_BUF_READ(in->in_cpkt),
263 					  &tmp_offset);
264 			if (ret < 0) {
265 				goto error;
266 			}
267 
268 			len_pos++;
269 			tlv_len = tlv_len << 8 | buf[1];
270 			len_type--;
271 		}
272 	}
273 
274 	/* and read out the data??? */
275 	tlv->length = tlv_len;
276 
277 	if (!dont_advance) {
278 		in->offset = tmp_offset;
279 	}
280 
281 	return len_pos + tlv_len;
282 
283 error:
284 	if (!dont_advance) {
285 		in->offset = tmp_offset;
286 	}
287 
288 	return ret;
289 }
290 
put_begin_tlv(struct lwm2m_output_context * out,uint16_t * mark_pos,uint8_t * writer_flags,int writer_flag)291 static int put_begin_tlv(struct lwm2m_output_context *out, uint16_t *mark_pos,
292 			 uint8_t *writer_flags, int writer_flag)
293 {
294 	/* set flags */
295 	*writer_flags |= writer_flag;
296 
297 	/*
298 	 * store position for inserting TLV when we know the length
299 	 */
300 	*mark_pos = out->out_cpkt->offset;
301 
302 	return 0;
303 }
304 
put_end_tlv(struct lwm2m_output_context * out,uint16_t mark_pos,uint8_t * writer_flags,uint8_t writer_flag,int tlv_type,int tlv_id)305 static int put_end_tlv(struct lwm2m_output_context *out, uint16_t mark_pos,
306 		       uint8_t *writer_flags, uint8_t writer_flag, int tlv_type,
307 		       int tlv_id)
308 {
309 	struct tlv_out_formatter_data *fd;
310 	struct oma_tlv tlv;
311 	int len;
312 
313 	fd = engine_get_out_user_data(out);
314 	if (!fd) {
315 		return -EINVAL;
316 	}
317 
318 	*writer_flags &= ~writer_flag;
319 
320 	len = out->out_cpkt->offset - mark_pos;
321 
322 	/* use stored location */
323 	fd->mark_pos = mark_pos;
324 
325 	/* set instance length */
326 	tlv_setup(&tlv, tlv_type, tlv_id, len);
327 	len = oma_tlv_put(&tlv, out, NULL, true) - tlv.length;
328 
329 	return len;
330 }
331 
put_begin_oi(struct lwm2m_output_context * out,struct lwm2m_obj_path * path)332 static int put_begin_oi(struct lwm2m_output_context *out,
333 			struct lwm2m_obj_path *path)
334 {
335 	struct tlv_out_formatter_data *fd;
336 
337 	/* No need for oi level TLV constructs */
338 	if (path->level > LWM2M_PATH_LEVEL_OBJECT) {
339 		return 0;
340 	}
341 
342 	fd = engine_get_out_user_data(out);
343 	if (!fd) {
344 		return -EINVAL;
345 	}
346 
347 	return put_begin_tlv(out, &fd->mark_pos_oi, &fd->writer_flags, 0);
348 }
349 
put_end_oi(struct lwm2m_output_context * out,struct lwm2m_obj_path * path)350 static int put_end_oi(struct lwm2m_output_context *out,
351 		      struct lwm2m_obj_path *path)
352 {
353 	struct tlv_out_formatter_data *fd;
354 
355 	if (path->level > LWM2M_PATH_LEVEL_OBJECT) {
356 		return 0;
357 	}
358 
359 	fd = engine_get_out_user_data(out);
360 	if (!fd) {
361 		return -EINVAL;
362 	}
363 
364 	return put_end_tlv(out, fd->mark_pos_oi, &fd->writer_flags, 0,
365 			   OMA_TLV_TYPE_OBJECT_INSTANCE, path->obj_inst_id);
366 }
367 
put_begin_ri(struct lwm2m_output_context * out,struct lwm2m_obj_path * path)368 static int put_begin_ri(struct lwm2m_output_context *out,
369 			struct lwm2m_obj_path *path)
370 {
371 	struct tlv_out_formatter_data *fd;
372 
373 	fd = engine_get_out_user_data(out);
374 	if (!fd) {
375 		return -EINVAL;
376 	}
377 
378 	return put_begin_tlv(out, &fd->mark_pos_ri, &fd->writer_flags,
379 			     WRITER_RESOURCE_INSTANCE);
380 }
381 
put_end_ri(struct lwm2m_output_context * out,struct lwm2m_obj_path * path)382 static int put_end_ri(struct lwm2m_output_context *out,
383 		      struct lwm2m_obj_path *path)
384 {
385 	struct tlv_out_formatter_data *fd;
386 
387 	fd = engine_get_out_user_data(out);
388 	if (!fd) {
389 		return -EINVAL;
390 	}
391 
392 	/* Skip writing Multiple Resource TLV if path level is 4 */
393 	if (IS_ENABLED(CONFIG_LWM2M_VERSION_1_1) &&
394 		path->level == LWM2M_PATH_LEVEL_RESOURCE_INST) {
395 		return 0;
396 	}
397 
398 	return put_end_tlv(out, fd->mark_pos_ri, &fd->writer_flags,
399 			   WRITER_RESOURCE_INSTANCE,
400 			   OMA_TLV_TYPE_MULTI_RESOURCE, path->res_id);
401 }
402 
put_s8(struct lwm2m_output_context * out,struct lwm2m_obj_path * path,int8_t value)403 static int put_s8(struct lwm2m_output_context *out, struct lwm2m_obj_path *path,
404 		  int8_t value)
405 {
406 	struct tlv_out_formatter_data *fd;
407 	int len;
408 	struct oma_tlv tlv;
409 
410 	fd = engine_get_out_user_data(out);
411 	if (!fd) {
412 		return -EINVAL;
413 	}
414 
415 	tlv_setup(&tlv, tlv_calc_type(fd->writer_flags),
416 		  tlv_calc_id(fd->writer_flags, path), sizeof(value));
417 
418 	len = oma_tlv_put(&tlv, out, (uint8_t *)&value, false);
419 	return len;
420 }
421 
put_s16(struct lwm2m_output_context * out,struct lwm2m_obj_path * path,int16_t value)422 static int put_s16(struct lwm2m_output_context *out,
423 		   struct lwm2m_obj_path *path, int16_t value)
424 {
425 	struct tlv_out_formatter_data *fd;
426 	int len;
427 	struct oma_tlv tlv;
428 	int16_t net_value;
429 
430 	if (INT8_MIN <= value && value <= INT8_MAX) {
431 		return put_s8(out, path, (int8_t)value);
432 	}
433 
434 	fd = engine_get_out_user_data(out);
435 	if (!fd) {
436 		return -EINVAL;
437 	}
438 
439 	net_value = sys_cpu_to_be16(value);
440 	tlv_setup(&tlv, tlv_calc_type(fd->writer_flags),
441 		  tlv_calc_id(fd->writer_flags, path), sizeof(net_value));
442 
443 	len = oma_tlv_put(&tlv, out, (uint8_t *)&net_value, false);
444 	return len;
445 }
446 
put_s32(struct lwm2m_output_context * out,struct lwm2m_obj_path * path,int32_t value)447 static int put_s32(struct lwm2m_output_context *out,
448 		   struct lwm2m_obj_path *path, int32_t value)
449 {
450 	struct tlv_out_formatter_data *fd;
451 	int len;
452 	struct oma_tlv tlv;
453 	int32_t net_value;
454 
455 	if (INT16_MIN <= value && value <= INT16_MAX) {
456 		return put_s16(out, path, (int16_t)value);
457 	}
458 
459 	fd = engine_get_out_user_data(out);
460 	if (!fd) {
461 		return -EINVAL;
462 	}
463 
464 	net_value = sys_cpu_to_be32(value);
465 	tlv_setup(&tlv, tlv_calc_type(fd->writer_flags),
466 		  tlv_calc_id(fd->writer_flags, path), sizeof(net_value));
467 
468 	len = oma_tlv_put(&tlv, out, (uint8_t *)&net_value, false);
469 
470 	return len;
471 }
472 
put_s64(struct lwm2m_output_context * out,struct lwm2m_obj_path * path,int64_t value)473 static int put_s64(struct lwm2m_output_context *out,
474 		   struct lwm2m_obj_path *path, int64_t value)
475 {
476 	struct tlv_out_formatter_data *fd;
477 	int len;
478 	struct oma_tlv tlv;
479 	int64_t net_value;
480 
481 	if (INT32_MIN <= value && value <= INT32_MAX) {
482 		return put_s32(out, path, (int32_t)value);
483 	}
484 
485 	fd = engine_get_out_user_data(out);
486 	if (!fd) {
487 		return -EINVAL;
488 	}
489 
490 	net_value = sys_cpu_to_be64(value);
491 	tlv_setup(&tlv, tlv_calc_type(fd->writer_flags),
492 		  tlv_calc_id(fd->writer_flags, path), sizeof(net_value));
493 
494 	len = oma_tlv_put(&tlv, out, (uint8_t *)&net_value, false);
495 	return len;
496 }
497 
498 
put_time(struct lwm2m_output_context * out,struct lwm2m_obj_path * path,time_t value)499 static int put_time(struct lwm2m_output_context *out, struct lwm2m_obj_path *path, time_t value)
500 {
501 	return put_s64(out, path, (int64_t)value);
502 }
503 
put_string(struct lwm2m_output_context * out,struct lwm2m_obj_path * path,char * buf,size_t buflen)504 static int put_string(struct lwm2m_output_context *out,
505 		      struct lwm2m_obj_path *path, char *buf, size_t buflen)
506 {
507 	struct tlv_out_formatter_data *fd;
508 	int len;
509 	struct oma_tlv tlv;
510 
511 	fd = engine_get_out_user_data(out);
512 	if (!fd) {
513 		return -EINVAL;
514 	}
515 
516 	tlv_setup(&tlv, tlv_calc_type(fd->writer_flags),
517 		  tlv_calc_id(fd->writer_flags, path), (uint32_t)buflen);
518 	len = oma_tlv_put(&tlv, out, (uint8_t *)buf, false);
519 	return len;
520 }
521 
put_float(struct lwm2m_output_context * out,struct lwm2m_obj_path * path,double * value)522 static int put_float(struct lwm2m_output_context *out,
523 		     struct lwm2m_obj_path *path, double *value)
524 {
525 	struct tlv_out_formatter_data *fd;
526 	int len;
527 	struct oma_tlv tlv;
528 	int ret;
529 	uint8_t b64[8];
530 
531 	fd = engine_get_out_user_data(out);
532 	if (!fd) {
533 		return -EINVAL;
534 	}
535 
536 	ret = lwm2m_float_to_b64(value, b64, sizeof(b64));
537 	if (ret < 0) {
538 		LOG_ERR("float32 conversion error: %d", ret);
539 		return ret;
540 	}
541 
542 	tlv_setup(&tlv, tlv_calc_type(fd->writer_flags),
543 		  tlv_calc_id(fd->writer_flags, path), sizeof(b64));
544 	len = oma_tlv_put(&tlv, out, b64, false);
545 	return len;
546 }
547 
put_bool(struct lwm2m_output_context * out,struct lwm2m_obj_path * path,bool value)548 static int put_bool(struct lwm2m_output_context *out,
549 		    struct lwm2m_obj_path *path, bool value)
550 {
551 	int8_t value_s8 = (value != 0 ? 1 : 0);
552 
553 	return put_s8(out, path, value_s8);
554 }
555 
put_opaque(struct lwm2m_output_context * out,struct lwm2m_obj_path * path,char * buf,size_t buflen)556 static int put_opaque(struct lwm2m_output_context *out,
557 		      struct lwm2m_obj_path *path, char *buf, size_t buflen)
558 {
559 	return put_string(out, path, buf, buflen);
560 }
561 
put_objlnk(struct lwm2m_output_context * out,struct lwm2m_obj_path * path,struct lwm2m_objlnk * value)562 static int put_objlnk(struct lwm2m_output_context *out,
563 		      struct lwm2m_obj_path *path, struct lwm2m_objlnk *value)
564 {
565 	struct tlv_out_formatter_data *fd;
566 	struct oma_tlv tlv;
567 	int32_t net_value = sys_cpu_to_be32(
568 				((value->obj_id) << 16) | value->obj_inst);
569 
570 	fd = engine_get_out_user_data(out);
571 	if (!fd) {
572 		return -EINVAL;
573 	}
574 
575 	tlv_setup(&tlv, tlv_calc_type(fd->writer_flags),
576 		  tlv_calc_id(fd->writer_flags, path), sizeof(net_value));
577 
578 	return oma_tlv_put(&tlv, out, (uint8_t *)&net_value, false);
579 }
580 
get_number(struct lwm2m_input_context * in,int64_t * value,uint8_t max_len)581 static int get_number(struct lwm2m_input_context *in, int64_t *value,
582 		      uint8_t max_len)
583 {
584 	struct oma_tlv tlv;
585 	int size;
586 	int64_t temp;
587 	int ret;
588 
589 	size = oma_tlv_get(&tlv, in, false);
590 	if (size < 0) {
591 		return size;
592 	}
593 
594 	if (tlv.length > max_len) {
595 		LOG_ERR("invalid length: %u", tlv.length);
596 		return -ENOMEM;
597 	}
598 
599 	ret = buf_read((uint8_t *)&temp, tlv.length,
600 		       CPKT_BUF_READ(in->in_cpkt), &in->offset);
601 	if (ret < 0) {
602 		return ret;
603 	}
604 
605 	switch (tlv.length) {
606 	case 1:
607 		*value = (int8_t)temp;
608 		break;
609 	case 2:
610 		*value = sys_cpu_to_be16((int16_t)temp);
611 		break;
612 	case 4:
613 		*value = sys_cpu_to_be32((int32_t)temp);
614 		break;
615 	case 8:
616 		*value = sys_cpu_to_be64(temp);
617 		break;
618 	default:
619 		LOG_ERR("invalid length: %u", tlv.length);
620 		return -EBADMSG;
621 	}
622 
623 	return size;
624 }
625 
get_s64(struct lwm2m_input_context * in,int64_t * value)626 static int get_s64(struct lwm2m_input_context *in, int64_t *value)
627 {
628 	return get_number(in, value, 8);
629 }
630 
get_time(struct lwm2m_input_context * in,time_t * value)631 static int get_time(struct lwm2m_input_context *in, time_t *value)
632 {
633 	int64_t temp64;
634 	int ret;
635 
636 	ret = get_number(in, &temp64, 8);
637 	*value = (time_t)temp64;
638 
639 	return ret;
640 }
641 
get_s32(struct lwm2m_input_context * in,int32_t * value)642 static int get_s32(struct lwm2m_input_context *in, int32_t *value)
643 {
644 	int64_t temp;
645 	int size;
646 
647 	size = get_number(in, &temp, 4);
648 	if (size < 0) {
649 		return size;
650 	}
651 
652 	*value = (int32_t)temp;
653 
654 	return size;
655 }
656 
get_string(struct lwm2m_input_context * in,uint8_t * buf,size_t buflen)657 static int get_string(struct lwm2m_input_context *in, uint8_t *buf,
658 		      size_t buflen)
659 {
660 	struct oma_tlv tlv;
661 	int size;
662 	int ret;
663 
664 	size = oma_tlv_get(&tlv, in, false);
665 	if (size < 0) {
666 		return size;
667 	}
668 
669 	if (buflen <= tlv.length) {
670 		return -ENOMEM;
671 	}
672 
673 	ret = buf_read(buf, tlv.length, CPKT_BUF_READ(in->in_cpkt),
674 		       &in->offset);
675 	if (ret < 0) {
676 		return ret;
677 	}
678 
679 	buf[tlv.length] = '\0';
680 
681 	return size;
682 }
683 
684 /* convert float to fixpoint */
get_float(struct lwm2m_input_context * in,double * value)685 static int get_float(struct lwm2m_input_context *in, double *value)
686 {
687 	struct oma_tlv tlv;
688 	int size;
689 	uint8_t buf[8];
690 	int ret;
691 
692 	size = oma_tlv_get(&tlv, in, false);
693 	if (size < 0) {
694 		return size;
695 	}
696 
697 	if (tlv.length != 4U && tlv.length != 8U) {
698 		LOG_ERR("Invalid float length: %d", tlv.length);
699 		return -EBADMSG;
700 	}
701 
702 	/* read float in network byte order */
703 	ret = buf_read(buf, tlv.length, CPKT_BUF_READ(in->in_cpkt),
704 		       &in->offset);
705 	if (ret < 0) {
706 		return ret;
707 	}
708 
709 	if (tlv.length == 4U) {
710 		ret = lwm2m_b32_to_float(buf, 4, value);
711 	} else {
712 		ret = lwm2m_b64_to_float(buf, 8, value);
713 	}
714 
715 	if (ret < 0) {
716 		LOG_ERR("binary%s conversion error: %d",
717 			tlv.length == 4U ? "32" : "64", ret);
718 		return ret;
719 	}
720 
721 	return size;
722 }
723 
get_bool(struct lwm2m_input_context * in,bool * value)724 static int get_bool(struct lwm2m_input_context *in, bool *value)
725 {
726 	int64_t temp;
727 	int size;
728 
729 	size = get_number(in, &temp, 2);
730 	if (size < 0) {
731 		return size;
732 
733 	}
734 
735 	*value = (temp != 0);
736 
737 	return size;
738 }
739 
get_opaque(struct lwm2m_input_context * in,uint8_t * value,size_t buflen,struct lwm2m_opaque_context * opaque,bool * last_block)740 static int get_opaque(struct lwm2m_input_context *in, uint8_t *value,
741 		      size_t buflen, struct lwm2m_opaque_context *opaque,
742 		      bool *last_block)
743 {
744 	struct oma_tlv tlv;
745 	int size;
746 
747 	/* Get the TLV header only on first read. */
748 	if (opaque->remaining == 0) {
749 		size = oma_tlv_get(&tlv, in, false);
750 		if (size < 0) {
751 			return size;
752 		}
753 
754 		opaque->len = tlv.length;
755 		opaque->remaining = tlv.length;
756 	}
757 
758 	return lwm2m_engine_get_opaque_more(in, value, buflen,
759 					    opaque, last_block);
760 }
761 
get_objlnk(struct lwm2m_input_context * in,struct lwm2m_objlnk * value)762 static int get_objlnk(struct lwm2m_input_context *in,
763 		      struct lwm2m_objlnk *value)
764 {
765 	int32_t value_s32;
766 	int size;
767 
768 	size = get_s32(in, &value_s32);
769 	if (size < 0) {
770 		return size;
771 	}
772 
773 	value->obj_id = (value_s32 >> 16) & 0xFFFF;
774 	value->obj_inst = value_s32 & 0xFFFF;
775 
776 	return size;
777 }
778 
779 const struct lwm2m_writer oma_tlv_writer = {
780 	.put_begin_oi = put_begin_oi,
781 	.put_end_oi = put_end_oi,
782 	.put_begin_ri = put_begin_ri,
783 	.put_end_ri = put_end_ri,
784 	.put_s8 = put_s8,
785 	.put_s16 = put_s16,
786 	.put_s32 = put_s32,
787 	.put_s64 = put_s64,
788 	.put_string = put_string,
789 	.put_float = put_float,
790 	.put_time = put_time,
791 	.put_bool = put_bool,
792 	.put_opaque = put_opaque,
793 	.put_objlnk = put_objlnk,
794 };
795 
796 const struct lwm2m_reader oma_tlv_reader = {
797 	.get_s32 = get_s32,
798 	.get_s64 = get_s64,
799 	.get_string = get_string,
800 	.get_time = get_time,
801 	.get_float = get_float,
802 	.get_bool = get_bool,
803 	.get_opaque = get_opaque,
804 	.get_objlnk = get_objlnk,
805 };
806 
do_read_op_tlv(struct lwm2m_message * msg,int content_format)807 int do_read_op_tlv(struct lwm2m_message *msg, int content_format)
808 {
809 	struct tlv_out_formatter_data fd;
810 	int ret;
811 
812 	(void)memset(&fd, 0, sizeof(fd));
813 	engine_set_out_user_data(&msg->out, &fd);
814 	ret = lwm2m_perform_read_op(msg, content_format);
815 	engine_clear_out_user_data(&msg->out);
816 	return ret;
817 }
818 
do_write_op_tlv_dummy_read(struct lwm2m_message * msg)819 static int do_write_op_tlv_dummy_read(struct lwm2m_message *msg)
820 {
821 	struct oma_tlv tlv;
822 	uint8_t read_char;
823 
824 	oma_tlv_get(&tlv, &msg->in, false);
825 	while (tlv.length--) {
826 		if (buf_read_u8(&read_char, CPKT_BUF_READ(msg->in.in_cpkt),
827 				&msg->in.offset) < 0) {
828 			break;
829 		}
830 	}
831 
832 	return 0;
833 }
834 
do_write_op_tlv_item(struct lwm2m_message * msg)835 static int do_write_op_tlv_item(struct lwm2m_message *msg)
836 {
837 	struct lwm2m_engine_obj_inst *obj_inst = NULL;
838 	struct lwm2m_engine_res *res = NULL;
839 	struct lwm2m_engine_res_inst *res_inst = NULL;
840 	struct lwm2m_engine_obj_field *obj_field = NULL;
841 	uint8_t created = 0U;
842 	int ret;
843 
844 	ret = lwm2m_get_or_create_engine_obj(msg, &obj_inst, &created);
845 	if (ret < 0) {
846 		goto error;
847 	}
848 
849 	ret = lwm2m_engine_validate_write_access(msg, obj_inst, &obj_field);
850 	if (ret < 0) {
851 		goto error;
852 	}
853 
854 	ret = lwm2m_engine_get_create_res_inst(&msg->path, &res, &res_inst);
855 
856 	if (ret < 0) {
857 		/* if OPTIONAL and BOOTSTRAP-WRITE or CREATE use ENOTSUP */
858 		if ((msg->ctx->bootstrap_mode ||
859 		     msg->operation == LWM2M_OP_CREATE) &&
860 		    LWM2M_HAS_PERM(obj_field, BIT(LWM2M_FLAG_OPTIONAL))) {
861 			ret = -ENOTSUP;
862 			goto error;
863 		}
864 
865 		ret = -ENOENT;
866 		goto error;
867 	}
868 
869 	ret = lwm2m_write_handler(obj_inst, res, res_inst, obj_field, msg);
870 	if (ret == -EACCES || ret == -ENOENT) {
871 		/* if read-only or non-existent data buffer move on */
872 		do_write_op_tlv_dummy_read(msg);
873 		ret = 0;
874 	}
875 
876 	return ret;
877 
878 error:
879 	do_write_op_tlv_dummy_read(msg);
880 	return ret;
881 }
882 
write_tlv_resource(struct lwm2m_message * msg,struct oma_tlv * tlv)883 static int write_tlv_resource(struct lwm2m_message *msg, struct oma_tlv *tlv)
884 {
885 	int ret;
886 
887 	if (msg->in.block_ctx) {
888 		msg->in.block_ctx->path.res_id = tlv->id;
889 	}
890 
891 	msg->path.res_id = tlv->id;
892 	msg->path.level = 3U;
893 	ret = do_write_op_tlv_item(msg);
894 
895 	/*
896 	 * ignore errors for CREATE op
897 	 * for OP_CREATE and BOOTSTRAP WRITE: errors on
898 	 * optional resources are ignored (ENOTSUP)
899 	 */
900 	if (ret < 0 &&
901 	    !((ret == -ENOTSUP) &&
902 	      (msg->ctx->bootstrap_mode ||
903 	       msg->operation == LWM2M_OP_CREATE))) {
904 		return ret;
905 	}
906 
907 	return 0;
908 }
909 
910 #if defined(CONFIG_LWM2M_VERSION_1_1)
write_tlv_resource_instance(struct lwm2m_message * msg,struct oma_tlv * tlv)911 static int write_tlv_resource_instance(struct lwm2m_message *msg, struct oma_tlv *tlv)
912 {
913 	int ret;
914 
915 	if (msg->in.block_ctx) {
916 		msg->in.block_ctx->path.res_inst_id = tlv->id;
917 	}
918 
919 	msg->path.res_inst_id = tlv->id;
920 	msg->path.level = LWM2M_PATH_LEVEL_RESOURCE_INST;
921 	ret = do_write_op_tlv_item(msg);
922 
923 	if (ret < 0) {
924 		return ret;
925 	}
926 
927 	return 0;
928 }
929 #endif
930 
lwm2m_multi_resource_tlv_parse(struct lwm2m_message * msg,struct oma_tlv * multi_resource_tlv)931 static int lwm2m_multi_resource_tlv_parse(struct lwm2m_message *msg,
932 					  struct oma_tlv *multi_resource_tlv)
933 {
934 	struct oma_tlv tlv_resource_instance;
935 	int len2;
936 	int pos = 0;
937 	int ret;
938 
939 	if (msg->in.block_ctx) {
940 		msg->in.block_ctx->path.res_id = multi_resource_tlv->id;
941 	}
942 
943 	if (multi_resource_tlv->length == 0U) {
944 		/* No data for resource instances, so create only a resource */
945 		return write_tlv_resource(msg, multi_resource_tlv);
946 	}
947 
948 	while (pos < multi_resource_tlv->length &&
949 	       (len2 = oma_tlv_get(&tlv_resource_instance, &msg->in, true))) {
950 		if (tlv_resource_instance.type != OMA_TLV_TYPE_RESOURCE_INSTANCE) {
951 			LOG_ERR("Multi resource id not supported %u %d", tlv_resource_instance.id,
952 				tlv_resource_instance.length);
953 			return -ENOTSUP;
954 		}
955 
956 		msg->path.res_id = multi_resource_tlv->id;
957 		msg->path.res_inst_id = tlv_resource_instance.id;
958 		msg->path.level = LWM2M_PATH_LEVEL_RESOURCE_INST;
959 		ret = do_write_op_tlv_item(msg);
960 
961 		/*
962 		 * Ignore errors on optional resources when doing
963 		 * BOOTSTRAP WRITE or CREATE operation.
964 		 */
965 		if (ret < 0 && !((ret == -ENOTSUP) &&
966 				 (msg->ctx->bootstrap_mode || msg->operation == LWM2M_OP_CREATE))) {
967 			return ret;
968 		}
969 
970 		pos += len2;
971 	}
972 
973 	return 0;
974 }
975 
do_write_op_tlv(struct lwm2m_message * msg)976 int do_write_op_tlv(struct lwm2m_message *msg)
977 {
978 	struct lwm2m_engine_obj_inst *obj_inst = NULL;
979 	int len;
980 	struct oma_tlv tlv;
981 	int ret;
982 
983 	/* In case of block transfer, check if there are any fragments
984 	 * left from the previous resource (instance). If this is the
985 	 * case, proceed directly to processing the message -
986 	 * consecutive blocks from the same resource do not carry the
987 	 * TLV header.
988 	 */
989 	if (msg->in.block_ctx != NULL && msg->in.block_ctx->ctx.current > 0) {
990 		msg->path.res_id = msg->in.block_ctx->path.res_id;
991 		msg->path.level = 3U;
992 		ret = do_write_op_tlv_item(msg);
993 		if (ret < 0) {
994 			return ret;
995 		}
996 
997 	}
998 
999 	while (true) {
1000 		/*
1001 		 * This initial read of TLV data won't advance frag/offset.
1002 		 * We need tlv.type to determine how to proceed.
1003 		 */
1004 		len = oma_tlv_get(&tlv, &msg->in, true);
1005 		if (len < 0) {
1006 			break;
1007 		}
1008 
1009 		if (tlv.type == OMA_TLV_TYPE_OBJECT_INSTANCE) {
1010 			struct oma_tlv tlv2;
1011 			int len2;
1012 			int pos = 0;
1013 
1014 			oma_tlv_get(&tlv, &msg->in, false);
1015 			msg->path.obj_inst_id = tlv.id;
1016 			if (tlv.length == 0U) {
1017 				/* Create only - no data */
1018 				ret = lwm2m_create_obj_inst(
1019 						msg->path.obj_id,
1020 						msg->path.obj_inst_id,
1021 						&obj_inst);
1022 				if (ret < 0) {
1023 					return ret;
1024 				}
1025 
1026 				if (!msg->ctx->bootstrap_mode) {
1027 					engine_trigger_update(true);
1028 				}
1029 			}
1030 
1031 			while (pos < tlv.length &&
1032 			       (len2 = oma_tlv_get(&tlv2, &msg->in, true))) {
1033 				if (tlv2.type == OMA_TLV_TYPE_RESOURCE) {
1034 					ret = write_tlv_resource(msg, &tlv2);
1035 					if (ret) {
1036 						return ret;
1037 					}
1038 				} else if (tlv2.type == OMA_TLV_TYPE_MULTI_RESOURCE) {
1039 					oma_tlv_get(&tlv2, &msg->in, false);
1040 					ret = lwm2m_multi_resource_tlv_parse(msg, &tlv2);
1041 					if (ret) {
1042 						return ret;
1043 					}
1044 				} else {
1045 					/* Skip Unsupported TLV type */
1046 					return -ENOTSUP;
1047 				}
1048 
1049 				pos += len2;
1050 			}
1051 		} else if (tlv.type == OMA_TLV_TYPE_RESOURCE) {
1052 			if (msg->path.level < LWM2M_PATH_LEVEL_OBJECT_INST) {
1053 				return -ENOTSUP;
1054 			}
1055 			ret = write_tlv_resource(msg, &tlv);
1056 			if (ret) {
1057 				return ret;
1058 			}
1059 		} else if (tlv.type == OMA_TLV_TYPE_MULTI_RESOURCE) {
1060 			if (msg->path.level < LWM2M_PATH_LEVEL_OBJECT_INST) {
1061 				return -ENOTSUP;
1062 			}
1063 			oma_tlv_get(&tlv, &msg->in, false);
1064 			ret = lwm2m_multi_resource_tlv_parse(msg, &tlv);
1065 			if (ret) {
1066 				return ret;
1067 			}
1068 #if defined(CONFIG_LWM2M_VERSION_1_1)
1069 		} else if (tlv.type == OMA_TLV_TYPE_RESOURCE_INSTANCE) {
1070 			if (msg->path.level < LWM2M_PATH_LEVEL_OBJECT_INST) {
1071 				return -ENOTSUP;
1072 			}
1073 			ret = write_tlv_resource_instance(msg, &tlv);
1074 			if (ret) {
1075 				return ret;
1076 			}
1077 #endif
1078 		} else {
1079 			return -ENOTSUP;
1080 		}
1081 	}
1082 
1083 	return 0;
1084 }
1085