mirror of
https://github.com/nyanmisaka/mpp.git
synced 2025-12-26 11:17:44 +01:00
1. Create mpp_cfg when define KMPP_OBJ_HIERARCHY_ENABLE 2. Add kmpp_objdef_get_cfg_root function 3. Fix mpp_cfg update flag judgment logic 4. Add mpp_enc_cfg extract and apply interface from configure file Signed-off-by: xiaoxu.chen <xiaoxu.chen@rock-chips.com> Change-Id: I3a02e7bbddd20a78c3284589fc9513a1c49cac18
3243 lines
84 KiB
C
3243 lines
84 KiB
C
/* SPDX-License-Identifier: Apache-2.0 OR MIT */
|
|
/*
|
|
* Copyright (c) 2025 Rockchip Electronics Co., Ltd.
|
|
*/
|
|
|
|
#define MODULE_TAG "mpp_cfg_io"
|
|
|
|
#include <errno.h>
|
|
#include <float.h>
|
|
#include <string.h>
|
|
#include <limits.h>
|
|
|
|
#include "mpp_env.h"
|
|
#include "mpp_mem.h"
|
|
#include "mpp_list.h"
|
|
#include "mpp_debug.h"
|
|
#include "mpp_common.h"
|
|
|
|
#include "mpp_trie.h"
|
|
#include "mpp_cfg.h"
|
|
#include "mpp_cfg_io.h"
|
|
#include "rk_venc_cfg.h"
|
|
|
|
#define MAX_CFG_DEPTH (64)
|
|
|
|
#define CFG_IO_DBG_FLOW (0x00000001)
|
|
#define CFG_IO_DBG_BYTE (0x00000002)
|
|
#define CFG_IO_DBG_TO (0x00000004)
|
|
#define CFG_IO_DBG_FROM (0x00000008)
|
|
#define CFG_IO_DBG_FREE (0x00000010)
|
|
#define CFG_IO_DBG_NAME (0x00000020)
|
|
#define CFG_IO_DBG_SHOW (0x00000040)
|
|
#define CFG_IO_DBG_INFO (0x00000080)
|
|
|
|
#define cfg_io_dbg(flag, fmt, ...) _mpp_dbg(mpp_cfg_io_debug, flag, fmt, ## __VA_ARGS__)
|
|
#define cfg_io_dbg_f(flag, fmt, ...) _mpp_dbg_f(mpp_cfg_io_debug, flag, fmt, ## __VA_ARGS__)
|
|
|
|
#define cfg_io_dbg_flow(fmt, ...) cfg_io_dbg(CFG_IO_DBG_FLOW, fmt, ## __VA_ARGS__)
|
|
#define cfg_io_dbg_byte(fmt, ...) cfg_io_dbg(CFG_IO_DBG_BYTE, fmt, ## __VA_ARGS__)
|
|
#define cfg_io_dbg_to(fmt, ...) cfg_io_dbg(CFG_IO_DBG_TO, fmt, ## __VA_ARGS__)
|
|
#define cfg_io_dbg_from(fmt, ...) cfg_io_dbg(CFG_IO_DBG_FROM, fmt, ## __VA_ARGS__)
|
|
#define cfg_io_dbg_free(fmt, ...) cfg_io_dbg(CFG_IO_DBG_FREE, fmt, ## __VA_ARGS__)
|
|
#define cfg_io_dbg_name(fmt, ...) cfg_io_dbg(CFG_IO_DBG_NAME, fmt, ## __VA_ARGS__)
|
|
#define cfg_io_dbg_show(fmt, ...) cfg_io_dbg_f(CFG_IO_DBG_SHOW, fmt, ## __VA_ARGS__)
|
|
#define cfg_io_dbg_info(fmt, ...) cfg_io_dbg(CFG_IO_DBG_INFO, fmt, ## __VA_ARGS__)
|
|
|
|
typedef enum MppCfgParserType_e {
|
|
MPP_CFG_PARSER_TYPE_KEY = 0,
|
|
MPP_CFG_PARSER_TYPE_VALUE,
|
|
MPP_CFG_PARSER_TYPE_TABLE,
|
|
MPP_CFG_PARSER_TYPE_ARRAY_TABLE,
|
|
MPP_CFG_PARSER_TYPE_BUTT,
|
|
} MppCfgParserType;
|
|
|
|
typedef struct MppCfgIoImpl_t MppCfgIoImpl;
|
|
typedef void (*MppCfgIoFunc)(MppCfgIoImpl *obj, void *data);
|
|
|
|
struct MppCfgIoImpl_t {
|
|
/* list for bothers */
|
|
struct list_head list;
|
|
/* list for children */
|
|
struct list_head child;
|
|
/* parent of current object */
|
|
MppCfgIoImpl *parent;
|
|
/* valid condition callback for the current object */
|
|
MppCfgObjCond cond;
|
|
|
|
MppCfgType type;
|
|
MppCfgVal val;
|
|
|
|
rk_s32 buf_size;
|
|
/* depth in tree */
|
|
rk_s32 depth;
|
|
|
|
/* internal name storage */
|
|
char *name;
|
|
rk_s32 name_len;
|
|
rk_s32 name_buf_len;
|
|
|
|
/* location info for structure access */
|
|
MppTrie trie;
|
|
MppCfgInfo info;
|
|
|
|
union {
|
|
/* MPP_CFG_TYPE_STRING */
|
|
struct {
|
|
char *string;
|
|
rk_s32 str_len;
|
|
};
|
|
/* MPP_CFG_TYPE_ARRAY */
|
|
struct {
|
|
MppCfgIoImpl **elems;
|
|
rk_s32 array_size;
|
|
};
|
|
};
|
|
};
|
|
|
|
typedef struct MppCfgStrBuf_t {
|
|
char *buf;
|
|
rk_s32 buf_size;
|
|
rk_s32 offset;
|
|
rk_s32 depth;
|
|
MppCfgStrFmt type;
|
|
} MppCfgStrBuf;
|
|
|
|
static rk_u32 mpp_cfg_io_debug = 0;
|
|
|
|
static const char *strof_type(MppCfgType type)
|
|
{
|
|
static const char *str[MPP_CFG_TYPE_BUTT + 1] = {
|
|
[MPP_CFG_TYPE_INVALID] = "invalid",
|
|
[MPP_CFG_TYPE_NULL] = "null",
|
|
[MPP_CFG_TYPE_BOOL] = "bool",
|
|
[MPP_CFG_TYPE_s32] = "s32",
|
|
[MPP_CFG_TYPE_u32] = "u32",
|
|
[MPP_CFG_TYPE_s64] = "s64",
|
|
[MPP_CFG_TYPE_u64] = "u64",
|
|
[MPP_CFG_TYPE_f32] = "f32",
|
|
[MPP_CFG_TYPE_f64] = "f64",
|
|
[MPP_CFG_TYPE_STRING] = "string",
|
|
[MPP_CFG_TYPE_RAW] = "raw",
|
|
[MPP_CFG_TYPE_OBJECT] = "object",
|
|
[MPP_CFG_TYPE_ARRAY] = "array",
|
|
[MPP_CFG_TYPE_BUTT] = "unknown",
|
|
};
|
|
|
|
if (type < 0 || type > MPP_CFG_TYPE_BUTT)
|
|
type = MPP_CFG_TYPE_BUTT;
|
|
|
|
return str[type];
|
|
}
|
|
|
|
static char *dup_str(const char *str, rk_s32 len)
|
|
{
|
|
char *ret = NULL;
|
|
|
|
if (str && len > 0) {
|
|
ret = mpp_calloc_size(char, len + 1);
|
|
if (ret) {
|
|
memcpy(ret, str, len);
|
|
ret[len] = '\0';
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static rk_s32 get_full_name(MppCfgIoImpl *obj, char *buf, rk_s32 buf_size)
|
|
{
|
|
MppCfgIoImpl *curr = obj;
|
|
char *name[MAX_CFG_DEPTH];
|
|
char *delmiter = ":";
|
|
rk_s32 depth = 0;
|
|
rk_s32 len = 0;
|
|
rk_s32 i = 0;
|
|
|
|
while (curr && curr->parent) {
|
|
/* skip the root */
|
|
if (curr->name) {
|
|
/* Add delimiter on object */
|
|
if (curr->type >= MPP_CFG_TYPE_OBJECT)
|
|
name[i++] = delmiter;
|
|
|
|
name[i++] = curr->name;
|
|
}
|
|
|
|
curr = curr->parent;
|
|
depth++;
|
|
|
|
if (i >= MAX_CFG_DEPTH) {
|
|
mpp_loge_f("too deep depth %d\n", depth);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
if (!i) {
|
|
buf[0] = '\0';
|
|
return 0;
|
|
}
|
|
|
|
depth = i;
|
|
for (i = depth - 1; i >= 0; i--) {
|
|
len += snprintf(buf + len, buf_size - len, "%s", name[i]);
|
|
|
|
if (len >= buf_size) {
|
|
mpp_loge_f("buffer overflow len %d buf_size %d\n", len, buf_size);
|
|
break;
|
|
}
|
|
}
|
|
|
|
cfg_io_dbg_name("depth %d obj %-16s -> %s\n", obj->depth, obj->name, buf);
|
|
|
|
return len;
|
|
}
|
|
|
|
void loop_all_children(MppCfgIoImpl *impl, MppCfgIoFunc func, void *data)
|
|
{
|
|
MppCfgIoImpl *pos, *n;
|
|
|
|
func(impl, data);
|
|
|
|
list_for_each_entry_safe(pos, n, &impl->child, MppCfgIoImpl, list) {
|
|
loop_all_children(pos, func, data);
|
|
}
|
|
}
|
|
|
|
rk_s32 mpp_cfg_get_object(MppCfgObj *obj, const char *name, MppCfgType type, MppCfgVal *val)
|
|
{
|
|
MppCfgIoImpl *impl = NULL;
|
|
rk_s32 name_buf_len = 0;
|
|
rk_s32 name_len = 0;
|
|
rk_s32 buf_size = 0;
|
|
rk_s32 str_len = 0;
|
|
|
|
if (!obj || type <= MPP_CFG_TYPE_INVALID || type >= MPP_CFG_TYPE_BUTT) {
|
|
mpp_loge_f("invalid param obj %p name %s type %d val %p\n", obj, name, type, val);
|
|
return rk_nok;
|
|
}
|
|
|
|
mpp_env_get_u32("mpp_cfg_io_debug", &mpp_cfg_io_debug, mpp_cfg_io_debug);
|
|
|
|
if (*obj)
|
|
mpp_logw_f("obj %p overwrite\n", *obj);
|
|
|
|
*obj = NULL;
|
|
|
|
if (name) {
|
|
name_len = strlen(name);
|
|
name_buf_len = MPP_ALIGN(name_len + 1, 4);
|
|
}
|
|
|
|
if (type == MPP_CFG_TYPE_STRING && val && val->str)
|
|
str_len = MPP_ALIGN(strlen(val->str) + 1, 4);
|
|
|
|
buf_size = sizeof(MppCfgIoImpl) + name_buf_len + str_len;
|
|
impl = mpp_calloc_size(MppCfgIoImpl, buf_size);
|
|
|
|
if (!impl) {
|
|
mpp_loge_f("failed to alloc impl size %d\n", buf_size);
|
|
return rk_nok;
|
|
}
|
|
|
|
INIT_LIST_HEAD(&impl->list);
|
|
INIT_LIST_HEAD(&impl->child);
|
|
|
|
if (name_buf_len) {
|
|
impl->name = (char *)(impl + 1);
|
|
memcpy(impl->name, name, name_len);
|
|
impl->name[name_len] = '\0';
|
|
impl->name_len = name_len;
|
|
impl->name_buf_len = name_buf_len;
|
|
}
|
|
|
|
if (str_len) {
|
|
impl->string = (char *)(impl + 1) + name_buf_len;
|
|
strncpy(impl->string, val->str, str_len);
|
|
impl->str_len = str_len;
|
|
}
|
|
|
|
impl->type = type;
|
|
if (val)
|
|
impl->val = *val;
|
|
impl->buf_size = buf_size;
|
|
/* set invalid data type by default */
|
|
impl->info.data_type = CFG_FUNC_TYPE_BUTT;
|
|
|
|
if (type == MPP_CFG_TYPE_STRING)
|
|
impl->val.str = impl->string;
|
|
|
|
*obj = impl;
|
|
|
|
return rk_ok;
|
|
}
|
|
|
|
rk_s32 mpp_cfg_get_array(MppCfgObj *obj, const char *name, rk_s32 count)
|
|
{
|
|
MppCfgIoImpl *impl = NULL;
|
|
rk_s32 name_buf_len = 0;
|
|
rk_s32 name_len = 0;
|
|
rk_s32 buf_size = 0;
|
|
|
|
if (!obj) {
|
|
mpp_loge_f("invalid param obj %p name %s count %d\n", obj, name, count);
|
|
return rk_nok;
|
|
}
|
|
|
|
if (*obj)
|
|
mpp_logw_f("obj %p overwrite\n", *obj);
|
|
|
|
*obj = NULL;
|
|
|
|
if (name) {
|
|
name_len = strlen(name);
|
|
name_buf_len = MPP_ALIGN(name_len + 1, 4);
|
|
}
|
|
|
|
buf_size = sizeof(MppCfgIoImpl) + name_buf_len + count * sizeof(MppCfgObj);
|
|
impl = mpp_calloc_size(MppCfgIoImpl, buf_size);
|
|
|
|
if (!impl) {
|
|
mpp_loge_f("failed to alloc impl size %d\n", buf_size);
|
|
return rk_nok;
|
|
}
|
|
|
|
INIT_LIST_HEAD(&impl->list);
|
|
INIT_LIST_HEAD(&impl->child);
|
|
|
|
if (name_len) {
|
|
impl->name = (char *)(impl + 1);
|
|
memcpy(impl->name, name, name_len);
|
|
impl->name[name_len] = '\0';
|
|
impl->name_len = name_len;
|
|
impl->name_buf_len = name_buf_len;
|
|
}
|
|
|
|
impl->type = MPP_CFG_TYPE_ARRAY;
|
|
impl->buf_size = buf_size;
|
|
/* set invalid data type by default */
|
|
impl->info.data_type = CFG_FUNC_TYPE_BUTT;
|
|
|
|
if (count) {
|
|
impl->elems = (MppCfgIoImpl **)((char *)(impl + 1) + name_buf_len);
|
|
impl->array_size = count;
|
|
}
|
|
|
|
*obj = impl;
|
|
|
|
return rk_ok;
|
|
}
|
|
|
|
rk_s32 mpp_cfg_put(MppCfgObj obj)
|
|
{
|
|
MppCfgIoImpl *impl = (MppCfgIoImpl *)obj;
|
|
|
|
if (!obj) {
|
|
mpp_loge_f("invalid param obj %p\n", obj);
|
|
return rk_nok;
|
|
}
|
|
|
|
list_del_init(&impl->list);
|
|
|
|
{
|
|
MppCfgIoImpl *pos, *n;
|
|
|
|
list_for_each_entry_safe(pos, n, &impl->child, MppCfgIoImpl, list) {
|
|
list_del_init(&pos->list);
|
|
}
|
|
}
|
|
|
|
impl->parent = NULL;
|
|
|
|
mpp_free(impl);
|
|
|
|
return rk_ok;
|
|
}
|
|
|
|
static void mpp_cfg_put_all_child(MppCfgIoImpl *impl)
|
|
{
|
|
MppCfgIoImpl *pos, *n;
|
|
|
|
cfg_io_dbg_free("depth %d - %p free start type %d name %s\n",
|
|
impl->depth, impl, impl->type, impl->name);
|
|
|
|
list_for_each_entry_safe(pos, n, &impl->child, MppCfgIoImpl, list) {
|
|
list_del_init(&pos->list);
|
|
|
|
cfg_io_dbg_free("depth %d - %p free child %p type %d name %s\n",
|
|
impl->depth, impl, pos, pos->type, pos->name);
|
|
|
|
mpp_cfg_put_all_child(pos);
|
|
}
|
|
|
|
cfg_io_dbg_free("depth %d - %p free done type %d name %s\n",
|
|
impl->depth, impl, impl->type, impl->name);
|
|
|
|
mpp_free(impl);
|
|
}
|
|
|
|
rk_s32 mpp_cfg_put_all(MppCfgObj obj)
|
|
{
|
|
MppCfgIoImpl *impl = (MppCfgIoImpl *)obj;
|
|
MppCfgIoImpl *root;
|
|
|
|
if (!obj) {
|
|
mpp_loge_f("invalid param obj %p\n", obj);
|
|
return rk_nok;
|
|
}
|
|
|
|
if (impl->trie) {
|
|
mpp_trie_deinit(impl->trie);
|
|
impl->trie = NULL;
|
|
}
|
|
|
|
root = impl->parent;
|
|
do {
|
|
mpp_cfg_put_all_child(impl);
|
|
|
|
if (!root)
|
|
break;
|
|
|
|
impl = root;
|
|
root = impl->parent;
|
|
} while (impl);
|
|
|
|
return rk_ok;
|
|
}
|
|
|
|
static void update_depth(MppCfgIoImpl *impl, void *data)
|
|
{
|
|
(void)data;
|
|
|
|
if (impl->parent)
|
|
impl->depth = impl->parent->depth + 1;
|
|
}
|
|
|
|
rk_s32 mpp_cfg_add(MppCfgObj root, MppCfgObj leaf)
|
|
{
|
|
MppCfgIoImpl *root_impl = (MppCfgIoImpl *)root;
|
|
MppCfgIoImpl *leaf_impl = (MppCfgIoImpl *)leaf;
|
|
|
|
if (!root || !leaf) {
|
|
mpp_loge_f("invalid param root %p leaf %p\n", root, leaf);
|
|
return rk_nok;
|
|
}
|
|
|
|
if (root_impl->type <= MPP_CFG_TYPE_INVALID || root_impl->type >= MPP_CFG_TYPE_BUTT) {
|
|
mpp_loge_f("invalid root type %d\n", root_impl->type);
|
|
return rk_nok;
|
|
}
|
|
|
|
list_add_tail(&leaf_impl->list, &root_impl->child);
|
|
leaf_impl->parent = root_impl;
|
|
|
|
loop_all_children(root, update_depth, NULL);
|
|
|
|
if (root_impl->type == MPP_CFG_TYPE_ARRAY && root_impl->elems) {
|
|
rk_s32 i;
|
|
|
|
for (i = 0; i < root_impl->array_size; i++) {
|
|
if (!root_impl->elems[i]) {
|
|
root_impl->elems[i] = leaf_impl;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return rk_ok;
|
|
}
|
|
|
|
rk_s32 mpp_cfg_del(MppCfgObj obj)
|
|
{
|
|
MppCfgIoImpl *impl = (MppCfgIoImpl *)obj;
|
|
MppCfgIoImpl *parent;
|
|
|
|
if (!obj) {
|
|
mpp_loge_f("invalid param obj %p\n", obj);
|
|
return rk_nok;
|
|
}
|
|
|
|
parent = impl->parent;
|
|
if (parent) {
|
|
list_del_init(&impl->list);
|
|
|
|
if (parent->type == MPP_CFG_TYPE_ARRAY && parent->elems) {
|
|
rk_s32 i;
|
|
|
|
for (i = 0; i < parent->array_size; i++) {
|
|
if (parent->elems[i] == impl) {
|
|
parent->elems[i] = NULL;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
impl->parent = NULL;
|
|
impl->depth = 0;
|
|
loop_all_children(impl, update_depth, NULL);
|
|
}
|
|
|
|
return rk_ok;
|
|
}
|
|
|
|
rk_s32 mpp_cfg_find(MppCfgObj *obj, MppCfgObj root, char *name, rk_s32 type)
|
|
{
|
|
MppCfgIoImpl *impl = (MppCfgIoImpl *)root;
|
|
rk_s32 str_start = 0;
|
|
rk_s32 str_len = 0;
|
|
rk_s32 i;
|
|
char delimiter;
|
|
|
|
if (!obj || !root || !name) {
|
|
mpp_loge_f("invalid param obj %p root %p name %s\n", obj, root, name);
|
|
return rk_nok;
|
|
}
|
|
|
|
delimiter = (type == MPP_CFG_STR_FMT_TOML) ? '.' : ':';
|
|
str_len = strlen(name);
|
|
|
|
for (i = 0; i <= str_len; i++) {
|
|
if (name[i] == delimiter || name[i] == '\0') {
|
|
MppCfgIoImpl *pos, *n;
|
|
MppCfgIoImpl *last_array = NULL;
|
|
char bak = name[i];
|
|
rk_s32 found = 0;
|
|
|
|
name[i] = '\0';
|
|
mpp_logi("try match %s\n", name + str_start);
|
|
list_for_each_entry_safe(pos, n, &impl->child, MppCfgIoImpl, list) {
|
|
if (pos->name && !strcmp(pos->name, name + str_start)) {
|
|
impl = pos;
|
|
found = 1;
|
|
break;
|
|
}
|
|
|
|
/* if impl is array, find impl->chil is object and has no name, to match its child */
|
|
if (impl->type == MPP_CFG_TYPE_ARRAY && pos->type == MPP_CFG_TYPE_OBJECT && !pos->name)
|
|
last_array = pos;
|
|
}
|
|
|
|
if (last_array) {
|
|
MppCfgIoImpl *array_pos, *array_n;
|
|
|
|
list_for_each_entry_safe(array_pos, array_n, &last_array->child, MppCfgIoImpl, list) {
|
|
if (array_pos->name && !strcmp(array_pos->name, name + str_start)) {
|
|
impl = array_pos;
|
|
found = 1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
name[i] = bak;
|
|
|
|
if (!found) {
|
|
*obj = NULL;
|
|
return rk_nok;
|
|
}
|
|
|
|
str_start = i + 1;
|
|
}
|
|
}
|
|
|
|
*obj = impl;
|
|
return rk_ok;
|
|
}
|
|
|
|
rk_s32 mpp_cfg_set_info(MppCfgObj obj, MppCfgInfo *info)
|
|
{
|
|
MppCfgIoImpl *impl = (MppCfgIoImpl *)obj;
|
|
|
|
if (impl && info) {
|
|
cfg_io_dbg_info("obj %-16s set info type %s offset %d size %d\n",
|
|
impl->name, strof_cfg_type(info->data_type),
|
|
info->data_offset, info->data_size);
|
|
|
|
if (info->data_type < CFG_FUNC_TYPE_BUTT) {
|
|
memcpy(&impl->info, info, sizeof(impl->info));
|
|
|
|
switch (info->data_type) {
|
|
case CFG_FUNC_TYPE_s32 : {
|
|
impl->type = MPP_CFG_TYPE_s32;
|
|
} break;
|
|
case CFG_FUNC_TYPE_u32 : {
|
|
impl->type = MPP_CFG_TYPE_u32;
|
|
} break;
|
|
case CFG_FUNC_TYPE_s64 : {
|
|
impl->type = MPP_CFG_TYPE_s64;
|
|
} break;
|
|
case CFG_FUNC_TYPE_u64 : {
|
|
impl->type = MPP_CFG_TYPE_u64;
|
|
} break;
|
|
default : {
|
|
} break;
|
|
}
|
|
} else {
|
|
impl->info.data_type = CFG_FUNC_TYPE_BUTT;
|
|
}
|
|
|
|
return rk_ok;
|
|
}
|
|
|
|
return rk_nok;
|
|
}
|
|
|
|
rk_s32 mpp_cfg_set_cond(MppCfgObj obj, MppCfgObjCond cond)
|
|
{
|
|
MppCfgIoImpl *impl = (MppCfgIoImpl *)obj;
|
|
|
|
if (impl)
|
|
impl->cond = cond;
|
|
|
|
return rk_ok;
|
|
}
|
|
typedef struct MppCfgFullNameCtx_t {
|
|
MppTrie trie;
|
|
char *buf;
|
|
rk_s32 buf_size;
|
|
} MppCfgFullNameCtx;
|
|
|
|
static void add_obj_info(MppCfgIoImpl *impl, void *data)
|
|
{
|
|
/* NOTE: skip the root object and the invalid object */
|
|
if (impl->info.data_type < CFG_FUNC_TYPE_BUTT && impl->parent) {
|
|
MppCfgFullNameCtx *ctx = (MppCfgFullNameCtx *)data;
|
|
|
|
get_full_name(impl, ctx->buf, ctx->buf_size);
|
|
mpp_trie_add_info(ctx->trie, ctx->buf, &impl->info, sizeof(impl->info));
|
|
}
|
|
}
|
|
|
|
MppTrie mpp_cfg_to_trie(MppCfgObj obj)
|
|
{
|
|
MppCfgIoImpl *impl = (MppCfgIoImpl *)obj;
|
|
MppTrie p = NULL;
|
|
|
|
do {
|
|
MppCfgFullNameCtx ctx;
|
|
rk_s32 ret = rk_nok;
|
|
char name[256];
|
|
|
|
if (!impl) {
|
|
mpp_loge_f("invalid param obj\n", impl);
|
|
break;
|
|
}
|
|
|
|
if (impl->parent) {
|
|
mpp_loge_f("invalid param obj %p not root\n", impl);
|
|
break;
|
|
}
|
|
|
|
if (impl->trie) {
|
|
p = impl->trie;
|
|
break;
|
|
}
|
|
|
|
ret = mpp_trie_init(&p, impl->name ? impl->name : "cfg_io");
|
|
if (ret || !p) {
|
|
mpp_loge_f("failed to init obj %s trie\n", impl->name ? impl->name : "cfg_io");
|
|
break;
|
|
}
|
|
|
|
ctx.trie = p;
|
|
ctx.buf = name;
|
|
ctx.buf_size = sizeof(name) - 1;
|
|
|
|
loop_all_children(impl, add_obj_info, &ctx);
|
|
mpp_trie_add_info(p, NULL, NULL, 0);
|
|
impl->trie = p;
|
|
} while (0);
|
|
|
|
return p;
|
|
}
|
|
|
|
/* read byte functions */
|
|
/* check valid len, get offset position */
|
|
#define test_byte_f(str, len) test_byte(str, len, __FUNCTION__)
|
|
/* check valid pos, get offset + pos position */
|
|
#define show_byte_f(str, pos) show_byte(str, pos, __FUNCTION__)
|
|
/* check valid len, get offset + len position and increase offset by len */
|
|
#define skip_byte_f(str, len) skip_byte(str, len, __FUNCTION__)
|
|
#define skip_ws_f(str) skip_ws(str, __FUNCTION__)
|
|
|
|
/* write byte functions */
|
|
#define write_byte_f(str, buf, size) write_byte(str, (void *)buf, size, __FUNCTION__)
|
|
#define write_indent_f(str) write_indent(str, __FUNCTION__)
|
|
/* revert comma for json */
|
|
#define revert_comma_f(str) revert_comma(str, __FUNCTION__)
|
|
|
|
static char *test_byte(MppCfgStrBuf *str, rk_s32 len, const char *caller)
|
|
{
|
|
char *ret = NULL;
|
|
|
|
if (str->offset + len >= str->buf_size) {
|
|
cfg_io_dbg_byte("str %p-[%p:%d] offset %d test %d get the end at %s\n",
|
|
str, str->buf, str->buf_size, str->offset, len, caller);
|
|
return ret;
|
|
}
|
|
|
|
ret = str->buf + str->offset;
|
|
|
|
cfg_io_dbg_byte("str %p-[%p:%d] offset %d test %d ret %p at %s\n",
|
|
str, str->buf, str->buf_size, str->offset, len, ret, caller);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static char *show_byte(MppCfgStrBuf *str, rk_s32 pos, const char *caller)
|
|
{
|
|
char *ret = NULL;
|
|
|
|
if (str->offset + pos >= str->buf_size) {
|
|
cfg_io_dbg_byte("str %p-[%p:%d] offset %d show pos %d get the end at %s\n",
|
|
str, str->buf, str->buf_size, str->offset, pos, caller);
|
|
return ret;
|
|
}
|
|
|
|
ret = str->buf + str->offset + pos;
|
|
|
|
cfg_io_dbg_byte("str %p-[%p:%d] offset %d show %d ret %p at %s\n",
|
|
str, str->buf, str->buf_size, str->offset, pos, ret, caller);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static char *skip_byte(MppCfgStrBuf *str, rk_s32 len, const char *caller)
|
|
{
|
|
char *ret = NULL;
|
|
|
|
if (str->offset + len >= str->buf_size) {
|
|
cfg_io_dbg_byte("str %p-[%p:%d] offset %d skip %d get the end at %s\n",
|
|
str, str->buf, str->buf_size, str->offset, len, caller);
|
|
return NULL;
|
|
}
|
|
|
|
ret = str->buf + str->offset + len;
|
|
|
|
cfg_io_dbg_byte("str %p-[%p:%d] offset %d skip %d ret %p at %s\n",
|
|
str, str->buf, str->buf_size, str->offset, len, ret, caller);
|
|
|
|
str->offset += len;
|
|
return ret;
|
|
}
|
|
|
|
static char *skip_ws(MppCfgStrBuf *str, const char *caller)
|
|
{
|
|
rk_s32 old = str->offset;
|
|
char *p;
|
|
|
|
cfg_io_dbg_byte("str %p-[%p:%d] offset %d skip ws start at %s\n",
|
|
str, str->buf, str->buf_size, old, caller);
|
|
|
|
while ((p = show_byte(str, 0, caller)) && p[0] <= 32)
|
|
str->offset++;
|
|
|
|
if (str->offset >= str->buf_size) {
|
|
cfg_io_dbg_byte("str %p-[%p:%d] offset %d skip ws to the end at %s\n",
|
|
str, str->buf, str->buf_size, str->offset, caller);
|
|
str->offset--;
|
|
return NULL;
|
|
}
|
|
|
|
cfg_io_dbg_byte("str %p-[%p:%d] offset %d skip ws to %d at %s\n",
|
|
str, str->buf, str->buf_size, old, str->offset, caller);
|
|
|
|
return str->buf + str->offset;
|
|
}
|
|
|
|
static rk_s32 write_byte(MppCfgStrBuf *str, void *buf, rk_s32 *size, const char *caller)
|
|
{
|
|
rk_s32 len = size[0];
|
|
|
|
if (!len)
|
|
return rk_ok;
|
|
|
|
if (str->offset + len >= str->buf_size) {
|
|
void *ptr = mpp_realloc_size(str->buf, void, str->buf_size * 2);
|
|
|
|
if (!ptr) {
|
|
mpp_loge("failed to realloc buf size %d -> %d at %s\n",
|
|
str->buf_size, str->buf_size * 2, caller);
|
|
return rk_nok;
|
|
}
|
|
|
|
cfg_io_dbg_byte("str %p-[%p:%d] enlarger buffer to [%p:%d] at %s\n",
|
|
str, str->buf, str->buf_size, ptr, str->buf_size * 2, caller);
|
|
|
|
str->buf = ptr;
|
|
str->buf_size *= 2;
|
|
}
|
|
|
|
cfg_io_dbg_byte("str %p-[%p:%d] write offset %d from [%p:%d] at %s\n",
|
|
str, str->buf, str->buf_size, str->offset, buf, len, caller);
|
|
|
|
memcpy(str->buf + str->offset, buf, len);
|
|
str->offset += len;
|
|
str->buf[str->offset] = '\0';
|
|
size[0] = 0;
|
|
|
|
return rk_ok;
|
|
}
|
|
|
|
static rk_s32 write_indent(MppCfgStrBuf *str, const char *caller)
|
|
{
|
|
rk_s32 depth;
|
|
|
|
cfg_io_dbg_byte("str %p-[%p:%d] write indent %d at %s\n",
|
|
str, str->buf, str->buf_size, str->depth, caller);
|
|
|
|
depth = str->depth;
|
|
if (str->type == MPP_CFG_STR_FMT_TOML) {
|
|
depth = depth - 1;
|
|
depth = depth >= 0 ? depth : 0;
|
|
}
|
|
|
|
if (depth) {
|
|
char space[17] = " ";
|
|
rk_s32 i;
|
|
|
|
for (i = 0; i < depth; i++) {
|
|
rk_s32 indent_width = 4;
|
|
|
|
if (write_byte_f(str, space, &indent_width))
|
|
return rk_nok;
|
|
}
|
|
}
|
|
|
|
return rk_ok;
|
|
}
|
|
|
|
static rk_s32 revert_comma(MppCfgStrBuf *str, const char *caller)
|
|
{
|
|
cfg_io_dbg_byte("str %p-[%p:%d] revert_comma %d at %s\n",
|
|
str, str->buf, str->buf_size, str->depth, caller);
|
|
|
|
if (str->offset <= 1) {
|
|
cfg_io_dbg_byte("str %p offset %d skip revert_comma at %s\n",
|
|
str, str->offset, caller);
|
|
return rk_ok;
|
|
}
|
|
|
|
if (str->buf[str->offset - 2] == ',') {
|
|
str->buf[str->offset - 2] = str->buf[str->offset - 1];
|
|
str->buf[str->offset - 1] = str->buf[str->offset];
|
|
str->offset--;
|
|
}
|
|
|
|
return rk_ok;
|
|
}
|
|
|
|
static rk_s32 mpp_cfg_to_log(MppCfgIoImpl *impl, MppCfgStrBuf *str)
|
|
{
|
|
MppCfgIoImpl *pos, *n;
|
|
char buf[256];
|
|
rk_s32 len = 0;
|
|
rk_s32 total = sizeof(buf) - 1;
|
|
rk_s32 ret = rk_ok;
|
|
|
|
write_indent_f(str);
|
|
|
|
/* leaf node write once and finish */
|
|
if (impl->type < MPP_CFG_TYPE_OBJECT) {
|
|
cfg_io_dbg_to("depth %d leaf write name %s type %d\n", str->depth, impl->name, impl->type);
|
|
|
|
if (impl->name)
|
|
len += snprintf(buf + len, total - len, "%s : ", impl->name);
|
|
|
|
switch (impl->type) {
|
|
case MPP_CFG_TYPE_NULL : {
|
|
len += snprintf(buf + len, total - len, "null\n");
|
|
} break;
|
|
case MPP_CFG_TYPE_BOOL : {
|
|
len += snprintf(buf + len, total - len, "%s\n", impl->val.b1 ? "true" : "false");
|
|
} break;
|
|
case MPP_CFG_TYPE_s32 : {
|
|
len += snprintf(buf + len, total - len, "%d\n", impl->val.s32);
|
|
} break;
|
|
case MPP_CFG_TYPE_u32 : {
|
|
len += snprintf(buf + len, total - len, "%u\n", impl->val.u32);
|
|
} break;
|
|
case MPP_CFG_TYPE_s64 : {
|
|
len += snprintf(buf + len, total - len, "%lld\n", impl->val.s64);
|
|
} break;
|
|
case MPP_CFG_TYPE_u64 : {
|
|
len += snprintf(buf + len, total - len, "%llu\n", impl->val.u64);
|
|
} break;
|
|
case MPP_CFG_TYPE_f32 : {
|
|
len += snprintf(buf + len, total - len, "%f\n", impl->val.f32);
|
|
} break;
|
|
case MPP_CFG_TYPE_f64 : {
|
|
len += snprintf(buf + len, total - len, "%lf\n", impl->val.f64);
|
|
} break;
|
|
case MPP_CFG_TYPE_STRING :
|
|
case MPP_CFG_TYPE_RAW : {
|
|
len += snprintf(buf + len, total - len, "\"%s\"\n", (char *)impl->val.str);
|
|
} break;
|
|
default : {
|
|
mpp_loge("invalid type %d\n", impl->type);
|
|
} break;
|
|
}
|
|
|
|
return write_byte_f(str, buf, &len);
|
|
}
|
|
|
|
cfg_io_dbg_to("depth %d branch write name %s type %d\n", str->depth, impl->name, impl->type);
|
|
|
|
if (impl->name)
|
|
len += snprintf(buf + len, total - len, "%s : ", impl->name);
|
|
|
|
if (list_empty(&impl->child)) {
|
|
len += snprintf(buf + len, total - len, "%s\n",
|
|
impl->type == MPP_CFG_TYPE_OBJECT ? "{}" : "[]");
|
|
return write_byte_f(str, buf, &len);
|
|
}
|
|
|
|
len += snprintf(buf + len, total - len, "%c\n",
|
|
impl->type == MPP_CFG_TYPE_OBJECT ? '{' : '[');
|
|
|
|
ret = write_byte_f(str, buf, &len);
|
|
if (ret)
|
|
return ret;
|
|
|
|
str->depth++;
|
|
|
|
list_for_each_entry_safe(pos, n, &impl->child, MppCfgIoImpl, list) {
|
|
cfg_io_dbg_to("depth %d child write name %s type %d\n", str->depth, pos->name, pos->type);
|
|
ret = mpp_cfg_to_log(pos, str);
|
|
if (ret)
|
|
break;
|
|
}
|
|
|
|
str->depth--;
|
|
|
|
write_indent_f(str);
|
|
|
|
len += snprintf(buf + len, total - len, "%c\n",
|
|
impl->type == MPP_CFG_TYPE_OBJECT ? '}' : ']');
|
|
|
|
return write_byte_f(str, buf, &len);
|
|
}
|
|
|
|
static rk_s32 mpp_cfg_to_json(MppCfgIoImpl *impl, MppCfgStrBuf *str)
|
|
{
|
|
MppCfgIoImpl *pos, *n;
|
|
char buf[256];
|
|
rk_s32 len = 0;
|
|
rk_s32 total = sizeof(buf) - 1;
|
|
rk_s32 ret = rk_ok;
|
|
|
|
write_indent_f(str);
|
|
|
|
/* leaf node write once and finish */
|
|
if (impl->type < MPP_CFG_TYPE_OBJECT) {
|
|
cfg_io_dbg_to("depth %d leaf write name %s type %d\n", str->depth, impl->name, impl->type);
|
|
|
|
if (impl->name)
|
|
len += snprintf(buf + len, total - len, "\"%s\" : ", impl->name);
|
|
|
|
switch (impl->type) {
|
|
case MPP_CFG_TYPE_NULL : {
|
|
len += snprintf(buf + len, total - len, "null,\n");
|
|
} break;
|
|
case MPP_CFG_TYPE_BOOL : {
|
|
len += snprintf(buf + len, total - len, "%s,\n", impl->val.b1 ? "true" : "false");
|
|
} break;
|
|
case MPP_CFG_TYPE_s32 : {
|
|
len += snprintf(buf + len, total - len, "%d,\n", impl->val.s32);
|
|
} break;
|
|
case MPP_CFG_TYPE_u32 : {
|
|
len += snprintf(buf + len, total - len, "%u,\n", impl->val.u32);
|
|
} break;
|
|
case MPP_CFG_TYPE_s64 : {
|
|
len += snprintf(buf + len, total - len, "%lld,\n", impl->val.s64);
|
|
} break;
|
|
case MPP_CFG_TYPE_u64 : {
|
|
len += snprintf(buf + len, total - len, "%llu,\n", impl->val.u64);
|
|
} break;
|
|
case MPP_CFG_TYPE_f32 : {
|
|
len += snprintf(buf + len, total - len, "%f,\n", impl->val.f32);
|
|
} break;
|
|
case MPP_CFG_TYPE_f64 : {
|
|
len += snprintf(buf + len, total - len, "%lf,\n", impl->val.f64);
|
|
} break;
|
|
case MPP_CFG_TYPE_STRING :
|
|
case MPP_CFG_TYPE_RAW : {
|
|
len += snprintf(buf + len, total - len, "\"%s\",\n", (char *)impl->val.str);
|
|
} break;
|
|
default : {
|
|
mpp_loge("invalid type %d\n", impl->type);
|
|
} break;
|
|
}
|
|
|
|
return write_byte_f(str, buf, &len);
|
|
}
|
|
|
|
cfg_io_dbg_to("depth %d branch write name %s type %d\n", str->depth, impl->name, impl->type);
|
|
|
|
if (impl->name)
|
|
len += snprintf(buf + len, total - len, "\"%s\" : ", impl->name);
|
|
|
|
if (list_empty(&impl->child)) {
|
|
len += snprintf(buf + len, total - len, "%s,\n",
|
|
impl->type == MPP_CFG_TYPE_OBJECT ? "{}" : "[]");
|
|
return write_byte_f(str, buf, &len);
|
|
}
|
|
|
|
len += snprintf(buf + len, total - len, "%c\n",
|
|
impl->type == MPP_CFG_TYPE_OBJECT ? '{' : '[');
|
|
|
|
ret = write_byte_f(str, buf, &len);
|
|
if (ret)
|
|
return ret;
|
|
|
|
str->depth++;
|
|
|
|
list_for_each_entry_safe(pos, n, &impl->child, MppCfgIoImpl, list) {
|
|
cfg_io_dbg_to("depth %d child write name %s type %d\n", str->depth, pos->name, pos->type);
|
|
ret = mpp_cfg_to_json(pos, str);
|
|
if (ret)
|
|
break;
|
|
}
|
|
|
|
revert_comma_f(str);
|
|
|
|
str->depth--;
|
|
|
|
write_indent_f(str);
|
|
|
|
if (str->depth)
|
|
len += snprintf(buf + len, total - len, "%c,\n",
|
|
impl->type == MPP_CFG_TYPE_OBJECT ? '}' : ']');
|
|
else
|
|
len += snprintf(buf + len, total - len, "%c\n",
|
|
impl->type == MPP_CFG_TYPE_OBJECT ? '}' : ']');
|
|
|
|
return write_byte_f(str, buf, &len);
|
|
}
|
|
|
|
static rk_s32 mpp_toml_parent_is_array_table(MppCfgIoImpl *impl, MppCfgStrBuf *str)
|
|
{
|
|
return str->depth == 1 && impl->type == MPP_CFG_TYPE_OBJECT &&
|
|
!impl->name && impl->parent->type == MPP_CFG_TYPE_ARRAY;
|
|
}
|
|
|
|
static rk_s32 mpp_toml_top(MppCfgIoImpl *impl, MppCfgStrBuf *str)
|
|
{
|
|
char buf[256];
|
|
rk_s32 len = 0;
|
|
rk_s32 total = sizeof(buf) - 1;
|
|
|
|
if (impl->name && impl->type == MPP_CFG_TYPE_OBJECT)
|
|
len += snprintf(buf + len, total - len, "\n[%s]\n", impl->name);
|
|
|
|
return write_byte_f(str, buf, &len);
|
|
}
|
|
|
|
static rk_s32 mpp_toml_non_top(MppCfgIoImpl *impl, MppCfgStrBuf *str)
|
|
{
|
|
char buf[256];
|
|
rk_s32 len = 0;
|
|
rk_s32 total = sizeof(buf) - 1;
|
|
|
|
if (impl->name)
|
|
len += snprintf(buf + len, total - len, "%s = ", impl->name);
|
|
|
|
if (list_empty(&impl->child)) {
|
|
len += snprintf(buf + len, total - len, "%s\n",
|
|
impl->type == MPP_CFG_TYPE_OBJECT ? "{}" : "[]");
|
|
return write_byte_f(str, buf, &len);
|
|
}
|
|
|
|
if (mpp_toml_parent_is_array_table(impl, str))
|
|
len += snprintf(buf + len, total - len, "\n[[%s]]\n", impl->parent->name);
|
|
else
|
|
len += snprintf(buf + len, total - len, "%c\n",
|
|
impl->type == MPP_CFG_TYPE_OBJECT ? '{' : '[');
|
|
|
|
return write_byte_f(str, buf, &len);
|
|
}
|
|
|
|
static rk_s32 mpp_cfg_to_toml(MppCfgIoImpl *impl, MppCfgStrBuf *str, rk_s32 first_time)
|
|
{
|
|
MppCfgIoImpl *pos, *n;
|
|
char buf[256];
|
|
rk_s32 len = 0;
|
|
rk_s32 total = sizeof(buf) - 1;
|
|
rk_s32 ret = rk_ok;
|
|
|
|
write_indent_f(str);
|
|
|
|
/* leaf node write once and finish */
|
|
if (impl->type < MPP_CFG_TYPE_OBJECT) {
|
|
cfg_io_dbg_to("depth %d leaf write name %s type %d\n", str->depth, impl->name, impl->type);
|
|
|
|
if (impl->name)
|
|
len += snprintf(buf + len, total - len, "%s = ", impl->name);
|
|
|
|
switch (impl->type) {
|
|
case MPP_CFG_TYPE_NULL : {
|
|
len += snprintf(buf + len, total - len, "null");
|
|
} break;
|
|
case MPP_CFG_TYPE_BOOL : {
|
|
len += snprintf(buf + len, total - len, "%s", impl->val.b1 ? "true" : "false");
|
|
} break;
|
|
case MPP_CFG_TYPE_s32 : {
|
|
len += snprintf(buf + len, total - len, "%d", impl->val.s32);
|
|
} break;
|
|
case MPP_CFG_TYPE_u32 : {
|
|
len += snprintf(buf + len, total - len, "%u", impl->val.u32);
|
|
} break;
|
|
case MPP_CFG_TYPE_s64 : {
|
|
len += snprintf(buf + len, total - len, "%lld", impl->val.s64);
|
|
} break;
|
|
case MPP_CFG_TYPE_u64 : {
|
|
len += snprintf(buf + len, total - len, "%llu", impl->val.u64);
|
|
} break;
|
|
case MPP_CFG_TYPE_f32 : {
|
|
len += snprintf(buf + len, total - len, "%f", impl->val.f32);
|
|
} break;
|
|
case MPP_CFG_TYPE_f64 : {
|
|
len += snprintf(buf + len, total - len, "%lf", impl->val.f64);
|
|
} break;
|
|
case MPP_CFG_TYPE_STRING :
|
|
case MPP_CFG_TYPE_RAW : {
|
|
len += snprintf(buf + len, total - len, "\"%s\"", (char *)impl->val.str);
|
|
} break;
|
|
default : {
|
|
mpp_loge("invalid type %d\n", impl->type);
|
|
} break;
|
|
}
|
|
|
|
if (str->depth > 1)
|
|
len += snprintf(buf + len, total - len, ",\n");
|
|
else
|
|
len += snprintf(buf + len, total - len, "\n");
|
|
|
|
return write_byte_f(str, buf, &len);
|
|
}
|
|
|
|
cfg_io_dbg_to("depth %d branch write name %s type %d\n", str->depth, impl->name, impl->type);
|
|
|
|
if (str->depth == 0) {
|
|
ret = mpp_toml_top(impl, str);
|
|
} else {
|
|
ret = mpp_toml_non_top(impl, str);
|
|
}
|
|
if (ret)
|
|
return ret;
|
|
|
|
if (list_empty(&impl->child))
|
|
return rk_ok;
|
|
|
|
if (!mpp_toml_parent_is_array_table(impl, str) && !first_time)
|
|
str->depth++;
|
|
|
|
list_for_each_entry_safe(pos, n, &impl->child, MppCfgIoImpl, list) {
|
|
cfg_io_dbg_to("depth %d child write name %s type %d\n", str->depth, pos->name, pos->type);
|
|
ret = mpp_cfg_to_toml(pos, str, 0);
|
|
if (ret)
|
|
break;
|
|
}
|
|
|
|
if (str->depth > 1)
|
|
revert_comma_f(str);
|
|
|
|
if (!mpp_toml_parent_is_array_table(impl, str) && !first_time)
|
|
str->depth--;
|
|
|
|
write_indent_f(str);
|
|
|
|
if (str->depth > 0 && !mpp_toml_parent_is_array_table(impl, str)) {
|
|
if (str->depth == 1)
|
|
len += snprintf(buf + len, total - len, "%c\n",
|
|
impl->type == MPP_CFG_TYPE_OBJECT ? '}' : ']');
|
|
else
|
|
len += snprintf(buf + len, total - len, "%c,\n",
|
|
impl->type == MPP_CFG_TYPE_OBJECT ? '}' : ']');
|
|
}
|
|
|
|
return write_byte_f(str, buf, &len);
|
|
}
|
|
|
|
static rk_s32 parse_number(MppCfgStrBuf *str, MppCfgType *type, MppCfgVal *val)
|
|
{
|
|
char *buf = NULL;
|
|
char tmp[64];
|
|
long double value;
|
|
rk_u32 i;
|
|
|
|
for (i = 0; i < sizeof(tmp) - 1; i++) {
|
|
buf = show_byte_f(str, 0);
|
|
if (!buf)
|
|
break;
|
|
|
|
switch (buf[0]) {
|
|
case '0' ... '9' :
|
|
case '.' :
|
|
case 'e' :
|
|
case 'E' :
|
|
case '+' :
|
|
case '-' : {
|
|
tmp[i] = buf[0];
|
|
} break;
|
|
default : {
|
|
tmp[i] = '\0';
|
|
goto done;
|
|
} break;
|
|
}
|
|
skip_byte_f(str, 1);
|
|
}
|
|
|
|
done:
|
|
if (!i)
|
|
return rk_nok;
|
|
|
|
errno = 0;
|
|
value = strtold(tmp, NULL);
|
|
if (errno) {
|
|
mpp_loge_f("failed to parse number %s errno %s\n", tmp, strerror(errno));
|
|
return rk_nok;
|
|
}
|
|
|
|
if (strstr(tmp, ".")) {
|
|
if (value >= FLT_MIN && value <= FLT_MAX) {
|
|
*type = MPP_CFG_TYPE_f32;
|
|
val->f32 = (float)value;
|
|
} else {
|
|
*type = MPP_CFG_TYPE_f64;
|
|
val->f64 = (double)value;
|
|
}
|
|
} else {
|
|
if (value >= INT_MIN && value <= INT_MAX) {
|
|
*type = MPP_CFG_TYPE_s32;
|
|
val->s32 = (int)value;
|
|
} else if (value >= 0 && value <= UINT_MAX) {
|
|
*type = MPP_CFG_TYPE_u32;
|
|
val->u32 = (unsigned int)value;
|
|
} else if (value >= (long double)LLONG_MIN && value <= (long double)LLONG_MAX) {
|
|
*type = MPP_CFG_TYPE_u64;
|
|
val->u64 = (unsigned long long)value;
|
|
} else if (value >= 0 && value <= (long double)ULLONG_MAX) {
|
|
*type = MPP_CFG_TYPE_s64;
|
|
val->s64 = (long long)value;
|
|
} else {
|
|
mpp_loge_f("invalid number %s\n", tmp);
|
|
return rk_nok;
|
|
}
|
|
}
|
|
|
|
return rk_ok;
|
|
}
|
|
|
|
static rk_s32 parse_log_string(MppCfgStrBuf *str, char **name, rk_s32 *len, rk_u32 type)
|
|
{
|
|
char *buf = NULL;
|
|
char *start = NULL;
|
|
rk_s32 name_len = 0;
|
|
char terminator = type ? '\"' : ' ';
|
|
|
|
*name = NULL;
|
|
*len = 0;
|
|
|
|
/* skip whitespace and find first double quotes */
|
|
buf = skip_ws_f(str);
|
|
if (!buf)
|
|
return -101;
|
|
|
|
if (type) {
|
|
if (buf[0] != '\"')
|
|
return -101;
|
|
|
|
buf = skip_byte_f(str, 1);
|
|
if (!buf)
|
|
return -102;
|
|
}
|
|
|
|
start = buf;
|
|
|
|
/* find the terminator */
|
|
while ((buf = show_byte_f(str, name_len)) && buf[0] != terminator) {
|
|
name_len++;
|
|
}
|
|
|
|
if (!buf || buf[0] != terminator)
|
|
return -103;
|
|
|
|
/* find complete string skip the string and terminator */
|
|
buf = skip_byte_f(str, name_len + 1);
|
|
if (!buf)
|
|
return -104;
|
|
|
|
*name = start;
|
|
*len = name_len;
|
|
|
|
return rk_ok;
|
|
}
|
|
|
|
static rk_s32 parse_log_value(MppCfgIoImpl *parent, const char *name, MppCfgStrBuf *str);
|
|
|
|
static rk_s32 parse_log_array(MppCfgIoImpl *obj, MppCfgStrBuf *str)
|
|
{
|
|
MppCfgIoImpl *parent = obj;
|
|
char *buf = NULL;
|
|
rk_s32 old = str->offset;
|
|
rk_s32 ret = rk_nok;
|
|
|
|
if (str->depth >= MAX_CFG_DEPTH) {
|
|
mpp_loge_f("depth %d reached max\n", MAX_CFG_DEPTH);
|
|
return rk_nok;
|
|
}
|
|
|
|
str->depth++;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d array parse start\n", str->depth, str->offset);
|
|
|
|
buf = test_byte_f(str, 0);
|
|
if (!buf || buf[0] != '[') {
|
|
ret = -2;
|
|
goto failed;
|
|
}
|
|
|
|
buf = skip_byte_f(str, 1);
|
|
if (!buf) {
|
|
ret = -3;
|
|
goto failed;
|
|
}
|
|
|
|
/* skip whitespace and check the end of buffer */
|
|
buf = skip_ws_f(str);
|
|
if (!buf) {
|
|
ret = -4;
|
|
goto failed;
|
|
}
|
|
|
|
/* check empty object */
|
|
if (buf[0] == ']') {
|
|
skip_byte_f(str, 1);
|
|
cfg_io_dbg_from("depth %d found empty array\n", str->depth);
|
|
str->depth--;
|
|
return rk_ok;
|
|
}
|
|
|
|
do {
|
|
/* find colon for separater */
|
|
buf = skip_ws_f(str);
|
|
if (!buf) {
|
|
ret = -5;
|
|
goto failed;
|
|
}
|
|
|
|
/* parse value */
|
|
ret = parse_log_value(parent, NULL, str);
|
|
if (ret) {
|
|
ret = -6;
|
|
goto failed;
|
|
}
|
|
|
|
buf = skip_ws_f(str);
|
|
if (!buf) {
|
|
ret = -7;
|
|
goto failed;
|
|
}
|
|
|
|
if (buf[0] == ']')
|
|
break;
|
|
} while (1);
|
|
|
|
if (!buf || buf[0] != ']') {
|
|
ret = -9;
|
|
goto failed;
|
|
}
|
|
|
|
skip_byte_f(str, 1);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d -> %d array parse success\n",
|
|
str->depth, old, str->offset);
|
|
|
|
str->depth--;
|
|
ret = rk_ok;
|
|
|
|
failed:
|
|
if (ret)
|
|
cfg_io_dbg_from("depth %d offset %d -> %d array parse failed ret %d\n",
|
|
str->depth, old, str->offset, ret);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static rk_s32 parse_log_object(MppCfgIoImpl *obj, MppCfgStrBuf *str);
|
|
|
|
static rk_s32 parse_log_value(MppCfgIoImpl *parent, const char *name, MppCfgStrBuf *str)
|
|
{
|
|
MppCfgObj obj = NULL;
|
|
char *buf = NULL;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: parse value\n", str->depth, str->offset);
|
|
|
|
buf = test_byte_f(str, 4);
|
|
if (buf && !strncmp(buf, "null", 4)) {
|
|
mpp_cfg_get_object(&obj, name, MPP_CFG_TYPE_NULL, NULL);
|
|
mpp_cfg_add(parent, obj);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value null\n", str->depth, str->offset);
|
|
skip_byte_f(str, 4);
|
|
return rk_ok;
|
|
}
|
|
|
|
if (buf && !strncmp(buf, "true", 4)) {
|
|
MppCfgVal val;
|
|
|
|
val.b1 = 1;
|
|
mpp_cfg_get_object(&obj, name, MPP_CFG_TYPE_BOOL, &val);
|
|
mpp_cfg_add(parent, obj);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value true\n", str->depth, str->offset);
|
|
skip_byte_f(str, 4);
|
|
return rk_ok;
|
|
}
|
|
|
|
buf = test_byte_f(str, 5);
|
|
if (buf && !strncmp(buf, "false", 5)) {
|
|
MppCfgVal val;
|
|
|
|
val.b1 = 0;
|
|
mpp_cfg_get_object(&obj, name, MPP_CFG_TYPE_BOOL, &val);
|
|
mpp_cfg_add(parent, obj);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value false\n", str->depth, str->offset);
|
|
skip_byte_f(str, 5);
|
|
return rk_ok;
|
|
}
|
|
|
|
buf = test_byte_f(str, 0);
|
|
if (buf && buf[0] == '\"') {
|
|
MppCfgVal val;
|
|
char *string = NULL;
|
|
rk_s32 len = 0;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value string start\n", str->depth, str->offset);
|
|
|
|
parse_log_string(str, &string, &len, MPP_CFG_PARSER_TYPE_VALUE);
|
|
if (!string)
|
|
return rk_nok;
|
|
|
|
val.str = dup_str(string, len);
|
|
mpp_cfg_get_object(&obj, name, MPP_CFG_TYPE_STRING, &val);
|
|
mpp_cfg_add(parent, obj);
|
|
MPP_FREE(val.str);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value string success\n", str->depth, str->offset);
|
|
return rk_ok;
|
|
}
|
|
|
|
if (buf && (buf[0] == '-' || (buf[0] >= '0' && buf[0] <= '9'))) {
|
|
MppCfgType type;
|
|
MppCfgVal val;
|
|
rk_s32 ret;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value number start\n",
|
|
str->depth, str->offset);
|
|
|
|
ret = parse_number(str, &type, &val);
|
|
if (ret)
|
|
return ret;
|
|
|
|
mpp_cfg_get_object(&obj, name, type, &val);
|
|
mpp_cfg_add(parent, obj);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value number success\n",
|
|
str->depth, str->offset);
|
|
return ret;
|
|
}
|
|
|
|
if (buf && buf[0] == '{') {
|
|
rk_s32 ret;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value object start\n",
|
|
str->depth, str->offset);
|
|
|
|
mpp_cfg_get_object(&obj, name, MPP_CFG_TYPE_OBJECT, NULL);
|
|
mpp_cfg_add(parent, obj);
|
|
|
|
ret = parse_log_object(obj, str);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value object ret %d\n",
|
|
str->depth, str->offset, ret);
|
|
return ret;
|
|
}
|
|
|
|
if (buf && buf[0] == '[') {
|
|
rk_s32 ret;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value array start\n",
|
|
str->depth, str->offset);
|
|
|
|
mpp_cfg_get_array(&obj, name, 0);
|
|
mpp_cfg_add(parent, obj);
|
|
|
|
ret = parse_log_array(obj, str);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value array ret %d\n",
|
|
str->depth, str->offset, ret);
|
|
return ret;
|
|
}
|
|
|
|
return rk_nok;
|
|
}
|
|
|
|
static rk_s32 parse_log_object(MppCfgIoImpl *obj, MppCfgStrBuf *str)
|
|
{
|
|
MppCfgIoImpl *parent = obj;
|
|
char *buf = NULL;
|
|
rk_s32 old = str->offset;
|
|
rk_s32 ret = rk_nok;
|
|
|
|
if (str->depth >= MAX_CFG_DEPTH) {
|
|
mpp_loge_f("depth %d reached max\n", MAX_CFG_DEPTH);
|
|
return rk_nok;
|
|
}
|
|
|
|
str->depth++;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d object parse start\n", str->depth, str->offset);
|
|
|
|
buf = test_byte_f(str, 0);
|
|
if (!buf || buf[0] != '{') {
|
|
ret = -2;
|
|
goto failed;
|
|
}
|
|
|
|
buf = skip_byte_f(str, 1);
|
|
if (!buf) {
|
|
ret = -3;
|
|
goto failed;
|
|
}
|
|
|
|
/* skip whitespace and check the end of buffer */
|
|
buf = skip_ws_f(str);
|
|
if (!buf) {
|
|
ret = -4;
|
|
goto failed;
|
|
}
|
|
|
|
/* check empty object */
|
|
if (buf[0] == '}') {
|
|
skip_byte_f(str, 1);
|
|
cfg_io_dbg_from("depth %d found empty object\n", str->depth);
|
|
str->depth--;
|
|
return rk_ok;
|
|
}
|
|
|
|
do {
|
|
rk_s32 name_len = 0;
|
|
char *name = NULL;
|
|
char *tmp = NULL;
|
|
|
|
/* support array without name */
|
|
if (buf[0] == '[') {
|
|
MppCfgObj object = NULL;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value array start\n",
|
|
str->depth, str->offset);
|
|
|
|
mpp_cfg_get_array(&object, NULL, 0);
|
|
mpp_cfg_add(parent, object);
|
|
|
|
ret = parse_log_array(object, str);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value array ret %d\n",
|
|
str->depth, str->offset, ret);
|
|
|
|
if (ret) {
|
|
mpp_cfg_put_all_child(object);
|
|
goto failed;
|
|
}
|
|
|
|
goto __next;
|
|
}
|
|
|
|
ret = parse_log_string(str, &name, &name_len, MPP_CFG_PARSER_TYPE_KEY);
|
|
if (ret) {
|
|
goto failed;
|
|
}
|
|
|
|
tmp = dup_str(name, name_len);
|
|
cfg_io_dbg_from("depth %d offset %d found object key %s len %d\n",
|
|
str->depth, str->offset, tmp, name_len);
|
|
MPP_FREE(tmp);
|
|
|
|
/* find colon for separater */
|
|
buf = skip_ws_f(str);
|
|
if (!buf || buf[0] != ':') {
|
|
ret = -5;
|
|
goto failed;
|
|
}
|
|
|
|
/* skip colon */
|
|
buf = skip_byte_f(str, 1);
|
|
if (!buf) {
|
|
ret = -6;
|
|
goto failed;
|
|
}
|
|
|
|
buf = skip_ws_f(str);
|
|
if (!buf) {
|
|
ret = -7;
|
|
goto failed;
|
|
}
|
|
|
|
tmp = dup_str(name, name_len);
|
|
if (!tmp) {
|
|
mpp_loge_f("failed to dup name\n");
|
|
ret = -8;
|
|
goto failed;
|
|
}
|
|
|
|
/* parse value */
|
|
ret = parse_log_value(parent, tmp, str);
|
|
MPP_FREE(tmp);
|
|
if (ret) {
|
|
ret = -9;
|
|
goto failed;
|
|
}
|
|
__next:
|
|
buf = skip_ws_f(str);
|
|
if (!buf) {
|
|
ret = -10;
|
|
goto failed;
|
|
}
|
|
|
|
if (buf[0] == '}')
|
|
break;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get next object\n", str->depth, str->offset);
|
|
} while (1);
|
|
|
|
skip_byte_f(str, 1);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d -> %d object parse success\n",
|
|
str->depth, old, str->offset);
|
|
|
|
str->depth--;
|
|
ret = rk_ok;
|
|
|
|
failed:
|
|
if (ret)
|
|
cfg_io_dbg_from("depth %d offset %d -> %d object parse failed ret %d\n",
|
|
str->depth, old, str->offset, ret);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static rk_s32 mpp_cfg_from_log(MppCfgObj *obj, MppCfgStrBuf *str)
|
|
{
|
|
MppCfgObj object = NULL;
|
|
char *buf = NULL;
|
|
rk_s32 ret = rk_ok;
|
|
|
|
/* skip white space and check the end of buffer */
|
|
buf = skip_ws_f(str);
|
|
if (!buf)
|
|
return rk_nok;
|
|
|
|
if (buf[0] == '{') {
|
|
ret = mpp_cfg_get_object(&object, NULL, MPP_CFG_TYPE_OBJECT, NULL);
|
|
if (ret || !object) {
|
|
mpp_loge_f("failed to create top object\n");
|
|
return rk_nok;
|
|
}
|
|
|
|
ret = parse_log_object(object, str);
|
|
} else if (buf[0] == '[') {
|
|
ret = mpp_cfg_get_array(&object, NULL, 0);
|
|
if (ret || !object) {
|
|
mpp_loge_f("failed to create top object\n");
|
|
return rk_nok;
|
|
}
|
|
|
|
ret = parse_log_array(object, str);
|
|
} else {
|
|
mpp_loge_f("invalid top element '%c' on offset %d\n", buf[0], str->offset);
|
|
}
|
|
|
|
*obj = object;
|
|
|
|
return ret;
|
|
}
|
|
|
|
static rk_s32 parse_json_string(MppCfgStrBuf *str, char **name, rk_s32 *len)
|
|
{
|
|
char *buf = NULL;
|
|
char *start = NULL;
|
|
rk_s32 name_len = 0;
|
|
|
|
*name = NULL;
|
|
*len = 0;
|
|
|
|
/* skip whitespace and find first double quotes */
|
|
buf = skip_ws_f(str);
|
|
if (!buf || buf[0] != '\"')
|
|
return -101;
|
|
|
|
buf = skip_byte_f(str, 1);
|
|
if (!buf)
|
|
return -102;
|
|
|
|
start = buf;
|
|
|
|
/* find the last double quotes */
|
|
while ((buf = show_byte_f(str, name_len)) && buf[0] != '\"') {
|
|
name_len++;
|
|
}
|
|
|
|
if (!buf || buf[0] != '\"')
|
|
return -103;
|
|
|
|
/* find complete string skip the string and double quotes */
|
|
buf = skip_byte_f(str, name_len + 1);
|
|
if (!buf)
|
|
return -104;
|
|
|
|
*name = start;
|
|
*len = name_len;
|
|
|
|
return rk_ok;
|
|
}
|
|
|
|
static rk_s32 parse_json_value(MppCfgIoImpl *parent, const char *name, MppCfgStrBuf *str);
|
|
static rk_s32 parse_json_array(MppCfgIoImpl *obj, MppCfgStrBuf *str);
|
|
|
|
static rk_s32 parse_json_object(MppCfgIoImpl *obj, MppCfgStrBuf *str)
|
|
{
|
|
MppCfgIoImpl *parent = obj;
|
|
char *buf = NULL;
|
|
rk_s32 old = str->offset;
|
|
rk_s32 ret = rk_nok;
|
|
|
|
if (str->depth >= MAX_CFG_DEPTH) {
|
|
mpp_loge_f("depth %d reached max\n", MAX_CFG_DEPTH);
|
|
return rk_nok;
|
|
}
|
|
|
|
str->depth++;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d object parse start\n", str->depth, str->offset);
|
|
|
|
buf = test_byte_f(str, 0);
|
|
if (!buf || buf[0] != '{') {
|
|
ret = -2;
|
|
goto failed;
|
|
}
|
|
|
|
buf = skip_byte_f(str, 1);
|
|
if (!buf) {
|
|
ret = -3;
|
|
goto failed;
|
|
}
|
|
|
|
/* skip whitespace and check the end of buffer */
|
|
buf = skip_ws_f(str);
|
|
if (!buf) {
|
|
ret = -4;
|
|
goto failed;
|
|
}
|
|
|
|
/* check empty object */
|
|
if (buf[0] == '}') {
|
|
skip_byte_f(str, 1);
|
|
cfg_io_dbg_from("depth %d found empty object\n", str->depth);
|
|
str->depth--;
|
|
return rk_ok;
|
|
}
|
|
|
|
do {
|
|
rk_s32 name_len = 0;
|
|
char *name = NULL;
|
|
char *tmp = NULL;
|
|
|
|
if (buf[0] == '[') {
|
|
MppCfgObj object = NULL;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value array start\n",
|
|
str->depth, str->offset);
|
|
|
|
mpp_cfg_get_array(&object, NULL, 0);
|
|
mpp_cfg_add(parent, object);
|
|
|
|
ret = parse_json_array(object, str);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value array ret %d\n",
|
|
str->depth, str->offset, ret);
|
|
|
|
if (ret) {
|
|
mpp_cfg_put_all_child(object);
|
|
goto failed;
|
|
}
|
|
|
|
goto __next;
|
|
}
|
|
|
|
ret = parse_json_string(str, &name, &name_len);
|
|
if (ret) {
|
|
goto failed;
|
|
}
|
|
|
|
/* find colon for separater */
|
|
buf = skip_ws_f(str);
|
|
if (!buf || buf[0] != ':') {
|
|
ret = -5;
|
|
goto failed;
|
|
}
|
|
|
|
/* skip colon */
|
|
buf = skip_byte_f(str, 1);
|
|
if (!buf) {
|
|
ret = -6;
|
|
goto failed;
|
|
}
|
|
|
|
buf = skip_ws_f(str);
|
|
if (!buf) {
|
|
ret = -7;
|
|
goto failed;
|
|
}
|
|
|
|
tmp = dup_str(name, name_len);
|
|
if (!tmp) {
|
|
mpp_loge_f("failed to dup name\n");
|
|
ret = -8;
|
|
goto failed;
|
|
}
|
|
|
|
/* parse value */
|
|
ret = parse_json_value(parent, tmp, str);
|
|
MPP_FREE(tmp);
|
|
if (ret) {
|
|
ret = -9;
|
|
goto failed;
|
|
}
|
|
__next:
|
|
buf = skip_ws_f(str);
|
|
if (!buf) {
|
|
ret = -10;
|
|
goto failed;
|
|
}
|
|
|
|
if (buf[0] == ',') {
|
|
buf = skip_byte_f(str, 1);
|
|
if (!buf) {
|
|
ret = -11;
|
|
goto failed;
|
|
}
|
|
|
|
buf = skip_ws_f(str);
|
|
if (buf[0] == '}')
|
|
break;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get next object\n", str->depth, str->offset);
|
|
continue;
|
|
}
|
|
|
|
break;
|
|
} while (1);
|
|
|
|
buf = skip_ws_f(str);
|
|
if (!buf || buf[0] != '}') {
|
|
ret = -12;
|
|
goto failed;
|
|
}
|
|
|
|
skip_byte_f(str, 1);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d -> %d object parse success\n",
|
|
str->depth, old, str->offset);
|
|
|
|
str->depth--;
|
|
ret = rk_ok;
|
|
|
|
failed:
|
|
if (ret)
|
|
cfg_io_dbg_from("depth %d offset %d -> %d object parse failed ret %d\n",
|
|
str->depth, old, str->offset, ret);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static rk_s32 parse_json_array(MppCfgIoImpl *obj, MppCfgStrBuf *str)
|
|
{
|
|
MppCfgIoImpl *parent = obj;
|
|
char *buf = NULL;
|
|
rk_s32 old = str->offset;
|
|
rk_s32 ret = rk_nok;
|
|
|
|
if (str->depth >= MAX_CFG_DEPTH) {
|
|
mpp_loge_f("depth %d reached max\n", MAX_CFG_DEPTH);
|
|
return rk_nok;
|
|
}
|
|
|
|
str->depth++;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d array parse start\n", str->depth, str->offset);
|
|
|
|
buf = test_byte_f(str, 0);
|
|
if (!buf || buf[0] != '[') {
|
|
ret = -2;
|
|
goto failed;
|
|
}
|
|
|
|
buf = skip_byte_f(str, 1);
|
|
if (!buf) {
|
|
ret = -3;
|
|
goto failed;
|
|
}
|
|
|
|
/* skip whitespace and check the end of buffer */
|
|
buf = skip_ws_f(str);
|
|
if (!buf) {
|
|
ret = -4;
|
|
goto failed;
|
|
}
|
|
|
|
/* check empty object */
|
|
if (buf[0] == ']') {
|
|
skip_byte_f(str, 1);
|
|
cfg_io_dbg_from("depth %d found empty array\n", str->depth);
|
|
str->depth--;
|
|
return rk_ok;
|
|
}
|
|
|
|
do {
|
|
/* find colon for separater */
|
|
buf = skip_ws_f(str);
|
|
if (!buf) {
|
|
ret = -5;
|
|
goto failed;
|
|
}
|
|
|
|
/* parse value */
|
|
ret = parse_json_value(parent, NULL, str);
|
|
if (ret) {
|
|
ret = -6;
|
|
goto failed;
|
|
}
|
|
|
|
buf = skip_ws_f(str);
|
|
if (!buf) {
|
|
ret = -7;
|
|
goto failed;
|
|
}
|
|
|
|
if (buf[0] == ',') {
|
|
buf = skip_byte_f(str, 1);
|
|
if (!buf) {
|
|
ret = -8;
|
|
goto failed;
|
|
}
|
|
|
|
buf = skip_ws_f(str);
|
|
if (buf[0] == '}')
|
|
break;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get next array\n", str->depth, str->offset);
|
|
continue;
|
|
}
|
|
break;
|
|
} while (1);
|
|
|
|
buf = skip_ws_f(str);
|
|
if (!buf || buf[0] != ']') {
|
|
ret = -9;
|
|
goto failed;
|
|
}
|
|
|
|
skip_byte_f(str, 1);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d -> %d array parse success\n",
|
|
str->depth, old, str->offset);
|
|
|
|
str->depth--;
|
|
ret = rk_ok;
|
|
|
|
failed:
|
|
if (ret)
|
|
cfg_io_dbg_from("depth %d offset %d -> %d array parse failed ret %d\n",
|
|
str->depth, old, str->offset, ret);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static rk_s32 parse_json_value(MppCfgIoImpl *parent, const char *name, MppCfgStrBuf *str)
|
|
{
|
|
MppCfgObj obj = NULL;
|
|
char *buf = NULL;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: parse value\n", str->depth, str->offset);
|
|
|
|
buf = test_byte_f(str, 4);
|
|
if (buf && !strncmp(buf, "null", 4)) {
|
|
mpp_cfg_get_object(&obj, name, MPP_CFG_TYPE_NULL, NULL);
|
|
mpp_cfg_add(parent, obj);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value null\n", str->depth, str->offset);
|
|
skip_byte_f(str, 4);
|
|
return rk_ok;
|
|
}
|
|
|
|
if (buf && !strncmp(buf, "true", 4)) {
|
|
MppCfgVal val;
|
|
|
|
val.b1 = 1;
|
|
mpp_cfg_get_object(&obj, name, MPP_CFG_TYPE_BOOL, &val);
|
|
mpp_cfg_add(parent, obj);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value true\n", str->depth, str->offset);
|
|
skip_byte_f(str, 4);
|
|
return rk_ok;
|
|
}
|
|
|
|
buf = test_byte_f(str, 5);
|
|
if (buf && !strncmp(buf, "false", 5)) {
|
|
MppCfgVal val;
|
|
|
|
val.b1 = 0;
|
|
mpp_cfg_get_object(&obj, name, MPP_CFG_TYPE_BOOL, &val);
|
|
mpp_cfg_add(parent, obj);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value false\n", str->depth, str->offset);
|
|
skip_byte_f(str, 5);
|
|
return rk_ok;
|
|
}
|
|
|
|
buf = test_byte_f(str, 0);
|
|
if (buf && buf[0] == '\"') {
|
|
MppCfgVal val;
|
|
char *string = NULL;
|
|
rk_s32 len = 0;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value string start\n", str->depth, str->offset);
|
|
|
|
parse_json_string(str, &string, &len);
|
|
if (!string)
|
|
return rk_nok;
|
|
|
|
val.str = dup_str(string, len);
|
|
mpp_cfg_get_object(&obj, name, MPP_CFG_TYPE_STRING, &val);
|
|
mpp_cfg_add(parent, obj);
|
|
MPP_FREE(val.str);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value string success\n", str->depth, str->offset);
|
|
return rk_ok;
|
|
}
|
|
|
|
if (buf && (buf[0] == '-' || (buf[0] >= '0' && buf[0] <= '9'))) {
|
|
MppCfgType type;
|
|
MppCfgVal val;
|
|
rk_s32 ret;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value number start\n",
|
|
str->depth, str->offset);
|
|
|
|
ret = parse_number(str, &type, &val);
|
|
if (ret)
|
|
return ret;
|
|
|
|
mpp_cfg_get_object(&obj, name, type, &val);
|
|
mpp_cfg_add(parent, obj);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value number success\n",
|
|
str->depth, str->offset);
|
|
return ret;
|
|
}
|
|
|
|
if (buf && buf[0] == '{') {
|
|
rk_s32 ret;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value object start\n",
|
|
str->depth, str->offset);
|
|
|
|
mpp_cfg_get_object(&obj, name, MPP_CFG_TYPE_OBJECT, NULL);
|
|
mpp_cfg_add(parent, obj);
|
|
|
|
ret = parse_json_object(obj, str);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value object ret %d\n",
|
|
str->depth, str->offset, ret);
|
|
return ret;
|
|
}
|
|
|
|
if (buf && buf[0] == '[') {
|
|
rk_s32 ret;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value array start\n",
|
|
str->depth, str->offset);
|
|
|
|
mpp_cfg_get_array(&obj, name, 0);
|
|
mpp_cfg_add(parent, obj);
|
|
|
|
ret = parse_json_array(obj, str);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value array ret %d\n",
|
|
str->depth, str->offset, ret);
|
|
return ret;
|
|
}
|
|
|
|
return rk_nok;
|
|
}
|
|
|
|
static rk_s32 mpp_cfg_from_json(MppCfgObj *obj, MppCfgStrBuf *str)
|
|
{
|
|
MppCfgObj object = NULL;
|
|
char *buf = NULL;
|
|
rk_s32 ret = rk_ok;
|
|
|
|
/* skip UTF-8 */
|
|
buf = test_byte_f(str, 4);
|
|
if (buf && !strncmp(buf, "\xEF\xBB\xBF", 3))
|
|
skip_byte_f(str, 3);
|
|
|
|
/* skip white space and check the end of buffer */
|
|
buf = skip_ws_f(str);
|
|
if (!buf)
|
|
return rk_nok;
|
|
|
|
if (buf[0] == '{') {
|
|
ret = mpp_cfg_get_object(&object, NULL, MPP_CFG_TYPE_OBJECT, NULL);
|
|
if (ret || !object) {
|
|
mpp_loge_f("failed to create top object\n");
|
|
return rk_nok;
|
|
}
|
|
|
|
ret = parse_json_object(object, str);
|
|
} else if (buf[0] == '[') {
|
|
ret = mpp_cfg_get_array(&object, NULL, 0);
|
|
if (ret || !object) {
|
|
mpp_loge_f("failed to create top object\n");
|
|
return rk_nok;
|
|
}
|
|
|
|
ret = parse_json_array(object, str);
|
|
} else {
|
|
mpp_loge_f("invalid top element '%c' on offset %d\n", buf[0], str->offset);
|
|
}
|
|
|
|
*obj = object;
|
|
|
|
return ret;
|
|
}
|
|
|
|
static rk_s32 parse_toml_nested_table(MppCfgIoImpl *root, MppCfgObj *object, char *name,
|
|
rk_s32 name_len)
|
|
{
|
|
MppCfgObj obj = NULL;
|
|
MppCfgIoImpl *parent = root;
|
|
rk_s32 i = 0;
|
|
char sub_name_offset = 0;
|
|
char sub_name_len = 0;
|
|
char sub_name[256] = {0};
|
|
rk_s32 ret = rk_ok;
|
|
|
|
for (i = 0; i <= name_len; i++) {
|
|
if (name[i] == '.' || name[i] == '\0') {
|
|
sub_name_len = i;
|
|
memcpy(sub_name, name, sub_name_len);
|
|
sub_name[i] = '\0';
|
|
obj = NULL;
|
|
mpp_cfg_find(&obj, root, sub_name, MPP_CFG_STR_FMT_TOML);
|
|
if (!obj) {
|
|
memcpy(sub_name, name + sub_name_offset, sub_name_len - sub_name_offset);
|
|
sub_name[sub_name_len - sub_name_offset] = '\0';
|
|
ret = mpp_cfg_get_object(&obj, sub_name, MPP_CFG_TYPE_OBJECT, NULL);
|
|
if (ret || !obj) {
|
|
mpp_loge_f("failed to create object %s\n", name);
|
|
ret = -101;
|
|
return ret;
|
|
}
|
|
mpp_cfg_add(parent, obj);
|
|
}
|
|
|
|
parent = obj;
|
|
sub_name_offset = i + 1;
|
|
}
|
|
}
|
|
|
|
*object = obj;
|
|
|
|
return ret;
|
|
}
|
|
|
|
static rk_s32 parse_toml_nested_array_table(MppCfgIoImpl *root, MppCfgObj *object, char *name,
|
|
rk_s32 name_len)
|
|
{
|
|
MppCfgObj obj = NULL;
|
|
MppCfgIoImpl *parent = root;
|
|
rk_s32 i = 0;
|
|
char sub_name_offset = 0;
|
|
char sub_name_len = 0;
|
|
char sub_name[256] = {0};
|
|
rk_s32 ret = rk_ok;
|
|
|
|
for (i = 0; i <= name_len; i++) {
|
|
if (name[i] == '.' || name[i] == '\0') {
|
|
sub_name_len = i;
|
|
memcpy(sub_name, name, sub_name_len);
|
|
sub_name[i] = '\0';
|
|
obj = NULL;
|
|
mpp_cfg_find(&obj, root, sub_name, MPP_CFG_STR_FMT_TOML);
|
|
if (!obj) {
|
|
memcpy(sub_name, name + sub_name_offset, sub_name_len - sub_name_offset);
|
|
sub_name[sub_name_len - sub_name_offset] = '\0';
|
|
|
|
/* if parent type is array, need get its last child as new parent */
|
|
if (parent->type == MPP_CFG_TYPE_ARRAY) {
|
|
MppCfgIoImpl *child_pos, *child_n;
|
|
MppCfgIoImpl *last_child = NULL;
|
|
list_for_each_entry_safe(child_pos, child_n, &parent->child, MppCfgIoImpl, list) {
|
|
if (!child_pos->name && child_pos->type == MPP_CFG_TYPE_OBJECT) {
|
|
last_child = child_pos;
|
|
}
|
|
}
|
|
if (!last_child) {
|
|
mpp_loge_f("failed to find last child\n");
|
|
ret = -111;
|
|
return ret;
|
|
}
|
|
parent = last_child;
|
|
}
|
|
if (name[i] == '\0') {
|
|
ret = mpp_cfg_get_array(&obj, sub_name, 0);
|
|
if (ret || !obj) {
|
|
mpp_loge_f("failed to create object %s\n", name);
|
|
ret = -112;
|
|
return ret;
|
|
}
|
|
mpp_cfg_add(parent, obj);
|
|
} else {
|
|
ret = mpp_cfg_get_object(&obj, sub_name, MPP_CFG_TYPE_OBJECT, NULL);
|
|
if (ret || !obj) {
|
|
mpp_loge_f("failed to create nested object %s\n", name);
|
|
ret = -113;
|
|
return ret;
|
|
}
|
|
mpp_cfg_add(parent, obj);
|
|
}
|
|
}
|
|
|
|
parent = obj;
|
|
sub_name_offset = i + 1;
|
|
}
|
|
}
|
|
|
|
*object = obj;
|
|
|
|
return ret;
|
|
}
|
|
|
|
static rk_s32 parse_toml_string(MppCfgStrBuf *str, char **name, rk_s32 *len, rk_u32 type)
|
|
{
|
|
char *buf = NULL;
|
|
char *start = NULL;
|
|
rk_s32 name_len = 0;
|
|
char terminator;
|
|
|
|
*name = NULL;
|
|
*len = 0;
|
|
|
|
/* skip whitespace and find first double quotes */
|
|
buf = skip_ws_f(str);
|
|
if (!buf)
|
|
return -201;
|
|
|
|
if (type == MPP_CFG_PARSER_TYPE_VALUE) {
|
|
terminator = '\"';
|
|
if (buf[0] != '\"')
|
|
return -202;
|
|
|
|
buf = skip_byte_f(str, 1);
|
|
if (!buf)
|
|
return -203;
|
|
} else if (type == MPP_CFG_PARSER_TYPE_KEY) {
|
|
terminator = ' ';
|
|
} else if (type == MPP_CFG_PARSER_TYPE_TABLE || type == MPP_CFG_PARSER_TYPE_ARRAY_TABLE) {
|
|
terminator = ']';
|
|
} else {
|
|
return -204;
|
|
}
|
|
|
|
start = buf;
|
|
|
|
/* find the terminator */
|
|
while ((buf = show_byte_f(str, name_len)) && buf[0] != terminator) {
|
|
name_len++;
|
|
}
|
|
|
|
if (!buf || buf[0] != terminator)
|
|
return -205;
|
|
|
|
/* find complete string skip the string */
|
|
if (type == MPP_CFG_PARSER_TYPE_VALUE)
|
|
buf = skip_byte_f(str, name_len + 1);
|
|
else
|
|
buf = skip_byte_f(str, name_len);
|
|
if (!buf)
|
|
return -206;
|
|
|
|
*name = start;
|
|
*len = name_len;
|
|
|
|
return rk_ok;
|
|
}
|
|
|
|
static rk_s32 parse_toml_value(MppCfgIoImpl *parent, const char *name, MppCfgStrBuf *str);
|
|
static rk_s32 parse_toml_object(MppCfgIoImpl *parent, MppCfgStrBuf *str, rk_s32 is_brace);
|
|
|
|
static rk_s32 parse_toml_array(MppCfgIoImpl *obj, MppCfgStrBuf *str)
|
|
{
|
|
MppCfgIoImpl *parent = obj;
|
|
char *buf = NULL;
|
|
rk_s32 old = str->offset;
|
|
rk_s32 ret = rk_nok;
|
|
|
|
if (str->depth >= MAX_CFG_DEPTH) {
|
|
mpp_loge_f("depth %d reached max\n", MAX_CFG_DEPTH);
|
|
return rk_nok;
|
|
}
|
|
|
|
str->depth++;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d array parse start\n", str->depth, str->offset);
|
|
|
|
buf = test_byte_f(str, 0);
|
|
if (!buf || buf[0] != '[') {
|
|
ret = -61;
|
|
goto failed;
|
|
}
|
|
|
|
buf = skip_byte_f(str, 1);
|
|
if (!buf) {
|
|
ret = -62;
|
|
goto failed;
|
|
}
|
|
|
|
/* skip whitespace and check the end of buffer */
|
|
buf = skip_ws_f(str);
|
|
if (!buf) {
|
|
ret = -63;
|
|
goto failed;
|
|
}
|
|
|
|
/* check empty object */
|
|
if (buf[0] == ']') {
|
|
skip_byte_f(str, 1);
|
|
cfg_io_dbg_from("depth %d found empty array\n", str->depth);
|
|
str->depth--;
|
|
return rk_ok;
|
|
}
|
|
|
|
do {
|
|
buf = skip_ws_f(str);
|
|
if (!buf) {
|
|
ret = -64;
|
|
goto failed;
|
|
}
|
|
|
|
/* parse value */
|
|
ret = parse_toml_value(parent, NULL, str);
|
|
if (ret) {
|
|
ret = -65;
|
|
goto failed;
|
|
}
|
|
|
|
buf = skip_ws_f(str);
|
|
if (!buf) {
|
|
ret = -66;
|
|
goto failed;
|
|
}
|
|
|
|
if (buf[0] == ',') {
|
|
buf = skip_byte_f(str, 1);
|
|
if (!buf) {
|
|
ret = -67;
|
|
goto failed;
|
|
}
|
|
|
|
buf = skip_ws_f(str);
|
|
if (buf[0] == '}')
|
|
break;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get next array\n", str->depth, str->offset);
|
|
continue;
|
|
}
|
|
break;
|
|
} while (1);
|
|
|
|
if (!buf || buf[0] != ']') {
|
|
ret = -68;
|
|
goto failed;
|
|
}
|
|
|
|
skip_byte_f(str, 1);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d -> %d array parse success\n",
|
|
str->depth, old, str->offset);
|
|
|
|
str->depth--;
|
|
ret = rk_ok;
|
|
|
|
failed:
|
|
if (ret)
|
|
cfg_io_dbg_from("depth %d offset %d -> %d array parse failed ret %d\n",
|
|
str->depth, old, str->offset, ret);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static rk_s32 parse_toml_value(MppCfgIoImpl *parent, const char *name, MppCfgStrBuf *str)
|
|
{
|
|
MppCfgObj obj = NULL;
|
|
char *buf = NULL;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: parse value\n", str->depth, str->offset);
|
|
|
|
buf = test_byte_f(str, 4);
|
|
if (buf && !strncmp(buf, "null", 4)) {
|
|
mpp_cfg_get_object(&obj, name, MPP_CFG_TYPE_NULL, NULL);
|
|
mpp_cfg_add(parent, obj);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value null\n", str->depth, str->offset);
|
|
skip_byte_f(str, 4);
|
|
return rk_ok;
|
|
}
|
|
|
|
if (buf && !strncmp(buf, "true", 4)) {
|
|
MppCfgVal val;
|
|
|
|
val.b1 = 1;
|
|
mpp_cfg_get_object(&obj, name, MPP_CFG_TYPE_BOOL, &val);
|
|
mpp_cfg_add(parent, obj);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value true\n", str->depth, str->offset);
|
|
skip_byte_f(str, 4);
|
|
return rk_ok;
|
|
}
|
|
|
|
buf = test_byte_f(str, 5);
|
|
if (buf && !strncmp(buf, "false", 5)) {
|
|
MppCfgVal val;
|
|
|
|
val.b1 = 0;
|
|
mpp_cfg_get_object(&obj, name, MPP_CFG_TYPE_BOOL, &val);
|
|
mpp_cfg_add(parent, obj);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value false\n", str->depth, str->offset);
|
|
skip_byte_f(str, 5);
|
|
return rk_ok;
|
|
}
|
|
|
|
buf = test_byte_f(str, 3);
|
|
if (buf && !strncmp(buf, "\"\"\"", 3)) {
|
|
MppCfgVal val;
|
|
char *string = NULL;
|
|
rk_s32 len = 0;
|
|
|
|
skip_byte_f(str, 2);
|
|
cfg_io_dbg_from("depth %d offset %d: get value multi line string start\n", str->depth, str->offset);
|
|
|
|
parse_toml_string(str, &string, &len, MPP_CFG_PARSER_TYPE_VALUE);
|
|
if (!string)
|
|
return rk_nok;
|
|
buf = test_byte_f(str, 1);
|
|
if (!buf || strncmp(buf, "\"\"", 2)) {
|
|
return rk_nok;
|
|
}
|
|
skip_byte_f(str, 2);
|
|
|
|
val.str = dup_str(string, len);
|
|
mpp_cfg_get_object(&obj, name, MPP_CFG_TYPE_STRING, &val);
|
|
mpp_cfg_add(parent, obj);
|
|
MPP_FREE(val.str);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value multi line string success\n", str->depth, str->offset);
|
|
return rk_ok;
|
|
}
|
|
|
|
buf = test_byte_f(str, 0);
|
|
if (buf && buf[0] == '\"') {
|
|
MppCfgVal val;
|
|
char *string = NULL;
|
|
rk_s32 len = 0;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value string start\n", str->depth, str->offset);
|
|
|
|
parse_toml_string(str, &string, &len, MPP_CFG_PARSER_TYPE_VALUE);
|
|
if (!string)
|
|
return rk_nok;
|
|
|
|
val.str = dup_str(string, len);
|
|
mpp_cfg_get_object(&obj, name, MPP_CFG_TYPE_STRING, &val);
|
|
mpp_cfg_add(parent, obj);
|
|
MPP_FREE(val.str);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value string success\n", str->depth, str->offset);
|
|
return rk_ok;
|
|
}
|
|
|
|
if (buf && (buf[0] == '-' || (buf[0] >= '0' && buf[0] <= '9'))) {
|
|
MppCfgType type;
|
|
MppCfgVal val;
|
|
rk_s32 ret;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value number start\n",
|
|
str->depth, str->offset);
|
|
|
|
ret = parse_number(str, &type, &val);
|
|
if (ret)
|
|
return ret;
|
|
|
|
mpp_cfg_get_object(&obj, name, type, &val);
|
|
mpp_cfg_add(parent, obj);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value number success\n",
|
|
str->depth, str->offset);
|
|
return ret;
|
|
}
|
|
|
|
if (buf && buf[0] == '{') {
|
|
rk_s32 ret;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value object start\n",
|
|
str->depth, str->offset);
|
|
|
|
mpp_cfg_get_object(&obj, name, MPP_CFG_TYPE_OBJECT, NULL);
|
|
mpp_cfg_add(parent, obj);
|
|
|
|
ret = parse_toml_object(obj, str, 1);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value object ret %d\n",
|
|
str->depth, str->offset, ret);
|
|
return ret;
|
|
}
|
|
|
|
if (buf && buf[0] == '[') {
|
|
rk_s32 ret;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value array start\n",
|
|
str->depth, str->offset);
|
|
|
|
mpp_cfg_get_array(&obj, name, 0);
|
|
mpp_cfg_add(parent, obj);
|
|
|
|
ret = parse_toml_array(obj, str);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value array ret %d\n",
|
|
str->depth, str->offset, ret);
|
|
return ret;
|
|
}
|
|
|
|
return rk_nok;
|
|
}
|
|
|
|
static rk_s32 parse_toml_object(MppCfgIoImpl *parent, MppCfgStrBuf *str, rk_s32 is_brace)
|
|
{
|
|
char *buf = NULL;
|
|
rk_s32 ret = rk_nok;
|
|
rk_s32 old = str->offset;
|
|
|
|
if (str->depth >= MAX_CFG_DEPTH) {
|
|
mpp_loge_f("depth %d reached max\n", MAX_CFG_DEPTH);
|
|
return rk_nok;
|
|
}
|
|
|
|
str->depth++;
|
|
/* skip whitespace and check the end of buffer */
|
|
if (is_brace) {
|
|
buf = test_byte_f(str, 0);
|
|
if (!buf || buf[0] != '{') {
|
|
ret = -31;
|
|
goto failed;
|
|
}
|
|
|
|
buf = skip_byte_f(str, 1);
|
|
if (!buf) {
|
|
ret = -32;
|
|
goto failed;
|
|
}
|
|
|
|
/* skip whitespace and check the end of buffer */
|
|
buf = skip_ws_f(str);
|
|
if (!buf) {
|
|
ret = -33;
|
|
goto failed;
|
|
}
|
|
|
|
/* check empty object */
|
|
if (buf[0] == '}') {
|
|
skip_byte_f(str, 1);
|
|
cfg_io_dbg_from("depth %d found empty object\n", str->depth);
|
|
str->depth--;
|
|
return rk_ok;
|
|
}
|
|
} else {
|
|
buf = skip_ws_f(str);
|
|
if (!buf) {
|
|
ret = -34;
|
|
goto failed;
|
|
}
|
|
}
|
|
|
|
do {
|
|
rk_s32 name_len = 0;
|
|
char *name = NULL;
|
|
char *tmp = NULL;
|
|
|
|
if (buf[0] == '[') {
|
|
MppCfgObj object = NULL;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value array start\n",
|
|
str->depth, str->offset);
|
|
|
|
mpp_cfg_get_array(&object, NULL, 0);
|
|
mpp_cfg_add(parent, object);
|
|
|
|
ret = parse_toml_array(object, str);
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get value array ret %d\n",
|
|
str->depth, str->offset, ret);
|
|
|
|
if (ret) {
|
|
mpp_cfg_put_all_child(object);
|
|
goto failed;
|
|
}
|
|
|
|
goto __next;
|
|
}
|
|
|
|
ret = parse_toml_string(str, &name, &name_len, MPP_CFG_PARSER_TYPE_KEY);
|
|
if (ret) {
|
|
ret = -35;
|
|
goto failed;
|
|
}
|
|
|
|
/* find equal for separater */
|
|
buf = skip_ws_f(str);
|
|
if (!buf || buf[0] != '=') {
|
|
ret = -36;
|
|
goto failed;
|
|
}
|
|
|
|
/* skip equal */
|
|
buf = skip_byte_f(str, 1);
|
|
if (!buf) {
|
|
ret = -37;
|
|
goto failed;
|
|
}
|
|
|
|
buf = skip_ws_f(str);
|
|
if (!buf) {
|
|
ret = -38;
|
|
goto failed;
|
|
}
|
|
|
|
tmp = dup_str(name, name_len);
|
|
if (!tmp) {
|
|
mpp_loge_f("failed to dup name\n");
|
|
ret = -39;
|
|
goto failed;
|
|
}
|
|
|
|
/* parse value */
|
|
ret = parse_toml_value(parent, tmp, str);
|
|
MPP_FREE(tmp);
|
|
if (ret) {
|
|
ret = -40;
|
|
goto failed;
|
|
}
|
|
__next:
|
|
buf = skip_ws_f(str);
|
|
if (!buf || buf[0] == '[' || buf[0] == '}')
|
|
break;
|
|
|
|
if (buf[0] == ',') {
|
|
buf = skip_byte_f(str, 1);
|
|
if (!buf) {
|
|
ret = -41;
|
|
goto failed;
|
|
}
|
|
|
|
buf = skip_ws_f(str);
|
|
if (buf[0] == '[' || buf[0] == '}')
|
|
break;
|
|
|
|
cfg_io_dbg_from("depth %d offset %d: get next object\n", str->depth, str->offset);
|
|
}
|
|
} while (1);
|
|
|
|
if (is_brace) {
|
|
if (buf && buf[0] == '}')
|
|
skip_byte_f(str, 1);
|
|
else {
|
|
ret = -42;
|
|
goto failed;
|
|
}
|
|
}
|
|
|
|
cfg_io_dbg_from("depth %d offset %d -> %d object parse success\n",
|
|
str->depth, old, str->offset);
|
|
|
|
str->depth--;
|
|
ret = rk_ok;
|
|
|
|
failed:
|
|
if (ret)
|
|
cfg_io_dbg_from("depth %d offset %d -> %d object parse failed ret %d\n",
|
|
str->depth, old, str->offset, ret);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static rk_s32 parse_toml_table(MppCfgIoImpl *parent, MppCfgStrBuf *str)
|
|
{
|
|
MppCfgObj obj = NULL;
|
|
char *buf = NULL;
|
|
rk_s32 ret = rk_nok;
|
|
rk_s32 name_len = 0;
|
|
char *name = NULL;
|
|
char *tmp = NULL;
|
|
|
|
ret = parse_toml_string(str, &name, &name_len, MPP_CFG_PARSER_TYPE_TABLE);
|
|
if (ret) {
|
|
ret = -11;
|
|
goto failed;
|
|
}
|
|
|
|
tmp = dup_str(name, name_len);
|
|
if (!tmp) {
|
|
mpp_loge_f("failed to dup tmp\n");
|
|
ret = -12;
|
|
goto failed;
|
|
}
|
|
|
|
if (strchr(tmp, '.')) {
|
|
ret = parse_toml_nested_table(parent, &obj, tmp, name_len);
|
|
MPP_FREE(tmp);
|
|
if (ret || !obj) {
|
|
return ret;
|
|
}
|
|
} else {
|
|
ret = mpp_cfg_get_object(&obj, tmp, MPP_CFG_TYPE_OBJECT, NULL);
|
|
MPP_FREE(tmp);
|
|
if (ret || !obj) {
|
|
mpp_loge_f("failed to create object %s\n", tmp);
|
|
ret = -13;
|
|
goto failed;
|
|
}
|
|
mpp_cfg_add(parent, obj);
|
|
}
|
|
|
|
buf = test_byte_f(str, 0);
|
|
if (!buf || buf[0] != ']') {
|
|
ret = -14;
|
|
goto failed;
|
|
}
|
|
|
|
buf = skip_byte_f(str, 1);
|
|
if (!buf) {
|
|
ret = -15;
|
|
goto failed;
|
|
}
|
|
|
|
buf = skip_ws_f(str);
|
|
if (!buf)
|
|
return rk_nok;
|
|
|
|
if (buf[0] == '[')
|
|
ret = rk_ok;
|
|
else
|
|
ret = parse_toml_object(obj, str, 0);
|
|
|
|
failed:
|
|
if (ret)
|
|
cfg_io_dbg_from("table parse failed ret %d\n", ret);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static rk_s32 parse_toml_array_table(MppCfgIoImpl *parent, MppCfgStrBuf *str)
|
|
{
|
|
MppCfgObj obj = NULL;
|
|
char *buf = NULL;
|
|
rk_s32 ret = rk_nok;
|
|
rk_s32 name_len = 0;
|
|
char *name = NULL;
|
|
char *tmp = NULL;
|
|
|
|
ret = parse_toml_string(str, &name, &name_len, MPP_CFG_PARSER_TYPE_ARRAY_TABLE);
|
|
if (ret) {
|
|
ret = -22;
|
|
goto failed;
|
|
}
|
|
|
|
tmp = dup_str(name, name_len);
|
|
if (!tmp) {
|
|
mpp_loge_f("failed to dup tmp\n");
|
|
ret = -23;
|
|
goto failed;
|
|
}
|
|
|
|
if (strchr(tmp, '.')) {
|
|
ret = parse_toml_nested_array_table(parent, &obj, tmp, name_len);
|
|
MPP_FREE(tmp);
|
|
if (ret || !obj) {
|
|
return ret;
|
|
}
|
|
} else {
|
|
mpp_cfg_find(&obj, parent, tmp, MPP_CFG_STR_FMT_TOML);
|
|
if (!obj) {
|
|
ret = mpp_cfg_get_array(&obj, tmp, 0);
|
|
MPP_FREE(tmp);
|
|
if (ret || !obj) {
|
|
mpp_loge_f("failed to create object %s\n", tmp);
|
|
ret = -24;
|
|
goto failed;
|
|
}
|
|
mpp_cfg_add(parent, obj);
|
|
} else {
|
|
MPP_FREE(tmp);
|
|
}
|
|
}
|
|
|
|
/* array object need create object as child */
|
|
parent = obj;
|
|
obj = NULL;
|
|
mpp_cfg_get_object(&obj, NULL, MPP_CFG_TYPE_OBJECT, NULL);
|
|
mpp_cfg_add(parent, obj);
|
|
|
|
buf = test_byte_f(str, 1);
|
|
if (!buf || strncmp(buf, "]]", 2)) {
|
|
ret = -25;
|
|
goto failed;
|
|
}
|
|
|
|
buf = skip_byte_f(str, 2);
|
|
if (!buf) {
|
|
ret = -26;
|
|
goto failed;
|
|
}
|
|
|
|
buf = skip_ws_f(str);
|
|
if (!buf)
|
|
return rk_nok;
|
|
|
|
if (buf[0] == '[')
|
|
ret = rk_ok;
|
|
else
|
|
ret = parse_toml_object(obj, str, 0);
|
|
|
|
failed:
|
|
if (ret)
|
|
cfg_io_dbg_from("array table parse failed ret %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
static rk_s32 parse_toml_section(MppCfgIoImpl *parent, MppCfgStrBuf *str)
|
|
{
|
|
char *buf = NULL;
|
|
rk_s32 ret = rk_nok;
|
|
rk_s32 old = str->offset;
|
|
|
|
if (str->depth >= MAX_CFG_DEPTH) {
|
|
mpp_loge_f("depth %d reached max\n", MAX_CFG_DEPTH);
|
|
return rk_nok;
|
|
}
|
|
|
|
buf = test_byte_f(str, 0);
|
|
if (!buf) {
|
|
ret = -2;
|
|
goto failed;
|
|
}
|
|
|
|
if (buf[0] == '[') {
|
|
str->depth++;
|
|
|
|
buf = skip_byte_f(str, 1);
|
|
if (!buf) {
|
|
ret = -3;
|
|
goto failed;
|
|
}
|
|
if (buf[0] != '[') {
|
|
ret = parse_toml_table(parent, str);
|
|
if (ret)
|
|
goto failed;
|
|
} else {
|
|
buf = skip_byte_f(str, 1);
|
|
if (!buf) {
|
|
ret = -4;
|
|
goto failed;
|
|
}
|
|
|
|
ret = parse_toml_array_table(parent, str);
|
|
if (ret)
|
|
goto failed;
|
|
}
|
|
str->depth--;
|
|
} else {
|
|
ret = parse_toml_object(parent, str, 0);
|
|
if (ret)
|
|
goto failed;
|
|
}
|
|
cfg_io_dbg_from("depth %d offset %d -> %d section parse success\n",
|
|
str->depth, old, str->offset);
|
|
|
|
ret = rk_ok;
|
|
|
|
failed:
|
|
if (ret)
|
|
cfg_io_dbg_from("depth %d offset %d -> %d section parse failed ret %d\n",
|
|
str->depth, old, str->offset, ret);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static rk_s32 mpp_cfg_from_toml(MppCfgObj *obj, MppCfgStrBuf *str)
|
|
{
|
|
MppCfgObj object = NULL;
|
|
char *buf = NULL;
|
|
rk_s32 ret = rk_ok;
|
|
|
|
/* skip white space and check the end of buffer */
|
|
buf = skip_ws_f(str);
|
|
if (!buf)
|
|
return rk_nok;
|
|
|
|
ret = mpp_cfg_get_object(&object, NULL, MPP_CFG_TYPE_OBJECT, NULL);
|
|
if (ret || !object) {
|
|
mpp_loge_f("failed to create top object\n");
|
|
return rk_nok;
|
|
}
|
|
|
|
do {
|
|
/* parse section */
|
|
ret = parse_toml_section(object, str);
|
|
if (ret) {
|
|
mpp_loge_f("failed to parse section, ret : %d.\n", ret);
|
|
return rk_nok;
|
|
}
|
|
|
|
buf = skip_ws_f(str);
|
|
if (!buf)
|
|
break;
|
|
} while (1);
|
|
|
|
*obj = object;
|
|
|
|
return ret;
|
|
}
|
|
|
|
void mpp_cfg_dump(MppCfgObj obj, const char *func)
|
|
{
|
|
MppCfgIoImpl *impl = (MppCfgIoImpl *)obj;
|
|
MppCfgStrBuf str;
|
|
rk_s32 ret;
|
|
|
|
if (!obj) {
|
|
mpp_loge_f("invalid param obj %p at %s\n", obj, func);
|
|
return;
|
|
}
|
|
|
|
mpp_logi_f("obj %s - %p at %s\n", impl->name ? impl->name : "n/a", impl, func);
|
|
|
|
str.buf_size = 4096;
|
|
str.buf = mpp_malloc_size(void, str.buf_size);
|
|
str.offset = 0;
|
|
str.depth = 0;
|
|
str.type = MPP_CFG_STR_FMT_LOG;
|
|
|
|
ret = mpp_cfg_to_log(impl, &str);
|
|
if (ret)
|
|
mpp_loge_f("failed to get log buffer\n");
|
|
else
|
|
mpp_cfg_print_string(str.buf);
|
|
|
|
MPP_FREE(str.buf);
|
|
}
|
|
|
|
rk_s32 mpp_cfg_to_string(MppCfgObj obj, MppCfgStrFmt fmt, char **buf)
|
|
{
|
|
MppCfgIoImpl *impl = (MppCfgIoImpl *)obj;
|
|
MppCfgStrBuf str;
|
|
rk_s32 ret = rk_nok;
|
|
|
|
if (!obj || !buf || fmt >= MPP_CFG_STR_FMT_BUTT) {
|
|
mpp_loge_f("invalid param obj %p fmt %d buf %p\n", obj, fmt, buf);
|
|
return ret;
|
|
}
|
|
|
|
mpp_env_get_u32("mpp_cfg_io_debug", &mpp_cfg_io_debug, mpp_cfg_io_debug);
|
|
|
|
str.buf_size = 4096;
|
|
str.buf = mpp_malloc_size(void, str.buf_size);
|
|
str.offset = 0;
|
|
str.depth = 0;
|
|
str.type = fmt;
|
|
|
|
switch (fmt) {
|
|
case MPP_CFG_STR_FMT_LOG : {
|
|
ret = mpp_cfg_to_log(impl, &str);
|
|
} break;
|
|
case MPP_CFG_STR_FMT_JSON : {
|
|
ret = mpp_cfg_to_json(impl, &str);
|
|
} break;
|
|
case MPP_CFG_STR_FMT_TOML : {
|
|
ret = mpp_cfg_to_toml(impl, &str, 1);
|
|
} break;
|
|
default : {
|
|
mpp_loge_f("invalid formoffset %d\n", fmt);
|
|
} break;
|
|
}
|
|
|
|
if (ret) {
|
|
mpp_loge_f("%p %s failed to get string buffer\n", impl, impl->name);
|
|
MPP_FREE(str.buf);
|
|
}
|
|
|
|
*buf = str.buf;
|
|
return ret;
|
|
}
|
|
|
|
rk_s32 mpp_cfg_from_string(MppCfgObj *obj, MppCfgStrFmt fmt, const char *buf)
|
|
{
|
|
MppCfgObj object = NULL;
|
|
rk_s32 size;
|
|
rk_s32 ret = rk_nok;
|
|
|
|
if (!obj || fmt >= MPP_CFG_STR_FMT_BUTT || !buf) {
|
|
mpp_loge_f("invalid param obj %p fmt %d buf %p\n", obj, fmt, buf);
|
|
return ret;
|
|
}
|
|
|
|
mpp_env_get_u32("mpp_cfg_io_debug", &mpp_cfg_io_debug, mpp_cfg_io_debug);
|
|
|
|
size = strlen(buf);
|
|
if (size) {
|
|
MppCfgStrBuf str;
|
|
|
|
size++;
|
|
|
|
str.buf = (char *)buf;
|
|
str.buf_size = size;
|
|
str.offset = 0;
|
|
str.depth = 0;
|
|
str.type = fmt;
|
|
|
|
cfg_io_dbg_from("buf %p size %d\n", buf, size);
|
|
cfg_io_dbg_from("%s", buf);
|
|
|
|
switch (fmt) {
|
|
case MPP_CFG_STR_FMT_LOG : {
|
|
ret = mpp_cfg_from_log(&object, &str);
|
|
} break;
|
|
case MPP_CFG_STR_FMT_JSON : {
|
|
ret = mpp_cfg_from_json(&object, &str);
|
|
} break;
|
|
case MPP_CFG_STR_FMT_TOML : {
|
|
ret = mpp_cfg_from_toml(&object, &str);
|
|
} break;
|
|
default : {
|
|
mpp_loge_f("invalid formoffset %d\n", fmt);
|
|
} break;
|
|
}
|
|
}
|
|
|
|
if (ret)
|
|
mpp_loge_f("buf %p size %d failed to get object\n", buf, size);
|
|
|
|
*obj = object;
|
|
return ret;
|
|
}
|
|
|
|
static void write_struct(MppCfgIoImpl *obj, MppTrie trie, MppCfgStrBuf *str, void *st)
|
|
{
|
|
MppCfgInfo *tbl = NULL;
|
|
|
|
if (obj->name) {
|
|
MppTrieInfo *info = NULL;
|
|
|
|
/* use string to index the location table */
|
|
get_full_name(obj, str->buf, str->buf_size);
|
|
|
|
info = mpp_trie_get_info(trie, str->buf);
|
|
if (info)
|
|
tbl = mpp_trie_info_ctx(info);
|
|
}
|
|
|
|
if (!tbl)
|
|
tbl = &obj->info;
|
|
|
|
cfg_io_dbg_show("depth %d obj type %s name %s -> info %s offset %d size %d\n",
|
|
obj->depth, strof_type(obj->type), obj->name ? str->buf : "null",
|
|
strof_cfg_type(tbl->data_type), tbl->data_offset, tbl->data_size);
|
|
|
|
if (tbl->data_type < CFG_FUNC_TYPE_BUTT) {
|
|
switch (tbl->data_type) {
|
|
case CFG_FUNC_TYPE_s32 : {
|
|
mpp_cfg_set_s32(tbl, st, obj->val.s32);
|
|
} break;
|
|
case CFG_FUNC_TYPE_u32 : {
|
|
mpp_cfg_set_u32(tbl, st, obj->val.u32);
|
|
} break;
|
|
case CFG_FUNC_TYPE_s64 : {
|
|
mpp_cfg_set_s64(tbl, st, obj->val.s64);
|
|
} break;
|
|
case CFG_FUNC_TYPE_u64 : {
|
|
mpp_cfg_set_u64(tbl, st, obj->val.u64);
|
|
} break;
|
|
default : {
|
|
} break;
|
|
}
|
|
}
|
|
|
|
{
|
|
MppCfgIoImpl *pos, *n;
|
|
|
|
list_for_each_entry_safe(pos, n, &obj->child, MppCfgIoImpl, list) {
|
|
write_struct(pos, trie, str, st);
|
|
}
|
|
}
|
|
}
|
|
|
|
rk_s32 mpp_cfg_to_struct(MppCfgObj obj, MppCfgObj type, void *st)
|
|
{
|
|
MppCfgIoImpl *orig;
|
|
MppCfgIoImpl *impl;
|
|
MppTrie trie;
|
|
MppCfgStrBuf str;
|
|
char name[256] = { 0 };
|
|
|
|
if (!obj || !st) {
|
|
mpp_loge_f("invalid param obj %p st %p\n", obj, st);
|
|
return rk_nok;
|
|
}
|
|
|
|
impl = (MppCfgIoImpl *)obj;
|
|
orig = (MppCfgIoImpl *)type;
|
|
trie = mpp_cfg_to_trie(orig);
|
|
|
|
str.buf = name;
|
|
str.buf_size = sizeof(name) - 1;
|
|
str.offset = 0;
|
|
str.depth = 0;
|
|
|
|
write_struct(impl, trie, &str, st + orig->info.data_offset);
|
|
|
|
return rk_ok;
|
|
}
|
|
|
|
static MppCfgObj read_struct(MppCfgIoImpl *impl, MppCfgObj parent, void *st)
|
|
{
|
|
MppCfgInfo *info = &impl->info;
|
|
MppCfgIoImpl *ret = NULL;
|
|
|
|
/* dup node first */
|
|
ret = mpp_calloc_size(MppCfgIoImpl, impl->buf_size);
|
|
if (!ret) {
|
|
mpp_loge_f("failed to alloc impl size %d\n", impl->buf_size);
|
|
return NULL;
|
|
}
|
|
|
|
INIT_LIST_HEAD(&ret->list);
|
|
INIT_LIST_HEAD(&ret->child);
|
|
|
|
ret->type = impl->type;
|
|
ret->buf_size = impl->buf_size;
|
|
|
|
if (impl->name_buf_len) {
|
|
ret->name = (char *)(ret + 1);
|
|
memcpy(ret->name, impl->name, impl->name_buf_len);
|
|
ret->name_len = impl->name_len;
|
|
ret->name_buf_len = impl->name_buf_len;
|
|
}
|
|
|
|
/* assign value by different type */
|
|
switch (info->data_type) {
|
|
case CFG_FUNC_TYPE_s32 :
|
|
case CFG_FUNC_TYPE_u32 :
|
|
case CFG_FUNC_TYPE_s64 :
|
|
case CFG_FUNC_TYPE_u64 : {
|
|
switch (info->data_type) {
|
|
case CFG_FUNC_TYPE_s32 : {
|
|
mpp_assert(impl->type == MPP_CFG_TYPE_s32);
|
|
mpp_cfg_get_s32(info, st, &ret->val.s32);
|
|
} break;
|
|
case CFG_FUNC_TYPE_u32 : {
|
|
mpp_assert(impl->type == MPP_CFG_TYPE_u32);
|
|
mpp_cfg_get_u32(info, st, &ret->val.u32);
|
|
} break;
|
|
case CFG_FUNC_TYPE_s64 : {
|
|
mpp_assert(impl->type == MPP_CFG_TYPE_s64);
|
|
mpp_cfg_get_s64(info, st, &ret->val.s64);
|
|
} break;
|
|
case CFG_FUNC_TYPE_u64 : {
|
|
mpp_assert(impl->type == MPP_CFG_TYPE_u64);
|
|
mpp_cfg_get_u64(info, st, &ret->val.u64);
|
|
} break;
|
|
default : {
|
|
} break;
|
|
}
|
|
} break;
|
|
case CFG_FUNC_TYPE_st :
|
|
case CFG_FUNC_TYPE_ptr : {
|
|
ret->val = impl->val;
|
|
} break;
|
|
default : {
|
|
} break;
|
|
}
|
|
|
|
cfg_io_dbg_show("depth %d obj type %s name %s\n", ret->depth,
|
|
strof_type(ret->type), ret->name);
|
|
|
|
if (parent)
|
|
mpp_cfg_add(parent, ret);
|
|
|
|
{
|
|
MppCfgIoImpl *pos, *n;
|
|
|
|
list_for_each_entry_safe(pos, n, &impl->child, MppCfgIoImpl, list) {
|
|
read_struct(pos, ret, st);
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
rk_s32 mpp_cfg_from_struct(MppCfgObj *obj, MppCfgObj type, void *st)
|
|
{
|
|
MppCfgIoImpl *orig = (MppCfgIoImpl *)type;
|
|
|
|
if (!obj || !type || !st) {
|
|
mpp_loge_f("invalid param obj %p type %p st %p\n", obj, type, st);
|
|
return rk_nok;
|
|
}
|
|
|
|
/* NOTE: update structure pointer by data_offset */
|
|
*obj = read_struct(orig, NULL, st + orig->info.data_offset);
|
|
|
|
return *obj ? rk_ok : rk_nok;
|
|
}
|
|
|
|
rk_s32 mpp_cfg_print_string(char *buf)
|
|
{
|
|
rk_s32 start = 0;
|
|
rk_s32 pos = 0;
|
|
rk_s32 len = strlen(buf);
|
|
|
|
/* it may be a very long string, split by \n to different line and print */
|
|
for (pos = 0; pos < len; pos++) {
|
|
if (buf[pos] == '\n') {
|
|
buf[pos] = '\0';
|
|
mpp_logi("%s\n", &buf[start]);
|
|
buf[pos] = '\n';
|
|
start = pos + 1;
|
|
}
|
|
}
|
|
|
|
return rk_ok;
|
|
}
|