1 /*
2 *******************************************************************************
3 *
4 *   Copyright (C) 2000, International Business Machines
5 *   Corporation and others.  All Rights Reserved.
6 *
7 *******************************************************************************
8 *   file name:  uoptions.c
9 *   encoding:   US-ASCII
10 *   tab size:   8 (not used)
11 *   indentation:4
12 *
13 *   created on: 2000apr17
14 *   created by: Markus W. Scherer
15 *
16 *   This file provides a command line argument parser.
17 */
18 
19 #include "unicode/utypes.h"
20 #include "cstring.h"
21 #include "uoptions.h"
22 
23 U_CAPI int U_EXPORT2
u_parseArgs(int argc,char * argv[],int optionCount,UOption options[])24 u_parseArgs(int argc, char* argv[],
25             int optionCount, UOption options[]) {
26     char *arg;
27     int i=1, remaining=1;
28     char c, stopOptions=0;
29 
30     while(i<argc) {
31         arg=argv[i];
32         if(!stopOptions && *arg=='-' && (c=arg[1])!=0) {
33             /* process an option */
34             UOption *option=NULL;
35             arg+=2;
36             if(c=='-') {
37                 /* process a long option */
38                 if(*arg==0) {
39                     /* stop processing options after "--" */
40                     stopOptions=1;
41                 } else {
42                     /* search for the option string */
43                     int j;
44                     for(j=0; j<optionCount; ++j) {
45                         if(options[j].longName && uprv_strcmp(arg, options[j].longName)==0) {
46                             option=options+j;
47                             break;
48                         }
49                     }
50                     if(option==NULL) {
51                         /* no option matches */
52                         return -i;
53                     }
54                     option->doesOccur=1;
55 
56                     if(option->hasArg!=UOPT_NO_ARG) {
57                         /* parse the argument for the option, if any */
58                         if(i+1<argc && !(argv[i+1][0]=='-' && argv[i+1][1]!=0)) {
59                             /* argument in the next argv[], and there is not an option in there */
60                             option->value=argv[++i];
61                         } else if(option->hasArg==UOPT_REQUIRES_ARG) {
62                             /* there is no argument, but one is required: return with error */
63                             return -i;
64                         }
65                     }
66                 }
67             } else {
68                 /* process one or more short options */
69                 do {
70                     /* search for the option letter */
71                     int j;
72                     for(j=0; j<optionCount; ++j) {
73                         if(c==options[j].shortName) {
74                             option=options+j;
75                             break;
76                         }
77                     }
78                     if(option==NULL) {
79                         /* no option matches */
80                         return -i;
81                     }
82                     option->doesOccur=1;
83 
84                     if(option->hasArg!=UOPT_NO_ARG) {
85                         /* parse the argument for the option, if any */
86                         if(*arg!=0) {
87                             /* argument following in the same argv[] */
88                             option->value=arg;
89                             /* do not process the rest of this arg as option letters */
90                             break;
91                         } else if(i+1<argc && !(argv[i+1][0]=='-' && argv[i+1][1]!=0)) {
92                             /* argument in the next argv[], and there is not an option in there */
93                             option->value=argv[++i];
94                             /* this break is redundant because we know that *arg==0 */
95                             break;
96                         } else if(option->hasArg==UOPT_REQUIRES_ARG) {
97                             /* there is no argument, but one is required: return with error */
98                             return -i;
99                         }
100                     }
101 
102                     /* get the next option letter */
103                     option=NULL;
104                     c=*arg++;
105                 } while(c!=0);
106             }
107 
108             if(option!=0 && option->optionFn!=0 && option->optionFn(option->context, option)<0) {
109                 /* the option function was called and returned an error */
110                 return -i;
111             }
112 
113             /* go to next argv[] */
114             ++i;
115         } else {
116             /* move a non-option up in argv[] */
117             argv[remaining++]=arg;
118             ++i;
119         }
120     }
121     return remaining;
122 }
123