2003-08-13 01:53:07 +00:00
/*
Unix SMB / CIFS implementation .
filename matching routine
2004-10-02 01:43:43 +00:00
Copyright ( C ) Andrew Tridgell 1992 - 2004
2003-08-13 01:53:07 +00:00
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 of the License , 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
2004-10-02 01:43:43 +00:00
Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
2003-08-13 01:53:07 +00:00
/*
This module was originally based on fnmatch . c copyright by the Free
2004-10-02 01:43:43 +00:00
Software Foundation . It bears little ( if any ) resemblence to that
code now
2003-08-13 01:53:07 +00:00
*/
# include "includes.h"
2004-10-08 08:13:00 +00:00
static int null_match ( const char * p )
2003-08-13 01:53:07 +00:00
{
2004-10-02 01:43:43 +00:00
for ( ; * p ; p + + ) {
2004-10-08 08:13:00 +00:00
if ( * p ! = ' * ' & &
* p ! = ' < ' & &
* p ! = ' " ' & &
* p ! = ' > ' ) return - 1 ;
2003-08-13 01:53:07 +00:00
}
return 0 ;
}
2004-10-02 01:43:43 +00:00
/*
the max_n structure is purely for efficiency , it doesn ' t contribute
to the matching algorithm except by ensuring that the algorithm does
not grow exponentially
*/
struct max_n {
2004-10-08 08:13:00 +00:00
const char * predot ;
const char * postdot ;
2004-10-02 01:43:43 +00:00
} ;
2003-08-13 01:53:07 +00:00
2004-10-02 01:43:43 +00:00
/*
p and n are the pattern and string being matched . The max_n array is
an optimisation only . The ldot pointer is NULL if the string does
not contain a ' . ' , otherwise it points at the last dot in ' n ' .
2003-08-13 01:53:07 +00:00
*/
2004-10-08 08:13:00 +00:00
static int ms_fnmatch_core ( const char * p , const char * n ,
struct max_n * max_n , const char * ldot )
2003-08-13 01:53:07 +00:00
{
2004-10-08 08:13:00 +00:00
codepoint_t c , c2 ;
2004-10-02 01:43:43 +00:00
int i ;
2004-10-08 08:13:00 +00:00
size_t size , size_n ;
while ( ( c = next_codepoint ( p , & size ) ) ) {
p + = size ;
2003-08-13 01:53:07 +00:00
switch ( c ) {
2004-10-08 08:13:00 +00:00
case ' * ' :
2004-10-02 01:43:43 +00:00
/* a '*' matches zero or more characters of any type */
if ( max_n - > predot & & max_n - > predot < = n ) {
return null_match ( p ) ;
}
2004-10-08 08:13:00 +00:00
for ( i = 0 ; n [ i ] ; i + = size_n ) {
next_codepoint ( n + i , & size_n ) ;
2004-10-02 01:43:43 +00:00
if ( ms_fnmatch_core ( p , n + i , max_n + 1 , ldot ) = = 0 ) {
return 0 ;
}
}
if ( ! max_n - > predot | | max_n - > predot > n ) max_n - > predot = n ;
return null_match ( p ) ;
2003-08-13 01:53:07 +00:00
2004-10-08 08:13:00 +00:00
case ' < ' :
2004-10-02 01:43:43 +00:00
/* a '<' matches zero or more characters of
any type , but stops matching at the last
' . ' in the string . */
if ( max_n - > predot & & max_n - > predot < = n ) {
return null_match ( p ) ;
}
if ( max_n - > postdot & & max_n - > postdot < = n & & n < = ldot ) {
2003-08-13 01:53:07 +00:00
return - 1 ;
}
2004-10-08 08:13:00 +00:00
for ( i = 0 ; n [ i ] ; i + = size_n ) {
next_codepoint ( n + i , & size_n ) ;
2004-10-02 01:43:43 +00:00
if ( ms_fnmatch_core ( p , n + i , max_n + 1 , ldot ) = = 0 ) return 0 ;
if ( n + i = = ldot ) {
2004-10-08 08:13:00 +00:00
if ( ms_fnmatch_core ( p , n + i + size_n , max_n + 1 , ldot ) = = 0 ) return 0 ;
2004-10-02 01:43:43 +00:00
if ( ! max_n - > postdot | | max_n - > postdot > n ) max_n - > postdot = n ;
return - 1 ;
}
2004-09-03 17:44:07 +00:00
}
2004-10-02 01:43:43 +00:00
if ( ! max_n - > predot | | max_n - > predot > n ) max_n - > predot = n ;
return null_match ( p ) ;
2004-10-08 08:13:00 +00:00
case ' ? ' :
2004-10-02 01:43:43 +00:00
/* a '?' matches any single character */
if ( ! * n ) {
return - 1 ;
2003-08-13 01:53:07 +00:00
}
2004-10-08 08:13:00 +00:00
next_codepoint ( n , & size_n ) ;
n + = size_n ;
2003-08-13 01:53:07 +00:00
break ;
2004-10-08 08:13:00 +00:00
case ' > ' :
/* a '?' matches any single character, but
treats ' . ' specially */
if ( n [ 0 ] = = ' . ' ) {
2004-10-02 01:43:43 +00:00
if ( ! n [ 1 ] & & null_match ( p ) = = 0 ) {
return 0 ;
2003-08-13 01:53:07 +00:00
}
2004-10-02 01:43:43 +00:00
break ;
2003-08-13 01:53:07 +00:00
}
2004-10-02 01:43:43 +00:00
if ( ! * n ) return null_match ( p ) ;
2004-10-08 08:13:00 +00:00
next_codepoint ( n , & size_n ) ;
n + = size_n ;
2003-08-13 01:53:07 +00:00
break ;
2004-10-08 08:13:00 +00:00
case ' " ' :
/* a bit like a soft '.' */
2004-10-02 01:43:43 +00:00
if ( * n = = 0 & & null_match ( p ) = = 0 ) {
return 0 ;
}
2004-10-08 08:13:00 +00:00
if ( * n ! = ' . ' ) return - 1 ;
next_codepoint ( n , & size_n ) ;
n + = size_n ;
2003-08-13 01:53:07 +00:00
break ;
default :
2004-10-08 08:13:00 +00:00
c2 = next_codepoint ( n , & size_n ) ;
if ( c ! = c2 & & codepoint_cmpi ( c , c2 ) ! = 0 ) {
2004-10-02 01:43:43 +00:00
return - 1 ;
}
2004-10-08 08:13:00 +00:00
n + = size_n ;
2004-10-02 01:43:43 +00:00
break ;
2003-08-13 01:53:07 +00:00
}
}
2004-10-02 01:43:43 +00:00
if ( ! * n ) {
return 0 ;
}
2003-08-13 01:53:07 +00:00
return - 1 ;
}
int ms_fnmatch ( const char * pattern , const char * string , enum protocol_types protocol )
{
2004-10-02 01:43:43 +00:00
int ret , count , i ;
struct max_n * max_n = NULL ;
if ( strcmp ( string , " .. " ) = = 0 ) {
string = " . " ;
}
2003-08-13 01:53:07 +00:00
2004-10-02 05:09:16 +00:00
if ( strpbrk ( pattern , " <>*? \" " ) = = NULL ) {
/* this is not just an optmisation - it is essential
for LANMAN1 correctness */
return StrCaseCmp ( pattern , string ) ;
}
2004-10-02 01:43:43 +00:00
if ( protocol < = PROTOCOL_LANMAN2 ) {
2004-10-08 08:13:00 +00:00
char * p = talloc_strdup ( NULL , pattern ) ;
if ( p = = NULL ) {
return - 1 ;
}
2004-10-02 01:43:43 +00:00
/*
for older negotiated protocols it is possible to
translate the pattern to produce a " new style "
pattern that exactly matches w2k behaviour
*/
for ( i = 0 ; p [ i ] ; i + + ) {
2004-10-08 08:13:00 +00:00
if ( p [ i ] = = ' ? ' ) {
p [ i ] = ' > ' ;
} else if ( p [ i ] = = ' . ' & &
( p [ i + 1 ] = = ' ? ' | |
p [ i + 1 ] = = ' * ' | |
2004-10-02 01:43:43 +00:00
p [ i + 1 ] = = 0 ) ) {
2004-10-08 08:13:00 +00:00
p [ i ] = ' " ' ;
} else if ( p [ i ] = = ' * ' & &
p [ i + 1 ] = = ' . ' ) {
p [ i ] = ' < ' ;
2004-10-02 01:43:43 +00:00
}
}
2004-10-08 08:13:00 +00:00
ret = ms_fnmatch ( p , string , PROTOCOL_NT1 ) ;
talloc_free ( p ) ;
return ret ;
2004-10-02 01:43:43 +00:00
}
2004-10-08 08:13:00 +00:00
for ( count = i = 0 ; pattern [ i ] ; i + + ) {
if ( pattern [ i ] = = ' * ' | | pattern [ i ] = = ' < ' ) count + + ;
2004-10-02 01:43:43 +00:00
}
2005-01-27 07:08:20 +00:00
max_n = talloc_array ( NULL , struct max_n , count ) ;
2004-10-02 01:43:43 +00:00
if ( ! max_n ) {
return - 1 ;
}
memset ( max_n , 0 , sizeof ( struct max_n ) * count ) ;
2004-10-08 08:13:00 +00:00
ret = ms_fnmatch_core ( pattern , string , max_n , strrchr ( string , ' . ' ) ) ;
2004-10-02 01:43:43 +00:00
talloc_free ( max_n ) ;
2003-08-13 01:53:07 +00:00
return ret ;
}
2004-10-02 01:43:43 +00:00
2003-08-13 01:53:07 +00:00
/* a generic fnmatch function - uses for non-CIFS pattern matching */
int gen_fnmatch ( const char * pattern , const char * string )
{
return ms_fnmatch ( pattern , string , PROTOCOL_NT1 ) ;
}