json.h 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666
  1. /*
  2. * Copyright (c) 2017 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #ifndef ZEPHYR_INCLUDE_DATA_JSON_H_
  7. #define ZEPHYR_INCLUDE_DATA_JSON_H_
  8. #include <sys/util.h>
  9. #include <stddef.h>
  10. #include <zephyr/types.h>
  11. #include <sys/types.h>
  12. #ifdef __cplusplus
  13. extern "C" {
  14. #endif
  15. /**
  16. * @brief Structured Data
  17. * @defgroup structured_data Structured Data
  18. */
  19. /**
  20. * @defgroup json JSON
  21. * @ingroup structured_data
  22. * @{
  23. */
  24. enum json_tokens {
  25. /* Before changing this enum, ensure that its maximum
  26. * value is still within 7 bits. See comment next to the
  27. * declaration of `type` in struct json_obj_descr.
  28. */
  29. JSON_TOK_NONE = '_',
  30. JSON_TOK_OBJECT_START = '{',
  31. JSON_TOK_OBJECT_END = '}',
  32. JSON_TOK_LIST_START = '[',
  33. JSON_TOK_LIST_END = ']',
  34. JSON_TOK_STRING = '"',
  35. JSON_TOK_COLON = ':',
  36. JSON_TOK_COMMA = ',',
  37. JSON_TOK_NUMBER = '0',
  38. JSON_TOK_TRUE = 't',
  39. JSON_TOK_FALSE = 'f',
  40. JSON_TOK_NULL = 'n',
  41. JSON_TOK_ERROR = '!',
  42. JSON_TOK_EOF = '\0',
  43. };
  44. struct json_obj_descr {
  45. const char *field_name;
  46. /* Alignment can be 1, 2, 4, or 8. The macros to create
  47. * a struct json_obj_descr will store the alignment's
  48. * power of 2 in order to keep this value in the 0-3 range
  49. * and thus use only 2 bits.
  50. */
  51. uint32_t align_shift : 2;
  52. /* 127 characters is more than enough for a field name. */
  53. uint32_t field_name_len : 7;
  54. /* Valid values here (enum json_tokens): JSON_TOK_STRING,
  55. * JSON_TOK_NUMBER, JSON_TOK_TRUE, JSON_TOK_FALSE,
  56. * JSON_TOK_OBJECT_START, JSON_TOK_LIST_START. (All others
  57. * ignored.) Maximum value is '}' (125), so this has to be 7 bits
  58. * long.
  59. */
  60. uint32_t type : 7;
  61. /* 65535 bytes is more than enough for many JSON payloads. */
  62. uint32_t offset : 16;
  63. union {
  64. struct {
  65. const struct json_obj_descr *sub_descr;
  66. size_t sub_descr_len;
  67. } object;
  68. struct {
  69. const struct json_obj_descr *element_descr;
  70. size_t n_elements;
  71. } array;
  72. };
  73. };
  74. /**
  75. * @brief Function pointer type to append bytes to a buffer while
  76. * encoding JSON data.
  77. *
  78. * @param bytes Contents to write to the output
  79. * @param len Number of bytes to append to output
  80. * @param data User-provided pointer
  81. *
  82. * @return This callback function should return a negative number on
  83. * error (which will be propagated to the return value of
  84. * json_obj_encode()), or 0 on success.
  85. */
  86. typedef int (*json_append_bytes_t)(const char *bytes, size_t len,
  87. void *data);
  88. #define Z_ALIGN_SHIFT(type) (__alignof__(type) == 1 ? 0 : \
  89. __alignof__(type) == 2 ? 1 : \
  90. __alignof__(type) == 4 ? 2 : 3)
  91. /**
  92. * @brief Helper macro to declare a descriptor for supported primitive
  93. * values.
  94. *
  95. * @param struct_ Struct packing the values
  96. * @param field_name_ Field name in the struct
  97. * @param type_ Token type for JSON value corresponding to a primitive
  98. * type. Must be one of: JSON_TOK_STRING for strings, JSON_TOK_NUMBER
  99. * for numbers, JSON_TOK_TRUE (or JSON_TOK_FALSE) for booleans.
  100. *
  101. * Here's an example of use:
  102. *
  103. * struct foo {
  104. * int some_int;
  105. * };
  106. *
  107. * struct json_obj_descr foo[] = {
  108. * JSON_OBJ_DESCR_PRIM(struct foo, some_int, JSON_TOK_NUMBER),
  109. * };
  110. */
  111. #define JSON_OBJ_DESCR_PRIM(struct_, field_name_, type_) \
  112. { \
  113. .field_name = (#field_name_), \
  114. .align_shift = Z_ALIGN_SHIFT(struct_), \
  115. .field_name_len = sizeof(#field_name_) - 1, \
  116. .type = type_, \
  117. .offset = offsetof(struct_, field_name_), \
  118. }
  119. /**
  120. * @brief Helper macro to declare a descriptor for an object value
  121. *
  122. * @param struct_ Struct packing the values
  123. * @param field_name_ Field name in the struct
  124. * @param sub_descr_ Array of json_obj_descr describing the subobject
  125. *
  126. * Here's an example of use:
  127. *
  128. * struct nested {
  129. * int foo;
  130. * struct {
  131. * int baz;
  132. * } bar;
  133. * };
  134. *
  135. * struct json_obj_descr nested_bar[] = {
  136. * { ... declare bar.baz descriptor ... },
  137. * };
  138. * struct json_obj_descr nested[] = {
  139. * { ... declare foo descriptor ... },
  140. * JSON_OBJ_DESCR_OBJECT(struct nested, bar, nested_bar),
  141. * };
  142. */
  143. #define JSON_OBJ_DESCR_OBJECT(struct_, field_name_, sub_descr_) \
  144. { \
  145. .field_name = (#field_name_), \
  146. .align_shift = Z_ALIGN_SHIFT(struct_), \
  147. .field_name_len = (sizeof(#field_name_) - 1), \
  148. .type = JSON_TOK_OBJECT_START, \
  149. .offset = offsetof(struct_, field_name_), \
  150. { \
  151. .object = { \
  152. .sub_descr = sub_descr_, \
  153. .sub_descr_len = ARRAY_SIZE(sub_descr_), \
  154. }, \
  155. }, \
  156. }
  157. /**
  158. * @brief Helper macro to declare a descriptor for an array of primitives
  159. *
  160. * @param struct_ Struct packing the values
  161. * @param field_name_ Field name in the struct
  162. * @param max_len_ Maximum number of elements in array
  163. * @param len_field_ Field name in the struct for the number of elements
  164. * in the array
  165. * @param elem_type_ Element type, must be a primitive type
  166. *
  167. * Here's an example of use:
  168. *
  169. * struct example {
  170. * int foo[10];
  171. * size_t foo_len;
  172. * };
  173. *
  174. * struct json_obj_descr array[] = {
  175. * JSON_OBJ_DESCR_ARRAY(struct example, foo, 10, foo_len,
  176. * JSON_TOK_NUMBER)
  177. * };
  178. */
  179. #define JSON_OBJ_DESCR_ARRAY(struct_, field_name_, max_len_, \
  180. len_field_, elem_type_) \
  181. { \
  182. .field_name = (#field_name_), \
  183. .align_shift = Z_ALIGN_SHIFT(struct_), \
  184. .field_name_len = sizeof(#field_name_) - 1, \
  185. .type = JSON_TOK_LIST_START, \
  186. .offset = offsetof(struct_, field_name_), \
  187. { \
  188. .array = { \
  189. .element_descr = (struct json_obj_descr[]) { { \
  190. .align_shift = \
  191. Z_ALIGN_SHIFT(struct_), \
  192. .type = elem_type_, \
  193. .offset = \
  194. offsetof(struct_, \
  195. len_field_), \
  196. } }, \
  197. .n_elements = (max_len_), \
  198. }, \
  199. }, \
  200. }
  201. /**
  202. * @brief Helper macro to declare a descriptor for an array of objects
  203. *
  204. * @param struct_ Struct packing the values
  205. * @param field_name_ Field name in the struct containing the array
  206. * @param max_len_ Maximum number of elements in the array
  207. * @param len_field_ Field name in the struct for the number of elements
  208. * in the array
  209. * @param elem_descr_ Element descriptor, pointer to a descriptor array
  210. * @param elem_descr_len_ Number of elements in elem_descr_
  211. *
  212. * Here's an example of use:
  213. *
  214. * struct person_height {
  215. * const char *name;
  216. * int height;
  217. * };
  218. *
  219. * struct people_heights {
  220. * struct person_height heights[10];
  221. * size_t heights_len;
  222. * };
  223. *
  224. * struct json_obj_descr person_height_descr[] = {
  225. * JSON_OBJ_DESCR_PRIM(struct person_height, name, JSON_TOK_STRING),
  226. * JSON_OBJ_DESCR_PRIM(struct person_height, height, JSON_TOK_NUMBER),
  227. * };
  228. *
  229. * struct json_obj_descr array[] = {
  230. * JSON_OBJ_DESCR_OBJ_ARRAY(struct people_heights, heights, 10,
  231. * heights_len, person_height_descr,
  232. * ARRAY_SIZE(person_height_descr)),
  233. * };
  234. */
  235. #define JSON_OBJ_DESCR_OBJ_ARRAY(struct_, field_name_, max_len_, \
  236. len_field_, elem_descr_, elem_descr_len_) \
  237. { \
  238. .field_name = (#field_name_), \
  239. .align_shift = Z_ALIGN_SHIFT(struct_), \
  240. .field_name_len = sizeof(#field_name_) - 1, \
  241. .type = JSON_TOK_LIST_START, \
  242. .offset = offsetof(struct_, field_name_), \
  243. { \
  244. .array = { \
  245. .element_descr = (struct json_obj_descr[]) { { \
  246. .align_shift = \
  247. Z_ALIGN_SHIFT(struct_), \
  248. .type = JSON_TOK_OBJECT_START, \
  249. .offset = offsetof(struct_, \
  250. len_field_), \
  251. { \
  252. .object = { \
  253. .sub_descr = \
  254. elem_descr_, \
  255. .sub_descr_len = \
  256. elem_descr_len_, \
  257. }, \
  258. }, \
  259. } }, \
  260. .n_elements = (max_len_), \
  261. }, \
  262. }, \
  263. }
  264. /**
  265. * @brief Helper macro to declare a descriptor for an array of array
  266. *
  267. * @param struct_ Struct packing the values
  268. * @param field_name_ Field name in the struct containing the array
  269. * @param max_len_ Maximum number of elements in the array
  270. * @param len_field_ Field name in the struct for the number of elements
  271. * in the array
  272. * @param elem_descr_ Element descriptor, pointer to a descriptor array
  273. * @param elem_descr_len_ Number of elements in elem_descr_
  274. *
  275. * Here's an example of use:
  276. *
  277. * struct person_height {
  278. * const char *name;
  279. * int height;
  280. * };
  281. *
  282. * struct person_heights_array {
  283. * struct person_height heights;
  284. * }
  285. *
  286. * struct people_heights {
  287. * struct person_height_array heights[10];
  288. * size_t heights_len;
  289. * };
  290. *
  291. * struct json_obj_descr person_height_descr[] = {
  292. * JSON_OBJ_DESCR_PRIM(struct person_height, name, JSON_TOK_STRING),
  293. * JSON_OBJ_DESCR_PRIM(struct person_height, height, JSON_TOK_NUMBER),
  294. * };
  295. *
  296. * struct json_obj_descr person_height_array_descr[] = {
  297. * JSON_OBJ_DESCR_OBJECT(struct person_heights_array,
  298. * heights, person_heigth_descr),
  299. * };
  300. *
  301. * struct json_obj_descr array_array[] = {
  302. * JSON_OBJ_DESCR_ARRAY_ARRAY(struct people_heights, heights, 10,
  303. * heights_len, person_height_array_descr,
  304. * ARRAY_SIZE(person_height_array_descr)),
  305. * };
  306. */
  307. #define JSON_OBJ_DESCR_ARRAY_ARRAY(struct_, field_name_, max_len_, len_field_, \
  308. elem_descr_, elem_descr_len_) \
  309. { \
  310. .field_name = (#field_name_), \
  311. .align_shift = Z_ALIGN_SHIFT(struct_), \
  312. .field_name_len = sizeof(#field_name_) - 1, \
  313. .type = JSON_TOK_LIST_START, \
  314. .offset = offsetof(struct_, field_name_), \
  315. { \
  316. .array = { \
  317. .element_descr = (struct json_obj_descr[]) { { \
  318. .align_shift = \
  319. Z_ALIGN_SHIFT(struct_), \
  320. .type = JSON_TOK_LIST_START, \
  321. .offset = offsetof(struct_, \
  322. len_field_), \
  323. { \
  324. .array = { \
  325. .element_descr = \
  326. elem_descr_, \
  327. .n_elements = \
  328. 1 + \
  329. ZERO_OR_COMPILE_ERROR( \
  330. elem_descr_len_ == 1 \
  331. ), \
  332. }, \
  333. }, \
  334. } }, \
  335. .n_elements = (max_len_), \
  336. }, \
  337. }, \
  338. }
  339. /**
  340. * @brief Variant of JSON_OBJ_DESCR_PRIM that can be used when the
  341. * structure and JSON field names differ.
  342. *
  343. * This is useful when the JSON field is not a valid C identifier.
  344. *
  345. * @param struct_ Struct packing the values.
  346. * @param json_field_name_ String, field name in JSON strings
  347. * @param struct_field_name_ Field name in the struct
  348. * @param type_ Token type for JSON value corresponding to a primitive
  349. * type.
  350. *
  351. * @see JSON_OBJ_DESCR_PRIM
  352. */
  353. #define JSON_OBJ_DESCR_PRIM_NAMED(struct_, json_field_name_, \
  354. struct_field_name_, type_) \
  355. { \
  356. .field_name = (json_field_name_), \
  357. .align_shift = Z_ALIGN_SHIFT(struct_), \
  358. .field_name_len = sizeof(json_field_name_) - 1, \
  359. .type = type_, \
  360. .offset = offsetof(struct_, struct_field_name_), \
  361. }
  362. /**
  363. * @brief Variant of JSON_OBJ_DESCR_OBJECT that can be used when the
  364. * structure and JSON field names differ.
  365. *
  366. * This is useful when the JSON field is not a valid C identifier.
  367. *
  368. * @param struct_ Struct packing the values
  369. * @param json_field_name_ String, field name in JSON strings
  370. * @param struct_field_name_ Field name in the struct
  371. * @param sub_descr_ Array of json_obj_descr describing the subobject
  372. *
  373. * @see JSON_OBJ_DESCR_OBJECT
  374. */
  375. #define JSON_OBJ_DESCR_OBJECT_NAMED(struct_, json_field_name_, \
  376. struct_field_name_, sub_descr_) \
  377. { \
  378. .field_name = (json_field_name_), \
  379. .align_shift = Z_ALIGN_SHIFT(struct_), \
  380. .field_name_len = (sizeof(json_field_name_) - 1), \
  381. .type = JSON_TOK_OBJECT_START, \
  382. .offset = offsetof(struct_, struct_field_name_), \
  383. { \
  384. .object = { \
  385. .sub_descr = sub_descr_, \
  386. .sub_descr_len = ARRAY_SIZE(sub_descr_), \
  387. }, \
  388. }, \
  389. }
  390. /**
  391. * @brief Variant of JSON_OBJ_DESCR_ARRAY that can be used when the
  392. * structure and JSON field names differ.
  393. *
  394. * This is useful when the JSON field is not a valid C identifier.
  395. *
  396. * @param struct_ Struct packing the values
  397. * @param json_field_name_ String, field name in JSON strings
  398. * @param struct_field_name_ Field name in the struct
  399. * @param max_len_ Maximum number of elements in array
  400. * @param len_field_ Field name in the struct for the number of elements
  401. * in the array
  402. * @param elem_type_ Element type, must be a primitive type
  403. *
  404. * @see JSON_OBJ_DESCR_ARRAY
  405. */
  406. #define JSON_OBJ_DESCR_ARRAY_NAMED(struct_, json_field_name_,\
  407. struct_field_name_, max_len_, len_field_, \
  408. elem_type_) \
  409. { \
  410. .field_name = (json_field_name_), \
  411. .align_shift = Z_ALIGN_SHIFT(struct_), \
  412. .field_name_len = sizeof(json_field_name_) - 1, \
  413. .type = JSON_TOK_LIST_START, \
  414. .offset = offsetof(struct_, struct_field_name_), \
  415. { \
  416. .array = { \
  417. .element_descr = (struct json_obj_descr[]) { { \
  418. .align_shift = \
  419. Z_ALIGN_SHIFT(struct_), \
  420. .type = elem_type_, \
  421. .offset = offsetof(struct_, \
  422. len_field_), \
  423. } }, \
  424. .n_elements = (max_len_), \
  425. }, \
  426. }, \
  427. }
  428. /**
  429. * @brief Variant of JSON_OBJ_DESCR_OBJ_ARRAY that can be used when
  430. * the structure and JSON field names differ.
  431. *
  432. * This is useful when the JSON field is not a valid C identifier.
  433. *
  434. * @param struct_ Struct packing the values
  435. * @param json_field_name_ String, field name of the array in JSON strings
  436. * @param struct_field_name_ Field name in the struct containing the array
  437. * @param max_len_ Maximum number of elements in the array
  438. * @param len_field_ Field name in the struct for the number of elements
  439. * in the array
  440. * @param elem_descr_ Element descriptor, pointer to a descriptor array
  441. * @param elem_descr_len_ Number of elements in elem_descr_
  442. *
  443. * Here's an example of use:
  444. *
  445. * struct person_height {
  446. * const char *name;
  447. * int height;
  448. * };
  449. *
  450. * struct people_heights {
  451. * struct person_height heights[10];
  452. * size_t heights_len;
  453. * };
  454. *
  455. * struct json_obj_descr person_height_descr[] = {
  456. * JSON_OBJ_DESCR_PRIM(struct person_height, name, JSON_TOK_STRING),
  457. * JSON_OBJ_DESCR_PRIM(struct person_height, height, JSON_TOK_NUMBER),
  458. * };
  459. *
  460. * struct json_obj_descr array[] = {
  461. * JSON_OBJ_DESCR_OBJ_ARRAY_NAMED(struct people_heights,
  462. * "people-heights", heights,
  463. * 10, heights_len,
  464. * person_height_descr,
  465. * ARRAY_SIZE(person_height_descr)),
  466. * };
  467. */
  468. #define JSON_OBJ_DESCR_OBJ_ARRAY_NAMED(struct_, json_field_name_, \
  469. struct_field_name_, max_len_, \
  470. len_field_, elem_descr_, \
  471. elem_descr_len_) \
  472. { \
  473. .field_name = json_field_name_, \
  474. .align_shift = Z_ALIGN_SHIFT(struct_), \
  475. .field_name_len = sizeof(json_field_name_) - 1, \
  476. .type = JSON_TOK_LIST_START, \
  477. .offset = offsetof(struct_, struct_field_name_), \
  478. { \
  479. .array = { \
  480. .element_descr = (struct json_obj_descr[]) { { \
  481. .align_shift = \
  482. Z_ALIGN_SHIFT(struct_), \
  483. .type = JSON_TOK_OBJECT_START, \
  484. .offset = offsetof(struct_, \
  485. len_field_), \
  486. { \
  487. .object = { \
  488. .sub_descr = \
  489. elem_descr_, \
  490. .sub_descr_len = \
  491. elem_descr_len_, \
  492. }, \
  493. }, \
  494. } }, \
  495. .n_elements = (max_len_), \
  496. }, \
  497. }, \
  498. }
  499. /**
  500. * @brief Parses the JSON-encoded object pointer to by @a json, with
  501. * size @a len, according to the descriptor pointed to by @a descr.
  502. * Values are stored in a struct pointed to by @a val. Set up the
  503. * descriptor like this:
  504. *
  505. * struct s { int foo; char *bar; }
  506. * struct json_obj_descr descr[] = {
  507. * JSON_OBJ_DESCR_PRIM(struct s, foo, JSON_TOK_NUMBER),
  508. * JSON_OBJ_DESCR_PRIM(struct s, bar, JSON_TOK_STRING),
  509. * };
  510. *
  511. * Since this parser is designed for machine-to-machine communications, some
  512. * liberties were taken to simplify the design:
  513. * (1) strings are not unescaped (but only valid escape sequences are
  514. * accepted);
  515. * (2) no UTF-8 validation is performed; and
  516. * (3) only integer numbers are supported (no strtod() in the minimal libc).
  517. *
  518. * @param json Pointer to JSON-encoded value to be parsed
  519. * @param len Length of JSON-encoded value
  520. * @param descr Pointer to the descriptor array
  521. * @param descr_len Number of elements in the descriptor array. Must be less
  522. * than 31 due to implementation detail reasons (if more fields are
  523. * necessary, use two descriptors)
  524. * @param val Pointer to the struct to hold the decoded values
  525. *
  526. * @return < 0 if error, bitmap of decoded fields on success (bit 0
  527. * is set if first field in the descriptor has been properly decoded, etc).
  528. */
  529. int json_obj_parse(char *json, size_t len,
  530. const struct json_obj_descr *descr, size_t descr_len,
  531. void *val);
  532. /**
  533. * @brief Escapes the string so it can be used to encode JSON objects
  534. *
  535. * @param str The string to escape; the escape string is stored the
  536. * buffer pointed to by this parameter
  537. * @param len Points to a size_t containing the size before and after
  538. * the escaping process
  539. * @param buf_size The size of buffer str points to
  540. *
  541. * @return 0 if string has been escaped properly, or -ENOMEM if there
  542. * was not enough space to escape the buffer
  543. */
  544. ssize_t json_escape(char *str, size_t *len, size_t buf_size);
  545. /**
  546. * @brief Calculates the JSON-escaped string length
  547. *
  548. * @param str The string to analyze
  549. * @param len String size
  550. *
  551. * @return The length str would have if it were escaped
  552. */
  553. size_t json_calc_escaped_len(const char *str, size_t len);
  554. /**
  555. * @brief Calculates the string length to fully encode an object
  556. *
  557. * @param descr Pointer to the descriptor array
  558. * @param descr_len Number of elements in the descriptor array
  559. * @param val Struct holding the values
  560. *
  561. * @return Number of bytes necessary to encode the values if >0,
  562. * an error code is returned.
  563. */
  564. ssize_t json_calc_encoded_len(const struct json_obj_descr *descr,
  565. size_t descr_len, const void *val);
  566. /**
  567. * @brief Encodes an object in a contiguous memory location
  568. *
  569. * @param descr Pointer to the descriptor array
  570. * @param descr_len Number of elements in the descriptor array
  571. * @param val Struct holding the values
  572. * @param buffer Buffer to store the JSON data
  573. * @param buf_size Size of buffer, in bytes, with space for the terminating
  574. * NUL character
  575. *
  576. * @return 0 if object has been successfully encoded. A negative value
  577. * indicates an error (as defined on errno.h).
  578. */
  579. int json_obj_encode_buf(const struct json_obj_descr *descr, size_t descr_len,
  580. const void *val, char *buffer, size_t buf_size);
  581. /**
  582. * @brief Encodes an array in a contiguous memory location
  583. *
  584. * @param descr Pointer to the descriptor array
  585. * @param val Struct holding the values
  586. * @param buffer Buffer to store the JSON data
  587. * @param buf_size Size of buffer, in bytes, with space for the terminating
  588. * NUL character
  589. *
  590. * @return 0 if object has been successfully encoded. A negative value
  591. * indicates an error (as defined on errno.h).
  592. */
  593. int json_arr_encode_buf(const struct json_obj_descr *descr, const void *val,
  594. char *buffer, size_t buf_size);
  595. /**
  596. * @brief Encodes an object using an arbitrary writer function
  597. *
  598. * @param descr Pointer to the descriptor array
  599. * @param descr_len Number of elements in the descriptor array
  600. * @param val Struct holding the values
  601. * @param append_bytes Function to append bytes to the output
  602. * @param data Data pointer to be passed to the append_bytes callback
  603. * function.
  604. *
  605. * @return 0 if object has been successfully encoded. A negative value
  606. * indicates an error.
  607. */
  608. int json_obj_encode(const struct json_obj_descr *descr, size_t descr_len,
  609. const void *val, json_append_bytes_t append_bytes,
  610. void *data);
  611. /**
  612. * @brief Encodes an array using an arbitrary writer function
  613. *
  614. * @param descr Pointer to the descriptor array
  615. * @param val Struct holding the values
  616. * @param append_bytes Function to append bytes to the output
  617. * @param data Data pointer to be passed to the append_bytes callback
  618. * function.
  619. *
  620. * @return 0 if object has been successfully encoded. A negative value
  621. * indicates an error.
  622. */
  623. int json_arr_encode(const struct json_obj_descr *descr, const void *val,
  624. json_append_bytes_t append_bytes, void *data);
  625. #ifdef __cplusplus
  626. }
  627. #endif
  628. /**
  629. * @}
  630. */
  631. #endif /* ZEPHYR_INCLUDE_DATA_JSON_H_ */