#include "parse.h"
#include "debug.h"
+#include "options.h"
static struct fio_option *fio_options;
extern unsigned int fio_get_kb_base(void *);
return 1;
}
+static inline int o_match(struct fio_option *o, const char *opt)
+{
+ if (!strcmp(o->name, opt))
+ return 1;
+ else if (o->alias && !strcmp(o->alias, opt))
+ return 1;
+
+ return 0;
+}
+
static struct fio_option *find_option(struct fio_option *options,
- const char *opt)
+ struct flist_head *eops, const char *opt)
{
+ struct flist_head *n;
struct fio_option *o;
- for (o = &options[0]; o->name; o++) {
- if (!strcmp(o->name, opt))
- return o;
- else if (o->alias && !strcmp(o->alias, opt))
+ for (o = &options[0]; o->name; o++)
+ if (o_match(o, opt))
return o;
+
+ if (!eops)
+ return NULL;
+
+ flist_for_each(n, eops) {
+ struct ext_option *eopt;
+
+ eopt = flist_entry(n, struct ext_option, list);
+ if (o_match(&eopt->o, opt))
+ return &eopt->o;
}
return NULL;
ret = 1;
if (!strncmp(vp->ival, ptr, strlen(vp->ival))) {
ret = 0;
- if (!o->off1)
- break;
- val_store(ilp, vp->oval, o->off1, data);
+ if (o->roff1)
+ *(unsigned int *) o->roff1 = vp->oval;
+ else {
+ if (!o->off1)
+ break;
+ val_store(ilp, vp->oval, o->off1, data);
+ }
break;
}
}
ret = fn(data, &ull);
else {
if (o->type == FIO_OPT_INT) {
- if (first)
- val_store(ilp, ull, o->off1, data);
- if (!more && o->off2)
- val_store(ilp, ull, o->off2, data);
+ if (first) {
+ if (o->roff1)
+ *(unsigned long long *) o->roff1 = ull;
+ else
+ val_store(ilp, ull, o->off1, data);
+ }
+ if (!more) {
+ if (o->roff2)
+ *(unsigned long long *) o->roff2 = ull;
+ else if (o->off2)
+ val_store(ilp, ull, o->off2, data);
+ }
} else {
- if (first)
- val_store(ullp, ull, o->off1, data);
- if (!more && o->off2)
- val_store(ullp, ull, o->off2, data);
+ if (first) {
+ if (o->roff1)
+ *(unsigned long long *) o->roff1 = ull;
+ else
+ val_store(ullp, ull, o->off1, data);
+ }
+ if (!more) {
+ if (o->roff2)
+ *(unsigned long long *) o->roff2 = ull;
+ else if (o->off2)
+ val_store(ullp, ull, o->off2, data);
+ }
}
}
break;
case FIO_OPT_STR_STORE: {
fio_opt_str_fn *fn = o->cb;
- cp = td_var(data, o->off1);
+ if (o->roff1)
+ cp = (char **) o->roff1;
+ else
+ cp = td_var(data, o->off1);
+
*cp = strdup(ptr);
if (fn) {
ret = fn(data, ptr);
}
if (first) {
- val_store(ilp, ul1, o->off1, data);
- val_store(ilp, ul2, o->off2, data);
+ if (o->roff1)
+ *(unsigned long *) o->roff1 = ul1;
+ else
+ val_store(ilp, ul1, o->off1, data);
+ if (o->roff2)
+ *(unsigned long *) o->roff2 = ul2;
+ else
+ val_store(ilp, ul2, o->off2, data);
}
- if (o->off3 && o->off4) {
+ if (o->roff3 && o->roff4) {
+ *(unsigned long *) o->roff3 = ul1;
+ *(unsigned long *) o->roff4 = ul2;
+ } else if (o->off3 && o->off4) {
val_store(ilp, ul1, o->off3, data);
val_store(ilp, ul2, o->off4, data);
}
if (fn)
ret = fn(data, &il);
else {
- if (first)
- val_store(ilp, il, o->off1, data);
- if (!more && o->off2)
- val_store(ilp, il, o->off2, data);
+ if (first) {
+ if (o->roff1)
+ *(unsigned int *)o->roff1 = il;
+ else
+ val_store(ilp, il, o->off1, data);
+ }
+ if (!more) {
+ if (o->roff2)
+ *(unsigned int *) o->roff2 = il;
+ else if (o->off2)
+ val_store(ilp, il, o->off2, data);
+ }
}
break;
}
if (fn)
ret = fn(data);
else {
- if (first)
- val_store(ilp, 1, o->off1, data);
- if (!more && o->off2)
- val_store(ilp, 1, o->off2, data);
+ if (first) {
+ if (o->roff1)
+ *(unsigned int *) o->roff1 = 1;
+ else
+ val_store(ilp, 1, o->off1, data);
+ }
+ if (!more) {
+ if (o->roff2)
+ *(unsigned int *) o->roff2 = 1;
+ else if (o->off2)
+ val_store(ilp, 1, o->off2, data);
+ }
}
break;
}
}
static struct fio_option *get_option(const char *opt,
- struct fio_option *options, char **post)
+ struct fio_option *options,
+ struct flist_head *eops, char **post)
{
struct fio_option *o;
char *ret;
ret = (char *) opt;
(*post)++;
strip_blank_end(ret);
- o = find_option(options, ret);
+ o = find_option(options, eops, ret);
} else {
- o = find_option(options, opt);
+ o = find_option(options, eops, opt);
*post = NULL;
}
s1 = strdup(*((char **) p1));
s2 = strdup(*((char **) p2));
- o1 = get_option(s1, fio_options, &foo);
- o2 = get_option(s2, fio_options, &foo);
+ o1 = get_option(s1, fio_options, NULL, &foo);
+ o2 = get_option(s2, fio_options, NULL, &foo);
prio1 = prio2 = 0;
if (o1)
}
int parse_cmd_option(const char *opt, const char *val,
- struct fio_option *options, void *data)
+ struct fio_option *options, struct flist_head *eops,
+ void *data)
{
struct fio_option *o;
- o = find_option(options, opt);
+ o = find_option(options, eops, opt);
if (!o) {
fprintf(stderr, "Bad option <%s>\n", opt);
return 1;
ssize_t nchr = OPT_LEN_MAX;
size_t envlen;
+ if (strlen(in) + 1 > OPT_LEN_MAX) {
+ fprintf(stderr, "OPT_LEN_MAX (%d) is too small\n", OPT_LEN_MAX);
+ return NULL;
+ }
+
in[OPT_LEN_MAX] = '\0';
strncpy(in, opt, OPT_LEN_MAX);
return strdup(out);
}
-int parse_option(const char *opt, struct fio_option *options, void *data)
+int parse_option(const char *opt, struct fio_option *options,
+ struct flist_head *ext_opt_list, void *data)
{
struct fio_option *o;
char *post, *tmp;
tmp = option_dup_subs(opt);
+ if (!tmp)
+ return 1;
- o = get_option(tmp, options, &post);
+ o = get_option(tmp, options, ext_opt_list, &post);
if (!o) {
fprintf(stderr, "Bad option <%s>\n", tmp);
free(tmp);
handle_option(o, o->def, data);
}
+void option_init(struct fio_option *o)
+{
+ if (o->type == FIO_OPT_DEPRECATED)
+ return;
+ if (o->type == FIO_OPT_BOOL) {
+ o->minval = 0;
+ o->maxval = 1;
+ }
+ if (o->type == FIO_OPT_STR_SET && o->def) {
+ fprintf(stderr, "Option %s: string set option with"
+ " default will always be true\n", o->name);
+ }
+ if (!o->cb && !o->off1) {
+ fprintf(stderr, "Option %s: neither cb nor offset given\n",
+ o->name);
+ }
+ if (o->type == FIO_OPT_STR || o->type == FIO_OPT_STR_STORE)
+ return;
+ if (o->cb && (o->off1 || o->off2 || o->off3 || o->off4)) {
+ fprintf(stderr, "Option %s: both cb and offset given\n",
+ o->name);
+ }
+}
+
/*
* Sanitize the options structure. For now it just sets min/max for bool
* values and whether both callback and offsets are given.
dprint(FD_PARSE, "init options\n");
- for (o = &options[0]; o->name; o++) {
- if (o->type == FIO_OPT_DEPRECATED)
- continue;
- if (o->type == FIO_OPT_BOOL) {
- o->minval = 0;
- o->maxval = 1;
- }
- if (o->type == FIO_OPT_STR_SET && o->def) {
- fprintf(stderr, "Option %s: string set option with"
- " default will always be true\n",
- o->name);
- }
- if (!o->cb && !o->off1) {
- fprintf(stderr, "Option %s: neither cb nor offset"
- " given\n", o->name);
- }
- if (o->type == FIO_OPT_STR || o->type == FIO_OPT_STR_STORE)
- continue;
- if (o->cb && (o->off1 || o->off2 || o->off3 || o->off4)) {
- fprintf(stderr, "Option %s: both cb and offset given\n",
- o->name);
- }
- }
+ for (o = &options[0]; o->name; o++)
+ option_init(o);
}