xref: /dflybsd-src/contrib/binutils-2.34/libiberty/fnmatch.c (revision b52ef7118d1621abed722c5bbbd542210290ecef)
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