2003-08-13 01:53:07 +00:00
/*
Unix SMB / CIFS implementation .
SMB torture tester - mangling test
Copyright ( C ) Andrew Tridgell 2002
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 0213 9 , USA .
*/
# include "includes.h"
static TDB_CONTEXT * tdb ;
# define NAME_LENGTH 20
static unsigned total , collisions , failures ;
static BOOL test_one ( struct cli_state * cli , const char * name )
{
int fnum ;
2003-11-28 09:05:44 +00:00
const char * shortname ;
2003-08-13 01:53:07 +00:00
fstring name2 ;
NTSTATUS status ;
TDB_DATA data ;
total + + ;
2004-02-08 00:51:07 +00:00
fnum = cli_open ( cli - > tree , name , O_RDWR | O_CREAT | O_EXCL , DENY_NONE ) ;
2003-08-13 01:53:07 +00:00
if ( fnum = = - 1 ) {
2004-02-08 00:51:07 +00:00
printf ( " open of %s failed (%s) \n " , name , cli_errstr ( cli - > tree ) ) ;
2003-08-13 01:53:07 +00:00
return False ;
}
2004-02-10 11:33:35 +00:00
if ( NT_STATUS_IS_ERR ( cli_close ( cli - > tree , fnum ) ) ) {
2004-02-08 00:51:07 +00:00
printf ( " close of %s failed (%s) \n " , name , cli_errstr ( cli - > tree ) ) ;
2003-08-13 01:53:07 +00:00
return False ;
}
/* get the short name */
2004-02-08 00:51:07 +00:00
status = cli_qpathinfo_alt_name ( cli - > tree , name , & shortname ) ;
2003-08-13 01:53:07 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2004-02-08 00:51:07 +00:00
printf ( " query altname of %s failed (%s) \n " , name , cli_errstr ( cli - > tree ) ) ;
2003-08-13 01:53:07 +00:00
return False ;
}
snprintf ( name2 , sizeof ( name2 ) , " \\ mangle_test \\ %s " , shortname ) ;
2004-02-10 11:33:35 +00:00
if ( NT_STATUS_IS_ERR ( cli_unlink ( cli - > tree , name2 ) ) ) {
2003-08-13 01:53:07 +00:00
printf ( " unlink of %s (%s) failed (%s) \n " ,
2004-02-08 00:51:07 +00:00
name2 , name , cli_errstr ( cli - > tree ) ) ;
2003-08-13 01:53:07 +00:00
return False ;
}
/* recreate by short name */
2004-02-08 00:51:07 +00:00
fnum = cli_open ( cli - > tree , name2 , O_RDWR | O_CREAT | O_EXCL , DENY_NONE ) ;
2003-08-13 01:53:07 +00:00
if ( fnum = = - 1 ) {
2004-02-08 00:51:07 +00:00
printf ( " open2 of %s failed (%s) \n " , name2 , cli_errstr ( cli - > tree ) ) ;
2003-08-13 01:53:07 +00:00
return False ;
}
2004-02-10 11:33:35 +00:00
if ( NT_STATUS_IS_ERR ( cli_close ( cli - > tree , fnum ) ) ) {
2004-02-08 00:51:07 +00:00
printf ( " close of %s failed (%s) \n " , name , cli_errstr ( cli - > tree ) ) ;
2003-08-13 01:53:07 +00:00
return False ;
}
/* and unlink by long name */
2004-02-10 11:33:35 +00:00
if ( NT_STATUS_IS_ERR ( cli_unlink ( cli - > tree , name ) ) ) {
2003-08-13 01:53:07 +00:00
printf ( " unlink2 of %s (%s) failed (%s) \n " ,
2004-02-08 00:51:07 +00:00
name , name2 , cli_errstr ( cli - > tree ) ) ;
2003-08-13 01:53:07 +00:00
failures + + ;
2004-02-08 00:51:07 +00:00
cli_unlink ( cli - > tree , name2 ) ;
2003-08-13 01:53:07 +00:00
return True ;
}
/* see if the short name is already in the tdb */
2003-12-10 03:02:12 +00:00
data = tdb_fetch_bystring ( tdb , shortname ) ;
2003-08-13 01:53:07 +00:00
if ( data . dptr ) {
/* maybe its a duplicate long name? */
if ( strcasecmp ( name , data . dptr ) ! = 0 ) {
/* we have a collision */
collisions + + ;
printf ( " Collision between %s and %s -> %s "
" (coll/tot: %u/%u) \n " ,
name , data . dptr , shortname , collisions , total ) ;
}
free ( data . dptr ) ;
} else {
TDB_DATA namedata ;
/* store it for later */
namedata . dptr = name ;
namedata . dsize = strlen ( name ) + 1 ;
2003-12-10 03:02:12 +00:00
tdb_store_bystring ( tdb , shortname , namedata , TDB_REPLACE ) ;
2003-08-13 01:53:07 +00:00
}
return True ;
}
static void gen_name ( char * name )
{
const char * chars = " 0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz._-$~... " ;
unsigned max_idx = strlen ( chars ) ;
unsigned len ;
int i ;
char * p ;
fstrcpy ( name , " \\ mangle_test \\ " ) ;
p = name + strlen ( name ) ;
len = 1 + random ( ) % NAME_LENGTH ;
for ( i = 0 ; i < len ; i + + ) {
p [ i ] = chars [ random ( ) % max_idx ] ;
}
p [ i ] = 0 ;
if ( strcmp ( p , " . " ) = = 0 | | strcmp ( p , " .. " ) = = 0 ) {
p [ 0 ] = ' _ ' ;
}
/* have a high probability of a common lead char */
if ( random ( ) % 2 = = 0 ) {
p [ 0 ] = ' A ' ;
}
/* and a medium probability of a common lead string */
if ( random ( ) % 10 = = 0 ) {
strncpy ( p , " ABCDE " , 5 ) ;
}
/* and a high probability of a good extension length */
if ( random ( ) % 2 = = 0 ) {
char * s = strrchr ( p , ' . ' ) ;
if ( s ) {
s [ 4 ] = 0 ;
}
}
}
BOOL torture_mangle ( int dummy )
{
extern int torture_numops ;
static struct cli_state * cli ;
int i ;
printf ( " starting mangle test \n " ) ;
if ( ! torture_open_connection ( & cli ) ) {
return False ;
}
/* we will use an internal tdb to store the names we have used */
tdb = tdb_open ( NULL , 100000 , TDB_INTERNAL , 0 , 0 ) ;
if ( ! tdb ) {
printf ( " ERROR: Failed to open tdb \n " ) ;
return False ;
}
2004-02-08 00:51:07 +00:00
cli_unlink ( cli - > tree , " \\ mangle_test \\ * " ) ;
cli_rmdir ( cli - > tree , " \\ mangle_test " ) ;
2003-08-13 01:53:07 +00:00
2004-02-10 11:33:35 +00:00
if ( NT_STATUS_IS_ERR ( cli_mkdir ( cli - > tree , " \\ mangle_test " ) ) ) {
2003-08-13 01:53:07 +00:00
printf ( " ERROR: Failed to make directory \n " ) ;
return False ;
}
for ( i = 0 ; i < torture_numops ; i + + ) {
fstring name ;
gen_name ( name ) ;
if ( ! test_one ( cli , name ) ) {
break ;
}
if ( total & & total % 100 = = 0 ) {
printf ( " collisions %u/%u - %.2f%% (%u failures) \r " ,
collisions , total , ( 100.0 * collisions ) / total , failures ) ;
}
}
2004-02-08 00:51:07 +00:00
cli_unlink ( cli - > tree , " \\ mangle_test \\ * " ) ;
2004-02-10 11:33:35 +00:00
if ( NT_STATUS_IS_ERR ( cli_rmdir ( cli - > tree , " \\ mangle_test " ) ) ) {
2003-08-13 01:53:07 +00:00
printf ( " ERROR: Failed to remove directory \n " ) ;
return False ;
}
printf ( " \n Total collisions %u/%u - %.2f%% (%u failures) \n " ,
collisions , total , ( 100.0 * collisions ) / total , failures ) ;
torture_close_connection ( cli ) ;
printf ( " mangle test finished \n " ) ;
return ( failures = = 0 ) ;
}