update from main archive 961008
[glibc.git] / posix / fnmatch.c
blobde45941f8f98c1064335bb1b1a01522c225d19ee
1 /* Copyright (C) 1991, 1992, 1993, 1996 Free Software Foundation, Inc.
3 This library is free software; you can redistribute it and/or
4 modify it under the terms of the GNU Library General Public License as
5 published by the Free Software Foundation; either version 2 of the
6 License, or (at your option) any later version.
8 This library is distributed in the hope that it will be useful,
9 but WITHOUT ANY WARRANTY; without even the implied warranty of
10 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
11 Library General Public License for more details.
13 You should have received a copy of the GNU Library General Public
14 License along with this library; see the file COPYING.LIB. If
15 not, write to the Free Software Foundation, Inc., 675 Mass Ave,
16 Cambridge, MA 02139, USA. */
18 #ifdef HAVE_CONFIG_H
19 #include <config.h>
20 #endif
22 /* Enable GNU extensions in fnmatch.h. */
23 #ifndef _GNU_SOURCE
24 #define _GNU_SOURCE 1
25 #endif
27 #include <errno.h>
28 #include <fnmatch.h>
29 #include <ctype.h>
32 /* Comment out all this code if we are using the GNU C Library, and are not
33 actually compiling the library itself. This code is part of the GNU C
34 Library, but also included in many other GNU distributions. Compiling
35 and linking in this code is a waste when using the GNU C library
36 (especially if it is a shared library). Rather than having every GNU
37 program understand `configure --with-gnu-libc' and omit the object files,
38 it is simpler to just do this in the source for each such file. */
40 #if defined (_LIBC) || !defined (__GNU_LIBRARY__)
43 #ifndef errno
44 extern int errno;
45 #endif
47 /* Match STRING against the filename pattern PATTERN, returning zero if
48 it matches, nonzero if not. */
49 int
50 fnmatch (pattern, string, flags)
51 const char *pattern;
52 const char *string;
53 int flags;
55 register const char *p = pattern, *n = string;
56 register char c;
58 /* Note that this evalutes C many times. */
59 #define FOLD(c) ((flags & FNM_CASEFOLD) && isupper (c) ? tolower (c) : (c))
61 while ((c = *p++) != '\0')
63 c = FOLD (c);
65 switch (c)
67 case '?':
68 if (*n == '\0')
69 return FNM_NOMATCH;
70 else if ((flags & FNM_FILE_NAME) && *n == '/')
71 return FNM_NOMATCH;
72 else if ((flags & FNM_PERIOD) && *n == '.' &&
73 (n == string || ((flags & FNM_FILE_NAME) && n[-1] == '/')))
74 return FNM_NOMATCH;
75 break;
77 case '\\':
78 if (!(flags & FNM_NOESCAPE))
80 c = *p++;
81 if (c == '\0')
82 /* Trailing \ loses. */
83 return FNM_NOMATCH;
84 c = FOLD (c);
86 if (FOLD (*n) != c)
87 return FNM_NOMATCH;
88 break;
90 case '*':
91 if ((flags & FNM_PERIOD) && *n == '.' &&
92 (n == string || ((flags & FNM_FILE_NAME) && n[-1] == '/')))
93 return FNM_NOMATCH;
95 for (c = *p++; c == '?' || c == '*'; c = *p++)
97 if ((flags & FNM_FILE_NAME) && *n == '/')
98 /* A slash does not match a wildcard under FNM_FILE_NAME. */
99 return FNM_NOMATCH;
100 else if (c == '?')
102 /* A ? needs to match one character. */
103 if (*n == '\0')
104 /* There isn't another character; no match. */
105 return FNM_NOMATCH;
106 else
107 /* One character of the string is consumed in matching
108 this ? wildcard, so *??? won't match if there are
109 less than three characters. */
110 ++n;
114 if (c == '\0')
115 return 0;
118 char c1 = (!(flags & FNM_NOESCAPE) && c == '\\') ? *p : c;
119 c1 = FOLD (c1);
120 for (--p; *n != '\0'; ++n)
121 if ((c == '[' || FOLD (*n) == c1) &&
122 fnmatch (p, n, flags & ~FNM_PERIOD) == 0)
123 return 0;
124 return FNM_NOMATCH;
127 case '[':
129 /* Nonzero if the sense of the character class is inverted. */
130 register int not;
132 if (*n == '\0')
133 return FNM_NOMATCH;
135 if ((flags & FNM_PERIOD) && *n == '.' &&
136 (n == string || ((flags & FNM_FILE_NAME) && n[-1] == '/')))
137 return FNM_NOMATCH;
139 not = (*p == '!' || *p == '^');
140 if (not)
141 ++p;
143 c = *p++;
144 for (;;)
146 register char cstart = c, cend = c;
148 if (!(flags & FNM_NOESCAPE) && c == '\\')
150 if (*p == '\0')
151 return FNM_NOMATCH;
152 cstart = cend = *p++;
155 cstart = cend = FOLD (cstart);
157 if (c == '\0')
158 /* [ (unterminated) loses. */
159 return FNM_NOMATCH;
161 c = *p++;
162 c = FOLD (c);
164 if ((flags & FNM_FILE_NAME) && c == '/')
165 /* [/] can never match. */
166 return FNM_NOMATCH;
168 if (c == '-' && *p != ']')
170 cend = *p++;
171 if (!(flags & FNM_NOESCAPE) && cend == '\\')
172 cend = *p++;
173 if (cend == '\0')
174 return FNM_NOMATCH;
175 cend = FOLD (cend);
177 c = *p++;
180 if (FOLD (*n) >= cstart && FOLD (*n) <= cend)
181 goto matched;
183 if (c == ']')
184 break;
186 if (!not)
187 return FNM_NOMATCH;
188 break;
190 matched:;
191 /* Skip the rest of the [...] that already matched. */
192 while (c != ']')
194 if (c == '\0')
195 /* [... (unterminated) loses. */
196 return FNM_NOMATCH;
198 c = *p++;
199 if (!(flags & FNM_NOESCAPE) && c == '\\')
201 if (*p == '\0')
202 return FNM_NOMATCH;
203 /* XXX 1003.2d11 is unclear if this is right. */
204 ++p;
207 if (not)
208 return FNM_NOMATCH;
210 break;
212 default:
213 if (c != FOLD (*n))
214 return FNM_NOMATCH;
217 ++n;
220 if (*n == '\0')
221 return 0;
223 if ((flags & FNM_LEADING_DIR) && *n == '/')
224 /* The FNM_LEADING_DIR flag says that "foo*" matches "foobar/frobozz". */
225 return 0;
227 return FNM_NOMATCH;
230 #endif /* _LIBC or not __GNU_LIBRARY__. */