OpenFPGA/vpr7_x2p/vpr/SRC/fpga_x2p/shell/read_opt.c

310 lines
7.7 KiB
C

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <assert.h>
#include "util.h"
#include "read_opt_types.h"
#include "read_opt.h"
/* External function */
void my_free(void* ptr);
char** fpga_spice_strtok(char* str,
char* delims,
int* len);
int my_strcmp(char* str1, char* str2);
/**
* Read the integer in option list
* Do simple check and convert it from char to
* integer
*/
int process_int_arg(INP char* arg) {
/*Check if the pointor of arg is NULL */
if (NULL == arg) {
vpr_printf(TIO_MESSAGE_ERROR,
"Error: Arg is NULL when processing integer!\n");
exit(1);
}
return atoi(arg);
}
/**
* Read the float in option list
* Do simple check and convert it from char to
* float
*/
float process_float_arg(INP char* arg) {
/*Check if the pointor of arg is NULL */
if (NULL == arg) {
vpr_printf(TIO_MESSAGE_ERROR,
"Error: Arg is NULL when processing float!\n");
exit(1);
}
return atof(arg);
}
/**
* Process the argument by comparing
* Store the options in struct
*/
boolean process_arg_opt(INP char** argv,
INOUTP int* iarg,
INP char* curarg,
t_opt_info* cur) {
int itok = 0;
int num_tokens = 0;
char** token = NULL;
while (0 != my_strcmp(LAST_OPT_TAG, cur->tag)) {
/* Tokenize the opt_name*/
token = fpga_spice_strtok(cur->name, ",", &num_tokens);
/*Process Match Arguments*/
for (itok = 0; itok < num_tokens; itok++) {
if (0 == my_strcmp(curarg, token[itok])) {
/* Check the defined flag if yes, return with error! */
if (OPT_DEF == cur->opt_def) {
vpr_printf(TIO_MESSAGE_ERROR,
"Intend to redefine the option(%s) with (%s)!\n",
cur->name, token[itok]);
return FALSE;
}
cur->opt_def = OPT_DEF;
/*A value is stored in next argument*/
if (OPT_WITHVAL == cur->with_val) {
*(iarg) += 1;
cur->val = my_strdup((argv[*iarg]));
return TRUE;
} else if (OPT_NONVAL == cur->with_val) {
/*Do not need next argument, return*/
return TRUE;
} else {
vpr_printf(TIO_MESSAGE_ERROR,
"(FILE:%s, [LINE%d]) Unknown type of Option with_Val! Abort.\n",
__FILE__, __LINE__);
return FALSE;
}
}
}
cur++;
/* Free */
for (itok = 0; itok < num_tokens; itok++) {
my_free(token[itok]);
}
my_free(token);
}
return FALSE;
}
char* convert_option_mandatory_to_str(enum opt_manda cur) {
switch (cur) {
case OPT_REQ:
return "Required";
case OPT_OPT:
return "Optional";
default:
vpr_printf(TIO_MESSAGE_ERROR,
"(FILE:%s, [LINE%d]) Unknown type of Option Mandatory! Abort.\n",
__FILE__, __LINE__);
return NULL;
}
}
void print_opt_info_help_desk(t_opt_info* cur_opt_info) {
int max_str_len = -1;
t_opt_info* cur = cur_opt_info;
char* str_fixed_len = NULL;
char* name_tag = "Option Names";
char str_end = '\0';
/* Get the maximum string length of options
* We can align to the longest string when outputing the help desk
*/
while (0 != my_strcmp(LAST_OPT_TAG, cur->tag)) {
if ( (-1 == max_str_len)
|| (max_str_len < strlen(cur->name)) ) {
max_str_len = strlen(cur->name) + 1;
}
cur++;
}
/* Minimum size is 5 */
if (max_str_len < strlen(name_tag) + 1) {
max_str_len = strlen(name_tag) + 1;
}
/* Malloc */
str_fixed_len = (char*)my_calloc(max_str_len, sizeof(char));
vpr_printf(TIO_MESSAGE_INFO, "Help Desk:\n");
memset(str_fixed_len, ' ', max_str_len);
strcpy(str_fixed_len, name_tag);
str_fixed_len[strlen(name_tag)] = ' ';
str_fixed_len[max_str_len] = str_end;
vpr_printf(TIO_MESSAGE_INFO, "%s Status Description\n", str_fixed_len);
cur = cur_opt_info;
while (0 != my_strcmp(LAST_OPT_TAG, cur->tag)) {
memset(str_fixed_len, ' ', max_str_len);
strcpy(str_fixed_len, cur->name);
str_fixed_len[strlen(cur->name)] = ' ';
str_fixed_len[max_str_len] = str_end;
vpr_printf(TIO_MESSAGE_INFO, "%s ", str_fixed_len);
vpr_printf(TIO_MESSAGE_INFO, "%s ", convert_option_mandatory_to_str(cur->mandatory));
vpr_printf(TIO_MESSAGE_INFO, "%s", cur->description);
vpr_printf(TIO_MESSAGE_INFO, "\n");
cur++;
}
vpr_printf(TIO_MESSAGE_INFO, "\n");
/* Free */
my_free(str_fixed_len);
return;
}
boolean read_options(INP int argc,
INP char **argv,
INOUTP t_opt_info* cur_opt_info) {
int iarg;
char* curarg = NULL;
boolean arg_processed = FALSE;
t_opt_info* cur = cur_opt_info;
vpr_printf(TIO_MESSAGE_INFO,
"Processing Options...\n");
/*Start from argv[1], the 1st argv is programme name*/
for (iarg = 1; iarg < argc;) {
curarg = argv[iarg];
/*Process the option start with hyphone*/
if (0 == strncmp("-", curarg, 1)) {
arg_processed = process_arg_opt(argv, &iarg, curarg, cur_opt_info);
if (FALSE == arg_processed) {
vpr_printf(TIO_MESSAGE_WARNING,
"Warning: Unknown Option(%s) detected!\n",
curarg);
print_opt_info_help_desk(cur_opt_info);
return FALSE;
}
iarg += 1; /*Move on to the next argument*/
} else {
iarg++;
}
}
/* Search the command help */
if (TRUE == is_opt_set(cur_opt_info, LAST_OPT_TAG, FALSE)) {
print_opt_info_help_desk(cur_opt_info);
return FALSE;
}
/* Check if REQUIRED options are processed */
while (0 != my_strcmp(LAST_OPT_TAG, cur->tag)) {
if ( (NULL == cur->val)
&& (OPT_REQ == cur->mandatory)) {
vpr_printf(TIO_MESSAGE_WARNING,
"Warning: Required Option(%s) is missing!\n",
cur->name);
print_opt_info_help_desk(cur_opt_info);
return FALSE;
}
cur++;
}
/* Confirm options */
/*
show_opt_list(cur_opt_info);
*/
return TRUE;
}
/**
* Show the options in opt_list after process.
* Only for debug use
*/
int show_opt_list(t_opt_info* cur) {
vpr_printf(TIO_MESSAGE_INFO,
"List Options:\n");
while (0 != my_strcmp(LAST_OPT_TAG, cur->tag)) {
vpr_printf(TIO_MESSAGE_INFO,
"Name=%s, Value=%s.\n",
cur->name, cur->val);
cur++;
}
return 1;
}
boolean is_opt_set(t_opt_info* opts, char* opt_name, boolean default_val) {
while (0 != my_strcmp(LAST_OPT_TAG, opts->tag)) {
if ( 0 != my_strcmp(opts->tag, opt_name)) {
opts++;
continue;
}
if (OPT_DEF == opts->opt_def) {
return TRUE;
} else {
return default_val;
}
opts++;
}
return default_val;
}
char* get_opt_val(t_opt_info* opts, char* opt_name) {
while (0 != my_strcmp(LAST_OPT_TAG, opts->tag)) {
if ( 0 != my_strcmp(opts->tag, opt_name)) {
opts++;
continue;
}
if (OPT_WITHVAL != opts->with_val) {
vpr_printf(TIO_MESSAGE_INFO,
"Try to get the val of an option(%s) which is defined to be without_val!\n",
opt_name);
}
if (NULL == opts->val) {
return NULL;
} else {
return my_strdup(opts->val);
}
opts++;
}
return NULL;
}
int get_opt_int_val(t_opt_info* opts, char* opt_name, int default_val) {
char* temp = get_opt_val(opts, opt_name);
int ret = default_val;
if (NULL != temp) {
ret = process_int_arg(temp);
}
/* Free */
my_free(temp);
return ret;
}
float get_opt_float_val(t_opt_info* opts, char* opt_name, float default_val) {
char* temp = get_opt_val(opts, opt_name);
float ret = default_val;
if (NULL != temp) {
ret = process_float_arg(temp);
}
/* Free */
my_free(temp);
return ret;
}