forked from wryun/es-shell
-
Notifications
You must be signed in to change notification settings - Fork 0
/
opt.c
99 lines (89 loc) · 1.89 KB
/
opt.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
/* opt.c -- option parsing ($Revision: 1.1.1.1 $) */
#include "es.h"
static const char *usage, *invoker;
static List *args;
static Term *termarg;
static int nextchar;
static Boolean throwonerr;
extern void esoptbegin(List *list, const char *caller, const char *usagemsg, Boolean throws) {
static Boolean initialized = FALSE;
if (!initialized) {
initialized = TRUE;
globalroot(&usage);
globalroot(&invoker);
globalroot(&args);
globalroot(&termarg);
}
assert(usage == NULL);
usage = usagemsg;
invoker = caller;
args = list;
termarg = NULL;
nextchar = 0;
throwonerr = throws;
}
extern int esopt(const char *options) {
int c;
const char *arg, *opt;
assert(!throwonerr || usage != NULL);
assert(termarg == NULL);
if (nextchar == 0) {
if (args == NULL)
return EOF;
assert(args->term != NULL);
arg = getstr(args->term);
if (*arg != '-')
return EOF;
if (arg[1] == '-' && arg[2] == '\0') {
args = args->next;
return EOF;
}
nextchar = 1;
} else {
assert(args != NULL && args->term != NULL);
arg = getstr(args->term);
}
c = arg[nextchar++];
opt = strchr(options, c);
if (opt == NULL) {
const char *msg = usage;
usage = NULL;
args = NULL;
nextchar = 0;
if (throwonerr)
fail(invoker, "illegal option: -%c -- usage: %s", c, msg);
else return '?';
}
if (arg[nextchar] == '\0') {
nextchar = 0;
args = args->next;
}
if (opt[1] == ':') {
if (args == NULL) {
const char *msg = usage;
if (throwonerr)
fail(invoker,
"option -%c expects an argument -- usage: %s",
c, msg);
else return ':';
}
termarg = (nextchar == 0)
? args->term
: mkstr(gcdup(arg + nextchar));
nextchar = 0;
args = args->next;
}
return c;
}
extern Term *esoptarg(void) {
Term *t = termarg;
assert(t != NULL);
termarg = NULL;
return t;
}
extern List *esoptend(void) {
List *result = args;
args = NULL;
usage = NULL;
return result;
}