r23784: use the GPLv3 boilerplate as recommended by the FSF and the license text
[tprouty/samba.git] / source / smbd / mangle_hash.c
index f38c2ae2c45b8554be9d0b855abb90e46c998fb7..4c02d9685fb7463a0efd5ce1a9626111bfb668be 100644 (file)
@@ -7,7 +7,7 @@
    
    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
+   the Free Software Foundation; either version 3 of the License, or
    (at your option) any later version.
    
    This program is distributed in the hope that it will be useful,
    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.
+   along with this program.  If not, see <http://www.gnu.org/licenses/>.
 */
 
-
-/* -------------------------------------------------------------------------- **
- * Notable problems...
- *
- *  March/April 1998  CRH
- *  - Many of the functions in this module overwrite string buffers passed to
- *    them.  This causes a variety of problems and is, generally speaking,
- *    dangerous and scarry.  See the kludge notes in name_map()
- *    below.
- *  - It seems that something is calling name_map() twice.  The
- *    first call is probably some sort of test.  Names which contain
- *    illegal characters are being doubly mangled.  I'm not sure, but
- *    I'm guessing the problem is in server.c.
- *
- * -------------------------------------------------------------------------- **
- */
-
-/* -------------------------------------------------------------------------- **
- * History...
- *
- *  March/April 1998  CRH
- *  Updated a bit.  Rewrote is_mangled() to be a bit more selective.
- *  Rewrote the mangled name cache.  Added comments here and there.
- *  &c.
- * -------------------------------------------------------------------------- **
- */
-
 #include "includes.h"
 
-
-/* -------------------------------------------------------------------------- **
- * External Variables...
- */
-
-extern int case_default;    /* Are conforming 8.3 names all upper or lower?   */
-extern BOOL case_mangle;    /* If true, all chars in 8.3 should be same case. */
-
 /* -------------------------------------------------------------------------- **
  * Other stuff...
  *
@@ -74,41 +38,17 @@ extern BOOL case_mangle;    /* If true, all chars in 8.3 should be same case. */
  *
  * chartest       - array 0..255.  The index range is the set of all possible
  *                  values of a byte.  For each byte value, the content is a
- *                  two nibble pair.  See BASECHAR_MASK and ILLEGAL_MASK,
- *                  below.
+ *                  two nibble pair.  See BASECHAR_MASK below.
  *
  * ct_initialized - False until the chartest array has been initialized via
  *                  a call to init_chartest().
  *
  * BASECHAR_MASK  - Masks the upper nibble of a one-byte value.
  *
- * ILLEGAL_MASK   - Masks the lower nibble of a one-byte value.
- *
  * isbasecahr()   - Given a character, check the chartest array to see
  *                  if that character is in the basechars set.  This is
  *                  faster than using strchr_m().
  *
- * isillegal()    - Given a character, check the chartest array to see
- *                  if that character is in the illegal characters set.
- *                  This is faster than using strchr_m().
- *
- * mangled_cache  - Cache header used for storing mangled -> original
- *                  reverse maps.
- *
- * mc_initialized - False until the mangled_cache structure has been
- *                  initialized via a call to reset_mangled_cache().
- *
- * MANGLED_CACHE_MAX_ENTRIES - Default maximum number of entries for the
- *                  cache.  A value of 0 indicates "infinite".
- *
- * MANGLED_CACHE_MAX_MEMORY  - Default maximum amount of memory for the
- *                  cache.  When the cache was kept as an array of 256
- *                  byte strings, the default cache size was 50 entries.
- *                  This required a fixed 12.5Kbytes of memory.  The
- *                  mangled stack parameter is no longer used (though
- *                  this might change).  We're now using a fixed 16Kbyte
- *                  maximum cache size.  This will probably be much more
- *                  than 50 entries.
  */
 
 char magic_char = '~';
@@ -121,35 +61,50 @@ static BOOL          ct_initialized = False;
 
 #define mangle(V) ((char)(basechars[(V) % MANGLE_BASE]))
 #define BASECHAR_MASK 0xf0
-#define ILLEGAL_MASK  0x0f
 #define isbasechar(C) ( (chartest[ ((C) & 0xff) ]) & BASECHAR_MASK )
-#define isillegal(C) ( (chartest[ ((C) & 0xff) ]) & ILLEGAL_MASK )
 
