/*
|
* Copyright 2020 Rockchip Electronics Co. LTD
|
*
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
* you may not use this file except in compliance with the License.
|
* You may obtain a copy of the License at
|
*
|
* http://www.apache.org/licenses/LICENSE-2.0
|
*
|
* Unless required by applicable law or agreed to in writing, software
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
* See the License for the specific language governing permissions and
|
* limitations under the License.
|
*/
|
|
#define MODULE_TAG "mpp_opt"
|
|
#include "mpp_mem.h"
|
#include "mpp_log.h"
|
#include "mpp_trie.h"
|
#include "mpp_common.h"
|
|
#include "mpp_opt.h"
|
|
typedef struct MppOptImpl_t {
|
void *ctx;
|
MppTrie trie;
|
RK_S32 node_cnt;
|
RK_S32 info_cnt;
|
} MppOptImpl;
|
|
MPP_RET mpp_opt_init(MppOpt *opt)
|
{
|
MppOptImpl *impl = mpp_calloc(MppOptImpl, 1);
|
|
*opt = impl;
|
|
return (impl) ? MPP_OK : MPP_NOK;
|
}
|
|
MPP_RET mpp_opt_deinit(MppOpt opt)
|
{
|
MppOptImpl *impl = (MppOptImpl *)opt;
|
|
if (NULL == impl)
|
return MPP_NOK;
|
|
if (impl->trie) {
|
mpp_trie_deinit(impl->trie);
|
impl->trie = NULL;
|
}
|
MPP_FREE(impl);
|
|
return MPP_OK;
|
}
|
|
MPP_RET mpp_opt_setup(MppOpt opt, void *ctx, RK_S32 node_cnt, RK_S32 opt_cnt)
|
{
|
MppOptImpl *impl = (MppOptImpl *)opt;
|
|
if (NULL == impl)
|
return MPP_NOK;
|
|
mpp_trie_init(&impl->trie, node_cnt, opt_cnt);
|
if (impl->trie) {
|
impl->ctx = ctx;
|
impl->node_cnt = node_cnt;
|
impl->info_cnt = opt_cnt;
|
return MPP_OK;
|
}
|
|
mpp_err_f("failed to setup node %d opt %d\n", node_cnt, opt_cnt);
|
|
return MPP_NOK;
|
}
|
|
MPP_RET mpp_opt_add(MppOpt opt, MppOptInfo *info)
|
{
|
MppOptImpl *impl = (MppOptImpl *)opt;
|
|
if (NULL == impl || NULL == impl->trie)
|
return MPP_NOK;
|
|
if (NULL == info) {
|
RK_S32 node_cnt = mpp_trie_get_node_count(impl->trie);
|
RK_S32 info_cnt = mpp_trie_get_info_count(impl->trie);
|
|
if (impl->node_cnt != node_cnt || impl->info_cnt != info_cnt)
|
mpp_log("setup:real node %d:%d info %d:%d\n",
|
impl->node_cnt, node_cnt, impl->info_cnt, info_cnt);
|
|
return MPP_OK;
|
}
|
|
return mpp_trie_add_info(impl->trie, &info->name);
|
}
|
|
MPP_RET mpp_opt_parse(MppOpt opt, int argc, char **argv)
|
{
|
MppOptImpl *impl = (MppOptImpl *)opt;
|
MPP_RET ret = MPP_NOK;
|
RK_S32 opt_idx = 0;
|
|
if (NULL == impl || NULL == impl->trie || argc < 2 || NULL == argv)
|
return ret;
|
|
ret = MPP_OK;
|
|
while (opt_idx <= argc) {
|
RK_S32 opt_next = opt_idx + 1;
|
char *opts = argv[opt_idx++];
|
char *next = (opt_next >= argc) ? NULL : argv[opt_next];
|
|
if (NULL == opts)
|
break;
|
|
if (opts[0] == '-' && opts[1] != '\0') {
|
MppOptInfo *info = NULL;
|
const char **name = mpp_trie_get_info(impl->trie, opts + 1);
|
RK_S32 step = 0;
|
|
if (NULL == name) {
|
mpp_err("invalid option %s\n", opts + 1);
|
continue;
|
}
|
|
info = container_of(name, MppOptInfo, name);
|
if (info->proc)
|
step = info->proc(impl->ctx, next);
|
|
/* option failure or help */
|
if (step < 0) {
|
ret = step;
|
break;
|
}
|
|
opt_idx += step;
|
}
|
}
|
|
return ret;
|
}
|