2017-02-17 00:51:31 +01:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2017 Intel Corporation
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <limits.h>
|
2017-03-15 00:50:33 +01:00
|
|
|
#include <misc/printk.h>
|
2017-02-17 00:51:31 +01:00
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "json.h"
|
|
|
|
|
|
|
|
struct token {
|
|
|
|
enum json_tokens type;
|
|
|
|
char *start;
|
|
|
|
char *end;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct lexer {
|
|
|
|
void *(*state)(struct lexer *lexer);
|
|
|
|
char *start;
|
|
|
|
char *pos;
|
|
|
|
char *end;
|
|
|
|
struct token token;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct json_obj {
|
|
|
|
struct lexer lexer;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct json_obj_key_value {
|
|
|
|
const char *key;
|
|
|
|
size_t key_len;
|
|
|
|
struct token value;
|
|
|
|
};
|
|
|
|
|
|
|
|
static bool lexer_consume(struct lexer *lexer, struct token *token,
|
|
|
|
enum json_tokens empty_token)
|
|
|
|
{
|
|
|
|
if (lexer->token.type == empty_token) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
*token = lexer->token;
|
|
|
|
lexer->token.type = empty_token;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool lexer_next(struct lexer *lexer, struct token *token)
|
|
|
|
{
|
|
|
|
while (lexer->state) {
|
|
|
|
if (lexer_consume(lexer, token, JSON_TOK_NONE)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
lexer->state = lexer->state(lexer);
|
|
|
|
}
|
|
|
|
|
|
|
|
return lexer_consume(lexer, token, JSON_TOK_EOF);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void *lexer_json(struct lexer *lexer);
|
|
|
|
|
|
|
|
static void emit(struct lexer *lexer, enum json_tokens token)
|
|
|
|
{
|
|
|
|
lexer->token.type = token;
|
|
|
|
lexer->token.start = lexer->start;
|
|
|
|
lexer->token.end = lexer->pos;
|
|
|
|
lexer->start = lexer->pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
static char next(struct lexer *lexer)
|
|
|
|
{
|
|
|
|
if (lexer->pos >= lexer->end) {
|
|
|
|
lexer->pos = lexer->end + 1;
|
|
|
|
|
|
|
|
return '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
return *lexer->pos++;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ignore(struct lexer *lexer)
|
|
|
|
{
|
|
|
|
lexer->start = lexer->pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void backup(struct lexer *lexer)
|
|
|
|
{
|
|
|
|
lexer->pos--;
|
|
|
|
}
|
|
|
|
|
|
|
|
static char peek(struct lexer *lexer)
|
|
|
|
{
|
|
|
|
char chr = next(lexer);
|
|
|
|
|
|
|
|
backup(lexer);
|
|
|
|
|
|
|
|
return chr;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void *lexer_string(struct lexer *lexer)
|
|
|
|
{
|
|
|
|
ignore(lexer);
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
char chr = next(lexer);
|
|
|
|
|
|
|
|
if (chr == '\0') {
|
|
|
|
emit(lexer, JSON_TOK_ERROR);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (chr == '\\') {
|
|
|
|
switch (next(lexer)) {
|
|
|
|
case '"':
|
|
|
|
case '\\':
|
|
|
|
case '/':
|
|
|
|
case 'b':
|
|
|
|
case 'f':
|
|
|
|
case 'n':
|
|
|
|
case 'r':
|
|
|
|
case 't':
|
|
|
|
continue;
|
|
|
|
case 'u':
|
|
|
|
if (!isxdigit(next(lexer))) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isxdigit(next(lexer))) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isxdigit(next(lexer))) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isxdigit(next(lexer))) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (chr == '"') {
|
|
|
|
backup(lexer);
|
|
|
|
emit(lexer, JSON_TOK_STRING);
|
|
|
|
|
|
|
|
next(lexer);
|
|
|
|
ignore(lexer);
|
|
|
|
|
|
|
|
return lexer_json;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
error:
|
|
|
|
emit(lexer, JSON_TOK_ERROR);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2017-03-16 18:11:00 +01:00
|
|
|
static int accept_run(struct lexer *lexer, const char *run)
|
|
|
|
{
|
|
|
|
for (; *run; run++) {
|
|
|
|
if (next(lexer) != *run) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-02-17 00:51:31 +01:00
|
|
|
static void *lexer_boolean(struct lexer *lexer)
|
|
|
|
{
|
|
|
|
backup(lexer);
|
|
|
|
|
|
|
|
switch (next(lexer)) {
|
|
|
|
case 't':
|
2017-03-16 18:11:00 +01:00
|
|
|
if (!accept_run(lexer, "rue")) {
|
|
|
|
emit(lexer, JSON_TOK_TRUE);
|
|
|
|
return lexer_json;
|
2017-02-17 00:51:31 +01:00
|
|
|
}
|
2017-03-16 18:11:00 +01:00
|
|
|
break;
|
2017-02-17 00:51:31 +01:00
|
|
|
case 'f':
|
2017-03-16 18:11:00 +01:00
|
|
|
if (!accept_run(lexer, "alse")) {
|
|
|
|
emit(lexer, JSON_TOK_FALSE);
|
|
|
|
return lexer_json;
|
2017-02-17 00:51:31 +01:00
|
|
|
}
|
2017-03-16 18:11:00 +01:00
|
|
|
break;
|
2017-02-17 00:51:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
emit(lexer, JSON_TOK_ERROR);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void *lexer_null(struct lexer *lexer)
|
|
|
|
{
|
2017-03-16 18:11:00 +01:00
|
|
|
if (accept_run(lexer, "ull") < 0) {
|
|
|
|
emit(lexer, JSON_TOK_ERROR);
|
|
|
|
return NULL;
|
2017-02-17 00:51:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
emit(lexer, JSON_TOK_NULL);
|
|
|
|
return lexer_json;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void *lexer_number(struct lexer *lexer)
|
|
|
|
{
|
|
|
|
while (true) {
|
|
|
|
char chr = next(lexer);
|
|
|
|
|
|
|
|
if (isdigit(chr) || chr == '.') {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
backup(lexer);
|
|
|
|
emit(lexer, JSON_TOK_NUMBER);
|
|
|
|
|
|
|
|
return lexer_json;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void *lexer_json(struct lexer *lexer)
|
|
|
|
{
|
|
|
|
while (true) {
|
|
|
|
char chr = next(lexer);
|
|
|
|
|
|
|
|
switch (chr) {
|
|
|
|
case '\0':
|
|
|
|
emit(lexer, JSON_TOK_EOF);
|
|
|
|
return NULL;
|
|
|
|
case '}':
|
|
|
|
case '{':
|
2017-03-21 21:51:15 +01:00
|
|
|
case '[':
|
|
|
|
case ']':
|
2017-02-17 00:51:31 +01:00
|
|
|
case ',':
|
|
|
|
case ':':
|
|
|
|
emit(lexer, (enum json_tokens)chr);
|
|
|
|
return lexer_json;
|
|
|
|
case '"':
|
|
|
|
return lexer_string;
|
|
|
|
case 'n':
|
|
|
|
return lexer_null;
|
|
|
|
case 't':
|
|
|
|
case 'f':
|
|
|
|
return lexer_boolean;
|
|
|
|
case '-':
|
|
|
|
if (isdigit(peek(lexer))) {
|
|
|
|
return lexer_number;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* fallthrough */
|
|
|
|
default:
|
|
|
|
if (isspace(chr)) {
|
2017-03-23 23:15:11 +01:00
|
|
|
ignore(lexer);
|
2017-02-17 00:51:31 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isdigit(chr)) {
|
|
|
|
return lexer_number;
|
|
|
|
}
|
|
|
|
|
|
|
|
emit(lexer, JSON_TOK_ERROR);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void lexer_init(struct lexer *lexer, char *data, size_t len)
|
|
|
|
{
|
|
|
|
lexer->state = lexer_json;
|
|
|
|
lexer->start = data;
|
|
|
|
lexer->pos = data;
|
|
|
|
lexer->end = data + len;
|
|
|
|
lexer->token.type = JSON_TOK_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int obj_init(struct json_obj *json, char *data, size_t len)
|
|
|
|
{
|
|
|
|
struct token token;
|
|
|
|
|
|
|
|
lexer_init(&json->lexer, data, len);
|
|
|
|
|
|
|
|
if (!lexer_next(&json->lexer, &token)) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (token.type != JSON_TOK_OBJECT_START) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-03-21 21:51:15 +01:00
|
|
|
static int element_token(enum json_tokens token)
|
|
|
|
{
|
|
|
|
switch (token) {
|
|
|
|
case JSON_TOK_OBJECT_START:
|
|
|
|
case JSON_TOK_LIST_START:
|
|
|
|
case JSON_TOK_STRING:
|
|
|
|
case JSON_TOK_NUMBER:
|
|
|
|
case JSON_TOK_TRUE:
|
|
|
|
case JSON_TOK_FALSE:
|
|
|
|
return 0;
|
|
|
|
default:
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int obj_next(struct json_obj *json,
|
|
|
|
struct json_obj_key_value *kv)
|
2017-02-17 00:51:31 +01:00
|
|
|
{
|
|
|
|
struct token token;
|
|
|
|
|
|
|
|
if (!lexer_next(&json->lexer, &token)) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Match end of object or next key */
|
|
|
|
switch (token.type) {
|
|
|
|
case JSON_TOK_OBJECT_END:
|
|
|
|
kv->key = NULL;
|
|
|
|
kv->key_len = 0;
|
|
|
|
kv->value = token;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
case JSON_TOK_COMMA:
|
|
|
|
if (!lexer_next(&json->lexer, &token)) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (token.type != JSON_TOK_STRING) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* fallthrough */
|
|
|
|
case JSON_TOK_STRING:
|
|
|
|
kv->key = token.start;
|
|
|
|
kv->key_len = (size_t)(token.end - token.start);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Match : after key */
|
|
|
|
if (!lexer_next(&json->lexer, &token)) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (token.type != JSON_TOK_COLON) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Match value */
|
|
|
|
if (!lexer_next(&json->lexer, &kv->value)) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2017-03-21 21:51:15 +01:00
|
|
|
return element_token(kv->value.type);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int arr_next(struct json_obj *json, struct token *value)
|
|
|
|
{
|
|
|
|
if (!lexer_next(&json->lexer, value)) {
|
2017-02-17 00:51:31 +01:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
2017-03-21 21:51:15 +01:00
|
|
|
|
|
|
|
if (value->type == JSON_TOK_LIST_END) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (value->type == JSON_TOK_COMMA) {
|
|
|
|
if (!lexer_next(&json->lexer, value)) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return element_token(value->type);
|
2017-02-17 00:51:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static int decode_num(const struct token *token, int32_t *num)
|
|
|
|
{
|
|
|
|
/* FIXME: strtod() is not available in newlib/minimal libc,
|
2017-03-21 23:49:25 +01:00
|
|
|
* so using strtol() here.
|
2017-02-17 00:51:31 +01:00
|
|
|
*/
|
|
|
|
char *endptr;
|
|
|
|
char prev_end;
|
|
|
|
|
|
|
|
prev_end = *token->end;
|
|
|
|
*token->end = '\0';
|
|
|
|
|
|
|
|
errno = 0;
|
|
|
|
*num = strtol(token->start, &endptr, 10);
|
|
|
|
|
|
|
|
*token->end = prev_end;
|
|
|
|
|
|
|
|
if (errno != 0) {
|
|
|
|
return -errno;
|
|
|
|
}
|
|
|
|
|
2017-03-21 23:49:25 +01:00
|
|
|
if (endptr != token->end) {
|
2017-02-17 00:51:31 +01:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool equivalent_types(enum json_tokens type1, enum json_tokens type2)
|
|
|
|
{
|
|
|
|
if (type1 == JSON_TOK_TRUE || type1 == JSON_TOK_FALSE) {
|
|
|
|
return type2 == JSON_TOK_TRUE || type2 == JSON_TOK_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return type1 == type2;
|
|
|
|
}
|
|
|
|
|
2017-03-21 21:51:15 +01:00
|
|
|
static int obj_parse(struct json_obj *obj,
|
|
|
|
const struct json_obj_descr *descr, size_t descr_len,
|
|
|
|
void *val);
|
|
|
|
static int arr_parse(struct json_obj *obj,
|
|
|
|
const struct json_obj_descr *elem_descr,
|
|
|
|
size_t max_elements, void *field, void *val);
|
|
|
|
|
|
|
|
static int decode_value(struct json_obj *obj,
|
|
|
|
const struct json_obj_descr *descr,
|
|
|
|
struct token *value, void *field, void *val)
|
2017-02-17 00:51:31 +01:00
|
|
|
{
|
|
|
|
|
2017-03-21 21:51:15 +01:00
|
|
|
if (!equivalent_types(value->type, descr->type)) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
2017-02-17 00:51:31 +01:00
|
|
|
|
2017-03-21 21:51:15 +01:00
|
|
|
switch (descr->type) {
|
|
|
|
case JSON_TOK_OBJECT_START:
|
|
|
|
return obj_parse(obj, descr->sub_descr,
|
|
|
|
descr->sub_descr_len,
|
|
|
|
field);
|
|
|
|
case JSON_TOK_LIST_START:
|
|
|
|
return arr_parse(obj, descr->element_descr,
|
|
|
|
descr->n_elements, field, val);
|
|
|
|
case JSON_TOK_FALSE:
|
|
|
|
case JSON_TOK_TRUE: {
|
|
|
|
bool *value = field;
|
|
|
|
|
|
|
|
*value = descr->type == JSON_TOK_TRUE;
|
|
|
|
|
|
|
|
return 0;
|
2017-02-17 00:51:31 +01:00
|
|
|
}
|
2017-03-21 21:51:15 +01:00
|
|
|
case JSON_TOK_NUMBER: {
|
|
|
|
int32_t *num = field;
|
2017-02-17 00:51:31 +01:00
|
|
|
|
2017-03-21 21:51:15 +01:00
|
|
|
return decode_num(value, num);
|
|
|
|
}
|
|
|
|
case JSON_TOK_STRING: {
|
|
|
|
char **str = field;
|
|
|
|
|
|
|
|
*value->end = '\0';
|
|
|
|
*str = value->start;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static ptrdiff_t get_elem_size(const struct json_obj_descr *descr)
|
|
|
|
{
|
|
|
|
switch (descr->type) {
|
|
|
|
case JSON_TOK_NUMBER:
|
|
|
|
return sizeof(int32_t);
|
|
|
|
case JSON_TOK_STRING:
|
|
|
|
return sizeof(char *);
|
|
|
|
case JSON_TOK_TRUE:
|
|
|
|
case JSON_TOK_FALSE:
|
|
|
|
return sizeof(bool);
|
|
|
|
case JSON_TOK_LIST_START:
|
|
|
|
return descr->n_elements * get_elem_size(descr->element_descr);
|
|
|
|
case JSON_TOK_OBJECT_START: {
|
|
|
|
ptrdiff_t total = 0;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < descr->sub_descr_len; i++) {
|
|
|
|
total += get_elem_size(&descr->sub_descr[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return total;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int arr_parse(struct json_obj *obj,
|
|
|
|
const struct json_obj_descr *elem_descr,
|
|
|
|
size_t max_elements, void *field, void *val)
|
|
|
|
{
|
|
|
|
ptrdiff_t elem_size = get_elem_size(elem_descr);
|
|
|
|
void *last_elem = (char *)field + elem_size * max_elements;
|
|
|
|
size_t *elements = (size_t *)((char *)val + elem_descr->offset);
|
|
|
|
struct token value;
|
|
|
|
|
|
|
|
assert(elem_size > 0);
|
|
|
|
|
|
|
|
*elements = 0;
|
|
|
|
|
|
|
|
while (!arr_next(obj, &value)) {
|
|
|
|
if (value.type == JSON_TOK_LIST_END) {
|
|
|
|
return 0;
|
|
|
|
}
|
2017-02-17 00:51:31 +01:00
|
|
|
|
2017-03-21 21:51:15 +01:00
|
|
|
if (decode_value(obj, elem_descr, &value, field, val) < 0) {
|
2017-02-17 00:51:31 +01:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2017-03-21 21:51:15 +01:00
|
|
|
(*elements)++;
|
|
|
|
|
|
|
|
field = (char *)field + elem_size;
|
|
|
|
if (field == last_elem) {
|
|
|
|
return -ENOSPC;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int obj_parse(struct json_obj *obj, const struct json_obj_descr *descr,
|
|
|
|
size_t descr_len, void *val)
|
|
|
|
{
|
|
|
|
struct json_obj_key_value kv;
|
|
|
|
int32_t decoded_fields = 0;
|
|
|
|
size_t i;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
while (!obj_next(obj, &kv)) {
|
|
|
|
if (kv.value.type == JSON_TOK_OBJECT_END) {
|
|
|
|
return decoded_fields;
|
|
|
|
}
|
|
|
|
|
2017-02-17 00:51:31 +01:00
|
|
|
for (i = 0; i < descr_len; i++) {
|
2017-03-21 21:51:15 +01:00
|
|
|
void *decode_field = (char *)val + descr[i].offset;
|
2017-02-17 00:51:31 +01:00
|
|
|
|
|
|
|
/* Field has been decoded already, skip */
|
|
|
|
if (decoded_fields & (1 << i)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check if it's the i-th field */
|
|
|
|
if (kv.key_len != descr[i].field_name_len) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (memcmp(kv.key, descr[i].field_name,
|
2017-03-21 21:51:15 +01:00
|
|
|
descr[i].field_name_len)) {
|
2017-02-17 00:51:31 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Store the decoded value */
|
2017-03-21 21:51:15 +01:00
|
|
|
ret = decode_value(obj, &descr[i], &kv.value,
|
|
|
|
decode_field, val);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
2017-02-17 00:51:31 +01:00
|
|
|
}
|
|
|
|
|
2017-03-21 21:51:15 +01:00
|
|
|
decoded_fields |= 1<<i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2017-02-17 00:51:31 +01:00
|
|
|
|
2017-03-21 21:51:15 +01:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
2017-02-17 00:51:31 +01:00
|
|
|
|
2017-03-21 21:51:15 +01:00
|
|
|
int json_obj_parse(char *payload, size_t len,
|
|
|
|
const struct json_obj_descr *descr, size_t descr_len,
|
|
|
|
void *val)
|
|
|
|
{
|
|
|
|
struct json_obj obj;
|
|
|
|
int ret;
|
2017-02-17 00:51:31 +01:00
|
|
|
|
2017-03-21 21:51:15 +01:00
|
|
|
assert(descr_len < (sizeof(ret) * CHAR_BIT - 1));
|
2017-02-17 00:51:31 +01:00
|
|
|
|
2017-03-21 21:51:15 +01:00
|
|
|
ret = obj_init(&obj, payload, len);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
2017-02-17 00:51:31 +01:00
|
|
|
}
|
|
|
|
|
2017-03-21 21:51:15 +01:00
|
|
|
return obj_parse(&obj, descr, descr_len, val);
|
2017-02-17 00:51:31 +01:00
|
|
|
}
|
|
|
|
|
2017-03-21 23:31:59 +01:00
|
|
|
static uint8_t escape_as(uint8_t chr)
|
|
|
|
{
|
|
|
|
switch (chr) {
|
|
|
|
case '"':
|
|
|
|
return '"';
|
|
|
|
case '\\':
|
|
|
|
return '\\';
|
|
|
|
case '\b':
|
|
|
|
return 'b';
|
|
|
|
case '\f':
|
|
|
|
return 'f';
|
|
|
|
case '\n':
|
|
|
|
return 'n';
|
|
|
|
case '\r':
|
|
|
|
return 'r';
|
|
|
|
case '\t':
|
|
|
|
return 't';
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2017-02-17 00:51:31 +01:00
|
|
|
|
2017-03-15 00:50:33 +01:00
|
|
|
static int json_escape_internal(const char *str,
|
|
|
|
json_append_bytes_t append_bytes,
|
|
|
|
void *data)
|
2017-02-17 00:51:31 +01:00
|
|
|
{
|
2017-03-21 23:31:59 +01:00
|
|
|
const char *cur;
|
2017-03-15 00:50:33 +01:00
|
|
|
int ret = 0;
|
2017-02-17 00:51:31 +01:00
|
|
|
|
2017-03-15 00:50:33 +01:00
|
|
|
for (cur = str; ret == 0 && *cur; cur++) {
|
2017-03-21 23:31:59 +01:00
|
|
|
uint8_t escaped = escape_as(*cur);
|
2017-03-15 00:50:33 +01:00
|
|
|
|
2017-03-21 23:31:59 +01:00
|
|
|
if (escaped) {
|
|
|
|
uint8_t bytes[2] = { '\\', escaped };
|
2017-03-15 00:50:33 +01:00
|
|
|
|
|
|
|
ret = append_bytes(bytes, 2, data);
|
2017-02-17 00:51:31 +01:00
|
|
|
} else {
|
2017-03-15 00:50:33 +01:00
|
|
|
ret = append_bytes(cur, 1, data);
|
2017-02-17 00:51:31 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-15 00:50:33 +01:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct appender {
|
|
|
|
char *buffer;
|
|
|
|
size_t used;
|
|
|
|
size_t size;
|
|
|
|
};
|
|
|
|
|
|
|
|
static int append_bytes_to_buf(const uint8_t *bytes, size_t len, void *data)
|
|
|
|
{
|
|
|
|
struct appender *appender = data;
|
|
|
|
|
|
|
|
if (len > appender->size - appender->used) {
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(appender->buffer + appender->used, bytes, len);
|
|
|
|
appender->used += len;
|
|
|
|
appender->buffer[appender->used] = '\0';
|
2017-02-17 00:51:31 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-03-15 00:50:33 +01:00
|
|
|
static int json_escape_buf(char *str, size_t *len, size_t buf_size)
|
|
|
|
{
|
|
|
|
char tmp_buf[buf_size + 1];
|
|
|
|
struct appender appender = { .buffer = tmp_buf, .size = buf_size };
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = json_escape_internal(str, append_bytes_to_buf, &appender);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = append_bytes_to_buf("", 1, &appender);
|
|
|
|
if (!ret) {
|
|
|
|
memcpy(str, tmp_buf, appender.size);
|
|
|
|
if (len) {
|
|
|
|
*len = appender.size;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2017-02-17 00:51:31 +01:00
|
|
|
size_t json_calc_escaped_len(const char *str, size_t len)
|
|
|
|
{
|
|
|
|
size_t escaped_len = len;
|
|
|
|
size_t pos;
|
|
|
|
|
|
|
|
for (pos = 0; pos < len; pos++) {
|
2017-03-21 23:31:59 +01:00
|
|
|
if (escape_as(str[pos])) {
|
2017-02-17 00:51:31 +01:00
|
|
|
escaped_len++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return escaped_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t json_escape(char *str, size_t *len, size_t buf_size)
|
|
|
|
{
|
|
|
|
size_t escaped_len;
|
|
|
|
|
|
|
|
escaped_len = json_calc_escaped_len(str, *len);
|
|
|
|
|
|
|
|
if (escaped_len == *len) {
|
|
|
|
/* If no escape is necessary, don't bother using up temporary
|
|
|
|
* stack space to copy the string.
|
|
|
|
*/
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (escaped_len >= buf_size) {
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
2017-03-15 00:50:33 +01:00
|
|
|
return json_escape_buf(str, len, escaped_len);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int encode(const struct json_obj_descr *descr, const void *val,
|
|
|
|
json_append_bytes_t append_bytes, void *data);
|
|
|
|
|
|
|
|
static int arr_encode(const struct json_obj_descr *descr, const void *field,
|
|
|
|
const void *val, json_append_bytes_t append_bytes,
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
struct json_obj_descr elem_descr = { .type = descr->type };
|
|
|
|
ptrdiff_t elem_size = get_elem_size(descr);
|
|
|
|
size_t n_elem = *(size_t *)((char *)val + descr->offset);
|
|
|
|
size_t i;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = append_bytes("[", 1, data);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < n_elem; i++) {
|
|
|
|
ret = encode(&elem_descr, field, append_bytes, data);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i < n_elem - 1) {
|
|
|
|
ret = append_bytes(",", 1, data);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
field = (char *)field + elem_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
return append_bytes("]", 1, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int str_encode(const char **str, json_append_bytes_t append_bytes,
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = append_bytes("\"", 1, data);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = json_escape_internal(*str, append_bytes, data);
|
|
|
|
if (!ret) {
|
|
|
|
return append_bytes("\"", 1, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int num_encode(const int32_t *num, json_append_bytes_t append_bytes,
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
char buf[3 * sizeof(int32_t)];
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = snprintk(buf, sizeof(buf), "%d", *num);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
if (ret >= (int)sizeof(buf)) {
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
return append_bytes(buf, (size_t)ret, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int bool_encode(const bool *value, json_append_bytes_t append_bytes,
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
if (*value) {
|
|
|
|
return append_bytes("true", 4, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
return append_bytes("false", 5, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int obj_encode(const struct json_obj_descr *descr, size_t descr_len,
|
|
|
|
const void *val, json_append_bytes_t append_bytes,
|
|
|
|
void *data);
|
|
|
|
|
|
|
|
static int encode(const struct json_obj_descr *descr, const void *val,
|
|
|
|
json_append_bytes_t append_bytes, void *data)
|
|
|
|
{
|
|
|
|
void *ptr = (char *)val + descr->offset;
|
|
|
|
|
|
|
|
switch (descr->type) {
|
|
|
|
case JSON_TOK_FALSE:
|
|
|
|
case JSON_TOK_TRUE:
|
|
|
|
return bool_encode(ptr, append_bytes, data);
|
|
|
|
case JSON_TOK_STRING:
|
|
|
|
return str_encode(ptr, append_bytes, data);
|
|
|
|
case JSON_TOK_LIST_START:
|
|
|
|
return arr_encode(descr->element_descr, ptr,
|
|
|
|
val, append_bytes, data);
|
|
|
|
case JSON_TOK_OBJECT_START:
|
|
|
|
return obj_encode(descr->sub_descr, descr->sub_descr_len,
|
|
|
|
ptr, append_bytes, data);
|
|
|
|
case JSON_TOK_NUMBER:
|
|
|
|
return num_encode(ptr, append_bytes, data);
|
|
|
|
default:
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int obj_encode(const struct json_obj_descr *descr, size_t descr_len,
|
|
|
|
const void *val, json_append_bytes_t append_bytes,
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = append_bytes("{", 1, data);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < descr_len; i++) {
|
|
|
|
ret = str_encode((const char **)&descr[i].field_name,
|
|
|
|
append_bytes, data);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = append_bytes(":", 1, data);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = encode(&descr[i], val, append_bytes, data);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i < descr_len - 1) {
|
|
|
|
ret = append_bytes(",", 1, data);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return append_bytes("}", 1, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
int json_obj_encode(const struct json_obj_descr *descr, size_t descr_len,
|
|
|
|
const void *val, json_append_bytes_t append_bytes,
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = obj_encode(descr, descr_len, val, append_bytes, data);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
return append_bytes("", 1, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
int json_obj_encode_buf(const struct json_obj_descr *descr, size_t descr_len,
|
|
|
|
const void *val, char *buffer, size_t buf_size)
|
|
|
|
{
|
|
|
|
struct appender appender = { .buffer = buffer, .size = buf_size };
|
|
|
|
|
|
|
|
return json_obj_encode(descr, descr_len, val, append_bytes_to_buf,
|
|
|
|
&appender);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int measure_bytes(const uint8_t *bytes, size_t len, void *data)
|
|
|
|
{
|
|
|
|
ssize_t *total = data;
|
|
|
|
|
|
|
|
*total += (ssize_t)len;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t json_calc_encoded_len(const struct json_obj_descr *descr,
|
|
|
|
size_t descr_len, const void *val)
|
|
|
|
{
|
|
|
|
ssize_t total = 0;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = json_obj_encode(descr, descr_len, val, measure_bytes, &total);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
return total;
|
2017-02-17 00:51:31 +01:00
|
|
|
}
|