-static ubi_cacheRoot mangled_cache[1] =  { { { 0, 0, 0, 0 }, 0, 0, 0, 0, 0, 0 } };
-static BOOL          mc_initialized   = False;
-#define MANGLED_CACHE_MAX_ENTRIES 0
-#define MANGLED_CACHE_MAX_MEMORY  16384
+static TDB_CONTEXT *tdb_mangled_cache;
 
+/* -------------------------------------------------------------------- */
 
-/* -------------------------------------------------------------------------- **
- * External Variables...
- */
+static NTSTATUS has_valid_83_chars(const smb_ucs2_t *s, BOOL allow_wildcards)
+{
+       if (!*s) {
+               return NT_STATUS_INVALID_PARAMETER;
+       }
 
-extern int case_default;    /* Are conforming 8.3 names all upper or lower?   */
-extern BOOL case_mangle;    /* If true, all chars in 8.3 should be same case. */
+       if (!allow_wildcards && ms_has_wild_w(s)) {
+               return NT_STATUS_UNSUCCESSFUL;
+       }
 
-/* -------------------------------------------------------------------- */
+       while (*s) {
+               if(!isvalid83_w(*s)) {
+                       return NT_STATUS_UNSUCCESSFUL;
+               }
+               s++;
+       }
 
-static NTSTATUS has_valid_chars(const smb_ucs2_t *s)
-{
-       if (!s || !*s) return NT_STATUS_INVALID_PARAMETER;
+       return NT_STATUS_OK;
+}
 
-       /* CHECK: this should not be necessary if the ms wild chars
-          are not valid in valid.dat  --- simo */
-       if (ms_has_wild_w(s)) return NT_STATUS_UNSUCCESSFUL;
+static NTSTATUS has_illegal_chars(const smb_ucs2_t *s, BOOL allow_wildcards)
+{
+       if (!allow_wildcards && ms_has_wild_w(s)) {
+               return NT_STATUS_UNSUCCESSFUL;
+       }
 
        while (*s) {
-               if(!isvalid83_w(*s)) return NT_STATUS_UNSUCCESSFUL;
+               if (*s <= 0x1f) {
+                       /* Control characters. */
+                       return NT_STATUS_UNSUCCESSFUL;
+               }
+               switch(*s) {
+                       case UCS2_CHAR('\\'):
+                       case UCS2_CHAR('/'):
+                       case UCS2_CHAR('|'):
+                       case UCS2_CHAR(':'):
+                               return NT_STATUS_UNSUCCESSFUL;
+               }
                s++;
        }
 
@@ -159,7 +114,9 @@ static NTSTATUS has_valid_chars(const smb_ucs2_t *s)
 /* return False if something fail and
  * return 2 alloced unicode strings that contain prefix and extension
  */
-static NTSTATUS mangle_get_prefix(const smb_ucs2_t *ucs2_string, smb_ucs2_t **prefix, smb_ucs2_t **extension)
+
+static NTSTATUS mangle_get_prefix(const smb_ucs2_t *ucs2_string, smb_ucs2_t **prefix,
+               smb_ucs2_t **extension, BOOL allow_wildcards)
 {
        size_t ext_len;
        smb_ucs2_t *p;
@@ -169,12 +126,10 @@ static NTSTATUS mangle_get_prefix(const smb_ucs2_t *ucs2_string, smb_ucs2_t **pr
        if (!*prefix) {
                return NT_STATUS_NO_MEMORY;
        }
-       if ((p = strrchr_w(*prefix, UCS2_CHAR('.'))))
-       {
+       if ((p = strrchr_w(*prefix, UCS2_CHAR('.')))) {
                ext_len = strlen_w(p+1);
                if ((ext_len > 0) && (ext_len < 4) && (p != *prefix) &&
-                   (NT_STATUS_IS_OK(has_valid_chars(p+1)))) /* check extension */
-               {
+                   (NT_STATUS_IS_OK(has_valid_83_chars(p+1,allow_wildcards)))) /* check extension */ {
                        *p = 0;
                        *extension = strdup_w(p+1);
                        if (!*extension) {
@@ -188,32 +143,56 @@ static NTSTATUS mangle_get_prefix(const smb_ucs2_t *ucs2_string, smb_ucs2_t **pr
 
 /* ************************************************************************** **
  * Return NT_STATUS_UNSUCCESSFUL if a name is a special msdos reserved name.
+ * or contains illegal characters.
  *
  *  Input:  fname - String containing the name to be tested.
  *
- *  Output: NT_STATUS_UNSUCCESSFUL, if the name matches one of the list of reserved names.
+ *  Output: NT_STATUS_UNSUCCESSFUL, if the condition above is true.
  *
  *  Notes:  This is a static function called by is_8_3(), below.
  *
  * ************************************************************************** **
  */
-static NTSTATUS is_valid_name(const smb_ucs2_t *fname)
+
+static NTSTATUS is_valid_name(const smb_ucs2_t *fname, BOOL allow_wildcards, BOOL only_8_3)
 {
        smb_ucs2_t *str, *p;
+       size_t num_ucs2_chars;
        NTSTATUS ret = NT_STATUS_OK;
 
-       if (!fname || !*fname) return NT_STATUS_INVALID_PARAMETER;
+       if (!fname || !*fname)
+               return NT_STATUS_INVALID_PARAMETER;
 
-       if (*fname == UCS2_CHAR('.')) return NT_STATUS_UNSUCCESSFUL;
-       
-       ret = has_valid_chars(fname);
-       if (NT_STATUS_IS_ERR(ret)) return ret;
+       /* . and .. are valid names. */
+       if (strcmp_wa(fname, ".")==0 || strcmp_wa(fname, "..")==0)
+               return NT_STATUS_OK;
+
+       if (only_8_3) {
+               ret = has_valid_83_chars(fname, allow_wildcards);
+               if (!NT_STATUS_IS_OK(ret))
+                       return ret;
+       }
+
+       ret = has_illegal_chars(fname, allow_wildcards);
+       if (!NT_STATUS_IS_OK(ret))
+               return ret;
+
+       /* Name can't end in '.' or ' ' */
+       num_ucs2_chars = strlen_w(fname);
+       if (fname[num_ucs2_chars-1] == UCS2_CHAR('.') || fname[num_ucs2_chars-1] == UCS2_CHAR(' ')) {
+               return NT_STATUS_UNSUCCESSFUL;
+       }
 
        str = strdup_w(fname);
+
+       /* Truncate copy after the first dot. */
        p = strchr_w(str, UCS2_CHAR('.'));
-       if (p) *p = 0;
+       if (p) {
+               *p = 0;
+       }
+
        strupper_w(str);
-       p = &(str[1]);
+       p = &str[1];
 
        switch(str[0])
        {
@@ -254,27 +233,38 @@ static NTSTATUS is_valid_name(const smb_ucs2_t *fname)
        return ret;
 }
 
-static NTSTATUS is_8_3_w(const smb_ucs2_t *fname)
+static NTSTATUS is_8_3_w(const smb_ucs2_t *fname, BOOL allow_wildcards)
 {
        smb_ucs2_t *pref = 0, *ext = 0;
        size_t plen;
        NTSTATUS ret = NT_STATUS_UNSUCCESSFUL;
 
-       if (!fname || !*fname) return NT_STATUS_INVALID_PARAMETER;
+       if (!fname || !*fname)
+               return NT_STATUS_INVALID_PARAMETER;
 
-       if (strlen_w(fname) > 12) return NT_STATUS_UNSUCCESSFUL;
+       if (strlen_w(fname) > 12)
+               return NT_STATUS_UNSUCCESSFUL;
        
        if (strcmp_wa(fname, ".") == 0 || strcmp_wa(fname, "..") == 0)
                return NT_STATUS_OK;
 
-       if (NT_STATUS_IS_ERR(is_valid_name(fname))) goto done;
+       /* Name cannot start with '.' */
+       if (*fname == UCS2_CHAR('.'))
+               return NT_STATUS_UNSUCCESSFUL;
+       
+       if (!NT_STATUS_IS_OK(is_valid_name(fname, allow_wildcards, True)))
+               goto done;
 
-       if (NT_STATUS_IS_ERR(mangle_get_prefix(fname, &pref, &ext))) goto done;
+       if (!NT_STATUS_IS_OK(mangle_get_prefix(fname, &pref, &ext, allow_wildcards)))
+               goto done;
        plen = strlen_w(pref);
 
-       if (strchr_wa(pref, '.')) goto done;
-       if (plen < 1 || plen > 8) goto done;
-       if (ext) if (strlen_w(ext) > 3) goto done;
+       if (strchr_wa(pref, '.'))
+               goto done;
+       if (plen < 1 || plen > 8)
+               goto done;
+       if (ext && (strlen_w(ext) > 3))
+               goto done;
 
        ret = NT_STATUS_OK;
 
@@ -284,26 +274,33 @@ done:
        return ret;
 }
 
-static BOOL is_8_3(const char *fname, BOOL check_case)
+static BOOL is_8_3(const char *fname, BOOL check_case, BOOL allow_wildcards,
+                  const struct share_params *p)
 {
        const char *f;
        smb_ucs2_t *ucs2name;
        NTSTATUS ret = NT_STATUS_UNSUCCESSFUL;
+       size_t size;
 
-       if (!fname || !*fname) return False;
-       if ((f = strrchr(fname, '/')) == NULL) f = fname;
-       else f++;
+       magic_char = lp_magicchar(p);
+
+       if (!fname || !*fname)
+               return False;
+       if ((f = strrchr(fname, '/')) == NULL)
+               f = fname;
+       else
+               f++;
 
-       if (strlen(f) > 12) return False;
+       if (strlen(f) > 12)
+               return False;
        
-       ucs2name = acnv_uxu2(f);
-       if (!ucs2name)
-       {
-               DEBUG(0,("is_8_3: internal error acnv_uxu2() failed!\n"));
+       size = push_ucs2_allocate(&ucs2name, f);
+       if (size == (size_t)-1) {
+               DEBUG(0,("is_8_3: internal error push_ucs2_allocate() failed!\n"));
                goto done;
        }
 
-       ret = is_8_3_w(ucs2name);
+       ret = is_8_3_w(ucs2name, allow_wildcards);
 
 done:
        SAFE_FREE(ucs2name);
@@ -334,21 +331,17 @@ done:
  * ************************************************************************** **
  */
 static void init_chartest( void )
-  {
-  char          *illegalchars = "*\\/?<>|\":";
-  unsigned char *s;
+{
+       const unsigned char *s;
   
-  memset( (char *)chartest, '\0', 256 );
-
-  for( s = (unsigned char *)illegalchars; *s; s++ )
-    chartest[*s] = ILLEGAL_MASK;
+       memset( (char *)chartest, '\0', 256 );
 
-  for( s = (unsigned char *)basechars; *s; s++ )
-    chartest[*s] |= BASECHAR_MASK;
-
-  ct_initialized = True;
-  } /* init_chartest */
+       for( s = (const unsigned char *)basechars; *s; s++ ) {
+               chartest[*s] |= BASECHAR_MASK;
+       }
 
+       ct_initialized = True;
+}
 
 /* ************************************************************************** **
  * Return True if the name *could be* a mangled name.
@@ -367,187 +360,95 @@ static void init_chartest( void )
  *
  * ************************************************************************** **
  */
-static BOOL is_mangled(const char *s)
-  {
-  char *magic;
-
-  if( !ct_initialized )
-    init_chartest();
-
-  magic = strchr_m( s, magic_char );
-  while( magic && magic[1] && magic[2] )          /* 3 chars, 1st is magic. */
-    {
-    if( ('.' == magic[3] || '/' == magic[3] || !(magic[3]))          /* Ends with '.' or nul or '/' ?  */
-     && isbasechar( toupper(magic[1]) )           /* is 2nd char basechar?  */
-     && isbasechar( toupper(magic[2]) ) )         /* is 3rd char basechar?  */
-      return( True );                           /* If all above, then true, */
-    magic = strchr_m( magic+1, magic_char );      /*    else seek next magic. */
-    }
-  return( False );
-  } /* is_mangled */
+static BOOL is_mangled(const char *s, const struct share_params *p)
+{
+       char *magic;
 
+       magic_char = lp_magicchar(p);
 
-/* ************************************************************************** **
- * Compare two cache keys and return a value indicating their ordinal
- * relationship.
- *
- *  Input:  ItemPtr - Pointer to a comparison key.  In this case, this will
- *                    be a mangled name string.
- *          NodePtr - Pointer to a node in the cache.  The node structure
- *                    will be followed in memory by a mangled name string.
- *
- *  Output: A signed integer, as follows:
- *            (x < 0)  <==> Key1 less than Key2
- *            (x == 0) <==> Key1 equals Key2
- *            (x > 0)  <==> Key1 greater than Key2
- *
- *  Notes:  This is a ubiqx-style comparison routine.  See ubi_BinTree for
- *          more info.
- *
- * ************************************************************************** **
- */
-static signed int cache_compare( ubi_btItemPtr ItemPtr, ubi_btNodePtr NodePtr )
-  {
-  char *Key1 = (char *)ItemPtr;
-  char *Key2 = (char *)(((ubi_cacheEntryPtr)NodePtr) + 1);
+       if( !ct_initialized )
+               init_chartest();
 
-  return( StrCaseCmp( Key1, Key2 ) );
-  } /* cache_compare */
+       magic = strchr_m( s, magic_char );
+       while( magic && magic[1] && magic[2] ) {         /* 3 chars, 1st is magic. */
+               if( ('.' == magic[3] || '/' == magic[3] || !(magic[3]))          /* Ends with '.' or nul or '/' ?  */
+                               && isbasechar( toupper_ascii(magic[1]) )           /* is 2nd char basechar?  */
+                               && isbasechar( toupper_ascii(magic[2]) ) )         /* is 3rd char basechar?  */
+                       return( True );                           /* If all above, then true, */
+               magic = strchr_m( magic+1, magic_char );      /*    else seek next magic. */
+       }
+       return( False );
+}
 
-/* ************************************************************************** **
- * Free a cache entry.
- *
- *  Input:  WarrenZevon - Pointer to the entry that is to be returned to
- *                        Nirvana.
- *  Output: none.
- *
- *  Notes:  This function gets around the possibility that the standard
- *          free() function may be implemented as a macro, or other evil
- *          subversions (oh, so much fun).
- *
- * ************************************************************************** **
- */
-static void cache_free_entry( ubi_trNodePtr WarrenZevon )
-  {
-         ZERO_STRUCTP(WarrenZevon);
-         SAFE_FREE( WarrenZevon );
-  } /* cache_free_entry */
+/***************************************************************************
+ Initializes or clears the mangled cache.
+***************************************************************************/
 
-/* ************************************************************************** **
- * Initializes or clears the mangled cache.
- *
- *  Input:  none.
- *  Output: none.
- *
- *  Notes:  There is a section below that is commented out.  It shows how
- *          one might use lp_ calls to set the maximum memory and entry size
- *          of the cache.  You might also want to remove the constants used
- *          in ubi_cacheInit() and replace them with lp_ calls.  If so, then
- *          the calls to ubi_cacheSetMax*() would be moved into the else
- *          clause.  Another option would be to pass in the max_entries and
- *          max_memory values as parameters.  crh 09-Apr-1998.
- *
- * ************************************************************************** **
- */
 static void mangle_reset( void )
-  {
-  if( !mc_initialized )
-    {
-    (void)ubi_cacheInit( mangled_cache,
-                         cache_compare,
-                         cache_free_entry,
-                         MANGLED_CACHE_MAX_ENTRIES,
-                         MANGLED_CACHE_MAX_MEMORY );
-    mc_initialized = True;
-    }
-  else
-    {
-    (void)ubi_cacheClear( mangled_cache );
-    }
-
-  /*
-  (void)ubi_cacheSetMaxEntries( mangled_cache, lp_mangled_cache_entries() );
-  (void)ubi_cacheSetMaxMemory(  mangled_cache, lp_mangled_cache_memory() );
-  */
-  } /* reset_mangled_cache  */
+{
+       /* We could close and re-open the tdb here... should we ? The old code did
+          the equivalent... JRA. */
+}
 
+/***************************************************************************
+ Add a mangled name into the cache.
+ If the extension of the raw name maps directly to the
+ extension of the mangled name, then we'll store both names
+ *without* extensions.  That way, we can provide consistent
+ reverse mangling for all names that match.  The test here is
+ a bit more careful than the one done in earlier versions of
+ mangle.c:
+
+    - the extension must exist on the raw name,
+    - it must be all lower case
+    - it must match the mangled extension (to prove that no
+      mangling occurred).
+  crh 07-Apr-1998
+**************************************************************************/
+
+static void cache_mangled_name( const char mangled_name[13], char *raw_name )
+{
+       TDB_DATA data_val;
+       char mangled_name_key[13];
+       char *s1;
+       char *s2;
+
+       /* If the cache isn't initialized, give up. */
+       if( !tdb_mangled_cache )
+               return;
+
+       /* Init the string lengths. */
+       safe_strcpy(mangled_name_key, mangled_name, sizeof(mangled_name_key)-1);
+
+       /* See if the extensions are unmangled.  If so, store the entry
+        * without the extension, thus creating a "group" reverse map.
+        */
+       s1 = strrchr( mangled_name_key, '.' );
+       if( s1 && (s2 = strrchr( raw_name, '.' )) ) {
+               size_t i = 1;
+               while( s1[i] && (tolower_ascii( s1[i] ) == s2[i]) )
+                       i++;
+               if( !s1[i] && !s2[i] ) {
+                       /* Truncate at the '.' */
+                       *s1 = '\0';
+                       *s2 = '\0';
+               }
+       }
 
-/* ************************************************************************** **
- * Add a mangled name into the cache.
- *
- *  Notes:  If the mangled cache has not been initialized, then the
- *          function will simply fail.  It could initialize the cache,
- *          but that's not the way it was done before I changed the
- *          cache mechanism, so I'm sticking with the old method.
- *
- *          If the extension of the raw name maps directly to the
- *          extension of the mangled name, then we'll store both names
- *          *without* extensions.  That way, we can provide consistent
- *          reverse mangling for all names that match.  The test here is
- *          a bit more careful than the one done in earlier versions of
- *          mangle.c:
- *
- *            - the extension must exist on the raw name,
- *            - it must be all lower case
- *            - it must match the mangled extension (to prove that no
- *              mangling occurred).
- *
- *  crh 07-Apr-1998
- *
- * ************************************************************************** **
- */
-static void cache_mangled_name( char *mangled_name, char *raw_name )
-  {
-  ubi_cacheEntryPtr new_entry;
-  char             *s1;
-  char             *s2;
-  size_t               mangled_len;
-  size_t               raw_len;
-  size_t               i;
-
-  /* If the cache isn't initialized, give up. */
-  if( !mc_initialized )
-    return;
-
-  /* Init the string lengths. */
-  mangled_len = strlen( mangled_name );
-  raw_len     = strlen( raw_name );
-
-  /* See if the extensions are unmangled.  If so, store the entry
-   * without the extension, thus creating a "group" reverse map.
-   */
-  s1 = strrchr( mangled_name, '.' );
-  if( s1 && (s2 = strrchr( raw_name, '.' )) )
-    {
-    i = 1;
-    while( s1[i] && (tolower( s1[i] ) == s2[i]) )
-      i++;
-    if( !s1[i] && !s2[i] )
-      {
-      mangled_len -= i;
-      raw_len     -= i;
-      }
-    }
-
-  /* Allocate a new cache entry.  If the allocation fails, just return. */
-  i = sizeof( ubi_cacheEntry ) + mangled_len + raw_len + 2;
-  new_entry = malloc( i );
-  if( !new_entry )
-    return;
-
-  /* Fill the new cache entry, and add it to the cache. */
-  s1 = (char *)(new_entry + 1);
-  s2 = (char *)&(s1[mangled_len + 1]);
-  (void)StrnCpy( s1, mangled_name, mangled_len );
-  (void)StrnCpy( s2, raw_name,     raw_len );
-  ubi_cachePut( mangled_cache, i, new_entry, s1 );
-  } /* cache_mangled_name */
+       /* Allocate a new cache entry.  If the allocation fails, just return. */
+       data_val = string_term_tdb_data(raw_name);
+       if (tdb_store_bystring(tdb_mangled_cache, mangled_name_key, data_val, TDB_REPLACE) != 0) {
+               DEBUG(0,("cache_mangled_name: Error storing entry %s -> %s\n", mangled_name_key, raw_name));
+       } else {
+               DEBUG(5,("cache_mangled_name: Stored entry %s -> %s\n", mangled_name_key, raw_name));
+       }
+}
 
 /* ************************************************************************** **
  * Check for a name on the mangled name stack
  *
  *  Input:  s - Input *and* output string buffer.
- *
+ *         maxlen - space in i/o string buffer.
  *  Output: True if the name was found in the cache, else False.
  *
  *  Notes:  If a reverse map is found, the function will overwrite the string
@@ -558,143 +459,125 @@ static void cache_mangled_name( char *mangled_name, char *raw_name )
  * ************************************************************************** **
  */
 
-static BOOL check_cache( char *s )
+static BOOL check_cache( char *s, size_t maxlen, const struct share_params *p )
 {
-  ubi_cacheEntryPtr FoundPtr;
-  char             *ext_start = NULL;
-  char             *found_name;
-  char             *saved_ext = NULL;
-
-  /* If the cache isn't initialized, give up. */
-  if( !mc_initialized )
-    return( False );
-
-  FoundPtr = ubi_cacheGet( mangled_cache, (ubi_trItemPtr)s );
-
-  /* If we didn't find the name *with* the extension, try without. */
-  if( !FoundPtr )
-  {
-    ext_start = strrchr( s, '.' );
-    if( ext_start )
-    {
-      if((saved_ext = strdup(ext_start)) == NULL)
-        return False;
-
-      *ext_start = '\0';
-      FoundPtr = ubi_cacheGet( mangled_cache, (ubi_trItemPtr)s );
-      /* 
-       * At this point s is the name without the
-       * extension. We re-add the extension if saved_ext
-       * is not null, before freeing saved_ext.
-       */
-    }
-  }
-
-  /* Okay, if we haven't found it we're done. */
-  if( !FoundPtr )
-  {
-    if(saved_ext)
-    {
-      /* Replace the saved_ext as it was truncated. */
-      (void)pstrcat( s, saved_ext );
-      SAFE_FREE(saved_ext);
-    }
-    return( False );
-  }
-
-  /* If we *did* find it, we need to copy it into the string buffer. */
-  found_name = (char *)(FoundPtr + 1);
-  found_name += (strlen( found_name ) + 1);
-
-  (void)pstrcpy( s, found_name );
-  if( saved_ext )
-  {
-    /* Replace the saved_ext as it was truncated. */
-    (void)pstrcat( s, saved_ext );
-    SAFE_FREE(saved_ext);
-  }
-
-  return( True );
-} /* check_mangled_cache */
+       TDB_DATA data_val;
+       char *ext_start = NULL;
+       char *saved_ext = NULL;
+
+       magic_char = lp_magicchar(p);
+
+       /* If the cache isn't initialized, give up. */
+       if( !tdb_mangled_cache )
+               return( False );
+
+       data_val = tdb_fetch_bystring(tdb_mangled_cache, s);
+
+       /* If we didn't find the name *with* the extension, try without. */
+       if(data_val.dptr == NULL || data_val.dsize == 0) {
+               ext_start = strrchr( s, '.' );
+               if( ext_start ) {
+                       if((saved_ext = SMB_STRDUP(ext_start)) == NULL)
+                               return False;
+
+                       *ext_start = '\0';
+                       data_val = tdb_fetch_bystring(tdb_mangled_cache, s);
+                       /* 
+                        * At this point s is the name without the
+                        * extension. We re-add the extension if saved_ext
+                        * is not null, before freeing saved_ext.
+                        */
+               }
+       }
 
+       /* Okay, if we haven't found it we're done. */
+       if(data_val.dptr == NULL || data_val.dsize == 0) {
+               if(saved_ext) {
+                       /* Replace the saved_ext as it was truncated. */
+                       (void)safe_strcat( s, saved_ext, maxlen );
+                       SAFE_FREE(saved_ext);
+               }
+               return( False );
+       }
+
+       /* If we *did* find it, we need to copy it into the string buffer. */
+       (void)safe_strcpy( s, (const char *)data_val.dptr, maxlen );
+       if( saved_ext ) {
+               /* Replace the saved_ext as it was truncated. */
+               (void)safe_strcat( s, saved_ext, maxlen );
+               SAFE_FREE(saved_ext);
+       }
+       SAFE_FREE(data_val.dptr);
+       return( True );
+}
 
 /*****************************************************************************
  * do the actual mangling to 8.3 format
  * the buffer must be able to hold 13 characters (including the null)
  *****************************************************************************
  */
-static void to_8_3(char *s)
-  {
-  int csum;
-  char *p;
-  char extension[4];
-  char base[9];
-  int baselen = 0;
-  int extlen = 0;
-
-  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 = '.';
-      }
-    else
-      csum = str_checksum(s);
-    }
-  else
-    csum = str_checksum(s);
-
-  strupper( s );
-
-  if( p )
-  {
-         if( p == s )
-                 safe_strcpy( extension, "___", 3 );
-         else
-         {
-                 *p++ = 0;
-                 while( *p && extlen < 3 )
-                 {
-                         if ( *p != '.') {
-                                 extension[extlen++] = p[0];
-                         }
-                         p++;
-                 }
-                 extension[extlen] = 0;
-      }
-  }
-  
-  p = s;
-
-  while( *p && baselen < 5 )
-  {
-         if (*p != '.') {
-                 base[baselen++] = p[0];
-         }
-         p++;
-  }
-  base[baselen] = 0;
+static void to_8_3(char *s, int default_case)
+{
+       int csum;
+       char *p;
+       char extension[4];
+       char base[9];
+       int baselen = 0;
+       int extlen = 0;
+
+       extension[0] = 0;
+       base[0] = 0;
+
+       p = strrchr(s,'.');  
+       if( p && (strlen(p+1) < (size_t)4) ) {
+               BOOL all_normal = ( strisnormal(p+1, default_case) ); /* XXXXXXXXX */
+
+               if( all_normal && p[1] != 0 ) {
+                       *p = 0;
+                       csum = str_checksum( s );
+                       *p = '.';
+               } else
+                       csum = str_checksum(s);
+       } else
+               csum = str_checksum(s);
+
+       strupper_m( s );
+
+       if( p ) {
+               if( p == s )
+                       safe_strcpy( extension, "___", 3 );
+               else {
+                       *p++ = 0;
+                       while( *p && extlen < 3 ) {
+                               if ( *p != '.') {
+                                       extension[extlen++] = p[0];
+                               }
+                               p++;
+                       }
+                       extension[extlen] = 0;
+               }
+       }
   
-  csum = csum % (MANGLE_BASE*MANGLE_BASE);
+       p = s;
+
+       while( *p && baselen < 5 ) {
+               if (isbasechar(*p)) {
+                       base[baselen++] = p[0];
+               }
+               p++;
+       }
+       base[baselen] = 0;
   
-  (void)slprintf(s, 12, "%s%c%c%c",
-                 base, magic_char, mangle( csum/MANGLE_BASE ), mangle( csum ) );
+       csum = csum % (MANGLE_BASE*MANGLE_BASE);
   
-  if( *extension )
-  {
-         (void)pstrcat( s, "." );
-         (void)pstrcat( s, extension );
-  }
+       (void)slprintf(s, 12, "%s%c%c%c",
+               base, magic_char, mangle( csum/MANGLE_BASE ), mangle( csum ) );
   
-  } /* mangle_name_83 */
+       if( *extension ) {
+               (void)pstrcat( s, "." );
+               (void)pstrcat( s, extension );
+       }
+}
 
 /*****************************************************************************
  * Convert a filename to DOS format.  Return True if successful.
@@ -721,26 +604,33 @@ static void to_8_3(char *s)
  *
  * ****************************************************************************
  */
-static BOOL name_map(char *OutName, BOOL need83, BOOL cache83)
+
+static void name_map(char *OutName, BOOL need83, BOOL cache83,
+                    int default_case, const struct share_params *p)
 {
        smb_ucs2_t *OutName_ucs2;
+       magic_char = lp_magicchar(p);
+
        DEBUG(5,("name_map( %s, need83 = %s, cache83 = %s)\n", OutName,
                 need83 ? "True" : "False", cache83 ? "True" : "False"));
        
-       if (push_ucs2_allocate((void **)&OutName_ucs2, OutName) < 0) {
+       if (push_ucs2_allocate(&OutName_ucs2, OutName) == (size_t)-1) {
                DEBUG(0, ("push_ucs2_allocate failed!\n"));
-               return False;
+               return;
        }
 
+       if( !need83 && !NT_STATUS_IS_OK(is_valid_name(OutName_ucs2, False, False)))
+               need83 = True;
+
        /* check if it's already in 8.3 format */
-       if (need83 && !NT_STATUS_IS_OK(is_8_3_w(OutName_ucs2))) {
+       if (need83 && !NT_STATUS_IS_OK(is_8_3_w(OutName_ucs2, False))) {
                char *tmp = NULL; 
 
                /* mangle it into 8.3 */
                if (cache83)
-                       tmp = strdup(OutName);
+                       tmp = SMB_STRDUP(OutName);
 
-               to_8_3(OutName);
+               to_8_3(OutName, default_case);
 
                if(tmp != NULL) {
                        cache_mangled_name(OutName, tmp);
@@ -750,18 +640,16 @@ static BOOL name_map(char *OutName, BOOL need83, BOOL cache83)
 
        DEBUG(5,("name_map() ==> [%s]\n", OutName));
        SAFE_FREE(OutName_ucs2);
-       return(True);
-} /* name_map */
-
+}
 
 /*
   the following provides the abstraction layer to make it easier
   to drop in an alternative mangling implementation
 */
 static struct mangle_fns mangle_fns = {
+       mangle_reset,
        is_mangled,
        is_8_3,
-       mangle_reset,
        check_cache,
        name_map
 };
@@ -771,5 +659,9 @@ struct mangle_fns *mangle_hash_init(void)
 {
        mangle_reset();
 
+       /* Create the in-memory tdb using our custom hash function. */
+       tdb_mangled_cache = tdb_open_ex("mangled_cache", 1031, TDB_INTERNAL,
+                               (O_RDWR|O_CREAT), 0644, NULL, fast_string_hash);
+
        return &mangle_fns;
 }