2017-01-20 00:20:31 +00:00
|
|
|
// © 2016 and later: Unicode, Inc. and others.
|
2016-06-15 18:58:17 +00:00
|
|
|
// License & terms of use: http://www.unicode.org/copyright.html
|
2000-04-17 21:13:31 +00:00
|
|
|
/*
|
|
|
|
*******************************************************************************
|
|
|
|
*
|
2016-05-31 21:45:07 +00:00
|
|
|
* Copyright (C) 2000-2015, International Business Machines
|
|
|
|
* Corporation and others. All Rights Reserved.
|
2000-04-17 21:13:31 +00:00
|
|
|
*
|
|
|
|
*******************************************************************************
|
|
|
|
* file name: uoptions.c
|
2017-02-03 18:57:23 +00:00
|
|
|
* encoding: UTF-8
|
2000-04-17 21:13:31 +00:00
|
|
|
* tab size: 8 (not used)
|
|
|
|
* indentation:4
|
|
|
|
*
|
|
|
|
* created on: 2000apr17
|
|
|
|
* created by: Markus W. Scherer
|
|
|
|
*
|
|
|
|
* This file provides a command line argument parser.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "unicode/utypes.h"
|
|
|
|
#include "cstring.h"
|
|
|
|
#include "uoptions.h"
|
|
|
|
|
2000-04-18 16:34:27 +00:00
|
|
|
U_CAPI int U_EXPORT2
|
2000-09-21 21:49:32 +00:00
|
|
|
u_parseArgs(int argc, char* argv[],
|
2000-04-17 21:13:31 +00:00
|
|
|
int optionCount, UOption options[]) {
|
2000-09-21 21:49:32 +00:00
|
|
|
char *arg;
|
2000-04-17 21:13:31 +00:00
|
|
|
int i=1, remaining=1;
|
|
|
|
char c, stopOptions=0;
|
|
|
|
|
|
|
|
while(i<argc) {
|
|
|
|
arg=argv[i];
|
|
|
|
if(!stopOptions && *arg=='-' && (c=arg[1])!=0) {
|
|
|
|
/* process an option */
|
|
|
|
UOption *option=NULL;
|
|
|
|
arg+=2;
|
|
|
|
if(c=='-') {
|
|
|
|
/* process a long option */
|
|
|
|
if(*arg==0) {
|
|
|
|
/* stop processing options after "--" */
|
|
|
|
stopOptions=1;
|
|
|
|
} else {
|
|
|
|
/* search for the option string */
|
|
|
|
int j;
|
|
|
|
for(j=0; j<optionCount; ++j) {
|
2002-01-29 20:02:14 +00:00
|
|
|
if(options[j].longName && uprv_strcmp(arg, options[j].longName)==0) {
|
2000-04-17 21:13:31 +00:00
|
|
|
option=options+j;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(option==NULL) {
|
|
|
|
/* no option matches */
|
|
|
|
return -i;
|
|
|
|
}
|
|
|
|
option->doesOccur=1;
|
|
|
|
|
|
|
|
if(option->hasArg!=UOPT_NO_ARG) {
|
|
|
|
/* parse the argument for the option, if any */
|
|
|
|
if(i+1<argc && !(argv[i+1][0]=='-' && argv[i+1][1]!=0)) {
|
|
|
|
/* argument in the next argv[], and there is not an option in there */
|
|
|
|
option->value=argv[++i];
|
|
|
|
} else if(option->hasArg==UOPT_REQUIRES_ARG) {
|
|
|
|
/* there is no argument, but one is required: return with error */
|
2015-09-04 18:38:25 +00:00
|
|
|
option->doesOccur=0;
|
2000-04-17 21:13:31 +00:00
|
|
|
return -i;
|
|
|
|
}
|
|
|
|
}
|
2015-09-04 18:38:25 +00:00
|
|
|
|
|
|
|
if(option->optionFn!=NULL && option->optionFn(option->context, option)<0) {
|
|
|
|
/* the option function was called and returned an error */
|
|
|
|
option->doesOccur=0;
|
|
|
|
return -i;
|
|
|
|
}
|
2000-04-17 21:13:31 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* process one or more short options */
|
|
|
|
do {
|
|
|
|
/* search for the option letter */
|
|
|
|
int j;
|
|
|
|
for(j=0; j<optionCount; ++j) {
|
|
|
|
if(c==options[j].shortName) {
|
|
|
|
option=options+j;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(option==NULL) {
|
|
|
|
/* no option matches */
|
|
|
|
return -i;
|
|
|
|
}
|
|
|
|
option->doesOccur=1;
|
|
|
|
|
|
|
|
if(option->hasArg!=UOPT_NO_ARG) {
|
|
|
|
/* parse the argument for the option, if any */
|
|
|
|
if(*arg!=0) {
|
|
|
|
/* argument following in the same argv[] */
|
|
|
|
option->value=arg;
|
|
|
|
/* do not process the rest of this arg as option letters */
|
|
|
|
break;
|
|
|
|
} else if(i+1<argc && !(argv[i+1][0]=='-' && argv[i+1][1]!=0)) {
|
|
|
|
/* argument in the next argv[], and there is not an option in there */
|
|
|
|
option->value=argv[++i];
|
|
|
|
/* this break is redundant because we know that *arg==0 */
|
|
|
|
break;
|
|
|
|
} else if(option->hasArg==UOPT_REQUIRES_ARG) {
|
|
|
|
/* there is no argument, but one is required: return with error */
|
2015-09-04 18:38:25 +00:00
|
|
|
option->doesOccur=0;
|
2000-04-17 21:13:31 +00:00
|
|
|
return -i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-04 18:38:25 +00:00
|
|
|
if(option->optionFn!=NULL && option->optionFn(option->context, option)<0) {
|
|
|
|
/* the option function was called and returned an error */
|
|
|
|
option->doesOccur=0;
|
|
|
|
return -i;
|
|
|
|
}
|
|
|
|
|
2000-04-17 21:13:31 +00:00
|
|
|
/* get the next option letter */
|
|
|
|
option=NULL;
|
|
|
|
c=*arg++;
|
|
|
|
} while(c!=0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* go to next argv[] */
|
|
|
|
++i;
|
|
|
|
} else {
|
|
|
|
/* move a non-option up in argv[] */
|
|
|
|
argv[remaining++]=arg;
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return remaining;
|
|
|
|
}
|