1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun * Copyright 2017 Rockchip Electronics Co. LTD
3*4882a593Smuzhiyun *
4*4882a593Smuzhiyun * Licensed under the Apache License, Version 2.0 (the "License");
5*4882a593Smuzhiyun * you may not use this file except in compliance with the License.
6*4882a593Smuzhiyun * You may obtain a copy of the License at
7*4882a593Smuzhiyun *
8*4882a593Smuzhiyun * http://www.apache.org/licenses/LICENSE-2.0
9*4882a593Smuzhiyun *
10*4882a593Smuzhiyun * Unless required by applicable law or agreed to in writing, software
11*4882a593Smuzhiyun * distributed under the License is distributed on an "AS IS" BASIS,
12*4882a593Smuzhiyun * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*4882a593Smuzhiyun * See the License for the specific language governing permissions and
14*4882a593Smuzhiyun * limitations under the License.
15*4882a593Smuzhiyun */
16*4882a593Smuzhiyun
17*4882a593Smuzhiyun #include <stdio.h>
18*4882a593Smuzhiyun #include <string.h>
19*4882a593Smuzhiyun
20*4882a593Smuzhiyun #include "mpp_parse_cfg.h"
21*4882a593Smuzhiyun
22*4882a593Smuzhiyun #include "mpp_log.h"
23*4882a593Smuzhiyun
24*4882a593Smuzhiyun #define ARRAY_SIZE(x) (sizeof(x) / sizeof(x[0]))
25*4882a593Smuzhiyun
26*4882a593Smuzhiyun enum CONFIG_TYPE {
27*4882a593Smuzhiyun CONFIG_TYPE_OPTION,
28*4882a593Smuzhiyun CONFIG_TYPE_EVENT,
29*4882a593Smuzhiyun
30*4882a593Smuzhiyun OPT_TYPE_INDEX_TYPE,
31*4882a593Smuzhiyun OPT_TYPE_LOOP_NUM,
32*4882a593Smuzhiyun
33*4882a593Smuzhiyun IDX_TYPE_FRM_NUM,
34*4882a593Smuzhiyun IDX_TYPE_MSEC,
35*4882a593Smuzhiyun };
36*4882a593Smuzhiyun
37*4882a593Smuzhiyun struct options_table {
38*4882a593Smuzhiyun int type;
39*4882a593Smuzhiyun const char *type_str;
40*4882a593Smuzhiyun };
41*4882a593Smuzhiyun
42*4882a593Smuzhiyun /* table for converting string to type */
43*4882a593Smuzhiyun static struct options_table op_tbl[] = {
44*4882a593Smuzhiyun {CONFIG_TYPE_OPTION, "[CONFIG]"},
45*4882a593Smuzhiyun {CONFIG_TYPE_EVENT, "[EVENT]"},
46*4882a593Smuzhiyun
47*4882a593Smuzhiyun {OPT_TYPE_INDEX_TYPE, "index"},
48*4882a593Smuzhiyun {OPT_TYPE_LOOP_NUM, "loop"},
49*4882a593Smuzhiyun
50*4882a593Smuzhiyun {IDX_TYPE_FRM_NUM, "frm"},
51*4882a593Smuzhiyun {IDX_TYPE_MSEC, "msec"},
52*4882a593Smuzhiyun };
53*4882a593Smuzhiyun
54*4882a593Smuzhiyun struct cfg_file {
55*4882a593Smuzhiyun FILE *file;
56*4882a593Smuzhiyun char cache[128];
57*4882a593Smuzhiyun RK_S32 cache_on;
58*4882a593Smuzhiyun };
59*4882a593Smuzhiyun
60*4882a593Smuzhiyun /* get rid of leading and following space from string */
string_trim(char * string)61*4882a593Smuzhiyun static char *string_trim(char *string)
62*4882a593Smuzhiyun {
63*4882a593Smuzhiyun char *p = string;
64*4882a593Smuzhiyun long len;
65*4882a593Smuzhiyun
66*4882a593Smuzhiyun if (string == NULL)
67*4882a593Smuzhiyun return NULL;
68*4882a593Smuzhiyun
69*4882a593Smuzhiyun while (*p == ' ')
70*4882a593Smuzhiyun p++;
71*4882a593Smuzhiyun
72*4882a593Smuzhiyun len = strlen(p);
73*4882a593Smuzhiyun
74*4882a593Smuzhiyun while (p[len - 1] == ' ')
75*4882a593Smuzhiyun len--;
76*4882a593Smuzhiyun p[len] = '\0';
77*4882a593Smuzhiyun
78*4882a593Smuzhiyun return p;
79*4882a593Smuzhiyun }
80*4882a593Smuzhiyun
read_cfg_line(struct cfg_file * cfg)81*4882a593Smuzhiyun static char *read_cfg_line(struct cfg_file *cfg)
82*4882a593Smuzhiyun {
83*4882a593Smuzhiyun int ch;
84*4882a593Smuzhiyun int i;
85*4882a593Smuzhiyun
86*4882a593Smuzhiyun while (!cfg->cache_on) {
87*4882a593Smuzhiyun i = 0;
88*4882a593Smuzhiyun
89*4882a593Smuzhiyun while (1) {
90*4882a593Smuzhiyun ch = fgetc(cfg->file);
91*4882a593Smuzhiyun if (i == 0 && ch == EOF)
92*4882a593Smuzhiyun return NULL;
93*4882a593Smuzhiyun else if (ch == EOF || feof(cfg->file) || ch == '\n')
94*4882a593Smuzhiyun break;
95*4882a593Smuzhiyun
96*4882a593Smuzhiyun cfg->cache[i++] = ch;
97*4882a593Smuzhiyun }
98*4882a593Smuzhiyun cfg->cache[i] = '\0';
99*4882a593Smuzhiyun
100*4882a593Smuzhiyun /* a note, ignore and get the next line */
101*4882a593Smuzhiyun if (cfg->cache[0] != '#')
102*4882a593Smuzhiyun break;
103*4882a593Smuzhiyun }
104*4882a593Smuzhiyun
105*4882a593Smuzhiyun cfg->cache_on = 1;
106*4882a593Smuzhiyun return string_trim(cfg->cache);
107*4882a593Smuzhiyun }
108*4882a593Smuzhiyun
invalid_cfg_cache(struct cfg_file * cfg)109*4882a593Smuzhiyun static inline void invalid_cfg_cache(struct cfg_file *cfg)
110*4882a593Smuzhiyun {
111*4882a593Smuzhiyun cfg->cache_on = 0;
112*4882a593Smuzhiyun }
113*4882a593Smuzhiyun
get_opt_value(char * line)114*4882a593Smuzhiyun static char *get_opt_value(char *line)
115*4882a593Smuzhiyun {
116*4882a593Smuzhiyun size_t i;
117*4882a593Smuzhiyun
118*4882a593Smuzhiyun for (i = 0; i < strlen(line); ++i) {
119*4882a593Smuzhiyun if (line[i] == ':')
120*4882a593Smuzhiyun return string_trim(&line[i + 1]);
121*4882a593Smuzhiyun }
122*4882a593Smuzhiyun
123*4882a593Smuzhiyun return NULL;
124*4882a593Smuzhiyun }
125*4882a593Smuzhiyun
126*4882a593Smuzhiyun /* convert string to index by look up map table */
lookup_opt_type(char * line)127*4882a593Smuzhiyun static int lookup_opt_type(char *line)
128*4882a593Smuzhiyun {
129*4882a593Smuzhiyun size_t i;
130*4882a593Smuzhiyun
131*4882a593Smuzhiyun for (i = 0; i < ARRAY_SIZE(op_tbl); ++i) {
132*4882a593Smuzhiyun if (!strncmp(op_tbl[i].type_str, line,
133*4882a593Smuzhiyun strlen(op_tbl[i].type_str))) {
134*4882a593Smuzhiyun mpp_log("option type %s find\n", op_tbl[i].type_str);
135*4882a593Smuzhiyun return op_tbl[i].type;
136*4882a593Smuzhiyun }
137*4882a593Smuzhiyun }
138*4882a593Smuzhiyun
139*4882a593Smuzhiyun return -1;
140*4882a593Smuzhiyun }
141*4882a593Smuzhiyun
scan_event_line(struct cfg_file * cfg,struct rc_event * event)142*4882a593Smuzhiyun static int scan_event_line(struct cfg_file *cfg, struct rc_event *event)
143*4882a593Smuzhiyun {
144*4882a593Smuzhiyun char *line = read_cfg_line(cfg);
145*4882a593Smuzhiyun
146*4882a593Smuzhiyun if (line != NULL && line[0] != '\n' && line[0] != '[') {
147*4882a593Smuzhiyun sscanf(line, "%d\t%d\t%f",
148*4882a593Smuzhiyun &event->idx, &event->bps, &event->fps);
149*4882a593Smuzhiyun mpp_log("idx: %d, bps %u, fps: %f\n",
150*4882a593Smuzhiyun event->idx, event->bps, event->fps);
151*4882a593Smuzhiyun invalid_cfg_cache(cfg);
152*4882a593Smuzhiyun } else {
153*4882a593Smuzhiyun return -1;
154*4882a593Smuzhiyun }
155*4882a593Smuzhiyun
156*4882a593Smuzhiyun return 0;
157*4882a593Smuzhiyun }
158*4882a593Smuzhiyun
parse_events(struct cfg_file * cfg,struct rc_test_config * ea)159*4882a593Smuzhiyun static int parse_events(struct cfg_file *cfg, struct rc_test_config *ea)
160*4882a593Smuzhiyun {
161*4882a593Smuzhiyun int ret;
162*4882a593Smuzhiyun int i;
163*4882a593Smuzhiyun
164*4882a593Smuzhiyun for (i = 0; i < 128; ++i) {
165*4882a593Smuzhiyun ret = scan_event_line(cfg, &ea->event[i]);
166*4882a593Smuzhiyun if (ret < 0) {
167*4882a593Smuzhiyun ea->event_cnt = i;
168*4882a593Smuzhiyun break;
169*4882a593Smuzhiyun }
170*4882a593Smuzhiyun }
171*4882a593Smuzhiyun
172*4882a593Smuzhiyun return 0;
173*4882a593Smuzhiyun }
174*4882a593Smuzhiyun
parse_options(struct cfg_file * cfg,struct rc_test_config * ea)175*4882a593Smuzhiyun static int parse_options(struct cfg_file *cfg, struct rc_test_config *ea)
176*4882a593Smuzhiyun {
177*4882a593Smuzhiyun char *opt;
178*4882a593Smuzhiyun int type;
179*4882a593Smuzhiyun
180*4882a593Smuzhiyun while (1) {
181*4882a593Smuzhiyun opt = read_cfg_line(cfg);
182*4882a593Smuzhiyun
183*4882a593Smuzhiyun if (opt && opt[0] != '\n' && opt[0] != '[') {
184*4882a593Smuzhiyun type = lookup_opt_type(opt);
185*4882a593Smuzhiyun
186*4882a593Smuzhiyun switch (type) {
187*4882a593Smuzhiyun case OPT_TYPE_INDEX_TYPE:
188*4882a593Smuzhiyun ea->idx_type = lookup_opt_type(get_opt_value(opt));
189*4882a593Smuzhiyun break;
190*4882a593Smuzhiyun case OPT_TYPE_LOOP_NUM:
191*4882a593Smuzhiyun sscanf(get_opt_value(opt), "%d", &ea->loop);
192*4882a593Smuzhiyun mpp_log("loop num: %d\n", ea->loop);
193*4882a593Smuzhiyun break;
194*4882a593Smuzhiyun default:
195*4882a593Smuzhiyun break;
196*4882a593Smuzhiyun }
197*4882a593Smuzhiyun
198*4882a593Smuzhiyun invalid_cfg_cache(cfg);
199*4882a593Smuzhiyun } else {
200*4882a593Smuzhiyun break;
201*4882a593Smuzhiyun }
202*4882a593Smuzhiyun }
203*4882a593Smuzhiyun return 0;
204*4882a593Smuzhiyun }
205*4882a593Smuzhiyun
mpp_parse_config(char * cfg_url,struct rc_test_config * ea)206*4882a593Smuzhiyun int mpp_parse_config(char *cfg_url, struct rc_test_config *ea)
207*4882a593Smuzhiyun {
208*4882a593Smuzhiyun struct cfg_file cfg;
209*4882a593Smuzhiyun
210*4882a593Smuzhiyun if (cfg_url == NULL || strlen(cfg_url) == 0) {
211*4882a593Smuzhiyun mpp_err("invalid input config url\n");
212*4882a593Smuzhiyun return -1;
213*4882a593Smuzhiyun }
214*4882a593Smuzhiyun
215*4882a593Smuzhiyun cfg.file = fopen(cfg_url, "rb");
216*4882a593Smuzhiyun if (cfg.file == NULL) {
217*4882a593Smuzhiyun mpp_err("fopen %s failed\n", cfg_url);
218*4882a593Smuzhiyun return -1;
219*4882a593Smuzhiyun }
220*4882a593Smuzhiyun cfg.cache_on = 0;
221*4882a593Smuzhiyun
222*4882a593Smuzhiyun while (1) {
223*4882a593Smuzhiyun char *line = read_cfg_line(&cfg);
224*4882a593Smuzhiyun
225*4882a593Smuzhiyun if (!line)
226*4882a593Smuzhiyun break;
227*4882a593Smuzhiyun
228*4882a593Smuzhiyun invalid_cfg_cache(&cfg);
229*4882a593Smuzhiyun if (line[0] == '[') {
230*4882a593Smuzhiyun int type = lookup_opt_type(line);
231*4882a593Smuzhiyun
232*4882a593Smuzhiyun switch (type) {
233*4882a593Smuzhiyun case CONFIG_TYPE_EVENT:
234*4882a593Smuzhiyun parse_events(&cfg, ea);
235*4882a593Smuzhiyun break;
236*4882a593Smuzhiyun case CONFIG_TYPE_OPTION:
237*4882a593Smuzhiyun parse_options(&cfg, ea);
238*4882a593Smuzhiyun break;
239*4882a593Smuzhiyun default:
240*4882a593Smuzhiyun mpp_err("invalid config type find\n");
241*4882a593Smuzhiyun fclose(cfg.file);
242*4882a593Smuzhiyun return -1;
243*4882a593Smuzhiyun }
244*4882a593Smuzhiyun }
245*4882a593Smuzhiyun }
246*4882a593Smuzhiyun
247*4882a593Smuzhiyun fclose(cfg.file);
248*4882a593Smuzhiyun
249*4882a593Smuzhiyun return 0;
250*4882a593Smuzhiyun }
251*4882a593Smuzhiyun
252*4882a593Smuzhiyun #ifdef PARSE_CONFIG_TEST
main(int argc,char ** argv)253*4882a593Smuzhiyun int main(int argc, char **argv)
254*4882a593Smuzhiyun {
255*4882a593Smuzhiyun struct rc_test_config event_array;
256*4882a593Smuzhiyun
257*4882a593Smuzhiyun if (argc < 2) {
258*4882a593Smuzhiyun mpp_err("invalid input argument\n");
259*4882a593Smuzhiyun return -1;
260*4882a593Smuzhiyun }
261*4882a593Smuzhiyun
262*4882a593Smuzhiyun mpp_parse_config(argv[1], &event_array);
263*4882a593Smuzhiyun
264*4882a593Smuzhiyun return 0;
265*4882a593Smuzhiyun }
266*4882a593Smuzhiyun #endif
267