1*fae548d3Szrj /* Copyright (C) 1991-2020 Free Software Foundation, Inc.
2*fae548d3Szrj
3*fae548d3Szrj NOTE: This source is derived from an old version taken from the GNU C
4*fae548d3Szrj Library (glibc).
5*fae548d3Szrj
6*fae548d3Szrj This program is free software; you can redistribute it and/or modify it
7*fae548d3Szrj under the terms of the GNU General Public License as published by the
8*fae548d3Szrj Free Software Foundation; either version 2, or (at your option) any
9*fae548d3Szrj later version.
10*fae548d3Szrj
11*fae548d3Szrj This program is distributed in the hope that it will be useful,
12*fae548d3Szrj but WITHOUT ANY WARRANTY; without even the implied warranty of
13*fae548d3Szrj MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14*fae548d3Szrj GNU General Public License for more details.
15*fae548d3Szrj
16*fae548d3Szrj You should have received a copy of the GNU General Public License
17*fae548d3Szrj along with this program; if not, write to the Free Software
18*fae548d3Szrj Foundation, 51 Franklin Street - Fifth Floor,
19*fae548d3Szrj Boston, MA 02110-1301, USA. */
20*fae548d3Szrj
21*fae548d3Szrj #ifdef HAVE_CONFIG_H
22*fae548d3Szrj #if defined (CONFIG_BROKETS)
23*fae548d3Szrj /* We use <config.h> instead of "config.h" so that a compilation
24*fae548d3Szrj using -I. -I$srcdir will use ./config.h rather than $srcdir/config.h
25*fae548d3Szrj (which it would do because it found this file in $srcdir). */
26*fae548d3Szrj #include <config.h>
27*fae548d3Szrj #else
28*fae548d3Szrj #include "config.h"
29*fae548d3Szrj #endif
30*fae548d3Szrj #endif
31*fae548d3Szrj
32*fae548d3Szrj
33*fae548d3Szrj #ifndef _GNU_SOURCE
34*fae548d3Szrj #define _GNU_SOURCE
35*fae548d3Szrj #endif
36*fae548d3Szrj
37*fae548d3Szrj /* This code to undef const added in libiberty. */
38*fae548d3Szrj #ifndef __STDC__
39*fae548d3Szrj /* This is a separate conditional since some stdc systems
40*fae548d3Szrj reject `defined (const)'. */
41*fae548d3Szrj #ifndef const
42*fae548d3Szrj #define const
43*fae548d3Szrj #endif
44*fae548d3Szrj #endif
45*fae548d3Szrj
46*fae548d3Szrj #include <errno.h>
47*fae548d3Szrj #include <fnmatch.h>
48*fae548d3Szrj #include <safe-ctype.h>
49*fae548d3Szrj
50*fae548d3Szrj /* Comment out all this code if we are using the GNU C Library, and are not
51*fae548d3Szrj actually compiling the library itself. This code is part of the GNU C
52*fae548d3Szrj Library, but also included in many other GNU distributions. Compiling
53*fae548d3Szrj and linking in this code is a waste when using the GNU C library
54*fae548d3Szrj (especially if it is a shared library). Rather than having every GNU
55*fae548d3Szrj program understand `configure --with-gnu-libc' and omit the object files,
56*fae548d3Szrj it is simpler to just do this in the source for each such file. */
57*fae548d3Szrj
58*fae548d3Szrj #if defined (_LIBC) || !defined (__GNU_LIBRARY__)
59*fae548d3Szrj
60*fae548d3Szrj
61*fae548d3Szrj #if !defined(__GNU_LIBRARY__) && !defined(STDC_HEADERS)
62*fae548d3Szrj extern int errno;
63*fae548d3Szrj #endif
64*fae548d3Szrj
65*fae548d3Szrj /* Match STRING against the filename pattern PATTERN, returning zero if
66*fae548d3Szrj it matches, nonzero if not. */
67*fae548d3Szrj int
fnmatch(const char * pattern,const char * string,int flags)68*fae548d3Szrj fnmatch (const char *pattern, const char *string, int flags)
69*fae548d3Szrj {
70*fae548d3Szrj register const char *p = pattern, *n = string;
71*fae548d3Szrj register unsigned char c;
72*fae548d3Szrj
73*fae548d3Szrj #define FOLD(c) ((flags & FNM_CASEFOLD) ? TOLOWER (c) : (c))
74*fae548d3Szrj
75*fae548d3Szrj while ((c = *p++) != '\0')
76*fae548d3Szrj {
77*fae548d3Szrj c = FOLD (c);
78*fae548d3Szrj
79*fae548d3Szrj switch (c)
80*fae548d3Szrj {
81*fae548d3Szrj case '?':
82*fae548d3Szrj if (*n == '\0')
83*fae548d3Szrj return FNM_NOMATCH;
84*fae548d3Szrj else if ((flags & FNM_FILE_NAME) && *n == '/')
85*fae548d3Szrj return FNM_NOMATCH;
86*fae548d3Szrj else if ((flags & FNM_PERIOD) && *n == '.' &&
87*fae548d3Szrj (n == string || ((flags & FNM_FILE_NAME) && n[-1] == '/')))
88*fae548d3Szrj return FNM_NOMATCH;
89*fae548d3Szrj break;
90*fae548d3Szrj
91*fae548d3Szrj case '\\':
92*fae548d3Szrj if (!(flags & FNM_NOESCAPE))
93*fae548d3Szrj {
94*fae548d3Szrj c = *p++;
95*fae548d3Szrj c = FOLD (c);
96*fae548d3Szrj }
97*fae548d3Szrj if (FOLD ((unsigned char)*n) != c)
98*fae548d3Szrj return FNM_NOMATCH;
99*fae548d3Szrj break;
100*fae548d3Szrj
101*fae548d3Szrj case '*':
102*fae548d3Szrj if ((flags & FNM_PERIOD) && *n == '.' &&
103*fae548d3Szrj (n == string || ((flags & FNM_FILE_NAME) && n[-1] == '/')))
104*fae548d3Szrj return FNM_NOMATCH;
105*fae548d3Szrj
106*fae548d3Szrj for (c = *p++; c == '?' || c == '*'; c = *p++, ++n)
107*fae548d3Szrj if (((flags & FNM_FILE_NAME) && *n == '/') ||
108*fae548d3Szrj (c == '?' && *n == '\0'))
109*fae548d3Szrj return FNM_NOMATCH;
110*fae548d3Szrj
111*fae548d3Szrj if (c == '\0')
112*fae548d3Szrj return 0;
113*fae548d3Szrj
114*fae548d3Szrj {
115*fae548d3Szrj unsigned char c1 = (!(flags & FNM_NOESCAPE) && c == '\\') ? *p : c;
116*fae548d3Szrj c1 = FOLD (c1);
117*fae548d3Szrj for (--p; *n != '\0'; ++n)
118*fae548d3Szrj if ((c == '[' || FOLD ((unsigned char)*n) == c1) &&
119*fae548d3Szrj fnmatch (p, n, flags & ~FNM_PERIOD) == 0)
120*fae548d3Szrj return 0;
121*fae548d3Szrj return FNM_NOMATCH;
122*fae548d3Szrj }
123*fae548d3Szrj
124*fae548d3Szrj case '[':
125*fae548d3Szrj {
126*fae548d3Szrj /* Nonzero if the sense of the character class is inverted. */
127*fae548d3Szrj register int negate;
128*fae548d3Szrj
129*fae548d3Szrj if (*n == '\0')
130*fae548d3Szrj return FNM_NOMATCH;
131*fae548d3Szrj
132*fae548d3Szrj if ((flags & FNM_PERIOD) && *n == '.' &&
133*fae548d3Szrj (n == string || ((flags & FNM_FILE_NAME) && n[-1] == '/')))
134*fae548d3Szrj return FNM_NOMATCH;
135*fae548d3Szrj
136*fae548d3Szrj negate = (*p == '!' || *p == '^');
137*fae548d3Szrj if (negate)
138*fae548d3Szrj ++p;
139*fae548d3Szrj
140*fae548d3Szrj c = *p++;
141*fae548d3Szrj for (;;)
142*fae548d3Szrj {
143*fae548d3Szrj register unsigned char cstart = c, cend = c;
144*fae548d3Szrj
145*fae548d3Szrj if (!(flags & FNM_NOESCAPE) && c == '\\')
146*fae548d3Szrj cstart = cend = *p++;
147*fae548d3Szrj
148*fae548d3Szrj cstart = cend = FOLD (cstart);
149*fae548d3Szrj
150*fae548d3Szrj if (c == '\0')
151*fae548d3Szrj /* [ (unterminated) loses. */
152*fae548d3Szrj return FNM_NOMATCH;
153*fae548d3Szrj
154*fae548d3Szrj c = *p++;
155*fae548d3Szrj c = FOLD (c);
156*fae548d3Szrj
157*fae548d3Szrj if ((flags & FNM_FILE_NAME) && c == '/')
158*fae548d3Szrj /* [/] can never match. */
159*fae548d3Szrj return FNM_NOMATCH;
160*fae548d3Szrj
161*fae548d3Szrj if (c == '-' && *p != ']')
162*fae548d3Szrj {
163*fae548d3Szrj cend = *p++;
164*fae548d3Szrj if (!(flags & FNM_NOESCAPE) && cend == '\\')
165*fae548d3Szrj cend = *p++;
166*fae548d3Szrj if (cend == '\0')
167*fae548d3Szrj return FNM_NOMATCH;
168*fae548d3Szrj cend = FOLD (cend);
169*fae548d3Szrj
170*fae548d3Szrj c = *p++;
171*fae548d3Szrj }
172*fae548d3Szrj
173*fae548d3Szrj if (FOLD ((unsigned char)*n) >= cstart
174*fae548d3Szrj && FOLD ((unsigned char)*n) <= cend)
175*fae548d3Szrj goto matched;
176*fae548d3Szrj
177*fae548d3Szrj if (c == ']')
178*fae548d3Szrj break;
179*fae548d3Szrj }
180*fae548d3Szrj if (!negate)
181*fae548d3Szrj return FNM_NOMATCH;
182*fae548d3Szrj break;
183*fae548d3Szrj
184*fae548d3Szrj matched:;
185*fae548d3Szrj /* Skip the rest of the [...] that already matched. */
186*fae548d3Szrj while (c != ']')
187*fae548d3Szrj {
188*fae548d3Szrj if (c == '\0')
189*fae548d3Szrj /* [... (unterminated) loses. */
190*fae548d3Szrj return FNM_NOMATCH;
191*fae548d3Szrj
192*fae548d3Szrj c = *p++;
193*fae548d3Szrj if (!(flags & FNM_NOESCAPE) && c == '\\')
194*fae548d3Szrj /* XXX 1003.2d11 is unclear if this is right. */
195*fae548d3Szrj ++p;
196*fae548d3Szrj }
197*fae548d3Szrj if (negate)
198*fae548d3Szrj return FNM_NOMATCH;
199*fae548d3Szrj }
200*fae548d3Szrj break;
201*fae548d3Szrj
202*fae548d3Szrj default:
203*fae548d3Szrj if (c != FOLD ((unsigned char)*n))
204*fae548d3Szrj return FNM_NOMATCH;
205*fae548d3Szrj }
206*fae548d3Szrj
207*fae548d3Szrj ++n;
208*fae548d3Szrj }
209*fae548d3Szrj
210*fae548d3Szrj if (*n == '\0')
211*fae548d3Szrj return 0;
212*fae548d3Szrj
213*fae548d3Szrj if ((flags & FNM_LEADING_DIR) && *n == '/')
214*fae548d3Szrj /* The FNM_LEADING_DIR flag says that "foo*" matches "foobar/frobozz". */
215*fae548d3Szrj return 0;
216*fae548d3Szrj
217*fae548d3Szrj return FNM_NOMATCH;
218*fae548d3Szrj }
219*fae548d3Szrj
220*fae548d3Szrj #endif /* _LIBC or not __GNU_LIBRARY__. */
221