1
/* Copyright (C) 2002-2006 MySQL AB
3
This program is free software; you can redistribute it and/or modify
4
it under the terms of the GNU General Public License as published by
5
the Free Software Foundation; version 2 of the License.
7
This program is distributed in the hope that it will be useful,
8
but WITHOUT ANY WARRANTY; without even the implied warranty of
9
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10
GNU General Public License for more details.
12
You should have received a copy of the GNU General Public License
13
along with this program; if not, write to the Free Software
14
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
16
#include "mysys_priv.h"
17
#include <drizzled/gettext.h>
19
#include <mystrings/m_string.h>
20
#include <mysys/my_sys.h>
21
#include <mysys/mysys_err.h>
22
#include <mysys/my_getopt.h>
31
typedef void (*init_func_p)(const struct my_option *option, char **variable,
34
void default_reporter(enum loglevel level, const char *format, ...);
35
my_error_reporter my_getopt_error_reporter= &default_reporter;
37
static int findopt(char *optpat, uint32_t length,
38
const struct my_option **opt_res,
40
static int64_t getopt_ll(char *arg, const struct my_option *optp, int *err);
41
static uint64_t getopt_ull(char *arg, const struct my_option *optp,
43
static size_t getopt_size(char *arg, const struct my_option *optp, int *err);
44
static double getopt_double(char *arg, const struct my_option *optp, int *err);
45
static void init_variables(const struct my_option *options,
46
init_func_p init_one_value);
47
static void init_one_value(const struct my_option *option, char **variable,
49
static void fini_one_value(const struct my_option *option, char **variable,
51
static int setval(const struct my_option *opts, char* *value, char *argument,
52
bool set_maximum_value);
53
static char *check_struct_option(char *cur_arg, char *key_name);
56
The following three variables belong to same group and the number and
57
order of their arguments must correspond to each other.
59
static const char *special_opt_prefix[]=
60
{"skip", "disable", "enable", "maximum", "loose", 0};
61
static const uint32_t special_opt_prefix_lengths[]=
64
{ OPT_SKIP, OPT_DISABLE, OPT_ENABLE, OPT_MAXIMUM, OPT_LOOSE};
66
char *disabled_my_option= (char*) "0";
69
This is a flag that can be set in client programs. 0 means that
70
my_getopt will not print error messages, but the client should do
74
bool my_getopt_print_errors= 1;
77
This is a flag that can be set in client programs. 1 means that
78
my_getopt will skip over options it does not know how to handle.
81
bool my_getopt_skip_unknown= 0;
83
void default_reporter(enum loglevel level,
84
const char *format, ...)
87
va_start(args, format);
88
if (level == WARNING_LEVEL)
89
fprintf(stderr, "%s", _("Warning: "));
90
else if (level == INFORMATION_LEVEL)
91
fprintf(stderr, "%s", _("Info: "));
92
vfprintf(stderr, format, args);
99
function: handle_options
101
Sort options; put options first, until special end of options (--), or
102
until end of argv. Parse options; check that the given option matches with
103
one of the options in struct 'my_option', return error in case of ambiguous
104
or unknown option. Check that option was given an argument if it requires
105
one. Call function 'get_one_option()' once for each option.
108
static getopt_get_addr_func getopt_get_addr;
110
void my_getopt_register_get_addr(getopt_get_addr_func func_addr)
112
getopt_get_addr= func_addr;
115
int handle_options(int *argc, char ***argv,
116
const struct my_option *longopts,
117
my_get_one_option get_one_option)
119
uint32_t opt_found, argvpos= 0, length;
120
bool end_of_options= 0, must_be_var, set_maximum_value=false,
122
char **pos, **pos_end, *optend, *prev_found=NULL,
123
*opt_str, key_name[FN_REFLEN];
124
const struct my_option *optp;
128
/* handle_options() assumes arg0 (program name) always exists */
129
assert(argc && *argc >= 1);
130
assert(argv && *argv);
131
(*argc)--; /* Skip the program name */
132
(*argv)++; /* --- || ---- */
133
init_variables(longopts, init_one_value);
135
for (pos= *argv, pos_end=pos+ *argc; pos != pos_end ; pos++)
139
if (cur_arg[0] == '-' && cur_arg[1] && !end_of_options) /* must be opt */
143
set_maximum_value= 0;
146
cur_arg++; /* skip '-' */
147
if (*cur_arg == '-' || *cur_arg == 'O') /* check for long option, */
148
{ /* --set-variable, or -O */
153
if (!(*++cur_arg)) /* If not -Ovar=# */
155
/* the argument must be in next argv */
158
if (my_getopt_print_errors)
159
my_getopt_error_reporter(ERROR_LEVEL,
160
"%s: Option '-O' requires an argument",
162
return EXIT_ARGUMENT_REQUIRED;
168
else if (!getopt_compare_strings(cur_arg, "-set-variable", 13))
171
if (cur_arg[13] == '=')
176
if (my_getopt_print_errors)
177
my_getopt_error_reporter(ERROR_LEVEL,
178
"%s: Option '--set-variable' requires an argument",
180
return EXIT_ARGUMENT_REQUIRED;
183
else if (cur_arg[14]) /* garbage, or another option. break out */
187
/* the argument must be in next argv */
190
if (my_getopt_print_errors)
191
my_getopt_error_reporter(ERROR_LEVEL,
192
"%s: Option '--set-variable' requires an argument",
194
return EXIT_ARGUMENT_REQUIRED;
200
else if (!must_be_var)
202
if (!*++cur_arg) /* skip the double dash */
204
/* '--' means end of options, look no further */
210
opt_str= check_struct_option(cur_arg, key_name);
211
optend= strrchr(opt_str, '=');
214
length= (uint) (optend - opt_str);
219
length= strlen(opt_str);
224
Find first the right option. Return error in case of an ambiguous,
228
if (!(opt_found= findopt(opt_str, length, &optp, &prev_found)))
231
Didn't find any matching option. Let's see if someone called
232
option with a special option prefix
237
must_be_var= 1; /* option is followed by an argument */
238
for (i= 0; special_opt_prefix[i]; i++)
240
if (!getopt_compare_strings(special_opt_prefix[i], opt_str,
241
special_opt_prefix_lengths[i]) &&
242
(opt_str[special_opt_prefix_lengths[i]] == '-' ||
243
opt_str[special_opt_prefix_lengths[i]] == '_'))
246
We were called with a special prefix, we can reuse opt_found
248
opt_str+= special_opt_prefix_lengths[i] + 1;
249
length-= special_opt_prefix_lengths[i] + 1;
252
if ((opt_found= findopt(opt_str, length, &optp, &prev_found)))
256
if (my_getopt_print_errors)
257
my_getopt_error_reporter(ERROR_LEVEL,
258
"%s: ambiguous option '--%s-%s' (--%s-%s)",
259
my_progname, special_opt_prefix[i],
260
cur_arg, special_opt_prefix[i],
262
return EXIT_AMBIGUOUS_OPTION;
266
case OPT_DISABLE: /* fall through */
268
double negation is actually enable again,
269
for example: --skip-option=0 -> option = true
271
optend= (optend && *optend == '0' && !(*(optend + 1))) ?
272
(char*) "1" : disabled_my_option;
275
optend= (optend && *optend == '0' && !(*(optend + 1))) ?
276
disabled_my_option : (char*) "1";
279
set_maximum_value= true;
283
break; /* break from the inner loop, main loop continues */
285
i= -1; /* restart the loop */
291
if (my_getopt_skip_unknown)
294
preserve all the components of this unknown option, this may
295
occurr when the user provides options like: "-O foo" or
296
"--set-variable foo" (note that theres a space in there)
297
Generally, these kind of options are to be avoided
300
(*argv)[argvpos++]= *first++;
301
} while (first <= pos);
306
if (my_getopt_print_errors)
307
my_getopt_error_reporter(option_is_loose ?
308
WARNING_LEVEL : ERROR_LEVEL,
309
"%s: unknown variable '%s'",
310
my_progname, cur_arg);
311
if (!option_is_loose)
312
return EXIT_UNKNOWN_VARIABLE;
316
if (my_getopt_print_errors)
317
my_getopt_error_reporter(option_is_loose ?
318
WARNING_LEVEL : ERROR_LEVEL,
319
"%s: unknown option '--%s'",
320
my_progname, cur_arg);
321
if (!option_is_loose)
322
return EXIT_UNKNOWN_OPTION;
335
if (my_getopt_print_errors)
336
my_getopt_error_reporter(ERROR_LEVEL,
337
"%s: variable prefix '%s' is not unique",
338
my_progname, opt_str);
339
return EXIT_VAR_PREFIX_NOT_UNIQUE;
343
if (my_getopt_print_errors)
344
my_getopt_error_reporter(ERROR_LEVEL,
345
"%s: ambiguous option '--%s' (%s, %s)",
346
my_progname, opt_str, prev_found,
348
return EXIT_AMBIGUOUS_OPTION;
351
if ((optp->var_type & GET_TYPE_MASK) == GET_DISABLED)
353
if (my_getopt_print_errors)
355
_("%s: %s: Option '%s' used, but is disabled\n"), my_progname,
356
option_is_loose ? _("WARNING") : _("ERROR"), opt_str);
362
return EXIT_OPTION_DISABLED;
364
if (must_be_var && (optp->var_type & GET_TYPE_MASK) == GET_NO_ARG)
366
if (my_getopt_print_errors)
367
my_getopt_error_reporter(ERROR_LEVEL,
368
"%s: option '%s' cannot take an argument",
369
my_progname, optp->name);
370
return EXIT_NO_ARGUMENT_ALLOWED;
372
value= optp->var_type & GET_ASK_ADDR ?
373
(*getopt_get_addr)(key_name, (uint) strlen(key_name), optp) : optp->value;
375
if (optp->arg_type == NO_ARG)
377
if (optend && (optp->var_type & GET_TYPE_MASK) != GET_BOOL)
379
if (my_getopt_print_errors)
380
my_getopt_error_reporter(ERROR_LEVEL,
381
"%s: option '--%s' cannot take an argument",
382
my_progname, optp->name);
383
return EXIT_NO_ARGUMENT_ALLOWED;
385
if ((optp->var_type & GET_TYPE_MASK) == GET_BOOL)
388
Set bool to 1 if no argument or if the user has used
389
--enable-'option-name'.
390
*optend was set to '0' if one used --disable-option
393
if (!optend || *optend == '1' ||
394
!my_strcasecmp(&my_charset_utf8_general_ci, optend, "true"))
395
*((bool*) value)= (bool) 1;
396
else if (*optend == '0' ||
397
!my_strcasecmp(&my_charset_utf8_general_ci, optend, "false"))
398
*((bool*) value)= (bool) 0;
401
my_getopt_error_reporter(WARNING_LEVEL,
402
"%s: ignoring option '--%s' due to \
404
my_progname, optp->name, optend);
407
get_one_option(optp->id, optp,
409
(char*) "1" : disabled_my_option);
414
else if (optp->arg_type == OPT_ARG &&
415
(optp->var_type & GET_TYPE_MASK) == GET_BOOL)
417
if (optend == disabled_my_option)
418
*((bool*) value)= (bool) 0;
421
if (!optend) /* No argument -> enable option */
422
*((bool*) value)= (bool) 1;
427
else if (optp->arg_type == REQUIRED_ARG && !optend)
429
/* Check if there are more arguments after this one */
432
if (my_getopt_print_errors)
433
my_getopt_error_reporter(ERROR_LEVEL,
434
"%s: option '--%s' requires an argument",
435
my_progname, optp->name);
436
return EXIT_ARGUMENT_REQUIRED;
444
else /* must be short option */
446
for (optend= cur_arg; *optend; optend++)
449
for (optp= longopts; optp->id; optp++)
451
if (optp->id == (int) (unsigned char) *optend)
453
/* Option recognized. Find next what to do with it */
455
if ((optp->var_type & GET_TYPE_MASK) == GET_DISABLED)
457
if (my_getopt_print_errors)
459
_("%s: ERROR: Option '-%c' used, but is disabled\n"),
460
my_progname, optp->id);
461
return EXIT_OPTION_DISABLED;
463
if ((optp->var_type & GET_TYPE_MASK) == GET_BOOL &&
464
optp->arg_type == NO_ARG)
466
*((bool*) optp->value)= (bool) 1;
467
get_one_option(optp->id, optp, argument);
470
else if (optp->arg_type == REQUIRED_ARG ||
471
optp->arg_type == OPT_ARG)
475
/* The rest of the option is option argument */
476
argument= optend + 1;
477
/* This is in effect a jump out of the outer loop */
482
if (optp->arg_type == OPT_ARG)
484
if (optp->var_type == GET_BOOL)
485
*((bool*) optp->value)= (bool) 1;
486
get_one_option(optp->id, optp, argument);
489
/* Check if there are more arguments after this one */
492
if (my_getopt_print_errors)
493
my_getopt_error_reporter(ERROR_LEVEL,
494
"%s: option '-%c' requires an argument",
495
my_progname, optp->id);
496
return EXIT_ARGUMENT_REQUIRED;
500
/* the other loop will break, because *optend + 1 == 0 */
503
if ((error= setval(optp, optp->value, argument,
506
my_getopt_error_reporter(ERROR_LEVEL,
507
"%s: Error while setting value '%s' to '%s'",
508
my_progname, argument, optp->name);
511
get_one_option(optp->id, optp, argument);
517
if (my_getopt_print_errors)
518
my_getopt_error_reporter(ERROR_LEVEL,
519
"%s: unknown option '-%c'",
520
my_progname, *optend);
521
return EXIT_UNKNOWN_OPTION;
524
(*argc)--; /* option handled (short), decrease argument count */
527
if ((error= setval(optp, value, argument, set_maximum_value)))
529
my_getopt_error_reporter(ERROR_LEVEL,
530
"%s: Error while setting value '%s' to '%s'",
531
my_progname, argument, optp->name);
534
get_one_option(optp->id, optp, argument);
536
(*argc)--; /* option handled (short or long), decrease argument count */
538
else /* non-option found */
539
(*argv)[argvpos++]= cur_arg;
542
Destroy the first, already handled option, so that programs that look
543
for arguments in 'argv', without checking 'argc', know when to stop.
544
Items in argv, before the destroyed one, are all non-option -arguments
545
to the program, yet to be (possibly) handled.
553
function: check_struct_option
555
Arguments: Current argument under processing from argv and a variable
556
where to store the possible key name.
558
Return value: In case option is a struct option, returns a pointer to
559
the current argument at the position where the struct option (key_name)
560
ends, the next character after the dot. In case argument is not a struct
561
option, returns a pointer to the argument.
563
key_name will hold the name of the key, or 0 if not found.
566
static char *check_struct_option(char *cur_arg, char *key_name)
570
ptr= strrchr(cur_arg + 1, '.'); /* Skip the first character */
571
end= strrchr(cur_arg, '=');
574
If the first dot is after an equal sign, then it is part
575
of a variable value and the option is not a struct option.
576
Also, if the last character in the string before the ending
577
NULL, or the character right before equal sign is the first
578
dot found, the option is not a struct option.
580
if ((ptr != NULL) && (end != NULL) && (end - ptr > 1))
582
uint32_t len= (uint) (ptr - cur_arg);
583
set_if_smaller(len, FN_REFLEN-1);
584
strncpy(key_name, cur_arg, len);
594
Arguments: opts, argument
595
Will set the option value to given value
598
static int setval(const struct my_option *opts, char **value, char *argument,
599
bool set_maximum_value)
603
if (value && argument)
605
char* *result_pos= ((set_maximum_value) ?
606
opts->u_max_value : value);
609
return EXIT_NO_PTR_TO_VARIABLE;
611
switch ((opts->var_type & GET_TYPE_MASK)) {
612
case GET_BOOL: /* If argument differs from 0, enable option, else disable */
613
*((bool*) result_pos)= (bool) atoi(argument) != 0;
616
*((int32_t*) result_pos)= (int) getopt_ll(argument, opts, &err);
619
*((uint32_t*) result_pos)= (uint) getopt_ull(argument, opts, &err);
622
*((long*) result_pos)= (long) getopt_ll(argument, opts, &err);
625
*((long*) result_pos)= (long) getopt_ull(argument, opts, &err);
628
*((int64_t*) result_pos)= getopt_ll(argument, opts, &err);
631
*((uint64_t*) result_pos)= getopt_ull(argument, opts, &err);
634
*((size_t*) result_pos)= getopt_size(argument, opts, &err);
637
*((double*) result_pos)= getopt_double(argument, opts, &err);
640
*((char**) result_pos)= argument;
643
if ((*((char**) result_pos)))
644
free((*(char**) result_pos));
645
if (!(*((char**) result_pos)= strdup(argument)))
646
return EXIT_OUT_OF_MEMORY;
649
if (((*(int*)result_pos)= find_type(argument, opts->typelib, 2) - 1) < 0)
650
return EXIT_ARGUMENT_INVALID;
653
*((uint64_t*)result_pos)= find_typeset(argument, opts->typelib, &err);
655
return EXIT_ARGUMENT_INVALID;
657
default: /* dummy default to avoid compiler warnings */
661
return EXIT_UNKNOWN_SUFFIX;
672
optpat Prefix of option to find (with - or _)
673
length Length of optpat
675
ffname Place for pointer to first found name
678
Go through all options in the my_option struct. Return number
679
of options found that match the pattern and in the argument
680
list the option found, if any. In case of ambiguous option, store
681
the name in ffname argument
684
0 No matching options
685
# Number of matching options
686
ffname points to first matching option
689
static int findopt(char *optpat, uint32_t length,
690
const struct my_option **opt_res,
694
struct my_option *opt= (struct my_option *) *opt_res;
696
for (count= 0; opt->name; opt++)
698
if (!getopt_compare_strings(opt->name, optpat, length)) /* match found */
701
if (!opt->name[length]) /* Exact match */
706
*ffname= (char *) opt->name; /* We only need to know one prev */
708
else if (strcmp(*ffname, opt->name))
711
The above test is to not count same option twice
712
(see mysql.cc, option "help")
723
function: compare_strings
725
Works like strncmp, other than 1.) considers '-' and '_' the same.
726
2.) Returns -1 if strings differ, 0 if they are equal
729
bool getopt_compare_strings(register const char *s, register const char *t,
732
char const *end= s + length;
733
for (;s != end ; s++, t++)
735
if ((*s != '-' ? *s : '_') != (*t != '-' ? *t : '_'))
742
function: eval_num_suffix
744
Transforms a number with a suffix to real number. Suffix can
745
be k|K for kilo, m|M for mega or g|G for giga.
748
static int64_t eval_num_suffix(char *argument, int *error, char *option_name)
755
num= strtoll(argument, &endchar, 10);
758
my_getopt_error_reporter(ERROR_LEVEL,
759
"Incorrect integer value: '%s'", argument);
763
if (*endchar == 'k' || *endchar == 'K')
765
else if (*endchar == 'm' || *endchar == 'M')
767
else if (*endchar == 'g' || *endchar == 'G')
768
num*= 1024L * 1024L * 1024L;
772
_("Unknown suffix '%c' used for variable '%s' (value '%s')\n"),
773
*endchar, option_name, argument);
783
Evaluates and returns the value that user gave as an argument
784
to a variable. Recognizes (case insensitive) K as KILO, M as MEGA
785
and G as GIGA bytes. Some values must be in certain blocks, as
786
defined in the given my_option struct, this function will check
787
that those values are honored.
788
In case of an error, set error value in *err.
791
static int64_t getopt_ll(char *arg, const struct my_option *optp, int *err)
793
int64_t num=eval_num_suffix(arg, err, (char*) optp->name);
794
return getopt_ll_limit_value(num, optp, NULL);
798
function: getopt_ll_limit_value
800
Applies min/max/block_size to a numeric value of an option.
801
Returns "fixed" value.
804
int64_t getopt_ll_limit_value(int64_t num, const struct my_option *optp,
808
bool adjusted= false;
809
char buf1[255], buf2[255];
810
uint64_t block_size= (optp->block_size ? (uint64_t) optp->block_size : 1L);
812
if (num > 0 && ((uint64_t) num > (uint64_t) optp->max_value) &&
813
optp->max_value) /* if max value is not set -> no upper limit */
815
num= (uint64_t) optp->max_value;
819
switch ((optp->var_type & GET_TYPE_MASK)) {
821
if (num > (int64_t) INT_MAX)
823
num= ((int64_t) INT_MAX);
828
if (num > (int64_t) INT32_MAX)
830
num= ((int64_t) INT32_MAX);
835
assert((optp->var_type & GET_TYPE_MASK) == GET_LL);
839
num= ((num - optp->sub_size) / block_size);
840
num= (int64_t) (num * block_size);
842
if (num < optp->min_value)
844
num= optp->min_value;
851
my_getopt_error_reporter(WARNING_LEVEL,
852
"option '%s': signed value %s adjusted to %s",
853
optp->name, llstr(old, buf1), llstr(num, buf2));
860
This is the same as getopt_ll, but is meant for uint64_t
864
static uint64_t getopt_ull(char *arg, const struct my_option *optp, int *err)
866
uint64_t num= eval_num_suffix(arg, err, (char*) optp->name);
867
return getopt_ull_limit_value(num, optp, NULL);
871
static size_t getopt_size(char *arg, const struct my_option *optp, int *err)
873
return (size_t)getopt_ull(arg, optp, err);
878
uint64_t getopt_ull_limit_value(uint64_t num, const struct my_option *optp,
881
bool adjusted= false;
883
char buf1[255], buf2[255];
885
if ((uint64_t) num > (uint64_t) optp->max_value &&
886
optp->max_value) /* if max value is not set -> no upper limit */
888
num= (uint64_t) optp->max_value;
892
switch ((optp->var_type & GET_TYPE_MASK)) {
894
if (num > (uint64_t) UINT_MAX)
896
num= ((uint64_t) UINT_MAX);
901
if (num > (uint64_t) UINT32_MAX)
903
num= ((uint64_t) UINT32_MAX);
908
if (num > (uint64_t) SIZE_MAX)
910
num= ((uint64_t) SIZE_MAX);
915
assert((optp->var_type & GET_TYPE_MASK) == GET_ULL);
919
if (optp->block_size > 1)
921
num/= (uint64_t) optp->block_size;
922
num*= (uint64_t) optp->block_size;
925
if (num < (uint64_t) optp->min_value)
927
num= (uint64_t) optp->min_value;
934
my_getopt_error_reporter(WARNING_LEVEL,
935
"option '%s': unsigned value %s adjusted to %s",
936
optp->name, ullstr(old, buf1), ullstr(num, buf2));
943
Get double value withing ranges
945
Evaluates and returns the value that user gave as an argument to a variable.
950
In case of an error, prints an error message and sets *err to
951
EXIT_ARGUMENT_INVALID. Otherwise err is not touched
954
static double getopt_double(char *arg, const struct my_option *optp, int *err)
958
char *end= arg + 1000; /* Big enough as *arg is \0 terminated */
959
num= my_strtod(arg, &end, &error);
960
if (end[0] != 0 || error)
963
_("%s: ERROR: Invalid decimal value for option '%s'\n"),
964
my_progname, optp->name);
965
*err= EXIT_ARGUMENT_INVALID;
968
if (optp->max_value && num > (double) optp->max_value)
969
num= (double) optp->max_value;
970
return cmax(num, (double) optp->min_value);
974
Init one value to it's default values
978
option Option to initialize
979
value Pointer to variable
982
static void init_one_value(const struct my_option *option, char** variable,
985
switch ((option->var_type & GET_TYPE_MASK)) {
987
*((bool*) variable)= (bool) value;
990
*((int*) variable)= (int) value;
994
*((uint*) variable)= (uint) value;
997
*((long*) variable)= (long) value;
1000
*((uint32_t*) variable)= (uint32_t) value;
1003
*((int64_t*) variable)= (int64_t) value;
1006
*((size_t*) variable)= (size_t) value;
1009
*((uint64_t*) variable)= (uint64_t) value;
1012
*((double*) variable)= (double) value;
1016
Do not clear variable value if it has no default value.
1017
The default value may already be set.
1018
NOTE: To avoid compiler warnings, we first cast int64_t to intptr_t,
1019
so that the value has the same size as a pointer.
1021
if ((char*) (intptr_t) value)
1022
*((char**) variable)= (char*) (intptr_t) value;
1026
Do not clear variable value if it has no default value.
1027
The default value may already be set.
1028
NOTE: To avoid compiler warnings, we first cast int64_t to intptr_t,
1029
so that the value has the same size as a pointer.
1031
if ((char*) (intptr_t) value)
1033
free((*(char**) variable));
1034
char *tmpptr= strdup((char *) (intptr_t) value);
1036
*((char**) variable)= tmpptr;
1039
default: /* dummy default to avoid compiler warnings */
1047
Init one value to it's default values
1051
option Option to initialize
1052
value Pointer to variable
1055
static void fini_one_value(const struct my_option *option, char **variable,
1058
switch ((option->var_type & GET_TYPE_MASK)) {
1060
free((*(char**) variable));
1061
*((char**) variable)= NULL;
1063
default: /* dummy default to avoid compiler warnings */
1070
void my_cleanup_options(const struct my_option *options)
1072
init_variables(options, fini_one_value);
1077
initialize all variables to their default values
1081
options Array of options
1084
We will initialize the value that is pointed to by options->value.
1085
If the value is of type GET_ASK_ADDR, we will also ask for the address
1086
for a value and initialize.
1089
static void init_variables(const struct my_option *options,
1090
init_func_p init_one_value)
1092
for (; options->name; options++)
1096
We must set u_max_value first as for some variables
1097
options->u_max_value == options->value and in this case we want to
1098
set the value to default value.
1100
if (options->u_max_value)
1101
init_one_value(options, options->u_max_value, options->max_value);
1103
init_one_value(options, options->value, options->def_value);
1104
if (options->var_type & GET_ASK_ADDR &&
1105
(variable= (*getopt_get_addr)("", 0, options)))
1106
init_one_value(options, variable, options->def_value);
1113
function: my_print_options
1115
Print help for all options and variables.
1118
void my_print_help(const struct my_option *options)
1120
uint32_t col, name_space= 22, comment_space= 57;
1121
const char *line_end;
1122
const struct my_option *optp;
1124
for (optp= options; optp->id; optp++)
1128
printf(" -%c%s", optp->id, strlen(optp->name) ? ", " : " ");
1136
if (strlen(optp->name))
1138
printf("--%s", optp->name);
1139
col+= 2 + (uint) strlen(optp->name);
1140
if ((optp->var_type & GET_TYPE_MASK) == GET_STR ||
1141
(optp->var_type & GET_TYPE_MASK) == GET_STR_ALLOC)
1143
printf("%s=name%s ", optp->arg_type == OPT_ARG ? "[" : "",
1144
optp->arg_type == OPT_ARG ? "]" : "");
1145
col+= (optp->arg_type == OPT_ARG) ? 8 : 6;
1147
else if ((optp->var_type & GET_TYPE_MASK) == GET_NO_ARG ||
1148
(optp->var_type & GET_TYPE_MASK) == GET_BOOL)
1155
printf("%s=#%s ", optp->arg_type == OPT_ARG ? "[" : "",
1156
optp->arg_type == OPT_ARG ? "]" : "");
1157
col+= (optp->arg_type == OPT_ARG) ? 5 : 3;
1159
if (col > name_space && optp->comment && *optp->comment)
1165
for (; col < name_space; col++)
1167
if (optp->comment && *optp->comment)
1169
const char *comment= _(optp->comment), *end= strchr(comment, '\0');
1171
while ((uint) (end - comment) > comment_space)
1173
for (line_end= comment + comment_space; *line_end != ' '; line_end--)
1175
for (; comment != line_end; comment++)
1177
comment++; /* skip the space, as a newline will take it's place now */
1179
for (col= 0; col < name_space; col++)
1182
printf("%s", comment);
1185
if ((optp->var_type & GET_TYPE_MASK) == GET_NO_ARG ||
1186
(optp->var_type & GET_TYPE_MASK) == GET_BOOL)
1188
if (optp->def_value != 0)
1190
printf(_("%*s(Defaults to on; use --skip-%s to disable.)\n"), name_space, "", optp->name);
1198
function: my_print_options
1203
void my_print_variables(const struct my_option *options)
1205
uint32_t name_space= 34, length, nr;
1206
uint64_t bit, llvalue;
1208
const struct my_option *optp;
1210
printf(_("\nVariables (--variable-name=value)\n"
1211
"and boolean options {false|true} Value (after reading options)\n"
1212
"--------------------------------- -----------------------------\n"));
1213
for (optp= options; optp->id; optp++)
1215
char* *value= (optp->var_type & GET_ASK_ADDR ?
1216
(*getopt_get_addr)("", 0, optp) : optp->value);
1219
printf("%s ", optp->name);
1220
length= (uint) strlen(optp->name)+1;
1221
for (; length < name_space; length++)
1223
switch ((optp->var_type & GET_TYPE_MASK)) {
1225
if (!(llvalue= *(uint64_t*) value))
1226
printf("%s\n", _("(No default value)"));
1228
for (nr= 0, bit= 1; llvalue && nr < optp->typelib->count; nr++, bit<<=1)
1230
if (!(bit & llvalue))
1233
printf( llvalue ? "%s," : "%s\n", get_type(optp->typelib, nr));
1237
printf("%s\n", get_type(optp->typelib, *(uint*) value));
1240
case GET_STR_ALLOC: /* fall through */
1241
printf("%s\n", *((char**) value) ? *((char**) value) :
1242
_("(No default value)"));
1245
printf("%s\n", *((bool*) value) ? _("true") : _("false"));
1248
printf("%d\n", *((int*) value));
1251
printf("%d\n", *((uint*) value));
1254
printf("%ld\n", *((long*) value));
1257
printf("%u\n", *((uint32_t*) value));
1263
printf("%s\n", llstr(*((int64_t*) value), buff));
1266
int64_t2str(*((uint64_t*) value), buff, 10);
1267
printf("%s\n", buff);
1270
printf("%g\n", *(double*) value);
1273
printf(_("(Disabled)\n"));