xref: /aosp_15_r20/external/musl/src/regex/fnmatch.c (revision c9945492fdd68bbe62686c5b452b4dc1be3f8453)
1*c9945492SAndroid Build Coastguard Worker /*
2*c9945492SAndroid Build Coastguard Worker  * An implementation of what I call the "Sea of Stars" algorithm for
3*c9945492SAndroid Build Coastguard Worker  * POSIX fnmatch(). The basic idea is that we factor the pattern into
4*c9945492SAndroid Build Coastguard Worker  * a head component (which we match first and can reject without ever
5*c9945492SAndroid Build Coastguard Worker  * measuring the length of the string), an optional tail component
6*c9945492SAndroid Build Coastguard Worker  * (which only exists if the pattern contains at least one star), and
7*c9945492SAndroid Build Coastguard Worker  * an optional "sea of stars", a set of star-separated components
8*c9945492SAndroid Build Coastguard Worker  * between the head and tail. After the head and tail matches have
9*c9945492SAndroid Build Coastguard Worker  * been removed from the input string, the components in the "sea of
10*c9945492SAndroid Build Coastguard Worker  * stars" are matched sequentially by searching for their first
11*c9945492SAndroid Build Coastguard Worker  * occurrence past the end of the previous match.
12*c9945492SAndroid Build Coastguard Worker  *
13*c9945492SAndroid Build Coastguard Worker  * - Rich Felker, April 2012
14*c9945492SAndroid Build Coastguard Worker  */
15*c9945492SAndroid Build Coastguard Worker 
16*c9945492SAndroid Build Coastguard Worker #include <string.h>
17*c9945492SAndroid Build Coastguard Worker #include <fnmatch.h>
18*c9945492SAndroid Build Coastguard Worker #include <stdlib.h>
19*c9945492SAndroid Build Coastguard Worker #include <wchar.h>
20*c9945492SAndroid Build Coastguard Worker #include <wctype.h>
21*c9945492SAndroid Build Coastguard Worker #include "locale_impl.h"
22*c9945492SAndroid Build Coastguard Worker 
23*c9945492SAndroid Build Coastguard Worker #define END 0
24*c9945492SAndroid Build Coastguard Worker #define UNMATCHABLE -2
25*c9945492SAndroid Build Coastguard Worker #define BRACKET -3
26*c9945492SAndroid Build Coastguard Worker #define QUESTION -4
27*c9945492SAndroid Build Coastguard Worker #define STAR -5
28*c9945492SAndroid Build Coastguard Worker 
str_next(const char * str,size_t n,size_t * step)29*c9945492SAndroid Build Coastguard Worker static int str_next(const char *str, size_t n, size_t *step)
30*c9945492SAndroid Build Coastguard Worker {
31*c9945492SAndroid Build Coastguard Worker 	if (!n) {
32*c9945492SAndroid Build Coastguard Worker 		*step = 0;
33*c9945492SAndroid Build Coastguard Worker 		return 0;
34*c9945492SAndroid Build Coastguard Worker 	}
35*c9945492SAndroid Build Coastguard Worker 	if (str[0] >= 128U) {
36*c9945492SAndroid Build Coastguard Worker 		wchar_t wc;
37*c9945492SAndroid Build Coastguard Worker 		int k = mbtowc(&wc, str, n);
38*c9945492SAndroid Build Coastguard Worker 		if (k<0) {
39*c9945492SAndroid Build Coastguard Worker 			*step = 1;
40*c9945492SAndroid Build Coastguard Worker 			return -1;
41*c9945492SAndroid Build Coastguard Worker 		}
42*c9945492SAndroid Build Coastguard Worker 		*step = k;
43*c9945492SAndroid Build Coastguard Worker 		return wc;
44*c9945492SAndroid Build Coastguard Worker 	}
45*c9945492SAndroid Build Coastguard Worker 	*step = 1;
46*c9945492SAndroid Build Coastguard Worker 	return str[0];
47*c9945492SAndroid Build Coastguard Worker }
48*c9945492SAndroid Build Coastguard Worker 
pat_next(const char * pat,size_t m,size_t * step,int flags)49*c9945492SAndroid Build Coastguard Worker static int pat_next(const char *pat, size_t m, size_t *step, int flags)
50*c9945492SAndroid Build Coastguard Worker {
51*c9945492SAndroid Build Coastguard Worker 	int esc = 0;
52*c9945492SAndroid Build Coastguard Worker 	if (!m || !*pat) {
53*c9945492SAndroid Build Coastguard Worker 		*step = 0;
54*c9945492SAndroid Build Coastguard Worker 		return END;
55*c9945492SAndroid Build Coastguard Worker 	}
56*c9945492SAndroid Build Coastguard Worker 	*step = 1;
57*c9945492SAndroid Build Coastguard Worker 	if (pat[0]=='\\' && pat[1] && !(flags & FNM_NOESCAPE)) {
58*c9945492SAndroid Build Coastguard Worker 		*step = 2;
59*c9945492SAndroid Build Coastguard Worker 		pat++;
60*c9945492SAndroid Build Coastguard Worker 		esc = 1;
61*c9945492SAndroid Build Coastguard Worker 		goto escaped;
62*c9945492SAndroid Build Coastguard Worker 	}
63*c9945492SAndroid Build Coastguard Worker 	if (pat[0]=='[') {
64*c9945492SAndroid Build Coastguard Worker 		size_t k = 1;
65*c9945492SAndroid Build Coastguard Worker 		if (k<m) if (pat[k] == '^' || pat[k] == '!') k++;
66*c9945492SAndroid Build Coastguard Worker 		if (k<m) if (pat[k] == ']') k++;
67*c9945492SAndroid Build Coastguard Worker 		for (; k<m && pat[k] && pat[k]!=']'; k++) {
68*c9945492SAndroid Build Coastguard Worker 			if (k+1<m && pat[k+1] && pat[k]=='[' && (pat[k+1]==':' || pat[k+1]=='.' || pat[k+1]=='=')) {
69*c9945492SAndroid Build Coastguard Worker 				int z = pat[k+1];
70*c9945492SAndroid Build Coastguard Worker 				k+=2;
71*c9945492SAndroid Build Coastguard Worker 				if (k<m && pat[k]) k++;
72*c9945492SAndroid Build Coastguard Worker 				while (k<m && pat[k] && (pat[k-1]!=z || pat[k]!=']')) k++;
73*c9945492SAndroid Build Coastguard Worker 				if (k==m || !pat[k]) break;
74*c9945492SAndroid Build Coastguard Worker 			}
75*c9945492SAndroid Build Coastguard Worker 		}
76*c9945492SAndroid Build Coastguard Worker 		if (k==m || !pat[k]) {
77*c9945492SAndroid Build Coastguard Worker 			*step = 1;
78*c9945492SAndroid Build Coastguard Worker 			return '[';
79*c9945492SAndroid Build Coastguard Worker 		}
80*c9945492SAndroid Build Coastguard Worker 		*step = k+1;
81*c9945492SAndroid Build Coastguard Worker 		return BRACKET;
82*c9945492SAndroid Build Coastguard Worker 	}
83*c9945492SAndroid Build Coastguard Worker 	if (pat[0] == '*')
84*c9945492SAndroid Build Coastguard Worker 		return STAR;
85*c9945492SAndroid Build Coastguard Worker 	if (pat[0] == '?')
86*c9945492SAndroid Build Coastguard Worker 		return QUESTION;
87*c9945492SAndroid Build Coastguard Worker escaped:
88*c9945492SAndroid Build Coastguard Worker 	if (pat[0] >= 128U) {
89*c9945492SAndroid Build Coastguard Worker 		wchar_t wc;
90*c9945492SAndroid Build Coastguard Worker 		int k = mbtowc(&wc, pat, m);
91*c9945492SAndroid Build Coastguard Worker 		if (k<0) {
92*c9945492SAndroid Build Coastguard Worker 			*step = 0;
93*c9945492SAndroid Build Coastguard Worker 			return UNMATCHABLE;
94*c9945492SAndroid Build Coastguard Worker 		}
95*c9945492SAndroid Build Coastguard Worker 		*step = k + esc;
96*c9945492SAndroid Build Coastguard Worker 		return wc;
97*c9945492SAndroid Build Coastguard Worker 	}
98*c9945492SAndroid Build Coastguard Worker 	return pat[0];
99*c9945492SAndroid Build Coastguard Worker }
100*c9945492SAndroid Build Coastguard Worker 
casefold(int k)101*c9945492SAndroid Build Coastguard Worker static int casefold(int k)
102*c9945492SAndroid Build Coastguard Worker {
103*c9945492SAndroid Build Coastguard Worker 	int c = towupper(k);
104*c9945492SAndroid Build Coastguard Worker 	return c == k ? towlower(k) : c;
105*c9945492SAndroid Build Coastguard Worker }
106*c9945492SAndroid Build Coastguard Worker 
match_bracket(const char * p,int k,int kfold)107*c9945492SAndroid Build Coastguard Worker static int match_bracket(const char *p, int k, int kfold)
108*c9945492SAndroid Build Coastguard Worker {
109*c9945492SAndroid Build Coastguard Worker 	wchar_t wc;
110*c9945492SAndroid Build Coastguard Worker 	int inv = 0;
111*c9945492SAndroid Build Coastguard Worker 	p++;
112*c9945492SAndroid Build Coastguard Worker 	if (*p=='^' || *p=='!') {
113*c9945492SAndroid Build Coastguard Worker 		inv = 1;
114*c9945492SAndroid Build Coastguard Worker 		p++;
115*c9945492SAndroid Build Coastguard Worker 	}
116*c9945492SAndroid Build Coastguard Worker 	if (*p==']') {
117*c9945492SAndroid Build Coastguard Worker 		if (k==']') return !inv;
118*c9945492SAndroid Build Coastguard Worker 		p++;
119*c9945492SAndroid Build Coastguard Worker 	} else if (*p=='-') {
120*c9945492SAndroid Build Coastguard Worker 		if (k=='-') return !inv;
121*c9945492SAndroid Build Coastguard Worker 		p++;
122*c9945492SAndroid Build Coastguard Worker 	}
123*c9945492SAndroid Build Coastguard Worker 	wc = p[-1];
124*c9945492SAndroid Build Coastguard Worker 	for (; *p != ']'; p++) {
125*c9945492SAndroid Build Coastguard Worker 		if (p[0]=='-' && p[1]!=']') {
126*c9945492SAndroid Build Coastguard Worker 			wchar_t wc2;
127*c9945492SAndroid Build Coastguard Worker 			int l = mbtowc(&wc2, p+1, 4);
128*c9945492SAndroid Build Coastguard Worker 			if (l < 0) return 0;
129*c9945492SAndroid Build Coastguard Worker 			if (wc <= wc2)
130*c9945492SAndroid Build Coastguard Worker 				if ((unsigned)k-wc <= wc2-wc ||
131*c9945492SAndroid Build Coastguard Worker 				    (unsigned)kfold-wc <= wc2-wc)
132*c9945492SAndroid Build Coastguard Worker 					return !inv;
133*c9945492SAndroid Build Coastguard Worker 			p += l-1;
134*c9945492SAndroid Build Coastguard Worker 			continue;
135*c9945492SAndroid Build Coastguard Worker 		}
136*c9945492SAndroid Build Coastguard Worker 		if (p[0]=='[' && (p[1]==':' || p[1]=='.' || p[1]=='=')) {
137*c9945492SAndroid Build Coastguard Worker 			const char *p0 = p+2;
138*c9945492SAndroid Build Coastguard Worker 			int z = p[1];
139*c9945492SAndroid Build Coastguard Worker 			p+=3;
140*c9945492SAndroid Build Coastguard Worker 			while (p[-1]!=z || p[0]!=']') p++;
141*c9945492SAndroid Build Coastguard Worker 			if (z == ':' && p-1-p0 < 16) {
142*c9945492SAndroid Build Coastguard Worker 				char buf[16];
143*c9945492SAndroid Build Coastguard Worker 				memcpy(buf, p0, p-1-p0);
144*c9945492SAndroid Build Coastguard Worker 				buf[p-1-p0] = 0;
145*c9945492SAndroid Build Coastguard Worker 				if (iswctype(k, wctype(buf)) ||
146*c9945492SAndroid Build Coastguard Worker 				    iswctype(kfold, wctype(buf)))
147*c9945492SAndroid Build Coastguard Worker 					return !inv;
148*c9945492SAndroid Build Coastguard Worker 			}
149*c9945492SAndroid Build Coastguard Worker 			continue;
150*c9945492SAndroid Build Coastguard Worker 		}
151*c9945492SAndroid Build Coastguard Worker 		if (*p < 128U) {
152*c9945492SAndroid Build Coastguard Worker 			wc = (unsigned char)*p;
153*c9945492SAndroid Build Coastguard Worker 		} else {
154*c9945492SAndroid Build Coastguard Worker 			int l = mbtowc(&wc, p, 4);
155*c9945492SAndroid Build Coastguard Worker 			if (l < 0) return 0;
156*c9945492SAndroid Build Coastguard Worker 			p += l-1;
157*c9945492SAndroid Build Coastguard Worker 		}
158*c9945492SAndroid Build Coastguard Worker 		if (wc==k || wc==kfold) return !inv;
159*c9945492SAndroid Build Coastguard Worker 	}
160*c9945492SAndroid Build Coastguard Worker 	return inv;
161*c9945492SAndroid Build Coastguard Worker }
162*c9945492SAndroid Build Coastguard Worker 
fnmatch_internal(const char * pat,size_t m,const char * str,size_t n,int flags)163*c9945492SAndroid Build Coastguard Worker static int fnmatch_internal(const char *pat, size_t m, const char *str, size_t n, int flags)
164*c9945492SAndroid Build Coastguard Worker {
165*c9945492SAndroid Build Coastguard Worker 	const char *p, *ptail, *endpat;
166*c9945492SAndroid Build Coastguard Worker 	const char *s, *stail, *endstr;
167*c9945492SAndroid Build Coastguard Worker 	size_t pinc, sinc, tailcnt=0;
168*c9945492SAndroid Build Coastguard Worker 	int c, k, kfold;
169*c9945492SAndroid Build Coastguard Worker 
170*c9945492SAndroid Build Coastguard Worker 	if (flags & FNM_PERIOD) {
171*c9945492SAndroid Build Coastguard Worker 		if (*str == '.' && *pat != '.')
172*c9945492SAndroid Build Coastguard Worker 			return FNM_NOMATCH;
173*c9945492SAndroid Build Coastguard Worker 	}
174*c9945492SAndroid Build Coastguard Worker 	for (;;) {
175*c9945492SAndroid Build Coastguard Worker 		switch ((c = pat_next(pat, m, &pinc, flags))) {
176*c9945492SAndroid Build Coastguard Worker 		case UNMATCHABLE:
177*c9945492SAndroid Build Coastguard Worker 			return FNM_NOMATCH;
178*c9945492SAndroid Build Coastguard Worker 		case STAR:
179*c9945492SAndroid Build Coastguard Worker 			pat++;
180*c9945492SAndroid Build Coastguard Worker 			m--;
181*c9945492SAndroid Build Coastguard Worker 			break;
182*c9945492SAndroid Build Coastguard Worker 		default:
183*c9945492SAndroid Build Coastguard Worker 			k = str_next(str, n, &sinc);
184*c9945492SAndroid Build Coastguard Worker 			if (k <= 0)
185*c9945492SAndroid Build Coastguard Worker 				return (c==END) ? 0 : FNM_NOMATCH;
186*c9945492SAndroid Build Coastguard Worker 			str += sinc;
187*c9945492SAndroid Build Coastguard Worker 			n -= sinc;
188*c9945492SAndroid Build Coastguard Worker 			kfold = flags & FNM_CASEFOLD ? casefold(k) : k;
189*c9945492SAndroid Build Coastguard Worker 			if (c == BRACKET) {
190*c9945492SAndroid Build Coastguard Worker 				if (!match_bracket(pat, k, kfold))
191*c9945492SAndroid Build Coastguard Worker 					return FNM_NOMATCH;
192*c9945492SAndroid Build Coastguard Worker 			} else if (c != QUESTION && k != c && kfold != c) {
193*c9945492SAndroid Build Coastguard Worker 				return FNM_NOMATCH;
194*c9945492SAndroid Build Coastguard Worker 			}
195*c9945492SAndroid Build Coastguard Worker 			pat+=pinc;
196*c9945492SAndroid Build Coastguard Worker 			m-=pinc;
197*c9945492SAndroid Build Coastguard Worker 			continue;
198*c9945492SAndroid Build Coastguard Worker 		}
199*c9945492SAndroid Build Coastguard Worker 		break;
200*c9945492SAndroid Build Coastguard Worker 	}
201*c9945492SAndroid Build Coastguard Worker 
202*c9945492SAndroid Build Coastguard Worker 	/* Compute real pat length if it was initially unknown/-1 */
203*c9945492SAndroid Build Coastguard Worker 	m = strnlen(pat, m);
204*c9945492SAndroid Build Coastguard Worker 	endpat = pat + m;
205*c9945492SAndroid Build Coastguard Worker 
206*c9945492SAndroid Build Coastguard Worker 	/* Find the last * in pat and count chars needed after it */
207*c9945492SAndroid Build Coastguard Worker 	for (p=ptail=pat; p<endpat; p+=pinc) {
208*c9945492SAndroid Build Coastguard Worker 		switch (pat_next(p, endpat-p, &pinc, flags)) {
209*c9945492SAndroid Build Coastguard Worker 		case UNMATCHABLE:
210*c9945492SAndroid Build Coastguard Worker 			return FNM_NOMATCH;
211*c9945492SAndroid Build Coastguard Worker 		case STAR:
212*c9945492SAndroid Build Coastguard Worker 			tailcnt=0;
213*c9945492SAndroid Build Coastguard Worker 			ptail = p+1;
214*c9945492SAndroid Build Coastguard Worker 			break;
215*c9945492SAndroid Build Coastguard Worker 		default:
216*c9945492SAndroid Build Coastguard Worker 			tailcnt++;
217*c9945492SAndroid Build Coastguard Worker 			break;
218*c9945492SAndroid Build Coastguard Worker 		}
219*c9945492SAndroid Build Coastguard Worker 	}
220*c9945492SAndroid Build Coastguard Worker 
221*c9945492SAndroid Build Coastguard Worker 	/* Past this point we need not check for UNMATCHABLE in pat,
222*c9945492SAndroid Build Coastguard Worker 	 * because all of pat has already been parsed once. */
223*c9945492SAndroid Build Coastguard Worker 
224*c9945492SAndroid Build Coastguard Worker 	/* Compute real str length if it was initially unknown/-1 */
225*c9945492SAndroid Build Coastguard Worker 	n = strnlen(str, n);
226*c9945492SAndroid Build Coastguard Worker 	endstr = str + n;
227*c9945492SAndroid Build Coastguard Worker 	if (n < tailcnt) return FNM_NOMATCH;
228*c9945492SAndroid Build Coastguard Worker 
229*c9945492SAndroid Build Coastguard Worker 	/* Find the final tailcnt chars of str, accounting for UTF-8.
230*c9945492SAndroid Build Coastguard Worker 	 * On illegal sequences we may get it wrong, but in that case
231*c9945492SAndroid Build Coastguard Worker 	 * we necessarily have a matching failure anyway. */
232*c9945492SAndroid Build Coastguard Worker 	for (s=endstr; s>str && tailcnt; tailcnt--) {
233*c9945492SAndroid Build Coastguard Worker 		if (s[-1] < 128U || MB_CUR_MAX==1) s--;
234*c9945492SAndroid Build Coastguard Worker 		else while ((unsigned char)*--s-0x80U<0x40 && s>str);
235*c9945492SAndroid Build Coastguard Worker 	}
236*c9945492SAndroid Build Coastguard Worker 	if (tailcnt) return FNM_NOMATCH;
237*c9945492SAndroid Build Coastguard Worker 	stail = s;
238*c9945492SAndroid Build Coastguard Worker 
239*c9945492SAndroid Build Coastguard Worker 	/* Check that the pat and str tails match */
240*c9945492SAndroid Build Coastguard Worker 	p = ptail;
241*c9945492SAndroid Build Coastguard Worker 	for (;;) {
242*c9945492SAndroid Build Coastguard Worker 		c = pat_next(p, endpat-p, &pinc, flags);
243*c9945492SAndroid Build Coastguard Worker 		p += pinc;
244*c9945492SAndroid Build Coastguard Worker 		if ((k = str_next(s, endstr-s, &sinc)) <= 0) {
245*c9945492SAndroid Build Coastguard Worker 			if (c != END) return FNM_NOMATCH;
246*c9945492SAndroid Build Coastguard Worker 			break;
247*c9945492SAndroid Build Coastguard Worker 		}
248*c9945492SAndroid Build Coastguard Worker 		s += sinc;
249*c9945492SAndroid Build Coastguard Worker 		kfold = flags & FNM_CASEFOLD ? casefold(k) : k;
250*c9945492SAndroid Build Coastguard Worker 		if (c == BRACKET) {
251*c9945492SAndroid Build Coastguard Worker 			if (!match_bracket(p-pinc, k, kfold))
252*c9945492SAndroid Build Coastguard Worker 				return FNM_NOMATCH;
253*c9945492SAndroid Build Coastguard Worker 		} else if (c != QUESTION && k != c && kfold != c) {
254*c9945492SAndroid Build Coastguard Worker 			return FNM_NOMATCH;
255*c9945492SAndroid Build Coastguard Worker 		}
256*c9945492SAndroid Build Coastguard Worker 	}
257*c9945492SAndroid Build Coastguard Worker 
258*c9945492SAndroid Build Coastguard Worker 	/* We're all done with the tails now, so throw them out */
259*c9945492SAndroid Build Coastguard Worker 	endstr = stail;
260*c9945492SAndroid Build Coastguard Worker 	endpat = ptail;
261*c9945492SAndroid Build Coastguard Worker 
262*c9945492SAndroid Build Coastguard Worker 	/* Match pattern components until there are none left */
263*c9945492SAndroid Build Coastguard Worker 	while (pat<endpat) {
264*c9945492SAndroid Build Coastguard Worker 		p = pat;
265*c9945492SAndroid Build Coastguard Worker 		s = str;
266*c9945492SAndroid Build Coastguard Worker 		for (;;) {
267*c9945492SAndroid Build Coastguard Worker 			c = pat_next(p, endpat-p, &pinc, flags);
268*c9945492SAndroid Build Coastguard Worker 			p += pinc;
269*c9945492SAndroid Build Coastguard Worker 			/* Encountering * completes/commits a component */
270*c9945492SAndroid Build Coastguard Worker 			if (c == STAR) {
271*c9945492SAndroid Build Coastguard Worker 				pat = p;
272*c9945492SAndroid Build Coastguard Worker 				str = s;
273*c9945492SAndroid Build Coastguard Worker 				break;
274*c9945492SAndroid Build Coastguard Worker 			}
275*c9945492SAndroid Build Coastguard Worker 			k = str_next(s, endstr-s, &sinc);
276*c9945492SAndroid Build Coastguard Worker 			if (!k)
277*c9945492SAndroid Build Coastguard Worker 				return FNM_NOMATCH;
278*c9945492SAndroid Build Coastguard Worker 			kfold = flags & FNM_CASEFOLD ? casefold(k) : k;
279*c9945492SAndroid Build Coastguard Worker 			if (c == BRACKET) {
280*c9945492SAndroid Build Coastguard Worker 				if (!match_bracket(p-pinc, k, kfold))
281*c9945492SAndroid Build Coastguard Worker 					break;
282*c9945492SAndroid Build Coastguard Worker 			} else if (c != QUESTION && k != c && kfold != c) {
283*c9945492SAndroid Build Coastguard Worker 				break;
284*c9945492SAndroid Build Coastguard Worker 			}
285*c9945492SAndroid Build Coastguard Worker 			s += sinc;
286*c9945492SAndroid Build Coastguard Worker 		}
287*c9945492SAndroid Build Coastguard Worker 		if (c == STAR) continue;
288*c9945492SAndroid Build Coastguard Worker 		/* If we failed, advance str, by 1 char if it's a valid
289*c9945492SAndroid Build Coastguard Worker 		 * char, or past all invalid bytes otherwise. */
290*c9945492SAndroid Build Coastguard Worker 		k = str_next(str, endstr-str, &sinc);
291*c9945492SAndroid Build Coastguard Worker 		if (k > 0) str += sinc;
292*c9945492SAndroid Build Coastguard Worker 		else for (str++; str_next(str, endstr-str, &sinc)<0; str++);
293*c9945492SAndroid Build Coastguard Worker 	}
294*c9945492SAndroid Build Coastguard Worker 
295*c9945492SAndroid Build Coastguard Worker 	return 0;
296*c9945492SAndroid Build Coastguard Worker }
297*c9945492SAndroid Build Coastguard Worker 
fnmatch(const char * pat,const char * str,int flags)298*c9945492SAndroid Build Coastguard Worker int fnmatch(const char *pat, const char *str, int flags)
299*c9945492SAndroid Build Coastguard Worker {
300*c9945492SAndroid Build Coastguard Worker 	const char *s, *p;
301*c9945492SAndroid Build Coastguard Worker 	size_t inc;
302*c9945492SAndroid Build Coastguard Worker 	int c;
303*c9945492SAndroid Build Coastguard Worker 	if (flags & FNM_PATHNAME) for (;;) {
304*c9945492SAndroid Build Coastguard Worker 		for (s=str; *s && *s!='/'; s++);
305*c9945492SAndroid Build Coastguard Worker 		for (p=pat; (c=pat_next(p, -1, &inc, flags))!=END && c!='/'; p+=inc);
306*c9945492SAndroid Build Coastguard Worker 		if (c!=*s && (!*s || !(flags & FNM_LEADING_DIR)))
307*c9945492SAndroid Build Coastguard Worker 			return FNM_NOMATCH;
308*c9945492SAndroid Build Coastguard Worker 		if (fnmatch_internal(pat, p-pat, str, s-str, flags))
309*c9945492SAndroid Build Coastguard Worker 			return FNM_NOMATCH;
310*c9945492SAndroid Build Coastguard Worker 		if (!c) return 0;
311*c9945492SAndroid Build Coastguard Worker 		str = s+1;
312*c9945492SAndroid Build Coastguard Worker 		pat = p+inc;
313*c9945492SAndroid Build Coastguard Worker 	} else if (flags & FNM_LEADING_DIR) {
314*c9945492SAndroid Build Coastguard Worker 		for (s=str; *s; s++) {
315*c9945492SAndroid Build Coastguard Worker 			if (*s != '/') continue;
316*c9945492SAndroid Build Coastguard Worker 			if (!fnmatch_internal(pat, -1, str, s-str, flags))
317*c9945492SAndroid Build Coastguard Worker 				return 0;
318*c9945492SAndroid Build Coastguard Worker 		}
319*c9945492SAndroid Build Coastguard Worker 	}
320*c9945492SAndroid Build Coastguard Worker 	return fnmatch_internal(pat, -1, str, -1, flags);
321*c9945492SAndroid Build Coastguard Worker }
322