* cp-tree.h (struct lang_decl_flags): Add comdat.
[official-gcc.git] / libiberty / fnmatch.c
blob685d9e40edcc19285495f3c11251793a5f8c8267
1 /* Copyright (C) 1991, 1992, 1993 Free Software Foundation, Inc.
3 NOTE: The canonical source of this file is maintained with the GNU C Library.
4 Bugs can be reported to bug-glibc@prep.ai.mit.edu.
6 This program is free software; you can redistribute it and/or modify it
7 under the terms of the GNU General Public License as published by the
8 Free Software Foundation; either version 2, or (at your option) any
9 later version.
11 This program is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 You should have received a copy of the GNU General Public License
17 along with this program; if not, write to the Free Software
18 Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */
20 #ifdef HAVE_CONFIG_H
21 #if defined (CONFIG_BROKETS)
22 /* We use <config.h> instead of "config.h" so that a compilation
23 using -I. -I$srcdir will use ./config.h rather than $srcdir/config.h
24 (which it would do because it found this file in $srcdir). */
25 #include <config.h>
26 #else
27 #include "config.h"
28 #endif
29 #endif
32 #ifndef _GNU_SOURCE
33 #define _GNU_SOURCE
34 #endif
36 /* This code to undef const added in libiberty. */
37 #ifndef __STDC__
38 /* This is a separate conditional since some stdc systems
39 reject `defined (const)'. */
40 #ifndef const
41 #define const
42 #endif
43 #endif
45 #include <errno.h>
46 #include <fnmatch.h>
47 #include <ctype.h>
50 /* Comment out all this code if we are using the GNU C Library, and are not
51 actually compiling the library itself. This code is part of the GNU C
52 Library, but also included in many other GNU distributions. Compiling
53 and linking in this code is a waste when using the GNU C library
54 (especially if it is a shared library). Rather than having every GNU
55 program understand `configure --with-gnu-libc' and omit the object files,
56 it is simpler to just do this in the source for each such file. */
58 #if defined (_LIBC) || !defined (__GNU_LIBRARY__)
61 #if !defined(__GNU_LIBRARY__) && !defined(STDC_HEADERS)
62 extern int errno;
63 #endif
65 /* Match STRING against the filename pattern PATTERN, returning zero if
66 it matches, nonzero if not. */
67 int
68 fnmatch (pattern, string, flags)
69 const char *pattern;
70 const char *string;
71 int flags;
73 register const char *p = pattern, *n = string;
74 register char c;
76 /* Note that this evalutes C many times. */
77 #define FOLD(c) ((flags & FNM_CASEFOLD) && isupper (c) ? tolower (c) : (c))
79 while ((c = *p++) != '\0')
81 c = FOLD (c);
83 switch (c)
85 case '?':
86 if (*n == '\0')
87 return FNM_NOMATCH;
88 else if ((flags & FNM_FILE_NAME) && *n == '/')
89 return FNM_NOMATCH;
90 else if ((flags & FNM_PERIOD) && *n == '.' &&
91 (n == string || ((flags & FNM_FILE_NAME) && n[-1] == '/')))
92 return FNM_NOMATCH;
93 break;
95 case '\\':
96 if (!(flags & FNM_NOESCAPE))
98 c = *p++;
99 c = FOLD (c);
101 if (FOLD (*n) != c)
102 return FNM_NOMATCH;
103 break;
105 case '*':
106 if ((flags & FNM_PERIOD) && *n == '.' &&
107 (n == string || ((flags & FNM_FILE_NAME) && n[-1] == '/')))
108 return FNM_NOMATCH;
110 for (c = *p++; c == '?' || c == '*'; c = *p++, ++n)
111 if (((flags & FNM_FILE_NAME) && *n == '/') ||
112 (c == '?' && *n == '\0'))
113 return FNM_NOMATCH;
115 if (c == '\0')
116 return 0;
119 char c1 = (!(flags & FNM_NOESCAPE) && c == '\\') ? *p : c;
120 c1 = FOLD (c1);
121 for (--p; *n != '\0'; ++n)
122 if ((c == '[' || FOLD (*n) == c1) &&
123 fnmatch (p, n, flags & ~FNM_PERIOD) == 0)
124 return 0;
125 return FNM_NOMATCH;
128 case '[':
130 /* Nonzero if the sense of the character class is inverted. */
131 register int not;
133 if (*n == '\0')
134 return FNM_NOMATCH;
136 if ((flags & FNM_PERIOD) && *n == '.' &&
137 (n == string || ((flags & FNM_FILE_NAME) && n[-1] == '/')))
138 return FNM_NOMATCH;
140 not = (*p == '!' || *p == '^');
141 if (not)
142 ++p;
144 c = *p++;
145 for (;;)
147 register char cstart = c, cend = c;
149 if (!(flags & FNM_NOESCAPE) && c == '\\')
150 cstart = cend = *p++;
152 cstart = cend = FOLD (cstart);
154 if (c == '\0')
155 /* [ (unterminated) loses. */
156 return FNM_NOMATCH;
158 c = *p++;
159 c = FOLD (c);
161 if ((flags & FNM_FILE_NAME) && c == '/')
162 /* [/] can never match. */
163 return FNM_NOMATCH;
165 if (c == '-' && *p != ']')
167 cend = *p++;
168 if (!(flags & FNM_NOESCAPE) && cend == '\\')
169 cend = *p++;
170 if (cend == '\0')
171 return FNM_NOMATCH;
172 cend = FOLD (cend);
174 c = *p++;
177 if (FOLD (*n) >= cstart && FOLD (*n) <= cend)
178 goto matched;
180 if (c == ']')
181 break;
183 if (!not)
184 return FNM_NOMATCH;
185 break;
187 matched:;
188 /* Skip the rest of the [...] that already matched. */
189 while (c != ']')
191 if (c == '\0')
192 /* [... (unterminated) loses. */
193 return FNM_NOMATCH;
195 c = *p++;
196 if (!(flags & FNM_NOESCAPE) && c == '\\')
197 /* XXX 1003.2d11 is unclear if this is right. */
198 ++p;
200 if (not)
201 return FNM_NOMATCH;
203 break;
205 default:
206 if (c != FOLD (*n))
207 return FNM_NOMATCH;
210 ++n;
213 if (*n == '\0')
214 return 0;
216 if ((flags & FNM_LEADING_DIR) && *n == '/')
217 /* The FNM_LEADING_DIR flag says that "foo*" matches "foobar/frobozz". */
218 return 0;
220 return FNM_NOMATCH;
223 #endif /* _LIBC or not __GNU_LIBRARY__. */