www.pudn.com > sip-1[1].5.0.part02.rar > pmatch.c
/*
* Program: IMAP Wildcard Matching Routines (case-dependent)
*
* Author: Mark Crispin
* Networks and Distributed Computing
* Computing & Communications
* University of Washington
* Administration Building, AG-44
* Seattle, WA 98195
* Internet: MRC@CAC.Washington.EDU
*
* Date: 15 June 2000
* Last Edited: 15 June 2000
*
* Copyright 2000 by the University of Washington
*
* Permission to use, copy, modify, and distribute this software and its
* documentation for any purpose and without fee is hereby granted, provided
* that the above copyright notices appear in all copies and that both the
* above copyright notices and this permission notice appear in supporting
* documentation, and that the name of the University of Washington not be
* used in advertising or publicity pertaining to distribution of the software
* without specific, written prior permission. This software is made available
* "as is", and
* THE UNIVERSITY OF WASHINGTON DISCLAIMS ALL WARRANTIES, EXPRESS OR IMPLIED,
* WITH REGARD TO THIS SOFTWARE, INCLUDING WITHOUT LIMITATION ALL IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE, AND IN
* NO EVENT SHALL THE UNIVERSITY OF WASHINGTON BE LIABLE FOR ANY SPECIAL,
* INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
* LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, TORT
* (INCLUDING NEGLIGENCE) OR STRICT LIABILITY, ARISING OUT OF OR IN CONNECTION
* WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*
*/
/* Wildcard pattern match
* Accepts: base string
* pattern string
* delimiter character
* Returns: T if pattern matches base, else NIL
*/
long pmatch_full (char *s,char *pat,char delim)
{
switch (*pat) {
case '%': /* non-recursive */
/* % at end, OK if no inferiors */
if (!pat[1]) return (delim && strchr (s,delim)) ? NIL : T;
/* scan remainder of string until delimiter */
do if (pmatch_full (s,pat+1,delim)) return T;
while ((*s != delim) && *s++);
break;
case '*': /* match 0 or more characters */
if (!pat[1]) return T; /* * at end, unconditional match */
/* scan remainder of string */
do if (pmatch_full (s,pat+1,delim)) return T;
while (*s++);
break;
case '\0': /* end of pattern */
return *s ? NIL : T; /* success if also end of base */
default: /* match this character */
return (*pat == *s) ? pmatch_full (s+1,pat+1,delim) : NIL;
}
return NIL;
}
/* Directory pattern match
* Accepts: base string
* pattern string
* delimiter character
* Returns: T if base is a matching directory of pattern, else NIL
*/
long dmatch (char *s,char *pat,char delim)
{
switch (*pat) {
case '%': /* non-recursive */
if (!*s) return T; /* end of base means have a subset match */
if (!*++pat) return NIL; /* % at end, no inferiors permitted */
/* scan remainder of string until delimiter */
do if (dmatch (s,pat,delim)) return T;
while ((*s != delim) && *s++);
if (*s && !s[1]) return T; /* ends with delimiter, must be subset */
return dmatch (s,pat,delim);/* do new scan */
case '*': /* match 0 or more characters */
return T; /* unconditional match */
case '\0': /* end of pattern */
break;
default: /* match this character */
if (*s) return (*pat == *s) ? dmatch (s+1,pat+1,delim) : NIL;
/* end of base, return if at delimiter */
else if (*pat == delim) return T;
break;
}
return NIL;
}