* Copyright (C) 2004-2010 NVIDIA Corporation
*
* Permission is hereby granted, free of charge, to any person obtaining a
* copy of this software and associated documentation files (the "Software"),
* to deal in the Software without restriction, including without limitation
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
* and/or sell copies of the Software, and to permit persons to whom the
* Software is furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
* DEALINGS IN THE SOFTWARE.
*
*
* nvgetopt.c - portable getopt_long() replacement; removes the need
* for the stupid optstring argument.
*/
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <ctype.h>
#include <limits.h>
#include "nvgetopt.h"
#include "common-utils.h"
int nvgetopt(int argc,
char *argv[],
const NVGetoptOption *options,
char **strval,
int *boolval,
int *intval,
double *doubleval,
int *disable_val)
{
char *c, *a, *arg, *name = NULL, *argument=NULL;
int i, found = NVGETOPT_FALSE;
int ret = 0;
int negate = NVGETOPT_FALSE;
int disable = NVGETOPT_FALSE;
int double_dash = NVGETOPT_FALSE;
const NVGetoptOption *o = NULL;
static int argv_index = 0;
if (strval) *strval = NULL;
if (boolval) *boolval = NVGETOPT_FALSE;
if (intval) *intval = 0;
if (doubleval) *doubleval = 0.0;
if (disable_val) *disable_val = NVGETOPT_FALSE;
argv_index++;
if (argv_index >= argc) return -1;
arg = strdup(argv[argv_index]);
if ((arg[0] == '-') && (arg[1] == '-')) {
name = arg + 2;
double_dash = NVGETOPT_TRUE;
} else if (arg[0] == '-') {
name = arg + 1;
} else {
fprintf(stderr, "%s: invalid option: \"%s\"\n", argv[0], arg);
goto done;
}
* if there is an "=" in the string, then assign argument and zero
* out the equal sign so that name will match what is in the
* option table.
*/
c = name;
while (*c) {
if (*c == '=') {
argument = c + 1;
*c = '\0';
break;
}
c++;
}
* if there is no character after '--' then stop processing options.
* if the string is terminated after one character, interpret it
* as a short option. Otherwise, interpret it as a long option.
*/
if (name[0] == '\0') {
if (double_dash && argument == NULL) {
ret = -1;
goto done;
}
} else if (name[1] == '\0') {
for (i = 0; options[i].name; i++) {
if (options[i].val == name[0]) {
o = &options[i];
break;
}
}
} else {
for (i = 0; options[i].name; i++) {
const char *tmpname;
int tmp_negate;
* if this option allows negation by prepending with
* "--no-" (true for IS_BOOLEAN and ALLOW_DISABLE), then
* skip any leading "no-" in the argument
*/
if ((options[i].flags & (NVGETOPT_IS_BOOLEAN |
NVGETOPT_ALLOW_DISABLE)) &&
(name[0] == 'n') &&
(name[1] == 'o') &&
(name[2] == '-')) {
tmpname = name + 3;
tmp_negate = NVGETOPT_TRUE;
} else {
tmpname = name;
tmp_negate = NVGETOPT_FALSE;
}
if (strcmp(tmpname, options[i].name) == 0) {
o = &options[i];
negate = tmp_negate;
break;
}
}
}
* if we didn't find a match, maybe this is multiple short options
* stored together; is each character a short option?
*/
if (!o) {
for (c = name; *c; c++) {
found = NVGETOPT_FALSE;
for (i = 0; options[i].name; i++) {
if (options[i].val == *c) {
found = NVGETOPT_TRUE;
break;
}
}
if (!found) break;
}
if (found) {
* all characters individually are short options, so
* interpret them that way
*/
for (i = 0; options[i].name; i++) {
if (options[i].val == name[0]) {
* don't allow options with arguments to be
* processed in this way
*/
if (options[i].flags & NVGETOPT_HAS_ARGUMENT) break;
* remove the first short option from
* argv[argv_index]
*/
a = argv[argv_index];
if (a[0] == '-') a++;
if (a[0] == '-') a++;
if (a[0] == '+') a++;
while (a[0]) { a[0] = a[1]; a++; }
* decrement argv_index so that we process this
* entry again
*/
argv_index--;
o = &options[i];
break;
}
}
}
}
* If we still didn't find a match, maybe this is a short option
* with its argument value concatenated (e.g., "-j8"). For now,
* limit this to short options with integer argument values.
*/
if (!o && intval) {
int appendedInteger = NVGETOPT_FALSE;
if ((name[0] != '\0') && (name[1] != '\0')) {
char *endptr;
strtol(name + 1, &endptr, 0);
if (*endptr == '\0') {
* The only characters after the first character are
* parsable by strtol(3).
*/
appendedInteger = NVGETOPT_TRUE;
}
}
if (appendedInteger) {
for (i = 0; options[i].name; i++) {
if ((options[i].flags & NVGETOPT_INTEGER_ARGUMENT) == 0) {
continue;
}
if (options[i].val == name[0]) {
o = &options[i];
argument = name + 1;
break;
}
}
}
}
if (!o) {
fprintf(stderr, "%s: unrecognized option: \"%s\"\n", argv[0], arg);
goto done;
}
if (o->flags & NVGETOPT_IS_BOOLEAN) {
if (boolval) *boolval = !negate;
}
* if this option is flagged as "disable-able", then let the
* "--no-" prefix get interpreted to mean that the option should
* be disabled
*/
if ((o->flags & NVGETOPT_ALLOW_DISABLE) && (negate == NVGETOPT_TRUE)) {
disable = NVGETOPT_TRUE;
}
* if the option takes an argument (either string or integer), and
* we haven't already decided to disable the option, then we
* either need to use what was after the "=" in this argv[] entry,
* or we need to pull the next entry off of argv[]
*/
if ((o->flags & NVGETOPT_HAS_ARGUMENT) && !disable) {
if (argument) {
if (!argument[0]) {
fprintf(stderr, "%s: option \"%s\" requires an "
"argument.\n", argv[0], arg);
goto done;
}
} else {
* if the argument is optional, and we're either at the
* end of the argv list, or the next argv starts with '-',
* then assume there is no argument for this option
*/
if ((o->flags & NVGETOPT_ARGUMENT_IS_OPTIONAL) &&
((argv_index == (argc - 1)) ||
(argv[argv_index + 1][0] == '-'))) {
argument = NULL;
goto argument_processing_done;
} else {
argv_index++;
if (argv_index >= argc) {
fprintf(stderr, "%s: option \"%s\" requires an "
"argument.\n", argv[0], arg);
goto done;
}
argument = argv[argv_index];
}
}
if ((o->flags & NVGETOPT_INTEGER_ARGUMENT) && (intval)) {
char *endptr;
*intval = (int) strtol(argument, &endptr, 0);
if (*endptr) {
fprintf(stderr, "%s: \"%s\" is not a valid argument for "
"option \"%s\".\n", argv[0], argument, arg);
goto done;
}
} else if ((o->flags & NVGETOPT_STRING_ARGUMENT) && (strval)) {
*strval = strdup(argument);
} else if ((o->flags & NVGETOPT_DOUBLE_ARGUMENT) && (doubleval)) {
char *endptr;
*doubleval = (double) strtod(argument, &endptr);
if (*endptr) {
fprintf(stderr, "%s: \"%s\" is not a valid argument for "
"option \"%s\".\n", argv[0], argument, arg);
goto done;
}
} else {
fprintf(stderr, "%s: error while assigning argument for "
"option \"%s\".\n", argv[0], arg);
goto done;
}
} else {
if (argument) {
fprintf(stderr, "%s: option \"%s\" does not take an argument, but "
"was given an argument of \"%s\".\n",
argv[0], arg, argument);
goto done;
}
}
argument_processing_done:
ret = o->val;
done:
if (disable_val) *disable_val = disable;
free(arg);
return ret;
}
* cook_description() - the description string may contain text within
* special characters which are interpreted by the manpage generator.
* We want to omit those characters here.
*/
static char *cook_description(const char *description)
{
const char *src;
char *s, *dst;
if (!description) {
return NULL;
}
s = strdup(description);
if (!s) {
return NULL;
}
for (src = description, dst = s; *src; src++) {
if ((*src == '&') || (*src == '^')) {
continue;
}
*dst = *src;
dst++;
}
*dst = '\0';
return s;
}
void nvgetopt_print_help(const NVGetoptOption *options,
unsigned int include_mask,
nvgetopt_print_help_callback_ptr callback)
{
const NVGetoptOption *o;
int i;
for (i = 0; options[i].name; i++) {
char *msg = NULL, *arg = NULL, *description = NULL;
o = &options[i];
if (!o->description) {
continue;
}
if ((o->flags & include_mask) != include_mask) {
continue;
}
arg = NULL;
if (o->flags & NVGETOPT_HAS_ARGUMENT) {
if (o->arg_name) {
arg = strdup(o->arg_name);
} else {
char *tmp;
arg = strdup(o->name);
for (tmp = arg; tmp && *tmp; tmp++) {
*tmp = toupper(*tmp);
}
}
}
msg = NULL;
* create the long version of the option, possibly with an
* argument; e.g., "--foo" or "--foo=BAR"
*/
if (arg) {
msg = nvstrcat("--", o->name, "=", arg, NULL);
} else {
msg = nvstrcat("--", o->name, NULL);
}
* prepend the single character version of the option,
* possibly with an argument; e.g., "-f" or "-f BAR"
*/
if (o->val <= UCHAR_MAX && o->val >= 0 && isalpha(o->val)) {
char scratch[16];
char *tmp;
snprintf(scratch, sizeof(scratch), "%c", o->val);
if (arg) {
tmp = nvstrcat("-", scratch, " ", arg, ", ", msg, NULL);
} else {
tmp = nvstrcat("-", scratch, ", ", msg, NULL);
}
free(msg);
msg = tmp;
}
if (((o->flags & NVGETOPT_IS_BOOLEAN) &&
!(o->flags & NVGETOPT_HAS_ARGUMENT)) ||
(o->flags & NVGETOPT_ALLOW_DISABLE)) {
char *tmp = nvstrcat(msg, ", --no-", o->name, NULL);
free(msg);
msg = tmp;
}
description = cook_description(o->description);
callback(msg, description);
free(msg);
free(arg);
free(description);
}
}