2 * getopt_long() -- long options parser
4 * Portions Copyright (c) 1987, 1993, 1994
5 * The Regents of the University of California. All rights reserved.
7 * Portions Copyright (c) 2003
8 * PostgreSQL Global Development Group
10 * Redistribution and use in source and binary forms, with or without
11 * modification, are permitted provided that the following conditions
13 * 1. Redistributions of source code must retain the above copyright
14 * notice, this list of conditions and the following disclaimer.
15 * 2. Redistributions in binary form must reproduce the above copyright
16 * notice, this list of conditions and the following disclaimer in the
17 * documentation and/or other materials provided with the distribution.
18 * 3. Neither the name of the University nor the names of its contributors
19 * may be used to endorse or promote products derived from this software
20 * without specific prior written permission.
22 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
23 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
24 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
25 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
26 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
27 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
28 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
29 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
30 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
31 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
34 * src/port/getopt_long.c
39 #include "getopt_long.h"
48 * Parse argc/argv argument vector, with long options.
50 * This implementation does not use optreset. Instead, we guarantee that
51 * it can be restarted on a new argv array after a previous call returned -1,
52 * if the caller resets optind to 1 before the first call of the new series.
53 * (Internally, this means we must be sure to reset "place" to EMSG,
54 * "nonopt_start" to -1, and "force_nonopt" to false before returning -1.)
56 * Note that this routine reorders the pointers in argv (despite the const
57 * qualifier) so that all non-options will be at the end when -1 is returned.
60 getopt_long(int argc
, char *const argv
[],
61 const char *optstring
,
62 const struct option
*longopts
, int *longindex
)
64 static char *place
= EMSG
; /* option letter processing */
65 char *oli
; /* option letter list index */
66 static int nonopt_start
= -1;
67 static bool force_nonopt
= false;
70 { /* update scanning pointer */
71 char **args
= (char **) argv
;
76 * If we are out of arguments or only non-options remain, return -1.
78 if (optind
>= argc
|| optind
== nonopt_start
)
89 * An argument is a non-option if it meets any of the following
90 * criteria: it follows an argument that is equivalent to the string
91 * "--", it does not start with '-', or it is equivalent to the string
92 * "-". When we encounter a non-option, we move it to the end of argv
93 * (after shifting all remaining arguments over to make room), and
94 * then we try again with the next argument.
96 if (force_nonopt
|| place
[0] != '-' || place
[1] == '\0')
98 for (int i
= optind
; i
< argc
- 1; i
++)
99 args
[i
] = args
[i
+ 1];
100 args
[argc
- 1] = place
;
102 if (nonopt_start
== -1)
103 nonopt_start
= argc
- 1;
112 if (place
[0] == '-' && place
[1] == '\0')
114 /* found "--", treat it as end of options */
120 if (place
[0] == '-' && place
[1])
128 namelen
= strcspn(place
, "=");
129 for (i
= 0; longopts
[i
].name
!= NULL
; i
++)
131 if (strlen(longopts
[i
].name
) == namelen
132 && strncmp(place
, longopts
[i
].name
, namelen
) == 0)
134 int has_arg
= longopts
[i
].has_arg
;
136 if (has_arg
!= no_argument
)
138 if (place
[namelen
] == '=')
139 optarg
= place
+ namelen
+ 1;
140 else if (optind
< argc
- 1 &&
141 has_arg
== required_argument
)
144 optarg
= argv
[optind
];
148 if (optstring
[0] == ':')
151 if (opterr
&& has_arg
== required_argument
)
153 "%s: option requires an argument -- %s\n",
159 if (has_arg
== required_argument
)
167 if (place
[namelen
] != 0)
180 if (longopts
[i
].flag
== NULL
)
181 return longopts
[i
].val
;
184 *longopts
[i
].flag
= longopts
[i
].val
;
190 if (opterr
&& optstring
[0] != ':')
192 "%s: illegal option -- %s\n", argv
[0], place
);
200 optopt
= (int) *place
++;
202 oli
= strchr(optstring
, optopt
);
207 if (opterr
&& *optstring
!= ':')
209 "%s: illegal option -- %c\n", argv
[0], optopt
);
214 { /* don't need argument */
220 { /* need an argument */
221 if (*place
) /* no white space */
223 else if (argc
<= ++optind
)
226 if (*optstring
== ':')
230 "%s: option requires an argument -- %c\n",
236 optarg
= argv
[optind
];