blob: 39a7d1d29a3a9b57a3ae0e8c463bbab9cf5a9500 [file] [log] [blame]
Manuel Novoa III cad53642003-03-19 09:13:01 +00001/* vi: set sw=4 ts=4: */
2/*
Eric Andersen8876fb22003-06-20 09:01:58 +00003 * universal getopt_ulflags implementation for busybox
Manuel Novoa III cad53642003-03-19 09:13:01 +00004 *
Eric Andersen8876fb22003-06-20 09:01:58 +00005 * Copyright (C) 2003 Vladimir Oleynik <dzo@simtreas.ru>
Manuel Novoa III cad53642003-03-19 09:13:01 +00006 *
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License, or
10 * (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, write to the Free Software
19 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
20 *
21 */
22
23#include <getopt.h>
24#include <string.h>
Eric Andersen8876fb22003-06-20 09:01:58 +000025#include <assert.h>
26#include <stdlib.h>
Manuel Novoa III cad53642003-03-19 09:13:01 +000027#include "libbb.h"
28
Eric Andersen8876fb22003-06-20 09:01:58 +000029/*
30You can set bb_opt_complementaly as string with one or more
31complementaly or incongruously options.
32If sequential founded option haved from this string
33then your incongruously pairs unsets and complementaly make add sets.
34Format:
35one char - option for check,
36chars - complementaly option for add sets.
37- chars - option triggered for unsets.
38~ chars - option incongruously.
39* - option list, called add_to_list(*ptr_from_usaged, optarg)
40: - separator.
41Example: du applet can have options "-s" and "-d size"
42If getopt found -s then -d option flag unset or if found -d then -s unset.
43For this result you must set bb_opt_complementaly = "s-d:d-s".
44Result have last option flag only from called arguments.
45Warning! You can check returned flag, pointer to "d:" argument seted
46to own optarg always.
47Example two: cut applet must only one type of list may be specified,
48and -b, -c and -f incongruously option, overwited option is error also.
Glenn L McGrath85c51522004-01-22 07:10:13 +000049You must set bb_opt_complementaly = "b~cf:c~bf:f~bc".
Eric Andersen8876fb22003-06-20 09:01:58 +000050If called have more one specified, return value have error flag -
51high bite set (0x80000000UL).
52Example three: grep applet can have one or more "-e pattern" arguments.
53You should use bb_getopt_ulflags() as
54llist_t *paterns;
55bb_opt_complementaly = "e*";
56bb_getopt_ulflags (argc, argv, "e:", &paterns);
57*/
58
59const char *bb_opt_complementaly;
60
61typedef struct
62{
Glenn L McGrath850b05f2003-12-19 10:13:10 +000063 unsigned char opt;
Eric Andersen8876fb22003-06-20 09:01:58 +000064 char list_flg;
65 unsigned long switch_on;
66 unsigned long switch_off;
67 unsigned long incongruously;
68 void **optarg; /* char **optarg or llist_t **optarg */
69} t_complementaly;
70
71/* You can set bb_applet_long_options for parse called long options */
72
73static const struct option bb_default_long_options[] = {
74 /* { "help", 0, NULL, '?' }, */
75 { 0, 0, 0, 0 }
76};
77
78const struct option *bb_applet_long_options = bb_default_long_options;
79
80
81unsigned long
82bb_getopt_ulflags (int argc, char **argv, const char *applet_opts, ...)
Manuel Novoa III cad53642003-03-19 09:13:01 +000083{
Glenn L McGrath85c51522004-01-22 07:10:13 +000084 unsigned long flags = 0;
85 t_complementaly complementaly[sizeof(flags) * 8 + 1];
86 int c;
87 const unsigned char *s;
Eric Andersen8876fb22003-06-20 09:01:58 +000088 t_complementaly *on_off;
89 va_list p;
Manuel Novoa III cad53642003-03-19 09:13:01 +000090
Eric Andersen8876fb22003-06-20 09:01:58 +000091 va_start (p, applet_opts);
92
Glenn L McGrath9d1a33c2003-10-06 13:23:06 +000093 /* skip GNU extension */
94 s = applet_opts;
95 if(*s == '+' || *s == '-')
96 s++;
97
Eric Andersen8876fb22003-06-20 09:01:58 +000098 c = 0;
Glenn L McGrath85c51522004-01-22 07:10:13 +000099 on_off = complementaly;
Glenn L McGrath9d1a33c2003-10-06 13:23:06 +0000100 for (; *s; s++) {
Glenn L McGrath85c51522004-01-22 07:10:13 +0000101 if(c >= (sizeof(flags)*8))
102 break;
103 on_off->opt = *s;
Eric Andersen54426d52004-02-05 13:49:29 +0000104 on_off->switch_on = (1 << c);
Glenn L McGrath85c51522004-01-22 07:10:13 +0000105 on_off->list_flg = 0;
106 on_off->switch_off = 0;
107 on_off->incongruously = 0;
108 on_off->optarg = NULL;
Eric Andersen8876fb22003-06-20 09:01:58 +0000109 if (s[1] == ':') {
Glenn L McGrath85c51522004-01-22 07:10:13 +0000110 on_off->optarg = va_arg (p, void **);
Eric Andersen8876fb22003-06-20 09:01:58 +0000111 do
112 s++;
113 while (s[1] == ':');
Eric Andersen8876fb22003-06-20 09:01:58 +0000114 }
Glenn L McGrath85c51522004-01-22 07:10:13 +0000115 on_off++;
116 c++;
117 }
118 on_off->opt = 0;
Eric Andersen8876fb22003-06-20 09:01:58 +0000119 c = 0;
120 for (s = bb_opt_complementaly; s && *s; s++) {
121 t_complementaly *pair;
Manuel Novoa III cad53642003-03-19 09:13:01 +0000122
Eric Andersen8876fb22003-06-20 09:01:58 +0000123 if (*s == ':') {
124 c = 0;
125 continue;
126 }
127 if (c)
128 continue;
129 for (on_off = complementaly; on_off->opt; on_off++)
130 if (on_off->opt == *s)
131 break;
132 pair = on_off;
133 for(s++; *s && *s != ':'; s++) {
134 if (*s == '-' || *s == '~') {
135 c = *s;
136 } else if(*s == '*') {
137 pair->list_flg++;
138 } else {
139 unsigned long *pair_switch = &(pair->switch_on);
140
141 if(c)
142 pair_switch = c == '-' ? &(pair->switch_off) : &(pair->incongruously);
143 for (on_off = complementaly; on_off->opt; on_off++)
144 if (on_off->opt == *s) {
145 *pair_switch |= on_off->switch_on;
146 break;
147 }
148 }
149 }
150 s--;
151 }
152
153 while ((c = getopt_long (argc, argv, applet_opts,
154 bb_applet_long_options, NULL)) > 0) {
Eric Andersen8876fb22003-06-20 09:01:58 +0000155 for (on_off = complementaly; on_off->opt != c; on_off++) {
156 if(!on_off->opt)
Glenn L McGrath850b05f2003-12-19 10:13:10 +0000157 bb_show_usage ();
Eric Andersen8876fb22003-06-20 09:01:58 +0000158 }
159 if(flags & on_off->incongruously)
160 flags |= 0x80000000UL;
161 flags &= ~on_off->switch_off;
162 flags |= on_off->switch_on;
163 if(on_off->list_flg) {
164 *(llist_t **)(on_off->optarg) =
165 llist_add_to(*(llist_t **)(on_off->optarg), optarg);
166 } else if (on_off->optarg) {
167 *(char **)(on_off->optarg) = optarg;
168 }
169 }
Glenn L McGrath85c51522004-01-22 07:10:13 +0000170 return flags;
Manuel Novoa III cad53642003-03-19 09:13:01 +0000171}