/* Copyright (C) 1991, 1992, 1993, 1996, 1997 Free Software Foundation, Inc.
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2, or (at your option)
any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software Foundation,
Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
/* Modified version of fnmatch.c - Robert Nelson */
#include "bacula.h"
#include "enh_fnmatch.h"
# ifndef errno
extern int errno;
# endif
int
enh_fnmatch_sub(const char *pattern, const char *string, int patternlen, int flags)
{
register const char *p = pattern, *n = string;
register char c;
/* Note that this evaluates C many times. */
# define FOLD(c) ((flags & FNM_CASEFOLD) && B_ISUPPER (c) ? tolower (c) : (c))
while ((p - pattern) < patternlen)
{
c = *p++;
c = FOLD (c);
switch (c)
{
case '?':
if (*n == '\0')
return 0;
else if ((flags & FNM_FILE_NAME) && IsPathSeparator(*n))
return 0;
else if ((flags & FNM_PERIOD) && *n == '.' &&
(n == string || ((flags & FNM_FILE_NAME) && IsPathSeparator(n[-1]))))
return 0;
break;
case '\\':
if (!(flags & FNM_NOESCAPE))
{
if ((p - pattern) >= patternlen)
/* Trailing \ loses. */
return 0;
c = *p++;
c = FOLD(c);
}
if (FOLD (*n) != c)
return 0;
break;
case '*':
if ((flags & FNM_PERIOD) && *n == '.' &&
(n == string || ((flags & FNM_FILE_NAME) && IsPathSeparator(n[-1]))))
return FNM_NOMATCH;
if ((p - pattern) >= patternlen)
return patternlen;
for (c = *p++; ((p - pattern) <= patternlen) && (c == '?' || c == '*'); c = *p++)
{
if ((flags & FNM_FILE_NAME) && IsPathSeparator(*n))
/* A slash does not match a wildcard under FNM_FILE_NAME. */
return 0;
else if (c == '?')
{
/* A ? needs to match one character. */
if (*n == '\0')
/* There isn't another character; no match. */
return 0;
else
/* One character of the string is consumed in matching
this ? wildcard, so *??? won't match if there are
less than three characters. */
++n;
}
}
if ((p - pattern) >= patternlen)
return patternlen;
{
char c1 = (!(flags & FNM_NOESCAPE) && c == '\\') ? *p : c;
c1 = FOLD (c1);
for (--p; *n != '\0'; ++n)
{
if (c == '[' || c == '{' || FOLD((unsigned char)*n) == c1)
{
int len;
len = enh_fnmatch_sub(p, n, (int)(patternlen - (p - pattern)), flags & ~FNM_PERIOD);
if (len > 0 && n[len] == '\0')
return (int)(n - string + len);
}
else
{
if ((flags & FNM_FILE_NAME) && IsPathSeparator(*n))
return 0; /* A slash does not match a wildcard under FNM_FILE_NAME. */
}
}
return 0;
}
case '{':
{
const char *pstart = p;
while ((p - pattern) < patternlen)
{
c = *p++;
if (!(flags & FNM_NOESCAPE) && c == '\\')
{
if ((p - pattern) >= patternlen)
return 0;
++p;
continue;
}
if (c == ',' || c == '}')
{
int matchlen;
matchlen = enh_fnmatch_sub(pstart, n, (int)(p - pstart - 1), flags & ~FNM_PERIOD);
if (matchlen > 0)
{
n += matchlen - 1;
while (c != '}')
{
if (!(flags & FNM_NOESCAPE) && c == '\\')
{
if ((p - pattern) >= patternlen)
return 0;
++p;
}
if ((p - pattern) >= patternlen)
return 0;
c = *p++;
}
break;
}
if (c == '}')
return 0;
pstart = p;
}
}
break;
}
case '[':
{
/* Nonzero if the sense of the character class is inverted. */
register int nnot;
if (*n == '\0')
return 0;
if ((flags & FNM_PERIOD) && *n == '.' &&
(n == string || ((flags & FNM_FILE_NAME) && IsPathSeparator(n[-1]))))
return 0;
nnot = (*p == '!' || *p == '^');
if (nnot)
++p;
if ((p - pattern) >= patternlen)
/* [ (unterminated) loses. */
return 0;
c = *p++;
for (;;)
{
register char cstart, cend;
cstart = cend = FOLD (c);
if ((p - pattern) >= patternlen)
/* [ (unterminated) loses. */
return 0;
c = *p++;
c = FOLD (c);
if ((flags & FNM_FILE_NAME) && IsPathSeparator(c))
/* [/] can never match. */
return 0;
if (c == '-' && *p != ']')
{
if ((p - pattern) >= patternlen)
return 0;
cend = *p++;
cend = FOLD (cend);
if ((p - pattern) >= patternlen)
return 0;
c = *p++;
}
if (FOLD (*n) >= cstart && FOLD (*n) <= cend)
goto matched;
if (c == ']')
break;
}
if (!nnot)
return 0;
break;
matched:;
/* Skip the rest of the [...] that already matched. */
while (c != ']')
{
if ((p - pattern) >= patternlen)
return 0;
c = *p++;
}
if (nnot)
return 0;
}
break;
default:
if (c != FOLD (*n))
return 0;
break;
}
++n;
}
return (int)(n - string);
# undef FOLD
}
/* Match STRING against the filename pattern PATTERN, returning number of characters
in STRING that were matched if all of PATTERN matches, nonzero if not. */
int
enh_fnmatch(const char *pattern, const char *string, int flags)
{
int matchlen;
matchlen = enh_fnmatch_sub(pattern, string, (int)strlen(pattern), flags);
if (matchlen == 0)
return FNM_NOMATCH;
if (string[matchlen] == '\0')
return 0;
if ((flags & FNM_LEADING_DIR) && IsPathSeparator(string[matchlen]))
/* The FNM_LEADING_DIR flag says that "foo*" matches "foobar/frobozz". */
return 0;
return FNM_NOMATCH;
}
syntax highlighted by Code2HTML, v. 0.9.1