| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
/* |
|
2
|
|
|
|
|
|
|
* Copyright (C) the libgit2 contributors. All rights reserved. |
|
3
|
|
|
|
|
|
|
* |
|
4
|
|
|
|
|
|
|
* This file is part of libgit2, distributed under the GNU GPL v2 with |
|
5
|
|
|
|
|
|
|
* a Linking Exception. For full terms see the included COPYING file. |
|
6
|
|
|
|
|
|
|
* |
|
7
|
|
|
|
|
|
|
* Do shell-style pattern matching for ?, \, [], and * characters. |
|
8
|
|
|
|
|
|
|
* It is 8bit clean. |
|
9
|
|
|
|
|
|
|
* |
|
10
|
|
|
|
|
|
|
* Written by Rich $alz, mirror!rs, Wed Nov 26 19:03:17 EST 1986. |
|
11
|
|
|
|
|
|
|
* Rich $alz is now . |
|
12
|
|
|
|
|
|
|
* |
|
13
|
|
|
|
|
|
|
* Modified by Wayne Davison to special-case '/' matching, to make '**' |
|
14
|
|
|
|
|
|
|
* work differently than '*', and to fix the character-class code. |
|
15
|
|
|
|
|
|
|
* |
|
16
|
|
|
|
|
|
|
* Imported from git.git. |
|
17
|
|
|
|
|
|
|
*/ |
|
18
|
|
|
|
|
|
|
|
|
19
|
|
|
|
|
|
|
#include "wildmatch.h" |
|
20
|
|
|
|
|
|
|
|
|
21
|
|
|
|
|
|
|
#define GIT_SPACE 0x01 |
|
22
|
|
|
|
|
|
|
#define GIT_DIGIT 0x02 |
|
23
|
|
|
|
|
|
|
#define GIT_ALPHA 0x04 |
|
24
|
|
|
|
|
|
|
#define GIT_GLOB_SPECIAL 0x08 |
|
25
|
|
|
|
|
|
|
#define GIT_REGEX_SPECIAL 0x10 |
|
26
|
|
|
|
|
|
|
#define GIT_PATHSPEC_MAGIC 0x20 |
|
27
|
|
|
|
|
|
|
#define GIT_CNTRL 0x40 |
|
28
|
|
|
|
|
|
|
#define GIT_PUNCT 0x80 |
|
29
|
|
|
|
|
|
|
|
|
30
|
|
|
|
|
|
|
enum { |
|
31
|
|
|
|
|
|
|
S = GIT_SPACE, |
|
32
|
|
|
|
|
|
|
A = GIT_ALPHA, |
|
33
|
|
|
|
|
|
|
D = GIT_DIGIT, |
|
34
|
|
|
|
|
|
|
G = GIT_GLOB_SPECIAL, /* *, ?, [, \\ */ |
|
35
|
|
|
|
|
|
|
R = GIT_REGEX_SPECIAL, /* $, (, ), +, ., ^, {, | */ |
|
36
|
|
|
|
|
|
|
P = GIT_PATHSPEC_MAGIC, /* other non-alnum, except for ] and } */ |
|
37
|
|
|
|
|
|
|
X = GIT_CNTRL, |
|
38
|
|
|
|
|
|
|
U = GIT_PUNCT, |
|
39
|
|
|
|
|
|
|
Z = GIT_CNTRL | GIT_SPACE |
|
40
|
|
|
|
|
|
|
}; |
|
41
|
|
|
|
|
|
|
|
|
42
|
|
|
|
|
|
|
static const unsigned char sane_ctype[256] = { |
|
43
|
|
|
|
|
|
|
X, X, X, X, X, X, X, X, X, Z, Z, X, X, Z, X, X, /* 0.. 15 */ |
|
44
|
|
|
|
|
|
|
X, X, X, X, X, X, X, X, X, X, X, X, X, X, X, X, /* 16.. 31 */ |
|
45
|
|
|
|
|
|
|
S, P, P, P, R, P, P, P, R, R, G, R, P, P, R, P, /* 32.. 47 */ |
|
46
|
|
|
|
|
|
|
D, D, D, D, D, D, D, D, D, D, P, P, P, P, P, G, /* 48.. 63 */ |
|
47
|
|
|
|
|
|
|
P, A, A, A, A, A, A, A, A, A, A, A, A, A, A, A, /* 64.. 79 */ |
|
48
|
|
|
|
|
|
|
A, A, A, A, A, A, A, A, A, A, A, G, G, U, R, P, /* 80.. 95 */ |
|
49
|
|
|
|
|
|
|
P, A, A, A, A, A, A, A, A, A, A, A, A, A, A, A, /* 96..111 */ |
|
50
|
|
|
|
|
|
|
A, A, A, A, A, A, A, A, A, A, A, R, R, U, P, X, /* 112..127 */ |
|
51
|
|
|
|
|
|
|
/* Nothing in the 128.. range */ |
|
52
|
|
|
|
|
|
|
}; |
|
53
|
|
|
|
|
|
|
|
|
54
|
|
|
|
|
|
|
#define sane_istest(x,mask) ((sane_ctype[(unsigned char)(x)] & (mask)) != 0) |
|
55
|
|
|
|
|
|
|
#define is_glob_special(x) sane_istest(x,GIT_GLOB_SPECIAL) |
|
56
|
|
|
|
|
|
|
|
|
57
|
|
|
|
|
|
|
typedef unsigned char uchar; |
|
58
|
|
|
|
|
|
|
|
|
59
|
|
|
|
|
|
|
/* What character marks an inverted character class? */ |
|
60
|
|
|
|
|
|
|
#define NEGATE_CLASS '!' |
|
61
|
|
|
|
|
|
|
#define NEGATE_CLASS2 '^' |
|
62
|
|
|
|
|
|
|
|
|
63
|
|
|
|
|
|
|
#define CC_EQ(class, len, litmatch) ((len) == sizeof (litmatch)-1 \ |
|
64
|
|
|
|
|
|
|
&& *(class) == *(litmatch) \ |
|
65
|
|
|
|
|
|
|
&& strncmp((char*)class, litmatch, len) == 0) |
|
66
|
|
|
|
|
|
|
|
|
67
|
|
|
|
|
|
|
#if defined STDC_HEADERS || !defined isascii |
|
68
|
|
|
|
|
|
|
# define ISASCII(c) 1 |
|
69
|
|
|
|
|
|
|
#else |
|
70
|
|
|
|
|
|
|
# define ISASCII(c) isascii(c) |
|
71
|
|
|
|
|
|
|
#endif |
|
72
|
|
|
|
|
|
|
|
|
73
|
|
|
|
|
|
|
#ifdef isblank |
|
74
|
|
|
|
|
|
|
# define ISBLANK(c) (ISASCII(c) && isblank(c)) |
|
75
|
|
|
|
|
|
|
#else |
|
76
|
|
|
|
|
|
|
# define ISBLANK(c) ((c) == ' ' || (c) == '\t') |
|
77
|
|
|
|
|
|
|
#endif |
|
78
|
|
|
|
|
|
|
|
|
79
|
|
|
|
|
|
|
#ifdef isgraph |
|
80
|
|
|
|
|
|
|
# define ISGRAPH(c) (ISASCII(c) && isgraph(c)) |
|
81
|
|
|
|
|
|
|
#else |
|
82
|
|
|
|
|
|
|
# define ISGRAPH(c) (ISASCII(c) && isprint(c) && !isspace(c)) |
|
83
|
|
|
|
|
|
|
#endif |
|
84
|
|
|
|
|
|
|
|
|
85
|
|
|
|
|
|
|
#define ISPRINT(c) (ISASCII(c) && isprint(c)) |
|
86
|
|
|
|
|
|
|
#define ISDIGIT(c) (ISASCII(c) && isdigit(c)) |
|
87
|
|
|
|
|
|
|
#define ISALNUM(c) (ISASCII(c) && isalnum(c)) |
|
88
|
|
|
|
|
|
|
#define ISALPHA(c) (ISASCII(c) && isalpha(c)) |
|
89
|
|
|
|
|
|
|
#define ISCNTRL(c) (ISASCII(c) && iscntrl(c)) |
|
90
|
|
|
|
|
|
|
#define ISLOWER(c) (ISASCII(c) && islower(c)) |
|
91
|
|
|
|
|
|
|
#define ISPUNCT(c) (ISASCII(c) && ispunct(c)) |
|
92
|
|
|
|
|
|
|
#define ISSPACE(c) (ISASCII(c) && isspace(c)) |
|
93
|
|
|
|
|
|
|
#define ISUPPER(c) (ISASCII(c) && isupper(c)) |
|
94
|
|
|
|
|
|
|
#define ISXDIGIT(c) (ISASCII(c) && isxdigit(c)) |
|
95
|
|
|
|
|
|
|
|
|
96
|
|
|
|
|
|
|
/* Match pattern "p" against "text" */ |
|
97
|
2382
|
|
|
|
|
|
static int dowild(const uchar *p, const uchar *text, unsigned int flags) |
|
98
|
|
|
|
|
|
|
{ |
|
99
|
|
|
|
|
|
|
uchar p_ch; |
|
100
|
2382
|
|
|
|
|
|
const uchar *pattern = p; |
|
101
|
|
|
|
|
|
|
|
|
102
|
2609
|
100
|
|
|
|
|
for ( ; (p_ch = *p) != '\0'; text++, p++) { |
|
103
|
|
|
|
|
|
|
int matched, match_slash, negated; |
|
104
|
|
|
|
|
|
|
uchar t_ch, prev_ch; |
|
105
|
2595
|
100
|
|
|
|
|
if ((t_ch = *text) == '\0' && p_ch != '*') |
|
|
|
100
|
|
|
|
|
|
|
106
|
552
|
|
|
|
|
|
return WM_ABORT_ALL; |
|
107
|
2043
|
100
|
|
|
|
|
if ((flags & WM_CASEFOLD) && ISUPPER(t_ch)) |
|
|
|
50
|
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
108
|
0
|
|
|
|
|
|
t_ch = tolower(t_ch); |
|
109
|
2043
|
100
|
|
|
|
|
if ((flags & WM_CASEFOLD) && ISUPPER(p_ch)) |
|
|
|
50
|
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
110
|
8
|
|
|
|
|
|
p_ch = tolower(p_ch); |
|
111
|
2043
|
|
|
|
|
|
switch (p_ch) { |
|
112
|
|
|
|
|
|
|
case '\\': |
|
113
|
|
|
|
|
|
|
/* Literal match with following character. Note that the test |
|
114
|
|
|
|
|
|
|
* in "default" handles the p[1] == '\0' failure case. */ |
|
115
|
0
|
|
|
|
|
|
p_ch = *++p; |
|
116
|
|
|
|
|
|
|
/* FALLTHROUGH */ |
|
117
|
|
|
|
|
|
|
default: |
|
118
|
2016
|
100
|
|
|
|
|
if (t_ch != p_ch) |
|
119
|
1789
|
|
|
|
|
|
return WM_NOMATCH; |
|
120
|
227
|
|
|
|
|
|
continue; |
|
121
|
|
|
|
|
|
|
case '?': |
|
122
|
|
|
|
|
|
|
/* Match anything but '/'. */ |
|
123
|
0
|
0
|
|
|
|
|
if ((flags & WM_PATHNAME) && t_ch == '/') |
|
|
|
0
|
|
|
|
|
|
|
124
|
0
|
|
|
|
|
|
return WM_NOMATCH; |
|
125
|
0
|
|
|
|
|
|
continue; |
|
126
|
|
|
|
|
|
|
case '*': |
|
127
|
27
|
100
|
|
|
|
|
if (*++p == '*') { |
|
128
|
3
|
|
|
|
|
|
const uchar *prev_p = p - 2; |
|
129
|
3
|
50
|
|
|
|
|
while (*++p == '*') {} |
|
130
|
3
|
50
|
|
|
|
|
if (!(flags & WM_PATHNAME)) |
|
131
|
|
|
|
|
|
|
/* without WM_PATHNAME, '*' == '**' */ |
|
132
|
3
|
|
|
|
|
|
match_slash = 1; |
|
133
|
0
|
0
|
|
|
|
|
else if ((prev_p < pattern || *prev_p == '/') && |
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
134
|
0
|
0
|
|
|
|
|
(*p == '\0' || *p == '/' || |
|
|
|
0
|
|
|
|
|
|
|
135
|
0
|
0
|
|
|
|
|
(p[0] == '\\' && p[1] == '/'))) { |
|
136
|
|
|
|
|
|
|
/* |
|
137
|
|
|
|
|
|
|
* Assuming we already match 'foo/' and are at |
|
138
|
|
|
|
|
|
|
* , just assume it matches |
|
139
|
|
|
|
|
|
|
* nothing and go ahead match the rest of the |
|
140
|
|
|
|
|
|
|
* pattern with the remaining string. This |
|
141
|
|
|
|
|
|
|
* helps make foo/<*><*>/bar (<> because |
|
142
|
|
|
|
|
|
|
* otherwise it breaks C comment syntax) match |
|
143
|
|
|
|
|
|
|
* both foo/bar and foo/a/bar. |
|
144
|
|
|
|
|
|
|
*/ |
|
145
|
0
|
|
|
|
|
|
if (p[0] == '/' && |
|
146
|
0
|
|
|
|
|
|
dowild(p + 1, text, flags) == WM_MATCH) |
|
147
|
0
|
|
|
|
|
|
return WM_MATCH; |
|
148
|
0
|
|
|
|
|
|
match_slash = 1; |
|
149
|
|
|
|
|
|
|
} else /* WM_PATHNAME is set */ |
|
150
|
3
|
|
|
|
|
|
match_slash = 0; |
|
151
|
|
|
|
|
|
|
} else |
|
152
|
|
|
|
|
|
|
/* without WM_PATHNAME, '*' == '**' */ |
|
153
|
24
|
|
|
|
|
|
match_slash = flags & WM_PATHNAME ? 0 : 1; |
|
154
|
27
|
100
|
|
|
|
|
if (*p == '\0') { |
|
155
|
|
|
|
|
|
|
/* Trailing "**" matches everything. Trailing "*" matches |
|
156
|
|
|
|
|
|
|
* only if there are no more slash characters. */ |
|
157
|
24
|
50
|
|
|
|
|
if (!match_slash) { |
|
158
|
0
|
0
|
|
|
|
|
if (strchr((char*)text, '/') != NULL) |
|
159
|
0
|
|
|
|
|
|
return WM_NOMATCH; |
|
160
|
|
|
|
|
|
|
} |
|
161
|
24
|
|
|
|
|
|
return WM_MATCH; |
|
162
|
3
|
50
|
|
|
|
|
} else if (!match_slash && *p == '/') { |
|
|
|
0
|
|
|
|
|
|
|
163
|
|
|
|
|
|
|
/* |
|
164
|
|
|
|
|
|
|
* _one_ asterisk followed by a slash |
|
165
|
|
|
|
|
|
|
* with WM_PATHNAME matches the next |
|
166
|
|
|
|
|
|
|
* directory |
|
167
|
|
|
|
|
|
|
*/ |
|
168
|
0
|
|
|
|
|
|
const char *slash = strchr((char*)text, '/'); |
|
169
|
0
|
0
|
|
|
|
|
if (!slash) |
|
170
|
0
|
|
|
|
|
|
return WM_NOMATCH; |
|
171
|
0
|
|
|
|
|
|
text = (const uchar*)slash; |
|
172
|
|
|
|
|
|
|
/* the slash is consumed by the top-level for loop */ |
|
173
|
0
|
|
|
|
|
|
break; |
|
174
|
|
|
|
|
|
|
} |
|
175
|
|
|
|
|
|
|
while (1) { |
|
176
|
6
|
50
|
|
|
|
|
if (t_ch == '\0') |
|
177
|
0
|
|
|
|
|
|
break; |
|
178
|
|
|
|
|
|
|
/* |
|
179
|
|
|
|
|
|
|
* Try to advance faster when an asterisk is |
|
180
|
|
|
|
|
|
|
* followed by a literal. We know in this case |
|
181
|
|
|
|
|
|
|
* that the string before the literal |
|
182
|
|
|
|
|
|
|
* must belong to "*". |
|
183
|
|
|
|
|
|
|
* If match_slash is false, do not look past |
|
184
|
|
|
|
|
|
|
* the first slash as it cannot belong to '*'. |
|
185
|
|
|
|
|
|
|
*/ |
|
186
|
6
|
50
|
|
|
|
|
if (!is_glob_special(*p)) { |
|
187
|
6
|
|
|
|
|
|
p_ch = *p; |
|
188
|
6
|
50
|
|
|
|
|
if ((flags & WM_CASEFOLD) && ISUPPER(p_ch)) |
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
189
|
0
|
|
|
|
|
|
p_ch = tolower(p_ch); |
|
190
|
32
|
100
|
|
|
|
|
while ((t_ch = *text) != '\0' && |
|
|
|
50
|
|
|
|
|
|
|
191
|
0
|
0
|
|
|
|
|
(match_slash || t_ch != '/')) { |
|
192
|
30
|
50
|
|
|
|
|
if ((flags & WM_CASEFOLD) && ISUPPER(t_ch)) |
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
193
|
0
|
|
|
|
|
|
t_ch = tolower(t_ch); |
|
194
|
30
|
100
|
|
|
|
|
if (t_ch == p_ch) |
|
195
|
4
|
|
|
|
|
|
break; |
|
196
|
26
|
|
|
|
|
|
text++; |
|
197
|
|
|
|
|
|
|
} |
|
198
|
6
|
100
|
|
|
|
|
if (t_ch != p_ch) |
|
199
|
2
|
|
|
|
|
|
return WM_NOMATCH; |
|
200
|
|
|
|
|
|
|
} |
|
201
|
4
|
100
|
|
|
|
|
if ((matched = dowild(p, text, flags)) != WM_NOMATCH) { |
|
202
|
1
|
50
|
|
|
|
|
if (!match_slash || matched != WM_ABORT_TO_STARSTAR) |
|
|
|
50
|
|
|
|
|
|
|
203
|
1
|
|
|
|
|
|
return matched; |
|
204
|
3
|
50
|
|
|
|
|
} else if (!match_slash && t_ch == '/') |
|
|
|
0
|
|
|
|
|
|
|
205
|
0
|
|
|
|
|
|
return WM_ABORT_TO_STARSTAR; |
|
206
|
3
|
|
|
|
|
|
t_ch = *++text; |
|
207
|
3
|
|
|
|
|
|
} |
|
208
|
0
|
|
|
|
|
|
return WM_ABORT_ALL; |
|
209
|
|
|
|
|
|
|
case '[': |
|
210
|
0
|
|
|
|
|
|
p_ch = *++p; |
|
211
|
|
|
|
|
|
|
#ifdef NEGATE_CLASS2 |
|
212
|
0
|
0
|
|
|
|
|
if (p_ch == NEGATE_CLASS2) |
|
213
|
0
|
|
|
|
|
|
p_ch = NEGATE_CLASS; |
|
214
|
|
|
|
|
|
|
#endif |
|
215
|
|
|
|
|
|
|
/* Assign literal 1/0 because of "matched" comparison. */ |
|
216
|
0
|
|
|
|
|
|
negated = p_ch == NEGATE_CLASS ? 1 : 0; |
|
217
|
0
|
0
|
|
|
|
|
if (negated) { |
|
218
|
|
|
|
|
|
|
/* Inverted character class. */ |
|
219
|
0
|
|
|
|
|
|
p_ch = *++p; |
|
220
|
|
|
|
|
|
|
} |
|
221
|
0
|
|
|
|
|
|
prev_ch = 0; |
|
222
|
0
|
|
|
|
|
|
matched = 0; |
|
223
|
|
|
|
|
|
|
do { |
|
224
|
0
|
0
|
|
|
|
|
if (!p_ch) |
|
225
|
0
|
|
|
|
|
|
return WM_ABORT_ALL; |
|
226
|
0
|
0
|
|
|
|
|
if (p_ch == '\\') { |
|
227
|
0
|
|
|
|
|
|
p_ch = *++p; |
|
228
|
0
|
0
|
|
|
|
|
if (!p_ch) |
|
229
|
0
|
|
|
|
|
|
return WM_ABORT_ALL; |
|
230
|
0
|
0
|
|
|
|
|
if (t_ch == p_ch) |
|
231
|
0
|
|
|
|
|
|
matched = 1; |
|
232
|
0
|
0
|
|
|
|
|
} else if (p_ch == '-' && prev_ch && p[1] && p[1] != ']') { |
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
233
|
0
|
|
|
|
|
|
p_ch = *++p; |
|
234
|
0
|
0
|
|
|
|
|
if (p_ch == '\\') { |
|
235
|
0
|
|
|
|
|
|
p_ch = *++p; |
|
236
|
0
|
0
|
|
|
|
|
if (!p_ch) |
|
237
|
0
|
|
|
|
|
|
return WM_ABORT_ALL; |
|
238
|
|
|
|
|
|
|
} |
|
239
|
0
|
0
|
|
|
|
|
if (t_ch <= p_ch && t_ch >= prev_ch) |
|
|
|
0
|
|
|
|
|
|
|
240
|
0
|
|
|
|
|
|
matched = 1; |
|
241
|
0
|
0
|
|
|
|
|
else if ((flags & WM_CASEFOLD) && ISLOWER(t_ch)) { |
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
242
|
0
|
|
|
|
|
|
uchar t_ch_upper = toupper(t_ch); |
|
243
|
0
|
0
|
|
|
|
|
if (t_ch_upper <= p_ch && t_ch_upper >= prev_ch) |
|
|
|
0
|
|
|
|
|
|
|
244
|
0
|
|
|
|
|
|
matched = 1; |
|
245
|
|
|
|
|
|
|
} |
|
246
|
0
|
|
|
|
|
|
p_ch = 0; /* This makes "prev_ch" get set to 0. */ |
|
247
|
0
|
0
|
|
|
|
|
} else if (p_ch == '[' && p[1] == ':') { |
|
|
|
0
|
|
|
|
|
|
|
248
|
|
|
|
|
|
|
const uchar *s; |
|
249
|
|
|
|
|
|
|
int i; |
|
250
|
0
|
0
|
|
|
|
|
for (s = p += 2; (p_ch = *p) && p_ch != ']'; p++) {} /*SHARED ITERATOR*/ |
|
|
|
0
|
|
|
|
|
|
|
251
|
0
|
0
|
|
|
|
|
if (!p_ch) |
|
252
|
0
|
|
|
|
|
|
return WM_ABORT_ALL; |
|
253
|
0
|
|
|
|
|
|
i = (int)(p - s - 1); |
|
254
|
0
|
0
|
|
|
|
|
if (i < 0 || p[-1] != ':') { |
|
|
|
0
|
|
|
|
|
|
|
255
|
|
|
|
|
|
|
/* Didn't find ":]", so treat like a normal set. */ |
|
256
|
0
|
|
|
|
|
|
p = s - 2; |
|
257
|
0
|
|
|
|
|
|
p_ch = '['; |
|
258
|
0
|
0
|
|
|
|
|
if (t_ch == p_ch) |
|
259
|
0
|
|
|
|
|
|
matched = 1; |
|
260
|
0
|
|
|
|
|
|
continue; |
|
261
|
|
|
|
|
|
|
} |
|
262
|
0
|
0
|
|
|
|
|
if (CC_EQ(s,i, "alnum")) { |
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
263
|
0
|
0
|
|
|
|
|
if (ISALNUM(t_ch)) |
|
|
|
0
|
|
|
|
|
|
|
264
|
0
|
|
|
|
|
|
matched = 1; |
|
265
|
0
|
0
|
|
|
|
|
} else if (CC_EQ(s,i, "alpha")) { |
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
266
|
0
|
0
|
|
|
|
|
if (ISALPHA(t_ch)) |
|
|
|
0
|
|
|
|
|
|
|
267
|
0
|
|
|
|
|
|
matched = 1; |
|
268
|
0
|
0
|
|
|
|
|
} else if (CC_EQ(s,i, "blank")) { |
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
269
|
0
|
0
|
|
|
|
|
if (ISBLANK(t_ch)) |
|
|
|
0
|
|
|
|
|
|
|
270
|
0
|
|
|
|
|
|
matched = 1; |
|
271
|
0
|
0
|
|
|
|
|
} else if (CC_EQ(s,i, "cntrl")) { |
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
272
|
0
|
0
|
|
|
|
|
if (ISCNTRL(t_ch)) |
|
|
|
0
|
|
|
|
|
|
|
273
|
0
|
|
|
|
|
|
matched = 1; |
|
274
|
0
|
0
|
|
|
|
|
} else if (CC_EQ(s,i, "digit")) { |
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
275
|
0
|
0
|
|
|
|
|
if (ISDIGIT(t_ch)) |
|
|
|
0
|
|
|
|
|
|
|
276
|
0
|
|
|
|
|
|
matched = 1; |
|
277
|
0
|
0
|
|
|
|
|
} else if (CC_EQ(s,i, "graph")) { |
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
278
|
0
|
0
|
|
|
|
|
if (ISGRAPH(t_ch)) |
|
|
|
0
|
|
|
|
|
|
|
279
|
0
|
|
|
|
|
|
matched = 1; |
|
280
|
0
|
0
|
|
|
|
|
} else if (CC_EQ(s,i, "lower")) { |
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
281
|
0
|
0
|
|
|
|
|
if (ISLOWER(t_ch)) |
|
|
|
0
|
|
|
|
|
|
|
282
|
0
|
|
|
|
|
|
matched = 1; |
|
283
|
0
|
0
|
|
|
|
|
} else if (CC_EQ(s,i, "print")) { |
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
284
|
0
|
0
|
|
|
|
|
if (ISPRINT(t_ch)) |
|
|
|
0
|
|
|
|
|
|
|
285
|
0
|
|
|
|
|
|
matched = 1; |
|
286
|
0
|
0
|
|
|
|
|
} else if (CC_EQ(s,i, "punct")) { |
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
287
|
0
|
0
|
|
|
|
|
if (ISPUNCT(t_ch)) |
|
|
|
0
|
|
|
|
|
|
|
288
|
0
|
|
|
|
|
|
matched = 1; |
|
289
|
0
|
0
|
|
|
|
|
} else if (CC_EQ(s,i, "space")) { |
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
290
|
0
|
0
|
|
|
|
|
if (ISSPACE(t_ch)) |
|
|
|
0
|
|
|
|
|
|
|
291
|
0
|
|
|
|
|
|
matched = 1; |
|
292
|
0
|
0
|
|
|
|
|
} else if (CC_EQ(s,i, "upper")) { |
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
293
|
0
|
0
|
|
|
|
|
if (ISUPPER(t_ch)) |
|
|
|
0
|
|
|
|
|
|
|
294
|
0
|
|
|
|
|
|
matched = 1; |
|
295
|
0
|
0
|
|
|
|
|
else if ((flags & WM_CASEFOLD) && ISLOWER(t_ch)) |
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
296
|
0
|
|
|
|
|
|
matched = 1; |
|
297
|
0
|
0
|
|
|
|
|
} else if (CC_EQ(s,i, "xdigit")) { |
|
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
298
|
0
|
0
|
|
|
|
|
if (ISXDIGIT(t_ch)) |
|
|
|
0
|
|
|
|
|
|
|
299
|
0
|
|
|
|
|
|
matched = 1; |
|
300
|
|
|
|
|
|
|
} else /* malformed [:class:] string */ |
|
301
|
0
|
|
|
|
|
|
return WM_ABORT_ALL; |
|
302
|
0
|
|
|
|
|
|
p_ch = 0; /* This makes "prev_ch" get set to 0. */ |
|
303
|
0
|
0
|
|
|
|
|
} else if (t_ch == p_ch) |
|
304
|
0
|
|
|
|
|
|
matched = 1; |
|
305
|
0
|
0
|
|
|
|
|
} while (prev_ch = p_ch, (p_ch = *++p) != ']'); |
|
306
|
0
|
0
|
|
|
|
|
if (matched == negated || |
|
|
|
0
|
|
|
|
|
|
|
307
|
0
|
0
|
|
|
|
|
((flags & WM_PATHNAME) && t_ch == '/')) |
|
308
|
0
|
|
|
|
|
|
return WM_NOMATCH; |
|
309
|
0
|
|
|
|
|
|
continue; |
|
310
|
|
|
|
|
|
|
} |
|
311
|
|
|
|
|
|
|
} |
|
312
|
|
|
|
|
|
|
|
|
313
|
14
|
|
|
|
|
|
return *text ? WM_NOMATCH : WM_MATCH; |
|
314
|
|
|
|
|
|
|
} |
|
315
|
|
|
|
|
|
|
|
|
316
|
|
|
|
|
|
|
/* Match the "pattern" against the "text" string. */ |
|
317
|
2378
|
|
|
|
|
|
int wildmatch(const char *pattern, const char *text, unsigned int flags) |
|
318
|
|
|
|
|
|
|
{ |
|
319
|
2378
|
|
|
|
|
|
return dowild((const uchar*)pattern, (const uchar*)text, flags); |
|
320
|
|
|
|
|
|
|
} |