X-Git-Url: https://git.kernel.dk/?p=fio.git;a=blobdiff_plain;f=parse.c;h=18e8a400c3d85dc90e89a4347f3d390c934c458b;hp=c2c5bf29a51e8dbd0a5fbe99808e7e9ac277dbdc;hb=0dce9bc9b21a8d91aa55a29258dfaeffc57bd466;hpb=833491908a1afd67d27ce79257de3a4d80143d9f diff --git a/parse.c b/parse.c index c2c5bf29..18e8a400 100644 --- a/parse.c +++ b/parse.c @@ -44,24 +44,25 @@ static void posval_sort(struct fio_option *o, struct value_pair *vpmap) qsort(vpmap, entries, sizeof(struct value_pair), vp_cmp); } -static void show_option_range(struct fio_option *o, FILE *out) +static void show_option_range(struct fio_option *o, + int (*logger)(const char *format, ...)) { if (o->type == FIO_OPT_FLOAT_LIST){ if (isnan(o->minfp) && isnan(o->maxfp)) return; - fprintf(out, "%20s: min=%f", "range", o->minfp); + logger("%20s: min=%f", "range", o->minfp); if (!isnan(o->maxfp)) - fprintf(out, ", max=%f", o->maxfp); - fprintf(out, "\n"); + logger(", max=%f", o->maxfp); + logger("\n"); } else { if (!o->minval && !o->maxval) return; - fprintf(out, "%20s: min=%d", "range", o->minval); + logger("%20s: min=%d", "range", o->minval); if (o->maxval) - fprintf(out, ", max=%d", o->maxval); - fprintf(out, "\n"); + logger(", max=%d", o->maxval); + logger("\n"); } } @@ -75,17 +76,17 @@ static void show_option_values(struct fio_option *o) if (!vp->ival) continue; - printf("%20s: %-10s", i == 0 ? "valid values" : "", vp->ival); + log_info("%20s: %-10s", i == 0 ? "valid values" : "", vp->ival); if (vp->help) - printf(" %s", vp->help); - printf("\n"); + log_info(" %s", vp->help); + log_info("\n"); } if (i) - printf("\n"); + log_info("\n"); } -static void show_option_help(struct fio_option *o, FILE *out) +static void show_option_help(struct fio_option *o, int is_err) { const char *typehelp[] = { "invalid", @@ -101,15 +102,21 @@ static void show_option_help(struct fio_option *o, FILE *out) "no argument (opt)", "deprecated", }; + int (*logger)(const char *format, ...); + + if (is_err) + logger = log_err; + else + logger = log_info; if (o->alias) - fprintf(out, "%20s: %s\n", "alias", o->alias); + logger("%20s: %s\n", "alias", o->alias); - fprintf(out, "%20s: %s\n", "type", typehelp[o->type]); - fprintf(out, "%20s: %s\n", "default", o->def ? o->def : "no default"); + logger("%20s: %s\n", "type", typehelp[o->type]); + logger("%20s: %s\n", "default", o->def ? o->def : "no default"); if (o->prof_name) - fprintf(out, "%20s: only for profile '%s'\n", "valid", o->prof_name); - show_option_range(o, stdout); + logger("%20s: only for profile '%s'\n", "valid", o->prof_name); + show_option_range(o, logger); show_option_values(o); } @@ -187,16 +194,19 @@ static unsigned long long get_mult_bytes(const char *str, int len, void *data, int *percent) { const char *p = str; + int digit_seen = 0; if (len < 2) return __get_mult_bytes(str, data, percent); /* - * Go forward until we hit a non-digit + * Go forward until we hit a non-digit, or +/- sign */ while ((p - str) <= len) { - if (!isdigit((int) *p)) + if (!isdigit((int) *p) && + (((*p != '+') && (*p != '-')) || digit_seen)) break; + digit_seen |= isdigit((int) *p); p++; } @@ -263,6 +273,8 @@ void strip_blank_front(char **p) { char *s = *p; + if (!strlen(s)) + return; while (isspace((int) *s)) s++; @@ -273,6 +285,9 @@ void strip_blank_end(char *p) { char *start = p, *s; + if (!strlen(p)) + return; + s = strchr(p, ';'); if (s) *s = '\0'; @@ -346,12 +361,15 @@ static int __handle_option(struct fio_option *o, const char *ptr, void *data, double uf; char **cp; int ret = 0, is_time = 0; + const struct value_pair *vp; + struct value_pair posval[PARSE_MAX_VP]; + int i, all_skipped = 1; dprint(FD_PARSE, "__handle_option=%s, type=%d, ptr=%s\n", o->name, o->type, ptr); if (!ptr && o->type != FIO_OPT_STR_SET && o->type != FIO_OPT_STR) { - fprintf(stderr, "Option %s requires an argument\n", o->name); + log_err("Option %s requires an argument\n", o->name); return 1; } @@ -359,9 +377,6 @@ static int __handle_option(struct fio_option *o, const char *ptr, void *data, case FIO_OPT_STR: case FIO_OPT_STR_MULTI: { fio_opt_str_fn *fn = o->cb; - const struct value_pair *vp; - struct value_pair posval[PARSE_MAX_VP]; - int i, all_skipped = 1; posval_sort(o, posval); @@ -408,12 +423,12 @@ static int __handle_option(struct fio_option *o, const char *ptr, void *data, break; if (o->maxval && ull > o->maxval) { - fprintf(stderr, "max value out of range: %lld" + log_err("max value out of range: %lld" " (%d max)\n", ull, o->maxval); return 1; } if (o->minval && ull < o->minval) { - fprintf(stderr, "min value out of range: %lld" + log_err("min value out of range: %lld" " (%d min)\n", ull, o->minval); return 1; } @@ -459,22 +474,21 @@ static int __handle_option(struct fio_option *o, const char *ptr, void *data, *ilp = ul2; } if (curr >= o->maxlen) { - fprintf(stderr, "the list exceeding max length %d\n", + log_err("the list exceeding max length %d\n", o->maxlen); return 1; } if(!str_to_float(ptr, &uf)){ - fprintf(stderr, "not a floating point value: %s\n", - ptr); + log_err("not a floating point value: %s\n", ptr); return 1; } if (!isnan(o->maxfp) && uf > o->maxfp) { - fprintf(stderr, "value out of range: %f" + log_err("value out of range: %f" " (range max: %f)\n", uf, o->maxfp); return 1; } if (!isnan(o->minfp) && uf < o->minfp) { - fprintf(stderr, "value out of range: %f" + log_err("value out of range: %f" " (range min: %f)\n", uf, o->minfp); return 1; } @@ -487,19 +501,46 @@ static int __handle_option(struct fio_option *o, const char *ptr, void *data, case FIO_OPT_STR_STORE: { fio_opt_str_fn *fn = o->cb; - if (o->roff1) - cp = (char **) o->roff1; - else - cp = td_var(data, o->off1); + posval_sort(o, posval); - *cp = strdup(ptr); - if (fn) { - ret = fn(data, ptr); - if (ret) { - free(*cp); - *cp = NULL; + if (!o->posval[0].ival) { + vp = NULL; + goto match; + } + + ret = 1; + for (i = 0; i < PARSE_MAX_VP; i++) { + vp = &posval[i]; + if (!vp->ival || vp->ival[0] == '\0') + continue; + all_skipped = 0; + if (!strncmp(vp->ival, ptr, opt_len(ptr))) { + char *rest; + + ret = 0; + if (vp->cb) + fn = vp->cb; +match: + if (o->roff1) + cp = (char **) o->roff1; + else + cp = td_var(data, o->off1); + *cp = strdup(ptr); + rest = strstr(*cp, ":"); + if (rest) { + *rest = '\0'; + ptr = rest + 1; + } else if (vp && vp->cb) + ptr = NULL; + break; } } + + if (ret && !all_skipped) + show_option_values(o); + else if (fn && ptr) + ret = fn(data, ptr); + break; } case FIO_OPT_RANGE: { @@ -573,12 +614,12 @@ static int __handle_option(struct fio_option *o, const char *ptr, void *data, break; if (o->maxval && il > (int) o->maxval) { - fprintf(stderr, "max value out of range: %d (%d max)\n", + log_err("max value out of range: %d (%d max)\n", il, o->maxval); return 1; } if (o->minval && il < o->minval) { - fprintf(stderr, "min value out of range: %d (%d min)\n", + log_err("min value out of range: %d (%d min)\n", il, o->minval); return 1; } @@ -605,10 +646,10 @@ static int __handle_option(struct fio_option *o, const char *ptr, void *data, break; } case FIO_OPT_DEPRECATED: - fprintf(stdout, "Option %s is deprecated\n", o->name); + log_info("Option %s is deprecated\n", o->name); break; default: - fprintf(stderr, "Bad option type %u\n", o->type); + log_err("Bad option type %u\n", o->type); ret = 1; } @@ -618,9 +659,9 @@ static int __handle_option(struct fio_option *o, const char *ptr, void *data, if (o->verify) { ret = o->verify(o, data); if (ret) { - fprintf(stderr,"Correct format for offending option\n"); - fprintf(stderr, "%20s: %s\n", o->name, o->help); - show_option_help(o, stderr); + log_err("Correct format for offending option\n"); + log_err("%20s: %s\n", o->name, o->help); + show_option_help(o, 1); } } @@ -746,14 +787,14 @@ int parse_cmd_option(const char *opt, const char *val, o = find_option(options, opt); if (!o) { - fprintf(stderr, "Bad option <%s>\n", opt); + log_err("Bad option <%s>\n", opt); return 1; } if (!handle_option(o, val, data)) return 0; - fprintf(stderr, "fio: failed parsing %s=%s\n", opt, val); + log_err("fio: failed parsing %s=%s\n", opt, val); return 1; } @@ -774,7 +815,7 @@ static char *option_dup_subs(const char *opt) size_t envlen; if (strlen(opt) + 1 > OPT_LEN_MAX) { - fprintf(stderr, "OPT_LEN_MAX (%d) is too small\n", OPT_LEN_MAX); + log_err("OPT_LEN_MAX (%d) is too small\n", OPT_LEN_MAX); return NULL; } @@ -822,7 +863,7 @@ int parse_option(const char *opt, struct fio_option *options, void *data) o = get_option(tmp, options, &post); if (!o) { - fprintf(stderr, "Bad option <%s>\n", tmp); + log_err("Bad option <%s>\n", tmp); free(tmp); return 1; } @@ -832,7 +873,7 @@ int parse_option(const char *opt, struct fio_option *options, void *data) return 0; } - fprintf(stderr, "fio: failed parsing %s\n", opt); + log_err("fio: failed parsing %s\n", opt); free(tmp); return 1; } @@ -906,7 +947,7 @@ static void __print_option(struct fio_option *o, struct fio_option *org, sprintf(p, "%s", o->name); - printf("%-24s: %s\n", name, o->help); + log_info("%-24s: %s\n", name, o->help); } static void print_option(struct fio_option *o) @@ -971,7 +1012,7 @@ int show_cmd_help(struct fio_option *options, const char *name) if (show_all || match) { found = 1; if (match) - printf("%20s: %s\n", o->name, o->help); + log_info("%20s: %s\n", o->name, o->help); if (show_all) { if (!o->parent) print_option(o); @@ -982,24 +1023,24 @@ int show_cmd_help(struct fio_option *options, const char *name) if (!match) continue; - show_option_help(o, stdout); + show_option_help(o, 0); } if (found) return 0; - printf("No such command: %s", name); + log_err("No such command: %s", name); /* * Only print an appropriately close option, one where the edit * distance isn't too big. Otherwise we get crazy matches. */ if (closest && best_dist < 3) { - printf(" - showing closest match\n"); - printf("%20s: %s\n", closest->name, closest->help); - show_option_help(closest, stdout); + log_info(" - showing closest match\n"); + log_info("%20s: %s\n", closest->name, closest->help); + show_option_help(closest, 0); } else - printf("\n"); + log_info("\n"); return 1; } @@ -1031,20 +1072,17 @@ void option_init(struct fio_option *o) o->maxfp = NAN; } if (o->type == FIO_OPT_STR_SET && o->def) { - fprintf(stderr, "Option %s: string set option with" + log_err("Option %s: string set option with" " default will always be true\n", o->name); } - if (!o->cb && (!o->off1 && !o->roff1)) { - fprintf(stderr, "Option %s: neither cb nor offset given\n", - o->name); - } + if (!o->cb && (!o->off1 && !o->roff1)) + log_err("Option %s: neither cb nor offset given\n", o->name); if (o->type == FIO_OPT_STR || o->type == FIO_OPT_STR_STORE || o->type == FIO_OPT_STR_MULTI) return; if (o->cb && ((o->off1 || o->off2 || o->off3 || o->off4) || (o->roff1 || o->roff2 || o->roff3 || o->roff4))) { - fprintf(stderr, "Option %s: both cb and offset given\n", - o->name); + log_err("Option %s: both cb and offset given\n", o->name); } } @@ -1061,3 +1099,22 @@ void options_init(struct fio_option *options) for (o = &options[0]; o->name; o++) option_init(o); } + +void options_free(struct fio_option *options, void *data) +{ + struct fio_option *o; + char **ptr; + + dprint(FD_PARSE, "free options\n"); + + for (o = &options[0]; o->name; o++) { + if (o->type != FIO_OPT_STR_STORE) + continue; + + ptr = td_var(data, o->off1); + if (*ptr) { + free(*ptr); + *ptr = NULL; + } + } +}