in watchman/winbuild/getopt_long.cpp [50:175]
int getopt_long(
int argc,
char* const argv[],
const char* optstring,
const struct option* longopts,
int* longindex) {
static const char* place = EMSG; /* option letter processing */
const char* oli; /* option letter list index */
if (!*place) { /* update scanning pointer */
if (optind >= argc) {
place = EMSG;
return -1;
}
place = argv[optind];
if (place[0] != '-') {
place = EMSG;
return -1;
}
place++;
if (place[0] && place[0] == '-' && place[1] == '\0') { /* found "--" */
++optind;
place = EMSG;
return -1;
}
if (place[0] && place[0] == '-' && place[1]) {
/* long option */
size_t namelen;
int i;
place++;
namelen = strcspn(place, "=");
for (i = 0; longopts[i].name != NULL; i++) {
if (strlen(longopts[i].name) == namelen &&
strncmp(place, longopts[i].name, namelen) == 0) {
if (longopts[i].has_arg) {
if (place[namelen] == '=')
optarg = (char*)place + namelen + 1;
else if (optind < argc - 1) {
optind++;
optarg = argv[optind];
} else {
if (optstring[0] == ':')
return BADARG;
if (opterr)
fprintf(
stderr,
"%s: option requires an argument -- %s\n",
argv[0],
place);
place = EMSG;
optind++;
return BADCH;
}
} else {
optarg = NULL;
if (place[namelen] != 0) {
/* XXX error? */
}
}
optind++;
if (longindex)
*longindex = i;
place = EMSG;
if (longopts[i].flag == NULL)
return longopts[i].val;
else {
*longopts[i].flag = longopts[i].val;
return 0;
}
}
}
if (opterr && optstring[0] != ':')
fprintf(stderr, "%s: illegal option -- %s\n", argv[0], place);
place = EMSG;
optind++;
return BADCH;
}
}
/* short option */
optopt = (int)*place++;
oli = strchr(optstring, optopt);
if (!oli) {
if (!*place)
++optind;
if (opterr && *optstring != ':')
fprintf(stderr, "%s: illegal option -- %c\n", argv[0], optopt);
return BADCH;
}
if (oli[1] != ':') { /* don't need argument */
optarg = NULL;
if (!*place)
++optind;
} else { /* need an argument */
if (*place) /* no white space */
optarg = (char*)place;
else if (argc <= ++optind) { /* no arg */
place = EMSG;
if (*optstring == ':')
return BADARG;
if (opterr)
fprintf(
stderr, "%s: option requires an argument -- %c\n", argv[0], optopt);
return BADCH;
} else
/* white space */
optarg = argv[optind];
place = EMSG;
++optind;
}
return optopt;
}