1
0
mirror of https://github.com/samba-team/samba.git synced 2024-12-25 23:21:54 +03:00
samba-mirror/source3/smbd/mangle.c
Jeremy Allison f3eeb2f1fa Removed broken change I made to mangle.c (ooops. Andrew's original
algorithm was correct).
Finally (I think) fixed the mangled directory stack issue in
scan_directory() correctly. Mangled & non-mangled names are
now being checked correctly. Hurrah to Ulrik Dickow <ukd@kampsax.dk>
who helped isolate this one.
Jeremy.
(This used to be commit 37f5f7b557)
1998-03-04 22:24:07 +00:00

709 lines
19 KiB
C

/*
Unix SMB/Netbios implementation.
Version 1.9.
Name mangling
Copyright (C) Andrew Tridgell 1992-1998
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
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
#include "includes.h"
extern int DEBUGLEVEL;
extern int case_default;
extern BOOL case_mangle;
/****************************************************************************
* Provide a checksum on a string
*
* Input: s - the nul-terminated character string for which the checksum
* will be calculated.
* Output: The checksum value calculated for s.
*
****************************************************************************/
int str_checksum(char *s)
{
int res = 0;
int c;
int i=0;
while( *s )
{
c = *s;
res ^= (c << (i % 15)) ^ (c >> (15-(i%15)));
s++; i++;
}
return(res);
} /* str_checksum */
/****************************************************************************
return True if a name is a special msdos reserved name
****************************************************************************/
static BOOL is_reserved_msdos(char *fname)
{
char upperFname[13];
char *p;
StrnCpy (upperFname, fname, 12);
/* lpt1.txt and con.txt etc are also illegal */
p=strchr(upperFname,'.');
if (p)
*p='\0';
strupper (upperFname);
if ((strcmp(upperFname,"CLOCK$") == 0) ||
(strcmp(upperFname,"CON") == 0) ||
(strcmp(upperFname,"AUX") == 0) ||
(strcmp(upperFname,"COM1") == 0) ||
(strcmp(upperFname,"COM2") == 0) ||
(strcmp(upperFname,"COM3") == 0) ||
(strcmp(upperFname,"COM4") == 0) ||
(strcmp(upperFname,"LPT1") == 0) ||
(strcmp(upperFname,"LPT2") == 0) ||
(strcmp(upperFname,"LPT3") == 0) ||
(strcmp(upperFname,"NUL") == 0) ||
(strcmp(upperFname,"PRN") == 0))
return (True) ;
return (False);
} /* is_reserved_msdos */
/****************************************************************************
return True if a name is in 8.3 dos format
****************************************************************************/
BOOL is_8_3(char *fname, BOOL check_case)
{
int len;
char *dot_pos;
char *slash_pos = strrchr(fname,'/');
int l;
if( slash_pos )
fname = slash_pos+1;
len = strlen(fname);
DEBUG(5,("checking %s for 8.3\n",fname));
if( check_case && case_mangle )
{
switch (case_default)
{
case CASE_LOWER:
if (strhasupper(fname)) return(False);
break;
case CASE_UPPER:
if (strhaslower(fname)) return(False);
break;
}
}
/* can't be longer than 12 chars */
if( len == 0 || len > 12 )
return(False);
/* can't be an MS-DOS Special file such as lpt1 or even lpt1.txt */
if( is_reserved_msdos(fname) )
return(False);
/* can't contain invalid dos chars */
/* Windows use the ANSI charset.
But filenames are translated in the PC charset.
This Translation may be more or less relaxed depending
the Windows application. */
/* %%% A nice improvment to name mangling would be to translate
filename to ANSI charset on the smb server host */
dot_pos = strchr(fname,'.');
{
char *p = fname;
int skip;
dot_pos = 0;
while (*p)
{
if((skip = skip_multibyte_char( *p )) != 0)
p += skip;
else
{
if (*p == '.' && !dot_pos)
dot_pos = (char *) p;
if (!isdoschar(*p))
return(False);
p++;
}
}
}
/* no dot and less than 9 means OK */
if (!dot_pos)
return(len <= 8);
l = PTR_DIFF(dot_pos,fname);
/* base must be at least 1 char except special cases . and .. */
if( l == 0 )
return(strcmp(fname,".") == 0 || strcmp(fname,"..") == 0);
/* base can't be greater than 8 */
if( l > 8 )
return(False);
if( lp_strip_dot() &&
len - l == 1 &&
!strchr(dot_pos+1,'.') )
{
*dot_pos = 0;
return(True);
}
/* extension must be between 1 and 3 */
if( (len - l < 2 ) || (len - l > 4) )
return(False);
/* extension can't have a dot */
if( strchr(dot_pos+1,'.') )
return(False);
/* must be in 8.3 format */
return(True);
} /* is_8_3 */
/* -------------------------------------------------------------------------- **
* This section creates and maintains a stack of name mangling results.
* The original comments read: "keep a stack of name mangling results - just
* so file moves and copies have a chance of working" (whatever that means).
*
* There are three functions to manage the stack:
* reset_mangled_stack() -
* push_mangled_name() -
* check_mangled_stack() -
*/
fstring *mangled_stack = NULL;
int mangled_stack_size = 0;
int mangled_stack_len = 0;
/****************************************************************************
* create the mangled stack CRH
****************************************************************************/
void reset_mangled_stack( int size )
{
if( mangled_stack )
{
free(mangled_stack);
mangled_stack_size = 0;
mangled_stack_len = 0;
}
if( size > 0 )
{
mangled_stack = (fstring *)malloc( sizeof(fstring) * size );
if( mangled_stack )
mangled_stack_size = size;
}
else
mangled_stack = NULL;
} /* create_mangled_stack */
/****************************************************************************
* push a mangled name onto the stack CRH
****************************************************************************/
static void push_mangled_name(char *s)
{
int i;
char *p;
/* If the stack doesn't exist... Fail. */
if( !mangled_stack )
return;
/* If name <s> is already on the stack, move it to the top. */
for( i=0; i<mangled_stack_len; i++ )
{
if( strcmp( s, mangled_stack[i] ) == 0 )
{
array_promote( mangled_stack[0],sizeof(fstring), i );
return;
}
}
/* If name <s> wasn't already there, add it to the top of the stack. */
memmove( mangled_stack[1], mangled_stack[0],
sizeof(fstring) * MIN(mangled_stack_len, mangled_stack_size-1) );
strcpy( mangled_stack[0], s );
mangled_stack_len = MIN( mangled_stack_size, mangled_stack_len+1 );
/* Hmmm...
* Find the last dot '.' in the name,
* if there are any upper case characters past the last dot
* and there are no more than three characters past the last dot
* then terminate the name *at* the last dot.
*/
p = strrchr( mangled_stack[0], '.' );
if( p && (!strhasupper(p+1)) && (strlen(p+1) < (size_t)4) )
*p = 0;
} /* push_mangled_name */
/****************************************************************************
* check for a name on the mangled name stack CRH
****************************************************************************/
BOOL check_mangled_stack(char *s)
{
int i;
pstring tmpname;
char extension[5];
char *p = strrchr( s, '.' );
BOOL check_extension = False;
extension[0] = 0;
/* If the stack doesn't exist, fail. */
if( !mangled_stack )
return(False);
/* If there is a file extension, then we need to play with it, too. */
if( p )
{
check_extension = True;
StrnCpy( extension, p, 4 );
strlower( extension ); /* XXXXXXX */
}
for( i=0; i<mangled_stack_len; i++ )
{
strcpy(tmpname,mangled_stack[i]);
mangle_name_83(tmpname);
if( strequal(tmpname,s) )
{
strcpy(s,mangled_stack[i]);
break;
}
if( check_extension && !strchr(mangled_stack[i],'.') )
{
pstrcpy(tmpname,mangled_stack[i]);
strcat(tmpname,extension);
mangle_name_83(tmpname);
if( strequal(tmpname,s) )
{
strcpy(s,mangled_stack[i]);
strcat(s,extension);
break;
}
}
}
if( i < mangled_stack_len )
{
DEBUG(3,("Found %s on mangled stack as %s\n",s,mangled_stack[i]));
array_promote(mangled_stack[0],sizeof(fstring),i);
return(True);
}
return(False);
} /* check_mangled_stack */
/* End of the mangled stack section.
* -------------------------------------------------------------------------- **
*/
static char *map_filename( char *s, /* This is null terminated */
char *pattern, /* This isn't. */
int len ) /* This is the length of pattern. */
{
static pstring matching_bit; /* The bit of the string which matches */
/* a * in pattern if indeed there is a * */
char *sp; /* Pointer into s. */
char *pp; /* Pointer into p. */
char *match_start; /* Where the matching bit starts. */
pstring pat;
StrnCpy(pat, pattern, len); /* Get pattern into a proper string! */
pstrcpy(matching_bit,""); /* Match but no star gets this. */
pp = pat; /* Initialise the pointers. */
sp = s;
if( (len == 1) && (*pattern == '*') )
{
return NULL; /* Impossible, too ambiguous for */
} /* words! */
while ((*sp) /* Not the end of the string. */
&& (*pp) /* Not the end of the pattern. */
&& (*sp == *pp) /* The two match. */
&& (*pp != '*')) /* No wildcard. */
{
sp++; /* Keep looking. */
pp++;
}
if( !*sp && !*pp ) /* End of pattern. */
return( matching_bit ); /* Simple match. Return empty string. */
if (*pp == '*')
{
pp++; /* Always interrested in the chacter */
/* after the '*' */
if (!*pp) /* It is at the end of the pattern. */
{
StrnCpy(matching_bit, s, sp-s);
return matching_bit;
}
else
{
/* The next character in pattern must match a character further */
/* along s than sp so look for that character. */
match_start = sp;
while( (*sp) /* Not the end of s. */
&& (*sp != *pp)) /* Not the same */
sp++; /* Keep looking. */
if (!*sp) /* Got to the end without a match. */
{
return NULL;
} /* Still hope for a match. */
else
{
/* Now sp should point to a matching character. */
StrnCpy(matching_bit, match_start, sp-match_start);
/* Back to needing a stright match again. */
while( (*sp) /* Not the end of the string. */
&& (*pp) /* Not the end of the pattern. */
&& (*sp == *pp) ) /* The two match. */
{
sp++; /* Keep looking. */
pp++;
}
if (!*sp && !*pp) /* Both at end so it matched */
return matching_bit;
else
return NULL;
}
}
}
return NULL; /* No match. */
} /* map_filename */
/* this is the magic char used for mangling */
char magic_char = '~';
/****************************************************************************
return True if the name could be a mangled name
****************************************************************************/
BOOL is_mangled( char *s )
{
char *m = strchr(s,magic_char);
if( !m )
return(False);
/* we use two base 36 chars before the extension */
if( m[1] == '.' || m[1] == 0 ||
m[2] == '.' || m[2] == 0 ||
(m[3] != '.' && m[3] != 0) )
return( is_mangled(m+1) );
/* it could be */
return(True);
} /* is_mangled */
/****************************************************************************
return a base 36 character. v must be from 0 to 35.
****************************************************************************/
static char base36(unsigned int v)
{
static char basechars[] = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ";
return basechars[v % 36];
} /* base36 */
static void do_fwd_mangled_map(char *s, char *MangledMap)
{
/* MangledMap is a series of name pairs in () separated by spaces.
* If s matches the first of the pair then the name given is the
* second of the pair. A * means any number of any character and if
* present in the second of the pair as well as the first the
* matching part of the first string takes the place of the * in the
* second.
*
* I wanted this so that we could have RCS files which can be used
* by UNIX and DOS programs. My mapping string is (RCS rcs) which
* converts the UNIX RCS file subdirectory to lowercase thus
* preventing mangling.
*/
char *start=MangledMap; /* Use this to search for mappings. */
char *end; /* Used to find the end of strings. */
char *match_string;
pstring new_string; /* Make up the result here. */
char *np; /* Points into new_string. */
DEBUG(5,("Mangled Mapping '%s' map '%s'\n", s, MangledMap));
while (*start)
{
while ((*start) && (*start != '('))
start++;
if (!*start)
continue; /* Always check for the end. */
start++; /* Skip the ( */
end = start; /* Search for the ' ' or a ')' */
DEBUG(5,("Start of first in pair '%s'\n", start));
while ((*end) && !((*end == ' ') || (*end == ')')))
end++;
if (!*end)
{
start = end;
continue; /* Always check for the end. */
}
DEBUG(5,("End of first in pair '%s'\n", end));
if ((match_string = map_filename(s, start, end-start)))
{
DEBUG(5,("Found a match\n"));
/* Found a match. */
start = end+1; /* Point to start of what it is to become. */
DEBUG(5,("Start of second in pair '%s'\n", start));
end = start;
np = new_string;
while ((*end) /* Not the end of string. */
&& (*end != ')') /* Not the end of the pattern. */
&& (*end != '*')) /* Not a wildcard. */
*np++ = *end++;
if (!*end)
{
start = end;
continue; /* Always check for the end. */
}
if (*end == '*')
{
pstrcpy(np, match_string);
np += strlen(match_string);
end++; /* Skip the '*' */
while ((*end) /* Not the end of string. */
&& (*end != ')') /* Not the end of the pattern. */
&& (*end != '*')) /* Not a wildcard. */
*np++ = *end++;
}
if (!*end)
{
start = end;
continue; /* Always check for the end. */
}
*np++ = '\0'; /* NULL terminate it. */
DEBUG(5,("End of second in pair '%s'\n", end));
pstrcpy(s, new_string); /* Substitute with the new name. */
DEBUG(5,("s is now '%s'\n", s));
}
start = end; /* Skip a bit which cannot be wanted */
/* anymore. */
start++;
}
} /* do_fwd_mangled_map */
/****************************************************************************
do the actual mangling to 8.3 format
****************************************************************************/
void mangle_name_83(char *s)
{
int csum = str_checksum(s);
char *p;
char extension[4];
char base[9];
int baselen = 0;
int extlen = 0;
int skip;
extension[0]=0;
base[0]=0;
p = strrchr(s,'.');
if( p && (strlen(p+1) < (size_t)4) )
{
BOOL all_normal = (strisnormal(p+1)); /* XXXXXXXXX */
if (all_normal && p[1] != 0)
{
*p = 0;
csum = str_checksum(s);
*p = '.';
}
}
strupper(s);
DEBUG(5,("Mangling name %s to ",s));
if( p )
{
if (p == s)
strcpy(extension,"___");
else
{
*p++ = 0;
while (*p && extlen < 3)
{
skip = skip_multibyte_char(*p);
if (skip == 2)
{
if (extlen < 2)
{
extension[extlen++] = p[0];
extension[extlen++] = p[1];
}
else
{
extension[extlen++] = base36 (((unsigned char) *p) % 36);
}
p += 2;
}
else if( skip == 1 )
{
extension[extlen++] = p[0];
p++;
}
else
{
if (isdoschar (*p) && *p != '.')
extension[extlen++] = p[0];
p++;
}
}
extension[extlen] = 0;
}
}
p = s;
while (*p && baselen < 5)
{
skip = skip_multibyte_char(*p);
if (skip == 2)
{
if (baselen < 4)
{
base[baselen++] = p[0];
base[baselen++] = p[1];
}
else
{
base[baselen++] = base36 (((unsigned char) *p) % 36);
}
p += 2;
}
else if( skip == 1)
{
base[baselen++] = p[0];
p++;
}
else
{
if (isdoschar (*p) && *p != '.')
base[baselen++] = p[0];
p++;
}
}
base[baselen] = 0;
csum = csum % (36*36);
sprintf(s,"%s%c%c%c",base,magic_char,base36(csum/36),base36(csum%36));
if( *extension )
{
strcat(s,".");
strcat(s,extension);
}
DEBUG(5,("%s\n",s));
} /* mangle_name_83 */
/*******************************************************************
work out if a name is illegal, even for long names
******************************************************************/
static BOOL illegal_name(char *name)
{
static unsigned char illegal[256];
static BOOL initialised=False;
unsigned char *s;
int skip;
if( !initialised )
{
char *ill = "*\\/?<>|\":";
initialised = True;
bzero((char *)illegal,256);
for( s = (unsigned char *)ill; *s; s++ )
illegal[*s] = True;
}
for (s = (unsigned char *)name; *s;)
{
skip = skip_multibyte_char( *s );
if (skip != 0)
s += skip;
else
{
if (illegal[*s])
return(True);
else
s++;
}
}
return(False);
} /* illegal_name */
/****************************************************************************
convert a filename to DOS format. return True if successful.
****************************************************************************/
BOOL name_map_mangle(char *OutName,BOOL need83,int snum)
{
#ifdef MANGLE_LONG_FILENAMES
if( !need83 && illegal_name(OutName) )
need83 = True;
#endif
/* apply any name mappings */
{
char *map = lp_mangled_map(snum);
if (map && *map)
do_fwd_mangled_map(OutName,map);
}
/* check if it's already in 8.3 format */
if( need83 && !is_8_3(OutName, True) )
{
if( !lp_manglednames(snum) )
return(False);
/* mangle it into 8.3 */
push_mangled_name(OutName);
mangle_name_83(OutName);
}
return(True);
} /* name_map_mangle */