1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun * Copyright 2002-2003 Red Hat Inc., Durham, North Carolina.
3*4882a593Smuzhiyun *
4*4882a593Smuzhiyun * All Rights Reserved.
5*4882a593Smuzhiyun *
6*4882a593Smuzhiyun * Permission is hereby granted, free of charge, to any person obtaining
7*4882a593Smuzhiyun * a copy of this software and associated documentation files (the
8*4882a593Smuzhiyun * "Software"), to deal in the Software without restriction, including
9*4882a593Smuzhiyun * without limitation on the rights to use, copy, modify, merge,
10*4882a593Smuzhiyun * publish, distribute, sublicense, and/or sell copies of the Software,
11*4882a593Smuzhiyun * and to permit persons to whom the Software is furnished to do so,
12*4882a593Smuzhiyun * subject to the following conditions:
13*4882a593Smuzhiyun *
14*4882a593Smuzhiyun * The above copyright notice and this permission notice (including the
15*4882a593Smuzhiyun * next paragraph) shall be included in all copies or substantial
16*4882a593Smuzhiyun * portions of the Software.
17*4882a593Smuzhiyun *
18*4882a593Smuzhiyun * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
19*4882a593Smuzhiyun * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
20*4882a593Smuzhiyun * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
21*4882a593Smuzhiyun * NON-INFRINGEMENT. IN NO EVENT SHALL RED HAT AND/OR THEIR SUPPLIERS
22*4882a593Smuzhiyun * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
23*4882a593Smuzhiyun * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
24*4882a593Smuzhiyun * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
25*4882a593Smuzhiyun * SOFTWARE.
26*4882a593Smuzhiyun */
27*4882a593Smuzhiyun
28*4882a593Smuzhiyun /*
29*4882a593Smuzhiyun * Authors:
30*4882a593Smuzhiyun * Rickard E. (Rik) Faith <faith@redhat.com>
31*4882a593Smuzhiyun *
32*4882a593Smuzhiyun */
33*4882a593Smuzhiyun
34*4882a593Smuzhiyun /** \file
35*4882a593Smuzhiyun * Generic comma-delimited argument processing. */
36*4882a593Smuzhiyun
37*4882a593Smuzhiyun #ifdef HAVE_DMX_CONFIG_H
38*4882a593Smuzhiyun #include <dmx-config.h>
39*4882a593Smuzhiyun #endif
40*4882a593Smuzhiyun
41*4882a593Smuzhiyun #define DMX_ARG_TEST 0
42*4882a593Smuzhiyun
43*4882a593Smuzhiyun #include "dmx.h"
44*4882a593Smuzhiyun #include "dmxarg.h"
45*4882a593Smuzhiyun #include <stdio.h>
46*4882a593Smuzhiyun #include <string.h>
47*4882a593Smuzhiyun
48*4882a593Smuzhiyun #if DMX_ARG_TEST
49*4882a593Smuzhiyun #include <stdlib.h>
50*4882a593Smuzhiyun #endif
51*4882a593Smuzhiyun
52*4882a593Smuzhiyun /** Stores the parsed argument list. */
53*4882a593Smuzhiyun struct _dmxArg {
54*4882a593Smuzhiyun int argc; /**< Number of arguments in argv */
55*4882a593Smuzhiyun int argm; /**< Maximum number of arguments store-able in argv */
56*4882a593Smuzhiyun const char **argv; /**< Arguments */
57*4882a593Smuzhiyun };
58*4882a593Smuzhiyun
59*4882a593Smuzhiyun /** Create an (externally opaque) \a dmxArg object. */
60*4882a593Smuzhiyun dmxArg
dmxArgCreate(void)61*4882a593Smuzhiyun dmxArgCreate(void)
62*4882a593Smuzhiyun {
63*4882a593Smuzhiyun dmxArg a = malloc(sizeof(*a));
64*4882a593Smuzhiyun
65*4882a593Smuzhiyun a->argc = 0;
66*4882a593Smuzhiyun a->argm = 2;
67*4882a593Smuzhiyun a->argv = malloc(a->argm * sizeof(*a->argv));
68*4882a593Smuzhiyun a->argv[0] = NULL;
69*4882a593Smuzhiyun return a;
70*4882a593Smuzhiyun }
71*4882a593Smuzhiyun
72*4882a593Smuzhiyun /** Free the specified \a dmxArg object. */
73*4882a593Smuzhiyun void
dmxArgFree(dmxArg a)74*4882a593Smuzhiyun dmxArgFree(dmxArg a)
75*4882a593Smuzhiyun {
76*4882a593Smuzhiyun int i;
77*4882a593Smuzhiyun
78*4882a593Smuzhiyun for (i = 0; i < a->argc; i++)
79*4882a593Smuzhiyun free((char *) a->argv[i]);
80*4882a593Smuzhiyun free(a->argv);
81*4882a593Smuzhiyun free(a);
82*4882a593Smuzhiyun }
83*4882a593Smuzhiyun
84*4882a593Smuzhiyun /** Add the \a string as the next argument in the \a dmxArg object. */
85*4882a593Smuzhiyun void
dmxArgAdd(dmxArg a,const char * string)86*4882a593Smuzhiyun dmxArgAdd(dmxArg a, const char *string)
87*4882a593Smuzhiyun {
88*4882a593Smuzhiyun if (a->argm <= a->argc + 2)
89*4882a593Smuzhiyun a->argv = reallocarray(a->argv, (a->argm *= 2), sizeof(*a->argv));
90*4882a593Smuzhiyun a->argv[a->argc++] = strdup(string);
91*4882a593Smuzhiyun a->argv[a->argc] = NULL;
92*4882a593Smuzhiyun }
93*4882a593Smuzhiyun
94*4882a593Smuzhiyun /** Return the argument number \a item in the \a dmxArg object.
95*4882a593Smuzhiyun * Arguments are 0 based. NULL will be returned for values less than 0
96*4882a593Smuzhiyun * or equal to or greater than the number of arguments in the object. */
97*4882a593Smuzhiyun const char *
dmxArgV(dmxArg a,int item)98*4882a593Smuzhiyun dmxArgV(dmxArg a, int item)
99*4882a593Smuzhiyun {
100*4882a593Smuzhiyun if (item < 0 || item >= a->argc)
101*4882a593Smuzhiyun return NULL;
102*4882a593Smuzhiyun return a->argv[item];
103*4882a593Smuzhiyun }
104*4882a593Smuzhiyun
105*4882a593Smuzhiyun /** Return the number of arguments in the \a dmxArg object. */
106*4882a593Smuzhiyun int
dmxArgC(dmxArg a)107*4882a593Smuzhiyun dmxArgC(dmxArg a)
108*4882a593Smuzhiyun {
109*4882a593Smuzhiyun return a->argc;
110*4882a593Smuzhiyun }
111*4882a593Smuzhiyun
112*4882a593Smuzhiyun /** Parse a string into arguments delimited by commas. Return a new \a
113*4882a593Smuzhiyun * dmxArg object containing the arguments. */
114*4882a593Smuzhiyun dmxArg
dmxArgParse(const char * string)115*4882a593Smuzhiyun dmxArgParse(const char *string)
116*4882a593Smuzhiyun {
117*4882a593Smuzhiyun int i = 0;
118*4882a593Smuzhiyun dmxArg a = dmxArgCreate();
119*4882a593Smuzhiyun
120*4882a593Smuzhiyun if (!string)
121*4882a593Smuzhiyun return a;
122*4882a593Smuzhiyun
123*4882a593Smuzhiyun a->argv = (const char **)xstrtokenize(string, ",");
124*4882a593Smuzhiyun if (a->argv)
125*4882a593Smuzhiyun for (i = 0; a->argv[i] != NULL; i++);
126*4882a593Smuzhiyun a->argc = i;
127*4882a593Smuzhiyun
128*4882a593Smuzhiyun return a;
129*4882a593Smuzhiyun }
130*4882a593Smuzhiyun
131*4882a593Smuzhiyun #if DMX_ARG_TEST
132*4882a593Smuzhiyun static void
dmxArgPrint(dmxArg a)133*4882a593Smuzhiyun dmxArgPrint(dmxArg a)
134*4882a593Smuzhiyun {
135*4882a593Smuzhiyun int i;
136*4882a593Smuzhiyun
137*4882a593Smuzhiyun printf(" argc = %d\n", dmxArgC(a));
138*4882a593Smuzhiyun for (i = 0; i < dmxArgC(a); i++)
139*4882a593Smuzhiyun printf(" argv[%d] = \"%s\"\n", i, dmxArgV(a, i));
140*4882a593Smuzhiyun }
141*4882a593Smuzhiyun
142*4882a593Smuzhiyun static void
dmxArgTest(const char * string)143*4882a593Smuzhiyun dmxArgTest(const char *string)
144*4882a593Smuzhiyun {
145*4882a593Smuzhiyun dmxArg a;
146*4882a593Smuzhiyun
147*4882a593Smuzhiyun if (!string)
148*4882a593Smuzhiyun printf("Testing NULL\n");
149*4882a593Smuzhiyun else if (!strlen(string))
150*4882a593Smuzhiyun printf("Testing (empty)\n");
151*4882a593Smuzhiyun else
152*4882a593Smuzhiyun printf("Testing \"%s\"\n", string);
153*4882a593Smuzhiyun
154*4882a593Smuzhiyun a = dmxArgParse(string);
155*4882a593Smuzhiyun dmxArgPrint(a);
156*4882a593Smuzhiyun dmxArgFree(a);
157*4882a593Smuzhiyun }
158*4882a593Smuzhiyun
159*4882a593Smuzhiyun int
main(void)160*4882a593Smuzhiyun main(void)
161*4882a593Smuzhiyun {
162*4882a593Smuzhiyun dmxArgTest(NULL);
163*4882a593Smuzhiyun dmxArgTest("");
164*4882a593Smuzhiyun dmxArgTest(",");
165*4882a593Smuzhiyun
166*4882a593Smuzhiyun dmxArgTest("a");
167*4882a593Smuzhiyun dmxArgTest("a,");
168*4882a593Smuzhiyun dmxArgTest(",a");
169*4882a593Smuzhiyun
170*4882a593Smuzhiyun dmxArgTest("a,b");
171*4882a593Smuzhiyun dmxArgTest("a,b,");
172*4882a593Smuzhiyun dmxArgTest("a,b,,");
173*4882a593Smuzhiyun dmxArgTest("a,b,,c");
174*4882a593Smuzhiyun
175*4882a593Smuzhiyun return 0;
176*4882a593Smuzhiyun }
177*4882a593Smuzhiyun #endif
178