* doc/gnulib-tool.texi (Initial import): Update to match current
[gnulib.git] / lib / getsubopt.c
1 /* Parse comma separate list into words.
2    Copyright (C) 1996, 1997, 1999, 2004, 2007 Free Software Foundation, Inc.
3    This file is part of the GNU C Library.
4    Contributed by Ulrich Drepper <drepper@cygnus.com>, 1996.
5
6    This program is free software; you can redistribute it and/or modify
7    it under the terms of the GNU General Public License as published by
8    the Free Software Foundation; either version 2, or (at your option)
9    any later version.
10
11    This program is distributed in the hope that it will be useful,
12    but WITHOUT ANY WARRANTY; without even the implied warranty of
13    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14    GNU General Public License for more details.
15
16    You should have received a copy of the GNU General Public License along
17    with this program; if not, write to the Free Software Foundation,
18    Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.  */
19
20 #include <stdlib.h>
21 #include <string.h>
22
23 #if !_LIBC
24 /* This code is written for inclusion in gnu-libc, and uses names in
25    the namespace reserved for libc.  If we're compiling in gnulib,
26    define those names to be the normal ones instead.  */
27 # undef __strchrnul
28 # define __strchrnul strchrnul
29 #endif
30
31 /* Parse comma separated suboption from *OPTIONP and match against
32    strings in TOKENS.  If found return index and set *VALUEP to
33    optional value introduced by an equal sign.  If the suboption is
34    not part of TOKENS return in *VALUEP beginning of unknown
35    suboption.  On exit *OPTIONP is set to the beginning of the next
36    token or at the terminating NUL character.  */
37 int
38 getsubopt (char **optionp, char *const *tokens, char **valuep)
39 {
40   char *endp, *vstart;
41   int cnt;
42
43   if (**optionp == '\0')
44     return -1;
45
46   /* Find end of next token.  */
47   endp = __strchrnul (*optionp, ',');
48
49   /* Find start of value.  */
50   vstart = memchr (*optionp, '=', endp - *optionp);
51   if (vstart == NULL)
52     vstart = endp;
53
54   /* Try to match the characters between *OPTIONP and VSTART against
55      one of the TOKENS.  */
56   for (cnt = 0; tokens[cnt] != NULL; ++cnt)
57     if (strncmp (*optionp, tokens[cnt], vstart - *optionp) == 0
58         && tokens[cnt][vstart - *optionp] == '\0')
59       {
60         /* We found the current option in TOKENS.  */
61         *valuep = vstart != endp ? vstart + 1 : NULL;
62
63         if (*endp != '\0')
64           *endp++ = '\0';
65         *optionp = endp;
66
67         return cnt;
68       }
69
70   /* The current suboption does not match any option.  */
71   *valuep = *optionp;
72
73   if (*endp != '\0')
74     *endp++ = '\0';
75   *optionp = endp;
76
77   return -1;
78 }