2003-08-13 05:53:07 +04:00
/*
Unix SMB / CIFS implementation .
SMB torture tester
Copyright ( C ) Andrew Tridgell 1997 - 2003
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"
2004-11-03 03:17:12 +03:00
# include "dynconfig.h"
2004-11-02 15:15:17 +03:00
# include "client.h"
2004-11-02 05:57:18 +03:00
# include "lib/cmdline/popt_common.h"
2004-11-01 04:03:22 +03:00
# include "libcli/raw/libcliraw.h"
2004-11-02 03:24:21 +03:00
# include "system/time.h"
2004-11-02 07:51:57 +03:00
# include "system/wait.h"
2004-11-02 09:42:15 +03:00
# include "ioctl.h"
2003-08-13 05:53:07 +04:00
2004-03-09 05:13:13 +03:00
int torture_nprocs = 4 ;
2003-08-13 05:53:07 +04:00
int torture_numops = 100 ;
int torture_entries = 1000 ;
int torture_failures = 1 ;
2004-10-30 08:59:52 +04:00
int torture_seed = 0 ;
2003-08-13 05:53:07 +04:00
static int procnum ; /* records process count number when forking */
2004-08-04 17:23:35 +04:00
static struct smbcli_state * current_cli ;
2003-08-13 05:53:07 +04:00
static BOOL use_oplocks ;
static BOOL use_level_II_oplocks ;
static BOOL use_kerberos ;
BOOL torture_showall = False ;
# define CHECK_MAX_FAILURES(label) do { if (++failures >= torture_failures) goto label; } while (0)
2004-08-04 17:23:35 +04:00
static struct smbcli_state * open_nbt_connection ( void )
2003-08-13 05:53:07 +04:00
{
struct nmb_name called , calling ;
2004-08-04 17:23:35 +04:00
struct smbcli_state * cli ;
2004-06-14 03:50:55 +04:00
const char * host = lp_parm_string ( - 1 , " torture " , " host " ) ;
2003-08-13 05:53:07 +04:00
make_nmb_name ( & calling , lp_netbios_name ( ) , 0x0 ) ;
2004-08-19 00:13:08 +04:00
choose_called_name ( & called , host , 0x20 ) ;
2003-08-13 05:53:07 +04:00
2004-09-28 09:44:59 +04:00
cli = smbcli_state_init ( NULL ) ;
2003-08-13 05:53:07 +04:00
if ( ! cli ) {
2004-08-04 17:23:35 +04:00
printf ( " Failed initialize smbcli_struct to connect with %s \n " , host ) ;
2003-08-13 05:53:07 +04:00
return NULL ;
}
2004-08-19 00:13:08 +04:00
if ( ! smbcli_socket_connect ( cli , host ) ) {
2003-08-13 05:53:07 +04:00
printf ( " Failed to connect with %s \n " , host ) ;
return cli ;
}
cli - > transport - > socket - > timeout = 120000 ; /* set a really long timeout (2 minutes) */
2004-08-04 17:23:35 +04:00
if ( ! smbcli_transport_establish ( cli , & calling , & called ) ) {
2003-08-13 05:53:07 +04:00
/*
* Well , that failed , try * SMBSERVER . . .
* However , we must reconnect as well . . .
*/
2004-08-19 00:13:08 +04:00
if ( ! smbcli_socket_connect ( cli , host ) ) {
2003-08-13 05:53:07 +04:00
printf ( " Failed to connect with %s \n " , host ) ;
return False ;
}
make_nmb_name ( & called , " *SMBSERVER " , 0x20 ) ;
2004-08-04 17:23:35 +04:00
if ( ! smbcli_transport_establish ( cli , & calling , & called ) ) {
2003-08-13 05:53:07 +04:00
printf ( " %s rejected the session \n " , host ) ;
printf ( " We tried with a called name of %s & %s \n " ,
host , " *SMBSERVER " ) ;
2004-08-04 17:23:35 +04:00
smbcli_shutdown ( cli ) ;
2003-08-13 05:53:07 +04:00
return NULL ;
}
}
return cli ;
}
2004-08-04 17:23:35 +04:00
BOOL torture_open_connection_share ( struct smbcli_state * * c ,
2004-03-09 12:04:06 +03:00
const char * hostname ,
const char * sharename )
2003-08-13 05:53:07 +04:00
{
BOOL retry ;
int flags = 0 ;
NTSTATUS status ;
2004-06-14 03:50:55 +04:00
const char * username = lp_parm_string ( - 1 , " torture " , " username " ) ;
2004-08-25 18:31:59 +04:00
const char * userdomain = lp_parm_string ( - 1 , " torture " , " userdomain " ) ;
2004-06-14 03:50:55 +04:00
const char * password = lp_parm_string ( - 1 , " torture " , " password " ) ;
2003-08-13 05:53:07 +04:00
if ( use_kerberos )
2004-08-04 17:23:35 +04:00
flags | = SMBCLI_FULL_CONNECTION_USE_KERBEROS ;
2003-11-03 09:22:45 +03:00
2004-09-28 09:44:59 +04:00
status = smbcli_full_connection ( NULL ,
c , lp_netbios_name ( ) ,
2004-09-28 09:42:02 +04:00
hostname , NULL ,
sharename , " ????? " ,
username , username [ 0 ] ? userdomain : " " ,
password , flags , & retry ) ;
2003-08-13 05:53:07 +04:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2003-08-13 20:04:21 +04:00
printf ( " Failed to open connection - %s \n " , nt_errstr ( status ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
( * c ) - > transport - > options . use_oplocks = use_oplocks ;
( * c ) - > transport - > options . use_level2_oplocks = use_level_II_oplocks ;
( * c ) - > transport - > socket - > timeout = 120000 ;
return True ;
}
2004-08-04 17:23:35 +04:00
BOOL torture_open_connection ( struct smbcli_state * * c )
2004-03-09 12:04:06 +03:00
{
2004-06-14 03:50:55 +04:00
const char * host = lp_parm_string ( - 1 , " torture " , " host " ) ;
const char * share = lp_parm_string ( - 1 , " torture " , " share " ) ;
2004-03-09 12:04:06 +03:00
return torture_open_connection_share ( c , host , share ) ;
}
2004-08-04 17:23:35 +04:00
BOOL torture_close_connection ( struct smbcli_state * c )
2003-08-13 05:53:07 +04:00
{
BOOL ret = True ;
if ( ! c ) return True ;
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_tdis ( c ) ) ) {
printf ( " tdis failed (%s) \n " , smbcli_errstr ( c - > tree ) ) ;
2003-08-13 05:53:07 +04:00
ret = False ;
}
2004-08-04 17:23:35 +04:00
smbcli_shutdown ( c ) ;
2003-08-13 05:53:07 +04:00
return ret ;
}
2003-11-24 16:19:00 +03:00
2004-09-02 15:26:58 +04:00
/* open a rpc connection to the chosen binding string */
2003-11-24 16:19:00 +03:00
NTSTATUS torture_rpc_connection ( struct dcerpc_pipe * * p ,
const char * pipe_name ,
const char * pipe_uuid ,
2004-05-25 20:24:13 +04:00
uint32_t pipe_version )
2003-11-24 15:40:47 +03:00
{
NTSTATUS status ;
2004-06-14 03:50:55 +04:00
const char * binding = lp_parm_string ( - 1 , " torture " , " binding " ) ;
2003-11-26 04:16:41 +03:00
2003-12-15 06:29:55 +03:00
if ( ! binding ) {
printf ( " You must specify a ncacn binding string \n " ) ;
return NT_STATUS_INVALID_PARAMETER ;
2003-11-24 16:19:00 +03:00
}
2003-12-15 06:29:55 +03:00
status = dcerpc_pipe_connect ( p , binding , pipe_uuid , pipe_version ,
2004-08-25 18:31:59 +04:00
lp_parm_string ( - 1 , " torture " , " userdomain " ) ,
2003-12-15 06:29:55 +03:00
lp_parm_string ( - 1 , " torture " , " username " ) ,
lp_parm_string ( - 1 , " torture " , " password " ) ) ;
2003-11-24 15:40:47 +03:00
return status ;
}
2004-09-03 12:28:24 +04:00
/* open a rpc connection to a specific transport */
NTSTATUS torture_rpc_connection_transport ( struct dcerpc_pipe * * p ,
const char * pipe_name ,
const char * pipe_uuid ,
uint32_t pipe_version ,
enum dcerpc_transport_t transport )
2004-09-02 15:26:58 +04:00
{
NTSTATUS status ;
const char * binding = lp_parm_string ( - 1 , " torture " , " binding " ) ;
struct dcerpc_binding b ;
TALLOC_CTX * mem_ctx = talloc_init ( " torture_rpc_connection_smb " ) ;
if ( ! binding ) {
printf ( " You must specify a ncacn binding string \n " ) ;
return NT_STATUS_INVALID_PARAMETER ;
}
status = dcerpc_parse_binding ( mem_ctx , binding , & b ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " Failed to parse dcerpc binding '%s' \n " , binding ) ) ;
talloc_destroy ( mem_ctx ) ;
return status ;
}
2004-09-03 12:28:24 +04:00
b . transport = transport ;
2004-09-02 15:26:58 +04:00
status = dcerpc_pipe_connect_b ( p , & b , pipe_uuid , pipe_version ,
lp_parm_string ( - 1 , " torture " , " userdomain " ) ,
lp_parm_string ( - 1 , " torture " , " username " ) ,
lp_parm_string ( - 1 , " torture " , " password " ) ) ;
return status ;
}
2003-11-03 09:22:45 +03:00
/* close a rpc connection to a named pipe */
NTSTATUS torture_rpc_close ( struct dcerpc_pipe * p )
{
dcerpc_pipe_close ( p ) ;
2003-11-24 14:45:33 +03:00
return NT_STATUS_OK ;
2003-11-03 09:22:45 +03:00
}
2003-08-13 05:53:07 +04:00
/* check if the server produced the expected error code */
2004-10-17 06:52:37 +04:00
BOOL check_error ( const char * location , struct smbcli_state * c ,
uint8_t eclass , uint32_t ecode , NTSTATUS nterr )
2003-08-13 05:53:07 +04:00
{
2004-08-04 17:23:35 +04:00
if ( smbcli_is_dos_error ( c - > tree ) ) {
2004-05-25 21:50:17 +04:00
uint8_t class ;
2004-05-25 20:24:13 +04:00
uint32_t num ;
2003-08-13 05:53:07 +04:00
/* Check DOS error */
2004-08-04 17:23:35 +04:00
smbcli_dos_error ( c , & class , & num ) ;
2003-08-13 05:53:07 +04:00
if ( eclass ! = class | | ecode ! = num ) {
printf ( " unexpected error code class=%d code=%d \n " ,
( int ) class , ( int ) num ) ;
2004-10-17 06:52:37 +04:00
printf ( " expected %d/%d %s (at %s) \n " ,
( int ) eclass , ( int ) ecode , nt_errstr ( nterr ) , location ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
} else {
NTSTATUS status ;
/* Check NT error */
2004-08-04 17:23:35 +04:00
status = smbcli_nt_error ( c - > tree ) ;
2003-08-13 05:53:07 +04:00
if ( NT_STATUS_V ( nterr ) ! = NT_STATUS_V ( status ) ) {
printf ( " unexpected error code %s \n " , nt_errstr ( status ) ) ;
2004-10-17 06:52:37 +04:00
printf ( " expected %s (at %s) \n " , nt_errstr ( nterr ) , location ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
}
return True ;
}
2004-08-04 17:23:35 +04:00
static BOOL wait_lock ( struct smbcli_state * c , int fnum , uint32_t offset , uint32_t len )
2003-08-13 05:53:07 +04:00
{
2004-08-04 17:23:35 +04:00
while ( NT_STATUS_IS_ERR ( smbcli_lock ( c - > tree , fnum , offset , len , - 1 , WRITE_LOCK ) ) ) {
2004-10-17 06:52:37 +04:00
if ( ! check_error ( __location__ , c , ERRDOS , ERRlock , NT_STATUS_LOCK_NOT_GRANTED ) ) return False ;
2003-08-13 05:53:07 +04:00
}
return True ;
}
2004-08-04 17:23:35 +04:00
static BOOL rw_torture ( struct smbcli_state * c )
2003-08-13 05:53:07 +04:00
{
const char * lockfname = " \\ torture.lck " ;
char * fname ;
int fnum ;
int fnum2 ;
pid_t pid2 , pid = getpid ( ) ;
int i , j ;
char buf [ 1024 ] ;
BOOL correct = True ;
2004-08-04 17:23:35 +04:00
fnum2 = smbcli_open ( c - > tree , lockfname , O_RDWR | O_CREAT | O_EXCL ,
2003-08-13 05:53:07 +04:00
DENY_NONE ) ;
if ( fnum2 = = - 1 )
2004-08-04 17:23:35 +04:00
fnum2 = smbcli_open ( c - > tree , lockfname , O_RDWR , DENY_NONE ) ;
2003-08-13 05:53:07 +04:00
if ( fnum2 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " open of %s failed (%s) \n " , lockfname , smbcli_errstr ( c - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
for ( i = 0 ; i < torture_numops ; i + + ) {
2004-06-01 14:12:52 +04:00
uint_t n = ( uint_t ) sys_random ( ) % 10 ;
2003-08-13 05:53:07 +04:00
if ( i % 10 = = 0 ) {
printf ( " %d \r " , i ) ; fflush ( stdout ) ;
}
asprintf ( & fname , " \\ torture.%u " , n ) ;
if ( ! wait_lock ( c , fnum2 , n * sizeof ( int ) , sizeof ( int ) ) ) {
return False ;
}
2004-08-04 17:23:35 +04:00
fnum = smbcli_open ( c - > tree , fname , O_RDWR | O_CREAT | O_TRUNC , DENY_ALL ) ;
2003-08-13 05:53:07 +04:00
if ( fnum = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " open failed (%s) \n " , smbcli_errstr ( c - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
break ;
}
2004-08-04 17:23:35 +04:00
if ( smbcli_write ( c - > tree , fnum , 0 , ( char * ) & pid , 0 , sizeof ( pid ) ) ! = sizeof ( pid ) ) {
printf ( " write failed (%s) \n " , smbcli_errstr ( c - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
}
for ( j = 0 ; j < 50 ; j + + ) {
2004-08-04 17:23:35 +04:00
if ( smbcli_write ( c - > tree , fnum , 0 , ( char * ) buf ,
2003-08-13 05:53:07 +04:00
sizeof ( pid ) + ( j * sizeof ( buf ) ) ,
sizeof ( buf ) ) ! = sizeof ( buf ) ) {
2004-08-04 17:23:35 +04:00
printf ( " write failed (%s) \n " , smbcli_errstr ( c - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
}
}
pid2 = 0 ;
2004-08-04 17:23:35 +04:00
if ( smbcli_read ( c - > tree , fnum , ( char * ) & pid2 , 0 , sizeof ( pid ) ) ! = sizeof ( pid ) ) {
printf ( " read failed (%s) \n " , smbcli_errstr ( c - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
}
if ( pid2 ! = pid ) {
printf ( " data corruption! \n " ) ;
correct = False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( c - > tree , fnum ) ) ) {
printf ( " close failed (%s) \n " , smbcli_errstr ( c - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_unlink ( c - > tree , fname ) ) ) {
printf ( " unlink failed (%s) \n " , smbcli_errstr ( c - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_unlock ( c - > tree , fnum2 , n * sizeof ( int ) , sizeof ( int ) ) ) ) {
printf ( " unlock failed (%s) \n " , smbcli_errstr ( c - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
}
free ( fname ) ;
}
2004-08-04 17:23:35 +04:00
smbcli_close ( c - > tree , fnum2 ) ;
smbcli_unlink ( c - > tree , lockfname ) ;
2003-08-13 05:53:07 +04:00
printf ( " %d \n " , i ) ;
return correct ;
}
2004-08-04 17:23:35 +04:00
static BOOL run_torture ( struct smbcli_state * cli , int dummy )
2003-08-13 05:53:07 +04:00
{
BOOL ret ;
ret = rw_torture ( cli ) ;
if ( ! torture_close_connection ( cli ) ) {
ret = False ;
}
return ret ;
}
2004-08-04 17:23:35 +04:00
static BOOL rw_torture3 ( struct smbcli_state * c , const char * lockfname )
2003-08-13 05:53:07 +04:00
{
int fnum = - 1 ;
2004-06-01 12:12:45 +04:00
uint_t i = 0 ;
2003-08-13 05:53:07 +04:00
char buf [ 131072 ] ;
char buf_rd [ 131072 ] ;
2004-06-01 14:12:52 +04:00
uint_t count ;
uint_t countprev = 0 ;
2003-08-13 05:53:07 +04:00
ssize_t sent = 0 ;
BOOL correct = True ;
2004-05-25 20:24:13 +04:00
for ( i = 0 ; i < sizeof ( buf ) ; i + = sizeof ( uint32_t ) )
2003-08-13 05:53:07 +04:00
{
SIVAL ( buf , i , sys_random ( ) ) ;
}
if ( procnum = = 0 )
{
2004-08-04 17:23:35 +04:00
fnum = smbcli_open ( c - > tree , lockfname , O_RDWR | O_CREAT | O_EXCL ,
2004-02-08 03:51:07 +03:00
DENY_NONE ) ;
2003-08-13 05:53:07 +04:00
if ( fnum = = - 1 ) {
printf ( " first open read/write of %s failed (%s) \n " ,
2004-08-04 17:23:35 +04:00
lockfname , smbcli_errstr ( c - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
}
else
{
for ( i = 0 ; i < 500 & & fnum = = - 1 ; i + + )
{
2004-08-04 17:23:35 +04:00
fnum = smbcli_open ( c - > tree , lockfname , O_RDONLY ,
2004-02-08 03:51:07 +03:00
DENY_NONE ) ;
2003-08-13 05:53:07 +04:00
msleep ( 10 ) ;
}
if ( fnum = = - 1 ) {
printf ( " second open read-only of %s failed (%s) \n " ,
2004-08-04 17:23:35 +04:00
lockfname , smbcli_errstr ( c - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
}
i = 0 ;
for ( count = 0 ; count < sizeof ( buf ) ; count + = sent )
{
if ( count > = countprev ) {
printf ( " %d %8d \r " , i , count ) ;
fflush ( stdout ) ;
i + + ;
countprev + = ( sizeof ( buf ) / 20 ) ;
}
if ( procnum = = 0 )
{
2004-06-01 14:12:52 +04:00
sent = ( ( uint_t ) sys_random ( ) % ( 20 ) ) + 1 ;
2003-08-13 05:53:07 +04:00
if ( sent > sizeof ( buf ) - count )
{
sent = sizeof ( buf ) - count ;
}
2004-08-04 17:23:35 +04:00
if ( smbcli_write ( c - > tree , fnum , 0 , buf + count , count , ( size_t ) sent ) ! = sent ) {
printf ( " write failed (%s) \n " , smbcli_errstr ( c - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
}
}
else
{
2004-08-04 17:23:35 +04:00
sent = smbcli_read ( c - > tree , fnum , buf_rd + count , count ,
2004-02-08 03:51:07 +03:00
sizeof ( buf ) - count ) ;
2003-08-13 05:53:07 +04:00
if ( sent < 0 )
{
printf ( " read failed offset:%d size:%d (%s) \n " ,
count , sizeof ( buf ) - count ,
2004-08-04 17:23:35 +04:00
smbcli_errstr ( c - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
sent = 0 ;
}
if ( sent > 0 )
{
if ( memcmp ( buf_rd + count , buf + count , sent ) ! = 0 )
{
printf ( " read/write compare failed \n " ) ;
printf ( " offset: %d req %d recvd %d \n " ,
count , sizeof ( buf ) - count , sent ) ;
correct = False ;
break ;
}
}
}
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( c - > tree , fnum ) ) ) {
printf ( " close failed (%s) \n " , smbcli_errstr ( c - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
}
return correct ;
}
2004-08-04 17:23:35 +04:00
static BOOL rw_torture2 ( struct smbcli_state * c1 , struct smbcli_state * c2 )
2003-08-13 05:53:07 +04:00
{
const char * lockfname = " \\ torture2.lck " ;
int fnum1 ;
int fnum2 ;
int i ;
2004-06-01 12:30:34 +04:00
uint8_t buf [ 131072 ] ;
uint8_t buf_rd [ 131072 ] ;
2003-08-13 05:53:07 +04:00
BOOL correct = True ;
ssize_t bytes_read , bytes_written ;
2004-08-04 17:23:35 +04:00
if ( smbcli_deltree ( c1 - > tree , lockfname ) = = - 1 ) {
printf ( " unlink failed (%s) \n " , smbcli_errstr ( c1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
}
2004-08-04 17:23:35 +04:00
fnum1 = smbcli_open ( c1 - > tree , lockfname , O_RDWR | O_CREAT | O_EXCL ,
2003-08-13 05:53:07 +04:00
DENY_NONE ) ;
if ( fnum1 = = - 1 ) {
printf ( " first open read/write of %s failed (%s) \n " ,
2004-08-04 17:23:35 +04:00
lockfname , smbcli_errstr ( c1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
2004-08-04 17:23:35 +04:00
fnum2 = smbcli_open ( c2 - > tree , lockfname , O_RDONLY ,
2003-08-13 05:53:07 +04:00
DENY_NONE ) ;
if ( fnum2 = = - 1 ) {
printf ( " second open read-only of %s failed (%s) \n " ,
2004-08-04 17:23:35 +04:00
lockfname , smbcli_errstr ( c2 - > tree ) ) ;
smbcli_close ( c1 - > tree , fnum1 ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
printf ( " Checking data integrity over %d ops \n " , torture_numops ) ;
for ( i = 0 ; i < torture_numops ; i + + )
{
2004-06-01 14:12:52 +04:00
size_t buf_size = ( ( uint_t ) sys_random ( ) % ( sizeof ( buf ) - 1 ) ) + 1 ;
2003-08-13 05:53:07 +04:00
if ( i % 10 = = 0 ) {
printf ( " %d \r " , i ) ; fflush ( stdout ) ;
}
2004-07-14 16:14:07 +04:00
generate_random_buffer ( buf , buf_size ) ;
2003-08-13 05:53:07 +04:00
2004-08-04 17:23:35 +04:00
if ( ( bytes_written = smbcli_write ( c1 - > tree , fnum1 , 0 , buf , 0 , buf_size ) ) ! = buf_size ) {
printf ( " write failed (%s) \n " , smbcli_errstr ( c1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
printf ( " wrote %d, expected %d \n " , bytes_written , buf_size ) ;
correct = False ;
break ;
}
2004-08-04 17:23:35 +04:00
if ( ( bytes_read = smbcli_read ( c2 - > tree , fnum2 , buf_rd , 0 , buf_size ) ) ! = buf_size ) {
printf ( " read failed (%s) \n " , smbcli_errstr ( c2 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
printf ( " read %d, expected %d \n " , bytes_read , buf_size ) ;
correct = False ;
break ;
}
if ( memcmp ( buf_rd , buf , buf_size ) ! = 0 )
{
printf ( " read/write compare failed \n " ) ;
correct = False ;
break ;
}
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( c2 - > tree , fnum2 ) ) ) {
printf ( " close failed (%s) \n " , smbcli_errstr ( c2 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( c1 - > tree , fnum1 ) ) ) {
printf ( " close failed (%s) \n " , smbcli_errstr ( c1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_unlink ( c1 - > tree , lockfname ) ) ) {
printf ( " unlink failed (%s) \n " , smbcli_errstr ( c1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
}
return correct ;
}
2004-10-28 17:40:50 +04:00
static BOOL run_readwritetest ( void )
2003-08-13 05:53:07 +04:00
{
2004-08-04 17:23:35 +04:00
struct smbcli_state * cli1 , * cli2 ;
2003-08-13 05:53:07 +04:00
BOOL test1 , test2 = True ;
if ( ! torture_open_connection ( & cli1 ) | | ! torture_open_connection ( & cli2 ) ) {
return False ;
}
printf ( " starting readwritetest \n " ) ;
test1 = rw_torture2 ( cli1 , cli2 ) ;
printf ( " Passed readwritetest v1: %s \n " , BOOLSTR ( test1 ) ) ;
if ( test1 ) {
test2 = rw_torture2 ( cli1 , cli1 ) ;
printf ( " Passed readwritetest v2: %s \n " , BOOLSTR ( test2 ) ) ;
}
if ( ! torture_close_connection ( cli1 ) ) {
test1 = False ;
}
if ( ! torture_close_connection ( cli2 ) ) {
test2 = False ;
}
return ( test1 & & test2 ) ;
}
2004-08-04 17:23:35 +04:00
static BOOL run_readwritemulti ( struct smbcli_state * cli , int dummy )
2003-08-13 05:53:07 +04:00
{
BOOL test ;
2003-11-22 01:00:38 +03:00
test = rw_torture3 ( cli , " \\ multitest.txt " ) ;
2003-08-13 05:53:07 +04:00
if ( ! torture_close_connection ( cli ) ) {
test = False ;
}
return test ;
}
/*
this checks to see if a secondary tconx can use open files from an
earlier tconx
*/
2004-10-28 17:40:50 +04:00
static BOOL run_tcon_test ( void )
2003-08-13 05:53:07 +04:00
{
2004-08-04 17:23:35 +04:00
struct smbcli_state * cli ;
2003-08-13 05:53:07 +04:00
const char * fname = " \\ tcontest.tmp " ;
int fnum1 ;
2004-05-25 21:24:24 +04:00
uint16_t cnum1 , cnum2 , cnum3 ;
uint16_t vuid1 , vuid2 ;
2003-08-13 05:53:07 +04:00
char buf [ 4 ] ;
BOOL ret = True ;
2004-08-04 17:23:35 +04:00
struct smbcli_tree * tree1 ;
2004-06-14 03:50:55 +04:00
const char * host = lp_parm_string ( - 1 , " torture " , " host " ) ;
const char * share = lp_parm_string ( - 1 , " torture " , " share " ) ;
const char * password = lp_parm_string ( - 1 , " torture " , " password " ) ;
2003-08-13 05:53:07 +04:00
if ( ! torture_open_connection ( & cli ) ) {
return False ;
}
printf ( " starting tcontest \n " ) ;
2004-08-04 17:23:35 +04:00
if ( smbcli_deltree ( cli - > tree , fname ) = = - 1 ) {
printf ( " unlink of %s failed (%s) \n " , fname , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
}
2004-08-04 17:23:35 +04:00
fnum1 = smbcli_open ( cli - > tree , fname , O_RDWR | O_CREAT | O_EXCL , DENY_NONE ) ;
2003-08-13 05:53:07 +04:00
if ( fnum1 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " open of %s failed (%s) \n " , fname , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
cnum1 = cli - > tree - > tid ;
vuid1 = cli - > session - > vuid ;
memset ( & buf , 0 , 4 ) ; /* init buf so valgrind won't complain */
2004-08-04 17:23:35 +04:00
if ( smbcli_write ( cli - > tree , fnum1 , 0 , buf , 130 , 4 ) ! = 4 ) {
printf ( " initial write failed (%s) \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
tree1 = cli - > tree ; /* save old tree connection */
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_send_tconX ( cli , share , " ????? " , password ) ) ) {
2003-08-13 05:53:07 +04:00
printf ( " %s refused 2nd tree connect (%s) \n " , host ,
2004-08-04 17:23:35 +04:00
smbcli_errstr ( cli - > tree ) ) ;
smbcli_shutdown ( cli ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
cnum2 = cli - > tree - > tid ;
cnum3 = MAX ( cnum1 , cnum2 ) + 1 ; /* any invalid number */
vuid2 = cli - > session - > vuid + 1 ;
/* try a write with the wrong tid */
cli - > tree - > tid = cnum2 ;
2004-08-04 17:23:35 +04:00
if ( smbcli_write ( cli - > tree , fnum1 , 0 , buf , 130 , 4 ) = = 4 ) {
2003-08-13 05:53:07 +04:00
printf ( " * server allows write with wrong TID \n " ) ;
ret = False ;
} else {
2004-08-04 17:23:35 +04:00
printf ( " server fails write with wrong TID : %s \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
}
/* try a write with an invalid tid */
cli - > tree - > tid = cnum3 ;
2004-08-04 17:23:35 +04:00
if ( smbcli_write ( cli - > tree , fnum1 , 0 , buf , 130 , 4 ) = = 4 ) {
2003-08-13 05:53:07 +04:00
printf ( " * server allows write with invalid TID \n " ) ;
ret = False ;
} else {
2004-08-04 17:23:35 +04:00
printf ( " server fails write with invalid TID : %s \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
}
/* try a write with an invalid vuid */
cli - > session - > vuid = vuid2 ;
cli - > tree - > tid = cnum1 ;
2004-08-04 17:23:35 +04:00
if ( smbcli_write ( cli - > tree , fnum1 , 0 , buf , 130 , 4 ) = = 4 ) {
2003-08-13 05:53:07 +04:00
printf ( " * server allows write with invalid VUID \n " ) ;
ret = False ;
} else {
2004-08-04 17:23:35 +04:00
printf ( " server fails write with invalid VUID : %s \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
}
cli - > session - > vuid = vuid1 ;
cli - > tree - > tid = cnum1 ;
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli - > tree , fnum1 ) ) ) {
printf ( " close failed (%s) \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
cli - > tree - > tid = cnum2 ;
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_tdis ( cli ) ) ) {
printf ( " secondary tdis failed (%s) \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
cli - > tree = tree1 ; /* restore initial tree */
cli - > tree - > tid = cnum1 ;
if ( ! torture_close_connection ( cli ) ) {
return False ;
}
return ret ;
}
2004-08-04 17:23:35 +04:00
static BOOL tcon_devtest ( struct smbcli_state * cli ,
2003-08-13 05:53:07 +04:00
const char * myshare , const char * devtype ,
NTSTATUS expected_error )
{
BOOL status ;
BOOL ret ;
2004-06-14 03:50:55 +04:00
const char * password = lp_parm_string ( - 1 , " torture " , " password " ) ;
2003-08-13 05:53:07 +04:00
2004-08-04 17:23:35 +04:00
status = NT_STATUS_IS_OK ( smbcli_send_tconX ( cli , myshare , devtype ,
2004-02-10 14:33:35 +03:00
password ) ) ;
2003-08-13 05:53:07 +04:00
printf ( " Trying share %s with devtype %s \n " , myshare , devtype ) ;
if ( NT_STATUS_IS_OK ( expected_error ) ) {
if ( status ) {
ret = True ;
} else {
printf ( " tconX to share %s with type %s "
" should have succeeded but failed \n " ,
myshare , devtype ) ;
ret = False ;
}
2004-08-04 17:23:35 +04:00
smbcli_tdis ( cli ) ;
2003-08-13 05:53:07 +04:00
} else {
if ( status ) {
printf ( " tconx to share %s with type %s "
" should have failed but succeeded \n " ,
myshare , devtype ) ;
ret = False ;
} else {
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_EQUAL ( smbcli_nt_error ( cli - > tree ) ,
2003-08-13 05:53:07 +04:00
expected_error ) ) {
ret = True ;
} else {
printf ( " Returned unexpected error \n " ) ;
ret = False ;
}
}
}
return ret ;
}
/*
checks for correct tconX support
*/
2004-10-28 17:40:50 +04:00
static BOOL run_tcon_devtype_test ( void )
2003-08-13 05:53:07 +04:00
{
2004-08-04 17:23:35 +04:00
struct smbcli_state * cli1 = NULL ;
2003-08-13 05:53:07 +04:00
BOOL retry ;
int flags = 0 ;
NTSTATUS status ;
BOOL ret = True ;
2004-06-14 03:50:55 +04:00
const char * host = lp_parm_string ( - 1 , " torture " , " host " ) ;
const char * share = lp_parm_string ( - 1 , " torture " , " share " ) ;
const char * username = lp_parm_string ( - 1 , " torture " , " username " ) ;
2004-08-25 18:31:59 +04:00
const char * userdomain = lp_parm_string ( - 1 , " torture " , " userdomain " ) ;
2004-06-14 03:50:55 +04:00
const char * password = lp_parm_string ( - 1 , " torture " , " password " ) ;
2003-08-13 05:53:07 +04:00
2004-09-28 09:44:59 +04:00
status = smbcli_full_connection ( NULL ,
& cli1 , lp_netbios_name ( ) ,
2004-09-28 09:42:02 +04:00
host , NULL ,
share , " ????? " ,
username , userdomain ,
password , flags , & retry ) ;
2003-08-13 05:53:07 +04:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
printf ( " could not open connection \n " ) ;
return False ;
}
if ( ! tcon_devtest ( cli1 , " IPC$ " , " A: " , NT_STATUS_BAD_DEVICE_TYPE ) )
ret = False ;
if ( ! tcon_devtest ( cli1 , " IPC$ " , " ????? " , NT_STATUS_OK ) )
ret = False ;
if ( ! tcon_devtest ( cli1 , " IPC$ " , " LPT: " , NT_STATUS_BAD_DEVICE_TYPE ) )
ret = False ;
if ( ! tcon_devtest ( cli1 , " IPC$ " , " IPC " , NT_STATUS_OK ) )
ret = False ;
if ( ! tcon_devtest ( cli1 , " IPC$ " , " FOOBA " , NT_STATUS_BAD_DEVICE_TYPE ) )
ret = False ;
if ( ! tcon_devtest ( cli1 , share , " A: " , NT_STATUS_OK ) )
ret = False ;
if ( ! tcon_devtest ( cli1 , share , " ????? " , NT_STATUS_OK ) )
ret = False ;
if ( ! tcon_devtest ( cli1 , share , " LPT: " , NT_STATUS_BAD_DEVICE_TYPE ) )
ret = False ;
if ( ! tcon_devtest ( cli1 , share , " IPC " , NT_STATUS_BAD_DEVICE_TYPE ) )
ret = False ;
if ( ! tcon_devtest ( cli1 , share , " FOOBA " , NT_STATUS_BAD_DEVICE_TYPE ) )
ret = False ;
2004-08-04 17:23:35 +04:00
smbcli_shutdown ( cli1 ) ;
2003-08-13 05:53:07 +04:00
if ( ret )
printf ( " Passed tcondevtest \n " ) ;
return ret ;
}
/*
test whether fnums and tids open on one VC are available on another ( a major
security hole )
*/
2004-10-28 17:40:50 +04:00
static BOOL run_fdpasstest ( void )
2003-08-13 05:53:07 +04:00
{
2004-08-04 17:23:35 +04:00
struct smbcli_state * cli1 , * cli2 ;
2003-08-13 05:53:07 +04:00
const char * fname = " \\ fdpass.tst " ;
int fnum1 , oldtid ;
pstring buf ;
if ( ! torture_open_connection ( & cli1 ) | | ! torture_open_connection ( & cli2 ) ) {
return False ;
}
printf ( " starting fdpasstest \n " ) ;
2004-08-04 17:23:35 +04:00
smbcli_unlink ( cli1 - > tree , fname ) ;
2003-08-13 05:53:07 +04:00
printf ( " Opening a file on connection 1 \n " ) ;
2004-08-04 17:23:35 +04:00
fnum1 = smbcli_open ( cli1 - > tree , fname , O_RDWR | O_CREAT | O_EXCL , DENY_NONE ) ;
2003-08-13 05:53:07 +04:00
if ( fnum1 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " open of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
printf ( " writing to file on connection 1 \n " ) ;
2004-08-04 17:23:35 +04:00
if ( smbcli_write ( cli1 - > tree , fnum1 , 0 , " hello world \n " , 0 , 13 ) ! = 13 ) {
printf ( " write failed (%s) \n " , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
oldtid = cli2 - > tree - > tid ;
cli2 - > session - > vuid = cli1 - > session - > vuid ;
cli2 - > tree - > tid = cli1 - > tree - > tid ;
cli2 - > session - > pid = cli1 - > session - > pid ;
printf ( " reading from file on connection 2 \n " ) ;
2004-08-04 17:23:35 +04:00
if ( smbcli_read ( cli2 - > tree , fnum1 , buf , 0 , 13 ) = = 13 ) {
2003-08-13 05:53:07 +04:00
printf ( " read succeeded! nasty security hole [%s] \n " ,
buf ) ;
return False ;
}
2004-08-04 17:23:35 +04:00
smbcli_close ( cli1 - > tree , fnum1 ) ;
smbcli_unlink ( cli1 - > tree , fname ) ;
2003-08-13 05:53:07 +04:00
cli2 - > tree - > tid = oldtid ;
torture_close_connection ( cli1 ) ;
torture_close_connection ( cli2 ) ;
printf ( " finished fdpasstest \n " ) ;
return True ;
}
2004-06-08 23:25:26 +04:00
/*
2004-08-03 08:44:27 +04:00
test the timing of deferred open requests
2004-06-08 23:25:26 +04:00
*/
2004-08-04 17:23:35 +04:00
static BOOL run_deferopen ( struct smbcli_state * cli , int dummy )
2004-06-08 23:25:26 +04:00
{
2004-06-14 03:50:55 +04:00
const char * fname = " \\ defer_open_test.dat " ;
2004-06-08 23:25:26 +04:00
int retries = 4 ;
int i = 0 ;
BOOL correct = True ;
if ( retries < = 0 ) {
printf ( " failed to connect \n " ) ;
return False ;
}
printf ( " Testing deferred open requests. \n " ) ;
while ( i < 4 ) {
int fnum = - 1 ;
2004-06-09 04:07:59 +04:00
2004-06-08 23:25:26 +04:00
do {
2004-11-03 13:09:48 +03:00
struct timeval tv ;
tv = timeval_current ( ) ;
2004-08-04 17:23:35 +04:00
fnum = smbcli_nt_create_full ( cli - > tree , fname , 0 , GENERIC_RIGHTS_FILE_ALL_ACCESS ,
2004-06-08 23:25:26 +04:00
FILE_ATTRIBUTE_NORMAL , NTCREATEX_SHARE_ACCESS_NONE ,
NTCREATEX_DISP_OPEN_IF , 0 , 0 ) ;
if ( fnum ! = - 1 ) {
break ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_EQUAL ( smbcli_nt_error ( cli - > tree ) , NT_STATUS_SHARING_VIOLATION ) ) {
2004-11-03 13:09:48 +03:00
double e = timeval_elapsed ( & tv ) ;
if ( e < 0.5 | | e > 1.5 ) {
fprintf ( stderr , " Timing incorrect %.2f violation \n " ,
e ) ;
2004-06-09 04:07:59 +04:00
}
}
2004-08-04 17:23:35 +04:00
} while ( NT_STATUS_EQUAL ( smbcli_nt_error ( cli - > tree ) , NT_STATUS_SHARING_VIOLATION ) ) ;
2004-06-08 23:25:26 +04:00
if ( fnum = = - 1 ) {
2004-08-04 17:23:35 +04:00
fprintf ( stderr , " Failed to open %s, error=%s \n " , fname , smbcli_errstr ( cli - > tree ) ) ;
2004-06-08 23:25:26 +04:00
return False ;
}
printf ( " pid %u open %d \n " , getpid ( ) , i ) ;
sleep ( 10 ) ;
i + + ;
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli - > tree , fnum ) ) ) {
fprintf ( stderr , " Failed to close %s, error=%s \n " , fname , smbcli_errstr ( cli - > tree ) ) ;
2004-06-08 23:25:26 +04:00
return False ;
}
sleep ( 2 ) ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_unlink ( cli - > tree , fname ) ) ) {
2004-06-08 23:25:26 +04:00
/* All until the last unlink will fail with sharing violation. */
2004-08-04 17:23:35 +04:00
if ( ! NT_STATUS_EQUAL ( smbcli_nt_error ( cli - > tree ) , NT_STATUS_SHARING_VIOLATION ) ) {
printf ( " unlink of %s failed (%s) \n " , fname , smbcli_errstr ( cli - > tree ) ) ;
2004-06-08 23:25:26 +04:00
correct = False ;
}
}
printf ( " deferred test finished \n " ) ;
if ( ! torture_close_connection ( cli ) ) {
correct = False ;
}
return correct ;
}
2003-08-13 05:53:07 +04:00
/*
test how many open files this server supports on the one socket
*/
2004-08-04 17:23:35 +04:00
static BOOL run_maxfidtest ( struct smbcli_state * cli , int dummy )
2003-08-13 05:53:07 +04:00
{
2004-10-19 10:30:52 +04:00
# define MAXFID_TEMPLATE "\\maxfid\\fid%d\\maxfid.%d.%d"
2003-08-13 05:53:07 +04:00
char * fname ;
int fnums [ 0x11000 ] , i ;
2004-10-19 10:30:52 +04:00
int retries = 4 , maxfid ;
2003-08-13 05:53:07 +04:00
BOOL correct = True ;
if ( retries < = 0 ) {
printf ( " failed to connect \n " ) ;
return False ;
}
2004-10-19 10:30:52 +04:00
if ( smbcli_deltree ( cli - > tree , " \\ maxfid " ) = = - 1 ) {
printf ( " Failed to deltree \\ maxfid - %s \n " ,
smbcli_errstr ( cli - > tree ) ) ;
return False ;
}
if ( NT_STATUS_IS_ERR ( smbcli_mkdir ( cli - > tree , " \\ maxfid " ) ) ) {
printf ( " Failed to mkdir \\ maxfid, error=%s \n " ,
smbcli_errstr ( cli - > tree ) ) ;
return False ;
}
2003-08-13 05:53:07 +04:00
printf ( " Testing maximum number of open files \n " ) ;
for ( i = 0 ; i < 0x11000 ; i + + ) {
2004-10-19 10:30:52 +04:00
if ( i % 1000 = = 0 ) {
asprintf ( & fname , " \\ maxfid \\ fid%d " , i / 1000 ) ;
if ( NT_STATUS_IS_ERR ( smbcli_mkdir ( cli - > tree , fname ) ) ) {
printf ( " Failed to mkdir %s, error=%s \n " ,
fname , smbcli_errstr ( cli - > tree ) ) ;
return False ;
}
free ( fname ) ;
}
asprintf ( & fname , MAXFID_TEMPLATE , i / 1000 , i , ( int ) getpid ( ) ) ;
2004-08-04 17:23:35 +04:00
if ( ( fnums [ i ] = smbcli_open ( cli - > tree , fname ,
2003-08-13 05:53:07 +04:00
O_RDWR | O_CREAT | O_TRUNC , DENY_NONE ) ) = =
- 1 ) {
printf ( " open of %s failed (%s) \n " ,
2004-08-04 17:23:35 +04:00
fname , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
printf ( " maximum fnum is %d \n " , i ) ;
break ;
}
free ( fname ) ;
printf ( " %6d \r " , i ) ;
}
printf ( " %6d \n " , i ) ;
i - - ;
2004-10-19 10:30:52 +04:00
maxfid = i ;
2003-08-13 05:53:07 +04:00
printf ( " cleaning up \n " ) ;
2004-10-19 10:30:52 +04:00
for ( i = 0 ; i < maxfid / 2 ; i + + ) {
asprintf ( & fname , MAXFID_TEMPLATE , i / 1000 , i , ( int ) getpid ( ) ) ;
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli - > tree , fnums [ i ] ) ) ) {
printf ( " Close of fnum %d failed - %s \n " , fnums [ i ] , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_unlink ( cli - > tree , fname ) ) ) {
2003-08-13 05:53:07 +04:00
printf ( " unlink of %s failed (%s) \n " ,
2004-08-04 17:23:35 +04:00
fname , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
}
free ( fname ) ;
2004-10-19 10:30:52 +04:00
asprintf ( & fname , MAXFID_TEMPLATE , ( maxfid - i ) / 1000 , maxfid - i , ( int ) getpid ( ) ) ;
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli - > tree , fnums [ maxfid - i ] ) ) ) {
printf ( " Close of fnum %d failed - %s \n " , fnums [ maxfid - i ] , smbcli_errstr ( cli - > tree ) ) ;
}
if ( NT_STATUS_IS_ERR ( smbcli_unlink ( cli - > tree , fname ) ) ) {
printf ( " unlink of %s failed (%s) \n " ,
fname , smbcli_errstr ( cli - > tree ) ) ;
correct = False ;
}
free ( fname ) ;
printf ( " %6d %6d \r " , i , maxfid - i ) ;
2003-08-13 05:53:07 +04:00
}
printf ( " %6d \n " , 0 ) ;
2004-10-19 10:30:52 +04:00
if ( smbcli_deltree ( cli - > tree , " \\ maxfid " ) = = - 1 ) {
printf ( " Failed to deltree \\ maxfid - %s \n " ,
smbcli_errstr ( cli - > tree ) ) ;
return False ;
}
2003-08-13 05:53:07 +04:00
printf ( " maxfid test finished \n " ) ;
if ( ! torture_close_connection ( cli ) ) {
correct = False ;
}
return correct ;
2004-08-25 11:30:58 +04:00
# undef MAXFID_TEMPLATE
2003-08-13 05:53:07 +04:00
}
/* send smb negprot commands, not reading the response */
2004-10-28 17:40:50 +04:00
static BOOL run_negprot_nowait ( void )
2003-08-13 05:53:07 +04:00
{
int i ;
2004-08-04 17:23:35 +04:00
struct smbcli_state * cli , * cli2 ;
2003-08-13 05:53:07 +04:00
BOOL correct = True ;
printf ( " starting negprot nowait test \n " ) ;
cli = open_nbt_connection ( ) ;
if ( ! cli ) {
return False ;
}
2004-08-03 12:04:51 +04:00
printf ( " Filling send buffer \n " ) ;
2003-08-13 05:53:07 +04:00
2004-08-03 12:04:51 +04:00
for ( i = 0 ; i < 10000 ; i + + ) {
2004-08-04 17:23:35 +04:00
struct smbcli_request * req ;
2004-08-03 12:04:51 +04:00
time_t t1 = time ( NULL ) ;
req = smb_negprot_send ( cli - > transport , PROTOCOL_NT1 ) ;
2004-08-04 17:23:35 +04:00
while ( req - > state = = SMBCLI_REQUEST_SEND & & time ( NULL ) < t1 + 5 ) {
smbcli_transport_process ( cli - > transport ) ;
2004-08-03 12:04:51 +04:00
}
2004-08-04 17:23:35 +04:00
if ( req - > state = = SMBCLI_REQUEST_ERROR ) {
2004-08-03 12:04:51 +04:00
printf ( " Failed to fill pipe - %s \n " , nt_errstr ( req - > status ) ) ;
torture_close_connection ( cli ) ;
return correct ;
}
2004-08-04 17:23:35 +04:00
if ( req - > state = = SMBCLI_REQUEST_SEND ) {
2004-08-03 12:04:51 +04:00
break ;
}
}
if ( i = = 10000 ) {
printf ( " send buffer failed to fill \n " ) ;
if ( ! torture_close_connection ( cli ) ) {
correct = False ;
}
return correct ;
}
printf ( " send buffer filled after %d requests \n " , i ) ;
printf ( " Opening secondary connection \n " ) ;
if ( ! torture_open_connection ( & cli2 ) ) {
return False ;
2003-08-13 05:53:07 +04:00
}
if ( ! torture_close_connection ( cli ) ) {
correct = False ;
}
2004-08-03 12:04:51 +04:00
if ( ! torture_close_connection ( cli2 ) ) {
correct = False ;
}
2003-08-13 05:53:07 +04:00
printf ( " finished negprot nowait test \n " ) ;
return correct ;
}
/*
This checks how the getatr calls works
*/
2004-10-28 17:40:50 +04:00
static BOOL run_attrtest ( void )
2003-08-13 05:53:07 +04:00
{
2004-08-04 17:23:35 +04:00
struct smbcli_state * cli ;
2003-08-13 05:53:07 +04:00
int fnum ;
time_t t , t2 ;
const char * fname = " \\ attrib123456789.tst " ;
BOOL correct = True ;
printf ( " starting attrib test \n " ) ;
if ( ! torture_open_connection ( & cli ) ) {
return False ;
}
2004-08-04 17:23:35 +04:00
smbcli_unlink ( cli - > tree , fname ) ;
fnum = smbcli_open ( cli - > tree , fname ,
2003-08-13 05:53:07 +04:00
O_RDWR | O_CREAT | O_TRUNC , DENY_NONE ) ;
2004-08-04 17:23:35 +04:00
smbcli_close ( cli - > tree , fnum ) ;
2003-08-13 05:53:07 +04:00
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_getatr ( cli - > tree , fname , NULL , NULL , & t ) ) ) {
printf ( " getatr failed (%s) \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
}
printf ( " New file time is %s " , ctime ( & t ) ) ;
if ( abs ( t - time ( NULL ) ) > 60 * 60 * 24 * 10 ) {
printf ( " ERROR: SMBgetatr bug. time is %s " ,
ctime ( & t ) ) ;
t = time ( NULL ) ;
correct = False ;
}
t2 = t - 60 * 60 * 24 ; /* 1 day ago */
printf ( " Setting file time to %s " , ctime ( & t2 ) ) ;
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_setatr ( cli - > tree , fname , 0 , t2 ) ) ) {
printf ( " setatr failed (%s) \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = True ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_getatr ( cli - > tree , fname , NULL , NULL , & t ) ) ) {
printf ( " getatr failed (%s) \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = True ;
}
printf ( " Retrieved file time as %s " , ctime ( & t ) ) ;
if ( t ! = t2 ) {
printf ( " ERROR: getatr/setatr bug. times are \n %s " ,
ctime ( & t ) ) ;
printf ( " %s " , ctime ( & t2 ) ) ;
correct = True ;
}
2004-08-04 17:23:35 +04:00
smbcli_unlink ( cli - > tree , fname ) ;
2003-08-13 05:53:07 +04:00
if ( ! torture_close_connection ( cli ) ) {
correct = False ;
}
printf ( " attrib test finished \n " ) ;
return correct ;
}
/*
This checks a couple of trans2 calls
*/
2004-10-28 17:40:50 +04:00
static BOOL run_trans2test ( void )
2003-08-13 05:53:07 +04:00
{
2004-08-04 17:23:35 +04:00
struct smbcli_state * cli ;
2003-08-13 05:53:07 +04:00
int fnum ;
size_t size ;
time_t c_time , a_time , m_time , w_time , m_time2 ;
const char * fname = " \\ trans2.tst " ;
const char * dname = " \\ trans2 " ;
const char * fname2 = " \\ trans2 \\ trans2.tst " ;
const char * pname ;
BOOL correct = True ;
printf ( " starting trans2 test \n " ) ;
if ( ! torture_open_connection ( & cli ) ) {
return False ;
}
2004-08-04 17:23:35 +04:00
smbcli_unlink ( cli - > tree , fname ) ;
2003-08-13 05:53:07 +04:00
printf ( " Testing qfileinfo \n " ) ;
2004-08-04 17:23:35 +04:00
fnum = smbcli_open ( cli - > tree , fname ,
2003-08-13 05:53:07 +04:00
O_RDWR | O_CREAT | O_TRUNC , DENY_NONE ) ;
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_qfileinfo ( cli - > tree , fnum , NULL , & size , & c_time , & a_time , & m_time ,
2004-02-10 14:33:35 +03:00
NULL , NULL ) ) ) {
2004-08-04 17:23:35 +04:00
printf ( " ERROR: qfileinfo failed (%s) \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
}
printf ( " Testing NAME_INFO \n " ) ;
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_qfilename ( cli - > tree , fnum , & pname ) ) ) {
printf ( " ERROR: qfilename failed (%s) \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
}
if ( ! pname | | strcmp ( pname , fname ) ) {
printf ( " qfilename gave different name? [%s] [%s] \n " ,
fname , pname ) ;
correct = False ;
}
2004-08-04 17:23:35 +04:00
smbcli_close ( cli - > tree , fnum ) ;
smbcli_unlink ( cli - > tree , fname ) ;
2003-08-13 05:53:07 +04:00
2004-08-04 17:23:35 +04:00
fnum = smbcli_open ( cli - > tree , fname ,
2003-08-13 05:53:07 +04:00
O_RDWR | O_CREAT | O_TRUNC , DENY_NONE ) ;
if ( fnum = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " open of %s failed (%s) \n " , fname , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
2004-08-04 17:23:35 +04:00
smbcli_close ( cli - > tree , fnum ) ;
2003-08-13 05:53:07 +04:00
printf ( " Checking for sticky create times \n " ) ;
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_qpathinfo ( cli - > tree , fname , & c_time , & a_time , & m_time , & size , NULL ) ) ) {
printf ( " ERROR: qpathinfo failed (%s) \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
} else {
if ( c_time ! = m_time ) {
printf ( " create time=%s " , ctime ( & c_time ) ) ;
printf ( " modify time=%s " , ctime ( & m_time ) ) ;
printf ( " This system appears to have sticky create times \n " ) ;
}
if ( a_time % ( 60 * 60 ) = = 0 ) {
printf ( " access time=%s " , ctime ( & a_time ) ) ;
printf ( " This system appears to set a midnight access time \n " ) ;
correct = False ;
}
if ( abs ( m_time - time ( NULL ) ) > 60 * 60 * 24 * 7 ) {
printf ( " ERROR: totally incorrect times - maybe word reversed? mtime=%s " , ctime ( & m_time ) ) ;
correct = False ;
}
}
2004-08-04 17:23:35 +04:00
smbcli_unlink ( cli - > tree , fname ) ;
fnum = smbcli_open ( cli - > tree , fname ,
2003-08-13 05:53:07 +04:00
O_RDWR | O_CREAT | O_TRUNC , DENY_NONE ) ;
2004-08-04 17:23:35 +04:00
smbcli_close ( cli - > tree , fnum ) ;
if ( NT_STATUS_IS_ERR ( smbcli_qpathinfo2 ( cli - > tree , fname , & c_time , & a_time , & m_time , & w_time , & size , NULL , NULL ) ) ) {
printf ( " ERROR: qpathinfo2 failed (%s) \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
} else {
if ( w_time < 60 * 60 * 24 * 2 ) {
printf ( " write time=%s " , ctime ( & w_time ) ) ;
printf ( " This system appears to set a initial 0 write time \n " ) ;
correct = False ;
}
}
2004-08-04 17:23:35 +04:00
smbcli_unlink ( cli - > tree , fname ) ;
2003-08-13 05:53:07 +04:00
/* check if the server updates the directory modification time
when creating a new file */
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_mkdir ( cli - > tree , dname ) ) ) {
printf ( " ERROR: mkdir failed (%s) \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
}
sleep ( 3 ) ;
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_qpathinfo2 ( cli - > tree , " \\ trans2 \\ " , & c_time , & a_time , & m_time , & w_time , & size , NULL , NULL ) ) ) {
printf ( " ERROR: qpathinfo2 failed (%s) \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
}
2004-08-04 17:23:35 +04:00
fnum = smbcli_open ( cli - > tree , fname2 ,
2003-08-13 05:53:07 +04:00
O_RDWR | O_CREAT | O_TRUNC , DENY_NONE ) ;
2004-08-04 17:23:35 +04:00
smbcli_write ( cli - > tree , fnum , 0 , ( char * ) & fnum , 0 , sizeof ( fnum ) ) ;
smbcli_close ( cli - > tree , fnum ) ;
if ( NT_STATUS_IS_ERR ( smbcli_qpathinfo2 ( cli - > tree , " \\ trans2 \\ " , & c_time , & a_time , & m_time2 , & w_time , & size , NULL , NULL ) ) ) {
printf ( " ERROR: qpathinfo2 failed (%s) \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
} else {
if ( m_time2 = = m_time ) {
printf ( " This system does not update directory modification times \n " ) ;
correct = False ;
}
}
2004-08-04 17:23:35 +04:00
smbcli_unlink ( cli - > tree , fname2 ) ;
smbcli_rmdir ( cli - > tree , dname ) ;
2003-08-13 05:53:07 +04:00
if ( ! torture_close_connection ( cli ) ) {
correct = False ;
}
printf ( " trans2 test finished \n " ) ;
return correct ;
}
/*
print out server properties
*/
2004-10-28 17:40:50 +04:00
static BOOL run_properties ( void )
2003-08-13 05:53:07 +04:00
{
2004-08-04 17:23:35 +04:00
struct smbcli_state * cli ;
2003-08-13 05:53:07 +04:00
BOOL correct = True ;
printf ( " starting properties test \n " ) ;
ZERO_STRUCT ( cli ) ;
if ( ! torture_open_connection ( & cli ) ) {
return False ;
}
d_printf ( " Capabilities 0x%08x \n " , cli - > transport - > negotiate . capabilities ) ;
if ( ! torture_close_connection ( cli ) ) {
correct = False ;
}
return correct ;
}
/* FIRST_DESIRED_ACCESS 0xf019f */
# define FIRST_DESIRED_ACCESS SA_RIGHT_FILE_READ_DATA|SA_RIGHT_FILE_WRITE_DATA|SA_RIGHT_FILE_APPEND_DATA|\
SA_RIGHT_FILE_READ_EA | /* 0xf */ \
SA_RIGHT_FILE_WRITE_EA | SA_RIGHT_FILE_READ_ATTRIBUTES | /* 0x90 */ \
SA_RIGHT_FILE_WRITE_ATTRIBUTES | /* 0x100 */ \
STD_RIGHT_DELETE_ACCESS | STD_RIGHT_READ_CONTROL_ACCESS | \
STD_RIGHT_WRITE_DAC_ACCESS | STD_RIGHT_WRITE_OWNER_ACCESS /* 0xf0000 */
/* SECOND_DESIRED_ACCESS 0xe0080 */
# define SECOND_DESIRED_ACCESS SA_RIGHT_FILE_READ_ATTRIBUTES| /* 0x80 */ \
STD_RIGHT_READ_CONTROL_ACCESS | STD_RIGHT_WRITE_DAC_ACCESS | \
STD_RIGHT_WRITE_OWNER_ACCESS /* 0xe0000 */
#if 0
# define THIRD_DESIRED_ACCESS FILE_READ_ATTRIBUTES| /* 0x80 */ \
READ_CONTROL_ACCESS | WRITE_DAC_ACCESS | \
SA_RIGHT_FILE_READ_DATA | \
WRITE_OWNER_ACCESS /* */
# endif
/*
Test ntcreate calls made by xcopy
*/
2004-10-28 17:40:50 +04:00
static BOOL run_xcopy ( void )
2003-08-13 05:53:07 +04:00
{
2004-08-04 17:23:35 +04:00
struct smbcli_state * cli1 ;
2003-08-13 05:53:07 +04:00
const char * fname = " \\ test.txt " ;
BOOL correct = True ;
int fnum1 , fnum2 ;
printf ( " starting xcopy test \n " ) ;
if ( ! torture_open_connection ( & cli1 ) ) {
return False ;
}
2004-08-04 17:23:35 +04:00
fnum1 = smbcli_nt_create_full ( cli1 - > tree , fname , 0 ,
2003-08-13 05:53:07 +04:00
FIRST_DESIRED_ACCESS , FILE_ATTRIBUTE_ARCHIVE ,
NTCREATEX_SHARE_ACCESS_NONE , NTCREATEX_DISP_OVERWRITE_IF ,
0x4044 , 0 ) ;
if ( fnum1 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " First open failed - %s \n " , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
2004-08-04 17:23:35 +04:00
fnum2 = smbcli_nt_create_full ( cli1 - > tree , fname , 0 ,
2003-08-13 05:53:07 +04:00
SECOND_DESIRED_ACCESS , 0 ,
NTCREATEX_SHARE_ACCESS_READ | NTCREATEX_SHARE_ACCESS_WRITE | NTCREATEX_SHARE_ACCESS_DELETE , NTCREATEX_DISP_OPEN ,
0x200000 , 0 ) ;
if ( fnum2 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " second open failed - %s \n " , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
if ( ! torture_close_connection ( cli1 ) ) {
correct = False ;
}
return correct ;
}
2004-08-03 08:44:27 +04:00
/*
see how many RPC pipes we can open at once
*/
2004-10-28 17:40:50 +04:00
static BOOL run_pipe_number ( void )
2003-08-13 05:53:07 +04:00
{
2004-08-04 17:23:35 +04:00
struct smbcli_state * cli1 ;
2003-11-18 08:01:10 +03:00
const char * pipe_name = " \\ WKSSVC " ;
2003-08-13 05:53:07 +04:00
int fnum ;
int num_pipes = 0 ;
printf ( " starting pipenumber test \n " ) ;
if ( ! torture_open_connection ( & cli1 ) ) {
return False ;
}
while ( 1 ) {
2004-08-04 17:23:35 +04:00
fnum = smbcli_nt_create_full ( cli1 - > tree , pipe_name , 0 , SA_RIGHT_FILE_READ_DATA , FILE_ATTRIBUTE_NORMAL ,
2003-08-13 05:53:07 +04:00
NTCREATEX_SHARE_ACCESS_READ | NTCREATEX_SHARE_ACCESS_WRITE , NTCREATEX_DISP_OPEN_IF , 0 , 0 ) ;
if ( fnum = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " Open of pipe %s failed with error (%s) \n " , pipe_name , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
break ;
}
num_pipes + + ;
2004-08-03 08:44:27 +04:00
printf ( " %d \r " , num_pipes ) ;
fflush ( stdout ) ;
2003-08-13 05:53:07 +04:00
}
printf ( " pipe_number test - we can open %d %s pipes. \n " , num_pipes , pipe_name ) ;
torture_close_connection ( cli1 ) ;
return True ;
}
2004-02-21 07:02:55 +03:00
/*
open N connections to the server and just hold them open
used for testing performance when there are N idle users
already connected
*/
2004-10-28 17:40:50 +04:00
static BOOL torture_holdcon ( void )
2004-02-21 07:02:55 +03:00
{
int i ;
2004-08-04 17:23:35 +04:00
struct smbcli_state * * cli ;
2004-02-21 07:02:55 +03:00
int num_dead = 0 ;
printf ( " Opening %d connections \n " , torture_numops ) ;
2004-08-04 17:23:35 +04:00
cli = malloc ( sizeof ( struct smbcli_state * ) * torture_numops ) ;
2004-02-21 07:02:55 +03:00
for ( i = 0 ; i < torture_numops ; i + + ) {
if ( ! torture_open_connection ( & cli [ i ] ) ) {
return False ;
}
printf ( " opened %d connections \r " , i ) ;
fflush ( stdout ) ;
}
printf ( " \n Starting pings \n " ) ;
while ( 1 ) {
for ( i = 0 ; i < torture_numops ; i + + ) {
NTSTATUS status ;
if ( cli [ i ] ) {
2004-08-04 17:23:35 +04:00
status = smbcli_chkpath ( cli [ i ] - > tree , " \\ " ) ;
2004-02-21 07:02:55 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
printf ( " Connection %d is dead \n " , i ) ;
cli [ i ] = NULL ;
num_dead + + ;
}
usleep ( 100 ) ;
}
}
if ( num_dead = = torture_numops ) {
printf ( " All connections dead - finishing \n " ) ;
break ;
}
printf ( " . " ) ;
fflush ( stdout ) ;
}
return True ;
}
2004-04-28 16:45:16 +04:00
/*
Try with a wrong vuid and check error message .
*/
2004-10-28 17:40:50 +04:00
static BOOL run_vuidtest ( void )
2004-04-28 16:45:16 +04:00
{
2004-08-04 17:23:35 +04:00
struct smbcli_state * cli ;
2004-04-28 16:45:16 +04:00
const char * fname = " \\ vuid.tst " ;
int fnum ;
size_t size ;
2004-06-06 11:14:10 +04:00
time_t c_time , a_time , m_time ;
2004-04-28 16:45:16 +04:00
BOOL correct = True ;
2004-05-25 21:24:24 +04:00
uint16_t orig_vuid ;
2004-04-28 16:45:16 +04:00
NTSTATUS result ;
printf ( " starting vuid test \n " ) ;
if ( ! torture_open_connection ( & cli ) ) {
return False ;
}
2004-08-04 17:23:35 +04:00
smbcli_unlink ( cli - > tree , fname ) ;
2004-04-28 16:45:16 +04:00
2004-08-04 17:23:35 +04:00
fnum = smbcli_open ( cli - > tree , fname ,
2004-04-28 16:45:16 +04:00
O_RDWR | O_CREAT | O_TRUNC , DENY_NONE ) ;
orig_vuid = cli - > session - > vuid ;
cli - > session - > vuid + = 1234 ;
2004-02-21 07:02:55 +03:00
2004-04-28 16:45:16 +04:00
printf ( " Testing qfileinfo with wrong vuid \n " ) ;
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_OK ( result = smbcli_qfileinfo ( cli - > tree , fnum , NULL ,
2004-04-28 16:45:16 +04:00
& size , & c_time , & a_time ,
& m_time , NULL , NULL ) ) ) {
printf ( " ERROR: qfileinfo passed with wrong vuid \n " ) ;
correct = False ;
}
if ( ( cli - > transport - > error . etype ! = ETYPE_DOS ) | |
( cli - > transport - > error . e . dos . eclass ! = ERRSRV ) | |
( cli - > transport - > error . e . dos . ecode ! = ERRbaduid ) ) {
printf ( " ERROR: qfileinfo should have returned DOS error "
" ERRSRV:ERRbaduid \n but returned %s \n " ,
2004-08-04 17:23:35 +04:00
smbcli_errstr ( cli - > tree ) ) ;
2004-04-28 16:45:16 +04:00
correct = False ;
}
cli - > session - > vuid - = 1234 ;
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli - > tree , fnum ) ) ) {
printf ( " close failed (%s) \n " , smbcli_errstr ( cli - > tree ) ) ;
2004-04-28 16:45:16 +04:00
correct = False ;
}
2004-08-04 17:23:35 +04:00
smbcli_unlink ( cli - > tree , fname ) ;
2004-04-28 16:45:16 +04:00
if ( ! torture_close_connection ( cli ) ) {
correct = False ;
}
printf ( " vuid test finished \n " ) ;
return correct ;
}
2004-02-21 07:02:55 +03:00
2003-08-13 05:53:07 +04:00
/*
Test open mode returns on read - only files .
*/
2004-10-28 17:40:50 +04:00
static BOOL run_opentest ( void )
2003-08-13 05:53:07 +04:00
{
2004-08-04 17:23:35 +04:00
static struct smbcli_state * cli1 ;
static struct smbcli_state * cli2 ;
2003-08-13 05:53:07 +04:00
const char * fname = " \\ readonly.file " ;
int fnum1 , fnum2 ;
char buf [ 20 ] ;
size_t fsize ;
BOOL correct = True ;
char * tmp_path ;
int failures = 0 ;
printf ( " starting open test \n " ) ;
if ( ! torture_open_connection ( & cli1 ) ) {
return False ;
}
2004-08-04 17:23:35 +04:00
smbcli_setatr ( cli1 - > tree , fname , 0 , 0 ) ;
smbcli_unlink ( cli1 - > tree , fname ) ;
2003-08-13 05:53:07 +04:00
2004-08-04 17:23:35 +04:00
fnum1 = smbcli_open ( cli1 - > tree , fname , O_RDWR | O_CREAT | O_EXCL , DENY_NONE ) ;
2003-08-13 05:53:07 +04:00
if ( fnum1 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " open of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli1 - > tree , fnum1 ) ) ) {
printf ( " close2 failed (%s) \n " , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_setatr ( cli1 - > tree , fname , FILE_ATTRIBUTE_READONLY , 0 ) ) ) {
printf ( " smbcli_setatr failed (%s) \n " , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
CHECK_MAX_FAILURES ( error_test1 ) ;
return False ;
}
2004-08-04 17:23:35 +04:00
fnum1 = smbcli_open ( cli1 - > tree , fname , O_RDONLY , DENY_WRITE ) ;
2003-08-13 05:53:07 +04:00
if ( fnum1 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " open of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
CHECK_MAX_FAILURES ( error_test1 ) ;
return False ;
}
/* This will fail - but the error should be ERRnoaccess, not ERRbadshare. */
2004-08-04 17:23:35 +04:00
fnum2 = smbcli_open ( cli1 - > tree , fname , O_RDWR , DENY_ALL ) ;
2003-08-13 05:53:07 +04:00
2004-10-17 06:52:37 +04:00
if ( check_error ( __location__ , cli1 , ERRDOS , ERRnoaccess ,
2003-08-13 05:53:07 +04:00
NT_STATUS_ACCESS_DENIED ) ) {
printf ( " correct error code ERRDOS/ERRnoaccess returned \n " ) ;
}
printf ( " finished open test 1 \n " ) ;
error_test1 :
2004-08-04 17:23:35 +04:00
smbcli_close ( cli1 - > tree , fnum1 ) ;
2003-08-13 05:53:07 +04:00
/* Now try not readonly and ensure ERRbadshare is returned. */
2004-08-04 17:23:35 +04:00
smbcli_setatr ( cli1 - > tree , fname , 0 , 0 ) ;
2003-08-13 05:53:07 +04:00
2004-08-04 17:23:35 +04:00
fnum1 = smbcli_open ( cli1 - > tree , fname , O_RDONLY , DENY_WRITE ) ;
2003-08-13 05:53:07 +04:00
if ( fnum1 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " open of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
/* This will fail - but the error should be ERRshare. */
2004-08-04 17:23:35 +04:00
fnum2 = smbcli_open ( cli1 - > tree , fname , O_RDWR , DENY_ALL ) ;
2003-08-13 05:53:07 +04:00
2004-10-17 06:52:37 +04:00
if ( check_error ( __location__ , cli1 , ERRDOS , ERRbadshare ,
2003-08-13 05:53:07 +04:00
NT_STATUS_SHARING_VIOLATION ) ) {
printf ( " correct error code ERRDOS/ERRbadshare returned \n " ) ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli1 - > tree , fnum1 ) ) ) {
printf ( " close2 failed (%s) \n " , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
2004-08-04 17:23:35 +04:00
smbcli_unlink ( cli1 - > tree , fname ) ;
2003-08-13 05:53:07 +04:00
printf ( " finished open test 2 \n " ) ;
/* Test truncate open disposition on file opened for read. */
2004-08-04 17:23:35 +04:00
fnum1 = smbcli_open ( cli1 - > tree , fname , O_RDWR | O_CREAT | O_EXCL , DENY_NONE ) ;
2003-08-13 05:53:07 +04:00
if ( fnum1 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " (3) open (1) of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
/* write 20 bytes. */
memset ( buf , ' \0 ' , 20 ) ;
2004-08-04 17:23:35 +04:00
if ( smbcli_write ( cli1 - > tree , fnum1 , 0 , buf , 0 , 20 ) ! = 20 ) {
printf ( " write failed (%s) \n " , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
correct = False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli1 - > tree , fnum1 ) ) ) {
printf ( " (3) close1 failed (%s) \n " , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
/* Ensure size == 20. */
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_getatr ( cli1 - > tree , fname , NULL , & fsize , NULL ) ) ) {
printf ( " (3) getatr failed (%s) \n " , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
CHECK_MAX_FAILURES ( error_test3 ) ;
return False ;
}
if ( fsize ! = 20 ) {
printf ( " (3) file size != 20 \n " ) ;
CHECK_MAX_FAILURES ( error_test3 ) ;
return False ;
}
/* Now test if we can truncate a file opened for readonly. */
2004-08-04 17:23:35 +04:00
fnum1 = smbcli_open ( cli1 - > tree , fname , O_RDONLY | O_TRUNC , DENY_NONE ) ;
2003-08-13 05:53:07 +04:00
if ( fnum1 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " (3) open (2) of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
CHECK_MAX_FAILURES ( error_test3 ) ;
return False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli1 - > tree , fnum1 ) ) ) {
printf ( " close2 failed (%s) \n " , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
/* Ensure size == 0. */
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_getatr ( cli1 - > tree , fname , NULL , & fsize , NULL ) ) ) {
printf ( " (3) getatr failed (%s) \n " , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
CHECK_MAX_FAILURES ( error_test3 ) ;
return False ;
}
if ( fsize ! = 0 ) {
printf ( " (3) file size != 0 \n " ) ;
CHECK_MAX_FAILURES ( error_test3 ) ;
return False ;
}
printf ( " finished open test 3 \n " ) ;
error_test3 :
2004-08-04 17:23:35 +04:00
smbcli_unlink ( cli1 - > tree , fname ) ;
2003-08-13 05:53:07 +04:00
printf ( " testing ctemp \n " ) ;
2004-08-04 17:23:35 +04:00
fnum1 = smbcli_ctemp ( cli1 - > tree , " \\ " , & tmp_path ) ;
2003-08-13 05:53:07 +04:00
if ( fnum1 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " ctemp failed (%s) \n " , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
CHECK_MAX_FAILURES ( error_test4 ) ;
return False ;
}
printf ( " ctemp gave path %s \n " , tmp_path ) ;
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli1 - > tree , fnum1 ) ) ) {
printf ( " close of temp failed (%s) \n " , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_unlink ( cli1 - > tree , tmp_path ) ) ) {
printf ( " unlink of temp failed (%s) \n " , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
}
error_test4 :
/* Test the non-io opens... */
if ( ! torture_open_connection ( & cli2 ) ) {
return False ;
}
2004-08-04 17:23:35 +04:00
smbcli_setatr ( cli2 - > tree , fname , 0 , 0 ) ;
smbcli_unlink ( cli2 - > tree , fname ) ;
2003-08-13 05:53:07 +04:00
printf ( " TEST #1 testing 2 non-io opens (no delete) \n " ) ;
2004-08-04 17:23:35 +04:00
fnum1 = smbcli_nt_create_full ( cli1 - > tree , fname , 0 , SA_RIGHT_FILE_READ_ATTRIBUTES , FILE_ATTRIBUTE_NORMAL ,
2003-08-13 05:53:07 +04:00
NTCREATEX_SHARE_ACCESS_NONE , NTCREATEX_DISP_OVERWRITE_IF , 0 , 0 ) ;
if ( fnum1 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " test 1 open 1 of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
CHECK_MAX_FAILURES ( error_test10 ) ;
return False ;
}
2004-08-04 17:23:35 +04:00
fnum2 = smbcli_nt_create_full ( cli2 - > tree , fname , 0 , SA_RIGHT_FILE_READ_ATTRIBUTES , FILE_ATTRIBUTE_NORMAL ,
2003-08-13 05:53:07 +04:00
NTCREATEX_SHARE_ACCESS_NONE , NTCREATEX_DISP_OPEN_IF , 0 , 0 ) ;
if ( fnum2 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " test 1 open 2 of %s failed (%s) \n " , fname , smbcli_errstr ( cli2 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
CHECK_MAX_FAILURES ( error_test10 ) ;
return False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli1 - > tree , fnum1 ) ) ) {
printf ( " test 1 close 1 of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli2 - > tree , fnum2 ) ) ) {
printf ( " test 1 close 2 of %s failed (%s) \n " , fname , smbcli_errstr ( cli2 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
printf ( " non-io open test #1 passed. \n " ) ;
error_test10 :
2004-08-04 17:23:35 +04:00
smbcli_unlink ( cli1 - > tree , fname ) ;
2003-08-13 05:53:07 +04:00
printf ( " TEST #2 testing 2 non-io opens (first with delete) \n " ) ;
2004-08-04 17:23:35 +04:00
fnum1 = smbcli_nt_create_full ( cli1 - > tree , fname , 0 , STD_RIGHT_DELETE_ACCESS | SA_RIGHT_FILE_READ_ATTRIBUTES , FILE_ATTRIBUTE_NORMAL ,
2003-08-13 05:53:07 +04:00
NTCREATEX_SHARE_ACCESS_NONE , NTCREATEX_DISP_OVERWRITE_IF , 0 , 0 ) ;
if ( fnum1 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " test 2 open 1 of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
CHECK_MAX_FAILURES ( error_test20 ) ;
return False ;
}
2004-08-04 17:23:35 +04:00
fnum2 = smbcli_nt_create_full ( cli2 - > tree , fname , 0 , SA_RIGHT_FILE_READ_ATTRIBUTES , FILE_ATTRIBUTE_NORMAL ,
2003-08-13 05:53:07 +04:00
NTCREATEX_SHARE_ACCESS_NONE , NTCREATEX_DISP_OPEN_IF , 0 , 0 ) ;
if ( fnum2 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " test 2 open 2 of %s failed (%s) \n " , fname , smbcli_errstr ( cli2 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
CHECK_MAX_FAILURES ( error_test20 ) ;
return False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli1 - > tree , fnum1 ) ) ) {
printf ( " test 1 close 1 of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli2 - > tree , fnum2 ) ) ) {
printf ( " test 1 close 2 of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
printf ( " non-io open test #2 passed. \n " ) ;
error_test20 :
2004-08-04 17:23:35 +04:00
smbcli_unlink ( cli1 - > tree , fname ) ;
2003-08-13 05:53:07 +04:00
printf ( " TEST #3 testing 2 non-io opens (second with delete) \n " ) ;
2004-08-04 17:23:35 +04:00
fnum1 = smbcli_nt_create_full ( cli1 - > tree , fname , 0 , SA_RIGHT_FILE_READ_ATTRIBUTES , FILE_ATTRIBUTE_NORMAL ,
2003-08-13 05:53:07 +04:00
NTCREATEX_SHARE_ACCESS_NONE , NTCREATEX_DISP_OVERWRITE_IF , 0 , 0 ) ;
if ( fnum1 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " test 3 open 1 of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
CHECK_MAX_FAILURES ( error_test30 ) ;
return False ;
}
2004-08-04 17:23:35 +04:00
fnum2 = smbcli_nt_create_full ( cli2 - > tree , fname , 0 , STD_RIGHT_DELETE_ACCESS | SA_RIGHT_FILE_READ_ATTRIBUTES , FILE_ATTRIBUTE_NORMAL ,
2003-08-13 05:53:07 +04:00
NTCREATEX_SHARE_ACCESS_NONE , NTCREATEX_DISP_OPEN_IF , 0 , 0 ) ;
if ( fnum2 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " test 3 open 2 of %s failed (%s) \n " , fname , smbcli_errstr ( cli2 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
CHECK_MAX_FAILURES ( error_test30 ) ;
return False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli1 - > tree , fnum1 ) ) ) {
printf ( " test 3 close 1 of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli2 - > tree , fnum2 ) ) ) {
printf ( " test 3 close 2 of %s failed (%s) \n " , fname , smbcli_errstr ( cli2 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
printf ( " non-io open test #3 passed. \n " ) ;
error_test30 :
2004-08-04 17:23:35 +04:00
smbcli_unlink ( cli1 - > tree , fname ) ;
2003-08-13 05:53:07 +04:00
printf ( " TEST #4 testing 2 non-io opens (both with delete) \n " ) ;
2004-08-04 17:23:35 +04:00
fnum1 = smbcli_nt_create_full ( cli1 - > tree , fname , 0 , STD_RIGHT_DELETE_ACCESS | SA_RIGHT_FILE_READ_ATTRIBUTES , FILE_ATTRIBUTE_NORMAL ,
2003-08-13 05:53:07 +04:00
NTCREATEX_SHARE_ACCESS_NONE , NTCREATEX_DISP_OVERWRITE_IF , 0 , 0 ) ;
if ( fnum1 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " test 4 open 1 of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
CHECK_MAX_FAILURES ( error_test40 ) ;
return False ;
}
2004-08-04 17:23:35 +04:00
fnum2 = smbcli_nt_create_full ( cli2 - > tree , fname , 0 , STD_RIGHT_DELETE_ACCESS | SA_RIGHT_FILE_READ_ATTRIBUTES , FILE_ATTRIBUTE_NORMAL ,
2003-08-13 05:53:07 +04:00
NTCREATEX_SHARE_ACCESS_NONE , NTCREATEX_DISP_OPEN_IF , 0 , 0 ) ;
if ( fnum2 ! = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " test 4 open 2 of %s SUCCEEDED - should have failed (%s) \n " , fname , smbcli_errstr ( cli2 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
CHECK_MAX_FAILURES ( error_test40 ) ;
return False ;
}
2004-08-04 17:23:35 +04:00
printf ( " test 4 open 2 of %s gave %s (correct error should be %s) \n " , fname , smbcli_errstr ( cli2 - > tree ) , " sharing violation " ) ;
2003-08-13 05:53:07 +04:00
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli1 - > tree , fnum1 ) ) ) {
printf ( " test 4 close 1 of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
printf ( " non-io open test #4 passed. \n " ) ;
error_test40 :
2004-08-04 17:23:35 +04:00
smbcli_unlink ( cli1 - > tree , fname ) ;
2003-08-13 05:53:07 +04:00
printf ( " TEST #5 testing 2 non-io opens (both with delete - both with file share delete) \n " ) ;
2004-08-04 17:23:35 +04:00
fnum1 = smbcli_nt_create_full ( cli1 - > tree , fname , 0 , STD_RIGHT_DELETE_ACCESS | SA_RIGHT_FILE_READ_ATTRIBUTES , FILE_ATTRIBUTE_NORMAL ,
2003-08-13 05:53:07 +04:00
NTCREATEX_SHARE_ACCESS_DELETE , NTCREATEX_DISP_OVERWRITE_IF , 0 , 0 ) ;
if ( fnum1 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " test 5 open 1 of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
CHECK_MAX_FAILURES ( error_test50 ) ;
return False ;
}
2004-08-04 17:23:35 +04:00
fnum2 = smbcli_nt_create_full ( cli2 - > tree , fname , 0 , STD_RIGHT_DELETE_ACCESS | SA_RIGHT_FILE_READ_ATTRIBUTES , FILE_ATTRIBUTE_NORMAL ,
2003-08-13 05:53:07 +04:00
NTCREATEX_SHARE_ACCESS_DELETE , NTCREATEX_DISP_OPEN_IF , 0 , 0 ) ;
if ( fnum2 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " test 5 open 2 of %s failed (%s) \n " , fname , smbcli_errstr ( cli2 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
CHECK_MAX_FAILURES ( error_test50 ) ;
return False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli1 - > tree , fnum1 ) ) ) {
printf ( " test 5 close 1 of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli2 - > tree , fnum2 ) ) ) {
printf ( " test 5 close 2 of %s failed (%s) \n " , fname , smbcli_errstr ( cli2 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
printf ( " non-io open test #5 passed. \n " ) ;
error_test50 :
printf ( " TEST #6 testing 1 non-io open, one io open \n " ) ;
2004-08-04 17:23:35 +04:00
smbcli_unlink ( cli1 - > tree , fname ) ;
2003-08-13 05:53:07 +04:00
2004-08-04 17:23:35 +04:00
fnum1 = smbcli_nt_create_full ( cli1 - > tree , fname , 0 , SA_RIGHT_FILE_READ_DATA , FILE_ATTRIBUTE_NORMAL ,
2003-08-13 05:53:07 +04:00
NTCREATEX_SHARE_ACCESS_NONE , NTCREATEX_DISP_OVERWRITE_IF , 0 , 0 ) ;
if ( fnum1 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " test 6 open 1 of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
CHECK_MAX_FAILURES ( error_test60 ) ;
return False ;
}
2004-08-04 17:23:35 +04:00
fnum2 = smbcli_nt_create_full ( cli2 - > tree , fname , 0 , SA_RIGHT_FILE_READ_ATTRIBUTES , FILE_ATTRIBUTE_NORMAL ,
2003-08-13 05:53:07 +04:00
NTCREATEX_SHARE_ACCESS_READ , NTCREATEX_DISP_OPEN_IF , 0 , 0 ) ;
if ( fnum2 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " test 6 open 2 of %s failed (%s) \n " , fname , smbcli_errstr ( cli2 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
CHECK_MAX_FAILURES ( error_test60 ) ;
return False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli1 - > tree , fnum1 ) ) ) {
printf ( " test 6 close 1 of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli2 - > tree , fnum2 ) ) ) {
printf ( " test 6 close 2 of %s failed (%s) \n " , fname , smbcli_errstr ( cli2 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
printf ( " non-io open test #6 passed. \n " ) ;
error_test60 :
printf ( " TEST #7 testing 1 non-io open, one io open with delete \n " ) ;
2004-08-04 17:23:35 +04:00
smbcli_unlink ( cli1 - > tree , fname ) ;
2003-08-13 05:53:07 +04:00
2004-08-04 17:23:35 +04:00
fnum1 = smbcli_nt_create_full ( cli1 - > tree , fname , 0 , SA_RIGHT_FILE_READ_DATA , FILE_ATTRIBUTE_NORMAL ,
2003-08-13 05:53:07 +04:00
NTCREATEX_SHARE_ACCESS_NONE , NTCREATEX_DISP_OVERWRITE_IF , 0 , 0 ) ;
if ( fnum1 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " test 7 open 1 of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
CHECK_MAX_FAILURES ( error_test70 ) ;
return False ;
}
2004-08-04 17:23:35 +04:00
fnum2 = smbcli_nt_create_full ( cli2 - > tree , fname , 0 , STD_RIGHT_DELETE_ACCESS | SA_RIGHT_FILE_READ_ATTRIBUTES , FILE_ATTRIBUTE_NORMAL ,
2003-08-13 05:53:07 +04:00
NTCREATEX_SHARE_ACCESS_READ | NTCREATEX_SHARE_ACCESS_DELETE , NTCREATEX_DISP_OPEN_IF , 0 , 0 ) ;
if ( fnum2 ! = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " test 7 open 2 of %s SUCCEEDED - should have failed (%s) \n " , fname , smbcli_errstr ( cli2 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
CHECK_MAX_FAILURES ( error_test70 ) ;
return False ;
}
2004-08-04 17:23:35 +04:00
printf ( " test 7 open 2 of %s gave %s (correct error should be %s) \n " , fname , smbcli_errstr ( cli2 - > tree ) , " sharing violation " ) ;
2003-08-13 05:53:07 +04:00
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli1 - > tree , fnum1 ) ) ) {
printf ( " test 7 close 1 of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
printf ( " non-io open test #7 passed. \n " ) ;
2004-05-15 02:07:43 +04:00
2003-08-13 05:53:07 +04:00
error_test70 :
2004-05-15 02:07:43 +04:00
2004-05-25 03:40:50 +04:00
printf ( " TEST #8 testing one normal open, followed by lock, followed by open with truncate \n " ) ;
2004-08-04 17:23:35 +04:00
smbcli_unlink ( cli1 - > tree , fname ) ;
2004-05-25 03:40:50 +04:00
2004-08-04 17:23:35 +04:00
fnum1 = smbcli_open ( cli1 - > tree , fname , O_RDWR | O_CREAT , DENY_NONE ) ;
2004-05-25 03:40:50 +04:00
if ( fnum1 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " (8) open (1) of %s failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2004-05-25 03:40:50 +04:00
return False ;
}
/* write 20 bytes. */
memset ( buf , ' \0 ' , 20 ) ;
2004-08-04 17:23:35 +04:00
if ( smbcli_write ( cli1 - > tree , fnum1 , 0 , buf , 0 , 20 ) ! = 20 ) {
printf ( " (8) write failed (%s) \n " , smbcli_errstr ( cli1 - > tree ) ) ;
2004-05-25 03:40:50 +04:00
correct = False ;
}
/* Ensure size == 20. */
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_getatr ( cli1 - > tree , fname , NULL , & fsize , NULL ) ) ) {
printf ( " (8) getatr (1) failed (%s) \n " , smbcli_errstr ( cli1 - > tree ) ) ;
2004-05-25 03:40:50 +04:00
CHECK_MAX_FAILURES ( error_test80 ) ;
return False ;
}
if ( fsize ! = 20 ) {
printf ( " (8) file size != 20 \n " ) ;
CHECK_MAX_FAILURES ( error_test80 ) ;
return False ;
}
/* Get an exclusive lock on the open file. */
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_lock ( cli1 - > tree , fnum1 , 0 , 4 , 0 , WRITE_LOCK ) ) ) {
printf ( " (8) lock1 failed (%s) \n " , smbcli_errstr ( cli1 - > tree ) ) ;
2004-05-25 03:40:50 +04:00
CHECK_MAX_FAILURES ( error_test80 ) ;
return False ;
}
2004-08-04 17:23:35 +04:00
fnum2 = smbcli_open ( cli1 - > tree , fname , O_RDWR | O_TRUNC , DENY_NONE ) ;
2004-05-25 03:40:50 +04:00
if ( fnum1 = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " (8) open (2) of %s with truncate failed (%s) \n " , fname , smbcli_errstr ( cli1 - > tree ) ) ;
2004-05-25 03:40:50 +04:00
return False ;
}
/* Ensure size == 0. */
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_getatr ( cli1 - > tree , fname , NULL , & fsize , NULL ) ) ) {
printf ( " (8) getatr (2) failed (%s) \n " , smbcli_errstr ( cli1 - > tree ) ) ;
2004-05-25 03:40:50 +04:00
CHECK_MAX_FAILURES ( error_test80 ) ;
return False ;
}
if ( fsize ! = 0 ) {
printf ( " (8) file size != 0 \n " ) ;
CHECK_MAX_FAILURES ( error_test80 ) ;
return False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli1 - > tree , fnum1 ) ) ) {
printf ( " (8) close1 failed (%s) \n " , smbcli_errstr ( cli1 - > tree ) ) ;
2004-05-25 03:40:50 +04:00
return False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_close ( cli1 - > tree , fnum2 ) ) ) {
printf ( " (8) close1 failed (%s) \n " , smbcli_errstr ( cli1 - > tree ) ) ;
2004-05-25 03:40:50 +04:00
return False ;
}
error_test80 :
printf ( " open test #8 passed. \n " ) ;
2004-08-04 17:23:35 +04:00
smbcli_unlink ( cli1 - > tree , fname ) ;
2003-08-13 05:53:07 +04:00
if ( ! torture_close_connection ( cli1 ) ) {
correct = False ;
}
if ( ! torture_close_connection ( cli2 ) ) {
correct = False ;
}
return correct ;
}
/*
sees what IOCTLs are supported
*/
2004-10-28 17:40:50 +04:00
BOOL torture_ioctl_test ( void )
2003-08-13 05:53:07 +04:00
{
2004-08-04 17:23:35 +04:00
struct smbcli_state * cli ;
2004-05-25 21:24:24 +04:00
uint16_t device , function ;
2003-08-13 05:53:07 +04:00
int fnum ;
const char * fname = " \\ ioctl.dat " ;
NTSTATUS status ;
2003-12-04 05:03:06 +03:00
union smb_ioctl parms ;
2003-08-13 05:53:07 +04:00
TALLOC_CTX * mem_ctx ;
if ( ! torture_open_connection ( & cli ) ) {
return False ;
}
mem_ctx = talloc_init ( " ioctl_test " ) ;
printf ( " starting ioctl test \n " ) ;
2004-08-04 17:23:35 +04:00
smbcli_unlink ( cli - > tree , fname ) ;
2003-08-13 05:53:07 +04:00
2004-08-04 17:23:35 +04:00
fnum = smbcli_open ( cli - > tree , fname , O_RDWR | O_CREAT | O_EXCL , DENY_NONE ) ;
2003-08-13 05:53:07 +04:00
if ( fnum = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " open of %s failed (%s) \n " , fname , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
2003-12-04 05:03:06 +03:00
parms . ioctl . level = RAW_IOCTL_IOCTL ;
2004-08-03 08:44:27 +04:00
parms . ioctl . in . fnum = fnum ;
2003-12-04 05:03:06 +03:00
parms . ioctl . in . request = IOCTL_QUERY_JOB_INFO ;
2003-08-13 05:53:07 +04:00
status = smb_raw_ioctl ( cli - > tree , mem_ctx , & parms ) ;
2004-08-04 17:23:35 +04:00
printf ( " ioctl job info: %s \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
for ( device = 0 ; device < 0x100 ; device + + ) {
printf ( " testing device=0x%x \n " , device ) ;
for ( function = 0 ; function < 0x100 ; function + + ) {
2003-12-04 05:03:06 +03:00
parms . ioctl . in . request = ( device < < 16 ) | function ;
2003-08-13 05:53:07 +04:00
status = smb_raw_ioctl ( cli - > tree , mem_ctx , & parms ) ;
if ( NT_STATUS_IS_OK ( status ) ) {
printf ( " ioctl device=0x%x function=0x%x OK : %d bytes \n " ,
2003-12-04 05:03:06 +03:00
device , function , parms . ioctl . out . blob . length ) ;
2003-08-13 05:53:07 +04:00
}
}
}
if ( ! torture_close_connection ( cli ) ) {
return False ;
}
return True ;
}
/*
tries variants of chkpath
*/
2004-10-28 17:40:50 +04:00
BOOL torture_chkpath_test ( void )
2003-08-13 05:53:07 +04:00
{
2004-08-04 17:23:35 +04:00
struct smbcli_state * cli ;
2003-08-13 05:53:07 +04:00
int fnum ;
BOOL ret ;
if ( ! torture_open_connection ( & cli ) ) {
return False ;
}
printf ( " starting chkpath test \n " ) ;
printf ( " Testing valid and invalid paths \n " ) ;
/* cleanup from an old run */
2004-08-04 17:23:35 +04:00
smbcli_rmdir ( cli - > tree , " \\ chkpath.dir \\ dir2 " ) ;
smbcli_unlink ( cli - > tree , " \\ chkpath.dir \\ * " ) ;
smbcli_rmdir ( cli - > tree , " \\ chkpath.dir " ) ;
2003-08-13 05:53:07 +04:00
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_mkdir ( cli - > tree , " \\ chkpath.dir " ) ) ) {
printf ( " mkdir1 failed : %s \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_mkdir ( cli - > tree , " \\ chkpath.dir \\ dir2 " ) ) ) {
printf ( " mkdir2 failed : %s \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
2004-08-04 17:23:35 +04:00
fnum = smbcli_open ( cli - > tree , " \\ chkpath.dir \\ foo.txt " , O_RDWR | O_CREAT | O_EXCL , DENY_NONE ) ;
2003-08-13 05:53:07 +04:00
if ( fnum = = - 1 ) {
2004-08-04 17:23:35 +04:00
printf ( " open1 failed (%s) \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
return False ;
}
2004-08-04 17:23:35 +04:00
smbcli_close ( cli - > tree , fnum ) ;
2003-08-13 05:53:07 +04:00
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_chkpath ( cli - > tree , " \\ chkpath.dir " ) ) ) {
printf ( " chkpath1 failed: %s \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
ret = False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_chkpath ( cli - > tree , " \\ chkpath.dir \\ dir2 " ) ) ) {
printf ( " chkpath2 failed: %s \n " , smbcli_errstr ( cli - > tree ) ) ;
2003-08-13 05:53:07 +04:00
ret = False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_chkpath ( cli - > tree , " \\ chkpath.dir \\ foo.txt " ) ) ) {
2004-10-17 06:52:37 +04:00
ret = check_error ( __location__ , cli , ERRDOS , ERRbadpath ,
2003-08-13 05:53:07 +04:00
NT_STATUS_NOT_A_DIRECTORY ) ;
} else {
printf ( " * chkpath on a file should fail \n " ) ;
ret = False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_chkpath ( cli - > tree , " \\ chkpath.dir \\ bar.txt " ) ) ) {
2004-10-17 06:52:37 +04:00
ret = check_error ( __location__ , cli , ERRDOS , ERRbadfile ,
2003-08-13 05:53:07 +04:00
NT_STATUS_OBJECT_NAME_NOT_FOUND ) ;
} else {
printf ( " * chkpath on a non existent file should fail \n " ) ;
ret = False ;
}
2004-08-04 17:23:35 +04:00
if ( NT_STATUS_IS_ERR ( smbcli_chkpath ( cli - > tree , " \\ chkpath.dir \\ dirxx \\ bar.txt " ) ) ) {
2004-10-17 06:52:37 +04:00
ret = check_error ( __location__ , cli , ERRDOS , ERRbadpath ,
2003-08-13 05:53:07 +04:00
NT_STATUS_OBJECT_PATH_NOT_FOUND ) ;
} else {
printf ( " * chkpath on a non existent component should fail \n " ) ;
ret = False ;
}
2004-08-04 17:23:35 +04:00
smbcli_rmdir ( cli - > tree , " \\ chkpath.dir \\ dir2 " ) ;
smbcli_unlink ( cli - > tree , " \\ chkpath.dir \\ * " ) ;
smbcli_rmdir ( cli - > tree , " \\ chkpath.dir " ) ;
2003-08-13 05:53:07 +04:00
if ( ! torture_close_connection ( cli ) ) {
return False ;
}
return ret ;
}
2004-03-09 12:04:06 +03:00
/*
parse a //server/share type UNC name
*/
static BOOL parse_unc ( const char * unc_name , char * * hostname , char * * sharename )
{
char * p ;
if ( strncmp ( unc_name , " // " , 2 ) ) {
return False ;
}
* hostname = strdup ( & unc_name [ 2 ] ) ;
p = strchr_m ( & ( * hostname ) [ 2 ] , ' / ' ) ;
if ( ! p ) {
return False ;
}
* p = 0 ;
* sharename = strdup ( p + 1 ) ;
return True ;
}
2004-10-28 17:19:39 +04:00
static void sigcont ( int sig )
2003-10-29 07:21:58 +03:00
{
}
2004-08-04 17:23:35 +04:00
double torture_create_procs ( BOOL ( * fn ) ( struct smbcli_state * , int ) , BOOL * result )
2003-08-13 05:53:07 +04:00
{
int i , status ;
volatile pid_t * child_status ;
volatile BOOL * child_status_out ;
int synccount ;
int tries = 8 ;
2004-03-09 05:13:13 +03:00
double start_time_limit = 10 + ( torture_nprocs * 1.5 ) ;
2004-03-09 12:04:06 +03:00
char * * unc_list = NULL ;
2004-06-14 03:50:55 +04:00
const char * p ;
2004-03-09 12:04:06 +03:00
int num_unc_names = 0 ;
2004-11-03 13:09:48 +03:00
struct timeval tv ;
2003-08-13 05:53:07 +04:00
synccount = 0 ;
2003-10-29 07:21:58 +03:00
signal ( SIGCONT , sigcont ) ;
2004-03-09 05:13:13 +03:00
child_status = ( volatile pid_t * ) shm_setup ( sizeof ( pid_t ) * torture_nprocs ) ;
2003-08-13 05:53:07 +04:00
if ( ! child_status ) {
printf ( " Failed to setup shared memory \n " ) ;
return - 1 ;
}
2004-03-09 05:13:13 +03:00
child_status_out = ( volatile BOOL * ) shm_setup ( sizeof ( BOOL ) * torture_nprocs ) ;
2003-08-13 05:53:07 +04:00
if ( ! child_status_out ) {
printf ( " Failed to setup result status shared memory \n " ) ;
return - 1 ;
}
2004-03-09 12:04:06 +03:00
p = lp_parm_string ( - 1 , " torture " , " unclist " ) ;
if ( p ) {
unc_list = file_lines_load ( p , & num_unc_names ) ;
if ( ! unc_list | | num_unc_names < = 0 ) {
2004-09-10 06:16:41 +04:00
printf ( " Failed to load unc names list from '%s' \n " , p ) ;
2004-03-09 12:04:06 +03:00
exit ( 1 ) ;
}
}
2004-03-09 05:13:13 +03:00
for ( i = 0 ; i < torture_nprocs ; i + + ) {
2003-08-13 05:53:07 +04:00
child_status [ i ] = 0 ;
child_status_out [ i ] = True ;
}
2004-11-03 13:09:48 +03:00
tv = timeval_current ( ) ;
2003-08-13 05:53:07 +04:00
2004-03-09 05:13:13 +03:00
for ( i = 0 ; i < torture_nprocs ; i + + ) {
2003-08-13 05:53:07 +04:00
procnum = i ;
if ( fork ( ) = = 0 ) {
char * myname ;
2004-03-09 12:04:06 +03:00
char * hostname = NULL , * sharename ;
2003-08-13 05:53:07 +04:00
pid_t mypid = getpid ( ) ;
sys_srandom ( ( ( int ) mypid ) ^ ( ( int ) time ( NULL ) ) ) ;
asprintf ( & myname , " CLIENT%d " , i ) ;
lp_set_cmdline ( " netbios name " , myname ) ;
free ( myname ) ;
2004-03-09 12:04:06 +03:00
if ( unc_list ) {
if ( ! parse_unc ( unc_list [ i % num_unc_names ] ,
& hostname , & sharename ) ) {
printf ( " Failed to parse UNC name %s \n " ,
unc_list [ i % num_unc_names ] ) ;
exit ( 1 ) ;
}
}
2003-08-13 05:53:07 +04:00
while ( 1 ) {
2004-03-09 12:04:06 +03:00
if ( hostname ) {
if ( torture_open_connection_share ( & current_cli ,
hostname ,
sharename ) ) {
break ;
}
} else if ( torture_open_connection ( & current_cli ) ) {
break ;
}
2003-08-13 05:53:07 +04:00
if ( tries - - = = 0 ) {
printf ( " pid %d failed to start \n " , ( int ) getpid ( ) ) ;
_exit ( 1 ) ;
}
2003-09-29 10:04:23 +04:00
msleep ( 100 ) ;
2003-08-13 05:53:07 +04:00
}
child_status [ i ] = getpid ( ) ;
2003-10-29 07:21:58 +03:00
pause ( ) ;
2003-09-29 10:04:23 +04:00
if ( child_status [ i ] ) {
printf ( " Child %d failed to start! \n " , i ) ;
child_status_out [ i ] = 1 ;
_exit ( 1 ) ;
}
2003-08-13 05:53:07 +04:00
2004-03-09 05:13:13 +03:00
child_status_out [ i ] = fn ( current_cli , i ) ;
2003-08-13 05:53:07 +04:00
_exit ( 0 ) ;
}
}
do {
synccount = 0 ;
2004-03-09 05:13:13 +03:00
for ( i = 0 ; i < torture_nprocs ; i + + ) {
2003-08-13 05:53:07 +04:00
if ( child_status [ i ] ) synccount + + ;
}
2004-03-09 05:13:13 +03:00
if ( synccount = = torture_nprocs ) break ;
2003-09-29 10:04:23 +04:00
msleep ( 100 ) ;
2004-11-03 13:09:48 +03:00
} while ( timeval_elapsed ( & tv ) < start_time_limit ) ;
2003-08-13 05:53:07 +04:00
2004-03-09 05:13:13 +03:00
if ( synccount ! = torture_nprocs ) {
printf ( " FAILED TO START %d CLIENTS (started %d) \n " , torture_nprocs , synccount ) ;
2003-08-13 05:53:07 +04:00
* result = False ;
2004-11-03 13:09:48 +03:00
return timeval_elapsed ( & tv ) ;
2003-08-13 05:53:07 +04:00
}
2004-03-09 05:13:13 +03:00
printf ( " Starting %d clients \n " , torture_nprocs ) ;
2003-09-29 10:04:23 +04:00
2003-08-13 05:53:07 +04:00
/* start the client load */
2004-11-03 13:09:48 +03:00
tv = timeval_current ( ) ;
2004-03-09 05:13:13 +03:00
for ( i = 0 ; i < torture_nprocs ; i + + ) {
2003-08-13 05:53:07 +04:00
child_status [ i ] = 0 ;
}
2003-10-29 07:21:58 +03:00
kill ( 0 , SIGCONT ) ;
2003-08-13 05:53:07 +04:00
2004-03-09 05:13:13 +03:00
printf ( " %d clients started \n " , torture_nprocs ) ;
2003-08-13 05:53:07 +04:00
2004-03-09 05:13:13 +03:00
for ( i = 0 ; i < torture_nprocs ; i + + ) {
2003-09-29 10:04:23 +04:00
int ret ;
2004-11-02 02:37:12 +03:00
while ( ( ret = sys_waitpid ( 0 , & status , 0 ) ) = = - 1 & & errno = = EINTR ) /* noop */ ;
2003-09-29 10:04:23 +04:00
if ( ret = = - 1 | | WEXITSTATUS ( status ) ! = 0 ) {
* result = False ;
}
2003-08-13 05:53:07 +04:00
}
printf ( " \n " ) ;
2004-03-09 05:13:13 +03:00
for ( i = 0 ; i < torture_nprocs ; i + + ) {
2003-08-13 05:53:07 +04:00
if ( ! child_status_out [ i ] ) {
* result = False ;
}
}
2004-11-03 13:09:48 +03:00
return timeval_elapsed ( & tv ) ;
2003-08-13 05:53:07 +04:00
}
# define FLAG_MULTIPROC 1
static struct {
const char * name ;
2004-10-28 17:40:50 +04:00
BOOL ( * fn ) ( void ) ;
BOOL ( * multi_fn ) ( struct smbcli_state * , int ) ;
2003-08-13 05:53:07 +04:00
} torture_ops [ ] = {
2004-08-03 08:44:27 +04:00
/* base tests */
{ " BASE-FDPASS " , run_fdpasstest , 0 } ,
2004-10-17 06:52:37 +04:00
{ " BASE-LOCK1 " , torture_locktest1 , 0 } ,
{ " BASE-LOCK2 " , torture_locktest2 , 0 } ,
{ " BASE-LOCK3 " , torture_locktest3 , 0 } ,
{ " BASE-LOCK4 " , torture_locktest4 , 0 } ,
{ " BASE-LOCK5 " , torture_locktest5 , 0 } ,
{ " BASE-LOCK6 " , torture_locktest6 , 0 } ,
{ " BASE-LOCK7 " , torture_locktest7 , 0 } ,
2004-11-06 12:12:53 +03:00
{ " BASE-UNLINK " , torture_unlinktest , 0 } ,
2004-08-03 08:44:27 +04:00
{ " BASE-ATTR " , run_attrtest , 0 } ,
{ " BASE-TRANS2 " , run_trans2test , 0 } ,
{ " BASE-NEGNOWAIT " , run_negprot_nowait , 0 } ,
2004-10-25 11:24:46 +04:00
{ " BASE-DIR1 " , torture_dirtest1 , 0 } ,
{ " BASE-DIR2 " , torture_dirtest2 , 0 } ,
2004-08-03 08:44:27 +04:00
{ " BASE-DENY1 " , torture_denytest1 , 0 } ,
{ " BASE-DENY2 " , torture_denytest2 , 0 } ,
2004-10-25 11:24:46 +04:00
{ " BASE-DENY3 " , torture_denytest3 , 0 } ,
2004-10-30 08:59:52 +04:00
{ " BASE-NTDENY1 " , NULL , torture_ntdenytest1 } ,
{ " BASE-NTDENY2 " , torture_ntdenytest2 , 0 } ,
2004-08-03 08:44:27 +04:00
{ " BASE-TCON " , run_tcon_test , 0 } ,
{ " BASE-TCONDEV " , run_tcon_devtype_test , 0 } ,
{ " BASE-VUID " , run_vuidtest , 0 } ,
{ " BASE-RW1 " , run_readwritetest , 0 } ,
2004-10-28 17:40:50 +04:00
{ " BASE-RW2 " , NULL , run_readwritemulti } ,
2004-08-03 08:44:27 +04:00
{ " BASE-OPEN " , run_opentest , 0 } ,
2004-10-28 17:40:50 +04:00
{ " BASE-DEFER_OPEN " , NULL , run_deferopen } ,
2004-08-03 08:44:27 +04:00
{ " BASE-XCOPY " , run_xcopy , 0 } ,
2004-10-25 06:59:48 +04:00
{ " BASE-RENAME " , torture_test_rename , 0 } ,
2004-10-24 13:08:52 +04:00
{ " BASE-DELETE " , torture_test_delete , 0 } ,
2004-08-03 08:44:27 +04:00
{ " BASE-PROPERTIES " , run_properties , 0 } ,
{ " BASE-MANGLE " , torture_mangle , 0 } ,
2004-10-26 12:32:16 +04:00
{ " BASE-OPENATTR " , torture_openattrtest , 0 } ,
2004-08-03 08:44:27 +04:00
{ " BASE-CHARSET " , torture_charset , 0 } ,
{ " BASE-CHKPATH " , torture_chkpath_test , 0 } ,
2004-09-24 10:56:11 +04:00
{ " BASE-SECLEAK " , torture_sec_leak , 0 } ,
2004-08-03 08:44:27 +04:00
/* benchmarking tests */
{ " BENCH-HOLDCON " , torture_holdcon , 0 } ,
{ " BENCH-NBENCH " , torture_nbench , 0 } ,
2004-10-28 17:40:50 +04:00
{ " BENCH-TORTURE " , NULL , run_torture } ,
2004-08-03 08:44:27 +04:00
/* RAW smb tests */
2003-08-13 05:53:07 +04:00
{ " RAW-QFSINFO " , torture_raw_qfsinfo , 0 } ,
{ " RAW-QFILEINFO " , torture_raw_qfileinfo , 0 } ,
{ " RAW-SFILEINFO " , torture_raw_sfileinfo , 0 } ,
{ " RAW-SFILEINFO-BUG " , torture_raw_sfileinfo_bug , 0 } ,
{ " RAW-SEARCH " , torture_raw_search , 0 } ,
{ " RAW-CLOSE " , torture_raw_close , 0 } ,
{ " RAW-OPEN " , torture_raw_open , 0 } ,
{ " RAW-MKDIR " , torture_raw_mkdir , 0 } ,
{ " RAW-OPLOCK " , torture_raw_oplock , 0 } ,
{ " RAW-NOTIFY " , torture_raw_notify , 0 } ,
{ " RAW-MUX " , torture_raw_mux , 0 } ,
{ " RAW-IOCTL " , torture_raw_ioctl , 0 } ,
{ " RAW-CHKPATH " , torture_raw_chkpath , 0 } ,
{ " RAW-UNLINK " , torture_raw_unlink , 0 } ,
{ " RAW-READ " , torture_raw_read , 0 } ,
{ " RAW-WRITE " , torture_raw_write , 0 } ,
{ " RAW-LOCK " , torture_raw_lock , 0 } ,
{ " RAW-CONTEXT " , torture_raw_context , 0 } ,
{ " RAW-RENAME " , torture_raw_rename , 0 } ,
{ " RAW-SEEK " , torture_raw_seek , 0 } ,
2004-07-12 20:35:48 +04:00
{ " RAW-RAP " , torture_raw_rap , 0 } ,
2004-08-03 08:44:27 +04:00
/* protocol scanners */
2003-08-13 05:53:07 +04:00
{ " SCAN-TRANS2 " , torture_trans2_scan , 0 } ,
{ " SCAN-NTTRANS " , torture_nttrans_scan , 0 } ,
{ " SCAN-ALIASES " , torture_trans2_aliases , 0 } ,
2003-08-14 02:23:18 +04:00
{ " SCAN-SMB " , torture_smb_scan , 0 } ,
2004-10-28 17:40:50 +04:00
{ " SCAN-MAXFID " , NULL , run_maxfidtest } ,
2004-08-03 08:44:27 +04:00
{ " SCAN-UTABLE " , torture_utable , 0 } ,
{ " SCAN-CASETABLE " , torture_casetable , 0 } ,
{ " SCAN-PIPE_NUMBER " , run_pipe_number , 0 } ,
{ " SCAN-IOCTL " , torture_ioctl_test , 0 } ,
/* rpc testers */
2003-11-03 09:22:45 +03:00
{ " RPC-LSA " , torture_rpc_lsa , 0 } ,
2003-11-03 10:26:30 +03:00
{ " RPC-ECHO " , torture_rpc_echo , 0 } ,
2003-11-13 12:26:53 +03:00
{ " RPC-DFS " , torture_rpc_dfs , 0 } ,
2003-11-15 08:42:49 +03:00
{ " RPC-SPOOLSS " , torture_rpc_spoolss , 0 } ,
2003-11-15 09:00:21 +03:00
{ " RPC-SAMR " , torture_rpc_samr , 0 } ,
2003-12-01 04:41:38 +03:00
{ " RPC-NETLOGON " , torture_rpc_netlogon , 0 } ,
2004-06-06 11:14:10 +04:00
{ " RPC-SCHANNEL " , torture_rpc_schannel , 0 } ,
2003-11-17 15:43:18 +03:00
{ " RPC-WKSSVC " , torture_rpc_wkssvc , 0 } ,
2003-11-19 12:15:46 +03:00
{ " RPC-SRVSVC " , torture_rpc_srvsvc , 0 } ,
2004-08-02 04:24:04 +04:00
{ " RPC-SVCCTL " , torture_rpc_svcctl , 0 } ,
2003-11-20 06:27:56 +03:00
{ " RPC-ATSVC " , torture_rpc_atsvc , 0 } ,
2003-11-21 00:52:40 +03:00
{ " RPC-EVENTLOG " , torture_rpc_eventlog , 0 } ,
2003-11-21 16:14:17 +03:00
{ " RPC-EPMAPPER " , torture_rpc_epmapper , 0 } ,
2003-11-21 08:28:36 +03:00
{ " RPC-WINREG " , torture_rpc_winreg , 0 } ,
2004-09-28 23:20:00 +04:00
{ " RPC-OXIDRESOLVE " , torture_rpc_oxidresolve , 0 } ,
2004-10-31 21:37:59 +03:00
{ " RPC-REMACT " , torture_rpc_remact , 0 } ,
2003-11-24 04:24:29 +03:00
{ " RPC-MGMT " , torture_rpc_mgmt , 0 } ,
2003-11-27 08:34:28 +03:00
{ " RPC-SCANNER " , torture_rpc_scanner , 0 } ,
2003-11-28 11:51:09 +03:00
{ " RPC-AUTOIDL " , torture_rpc_autoidl , 0 } ,
2004-10-08 05:15:25 +04:00
{ " RPC-COUNTCALLS " , torture_rpc_countcalls , 0 } ,
2004-06-17 04:31:24 +04:00
{ " RPC-MULTIBIND " , torture_multi_bind , 0 } ,
2004-07-13 22:05:02 +04:00
{ " RPC-DRSUAPI " , torture_rpc_drsuapi , 0 } ,
2004-08-03 08:44:27 +04:00
2004-11-03 23:32:28 +03:00
/* Distributed COM testers */
{ " DCOM-SIMPLE " , torture_dcom_simple , 0 } ,
2004-09-01 08:39:06 +04:00
/* local (no server) testers */
{ " LOCAL-NTLMSSP " , torture_ntlmssp_self_check , 0 } ,
{ " LOCAL-ICONV " , torture_local_iconv , 0 } ,
2004-09-28 09:42:02 +04:00
{ " LOCAL-TALLOC " , torture_local_talloc , 0 } ,
2004-10-17 14:04:49 +04:00
{ " LOCAL-MESSAGING " , torture_local_messaging , 0 } ,
2004-10-24 17:00:51 +04:00
{ " LOCAL-BINDING " , torture_local_binding_string , 0 } ,
2004-10-22 10:46:04 +04:00
{ " LOCAL-IDTREE " , torture_local_idtree , 0 } ,
2004-08-03 08:44:27 +04:00
2004-08-12 12:00:45 +04:00
/* ldap testers */
{ " LDAP-BASIC " , torture_ldap_basic , 0 } ,
2003-08-13 05:53:07 +04:00
{ NULL , NULL , 0 } } ;
/****************************************************************************
run a specified test or " ALL "
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL run_test ( const char * name )
{
BOOL ret = True ;
int i ;
BOOL matched = False ;
if ( strequal ( name , " ALL " ) ) {
for ( i = 0 ; torture_ops [ i ] . name ; i + + ) {
if ( ! run_test ( torture_ops [ i ] . name ) ) {
ret = False ;
}
}
return ret ;
}
for ( i = 0 ; torture_ops [ i ] . name ; i + + ) {
if ( gen_fnmatch ( name , torture_ops [ i ] . name ) = = 0 ) {
double t ;
matched = True ;
2004-10-02 16:18:59 +04:00
init_iconv ( ) ;
2003-08-13 05:53:07 +04:00
printf ( " Running %s \n " , torture_ops [ i ] . name ) ;
2004-10-28 17:40:50 +04:00
if ( torture_ops [ i ] . multi_fn ) {
2003-08-13 05:53:07 +04:00
BOOL result ;
2004-10-28 17:40:50 +04:00
t = torture_create_procs ( torture_ops [ i ] . multi_fn ,
& result ) ;
2003-08-13 05:53:07 +04:00
if ( ! result ) {
ret = False ;
printf ( " TEST %s FAILED! \n " , torture_ops [ i ] . name ) ;
}
} else {
2004-11-03 13:09:48 +03:00
struct timeval tv = timeval_current ( ) ;
2004-10-28 17:40:50 +04:00
if ( ! torture_ops [ i ] . fn ( ) ) {
2003-08-13 05:53:07 +04:00
ret = False ;
printf ( " TEST %s FAILED! \n " , torture_ops [ i ] . name ) ;
}
2004-11-03 13:09:48 +03:00
t = timeval_elapsed ( & tv ) ;
2003-08-13 05:53:07 +04:00
}
printf ( " %s took %g secs \n \n " , torture_ops [ i ] . name , t ) ;
}
}
if ( ! matched ) {
printf ( " Unknown torture operation '%s' \n " , name ) ;
}
return ret ;
}
2004-08-13 02:25:01 +04:00
static void parse_dns ( const char * dns )
{
char * userdn , * basedn , * secret ;
char * p , * d ;
/* retrievieng the userdn */
p = strchr_m ( dns , ' # ' ) ;
if ( ! p ) {
lp_set_cmdline ( " torture:ldap_userdn " , " " ) ;
lp_set_cmdline ( " torture:ldap_basedn " , " " ) ;
lp_set_cmdline ( " torture:ldap_secret " , " " ) ;
return ;
}
userdn = strndup ( dns , p - dns ) ;
lp_set_cmdline ( " torture:ldap_userdn " , userdn ) ;
/* retrieve the basedn */
d = p + 1 ;
p = strchr_m ( d , ' # ' ) ;
if ( ! p ) {
lp_set_cmdline ( " torture:ldap_basedn " , " " ) ;
lp_set_cmdline ( " torture:ldap_secret " , " " ) ;
return ;
}
basedn = strndup ( d , p - d ) ;
lp_set_cmdline ( " torture:ldap_basedn " , basedn ) ;
/* retrieve the secret */
p = p + 1 ;
if ( ! p ) {
lp_set_cmdline ( " torture:ldap_secret " , " " ) ;
return ;
}
secret = strdup ( p ) ;
lp_set_cmdline ( " torture:ldap_secret " , secret ) ;
printf ( " %s - %s - %s \n " , userdn , basedn , secret ) ;
}
2004-08-25 18:31:59 +04:00
static void usage ( poptContext pc )
2003-08-13 05:53:07 +04:00
{
int i ;
2004-08-25 18:31:59 +04:00
int perline = 5 ;
2003-08-13 05:53:07 +04:00
2004-08-25 18:31:59 +04:00
poptPrintUsage ( pc , stdout , 0 ) ;
printf ( " \n " ) ;
2003-08-13 05:53:07 +04:00
printf ( " tests are: " ) ;
for ( i = 0 ; torture_ops [ i ] . name ; i + + ) {
2004-08-25 18:31:59 +04:00
if ( ( i % perline ) = = 0 ) {
printf ( " \n " ) ;
}
printf ( " %s " , torture_ops [ i ] . name ) ;
2003-08-13 05:53:07 +04:00
}
2004-08-25 18:31:59 +04:00
printf ( " \n \n " ) ;
2003-08-13 05:53:07 +04:00
printf ( " default test is ALL \n " ) ;
exit ( 1 ) ;
}
2004-10-18 19:33:34 +04:00
static BOOL is_binding_string ( const char * binding_string )
{
TALLOC_CTX * mem_ctx = talloc_init ( " is_binding_string " ) ;
struct dcerpc_binding binding_struct ;
NTSTATUS status ;
status = dcerpc_parse_binding ( mem_ctx , binding_string , & binding_struct ) ;
talloc_destroy ( mem_ctx ) ;
return NT_STATUS_IS_OK ( status ) ;
}
2003-08-13 05:53:07 +04:00
/****************************************************************************
main program
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
int main ( int argc , char * argv [ ] )
{
int opt , i ;
char * p ;
BOOL correct = True ;
2004-08-25 18:31:59 +04:00
int argc_new ;
char * * argv_new ;
poptContext pc ;
2004-09-10 06:16:41 +04:00
enum { OPT_LOADFILE = 1000 , OPT_UNCLIST , OPT_TIMELIMIT , OPT_DNS , OPT_DANGEROUS } ;
2004-08-25 18:31:59 +04:00
struct poptOption long_options [ ] = {
POPT_AUTOHELP
{ " smb-ports " , ' p ' , POPT_ARG_STRING , NULL , 0 , " SMB ports " , NULL } ,
2004-10-30 08:59:52 +04:00
{ " seed " , 0 , POPT_ARG_INT , & torture_seed , 0 , " seed " , NULL } ,
2004-09-10 07:39:11 +04:00
{ " num-progs " , 0 , POPT_ARG_INT , & torture_nprocs , 0 , " num progs " , NULL } ,
{ " num-ops " , 0 , POPT_ARG_INT , & torture_numops , 0 , " num ops " , NULL } ,
{ " entries " , 0 , POPT_ARG_INT , & torture_entries , 0 , " entries " , NULL } ,
2004-09-10 06:16:41 +04:00
{ " use-oplocks " , ' L ' , POPT_ARG_NONE , & use_oplocks , 0 , " use oplocks " , NULL } ,
2004-09-10 07:39:11 +04:00
{ " show-all " , 0 , POPT_ARG_NONE , & torture_showall , 0 , " show all " , NULL } ,
2004-09-10 06:16:41 +04:00
{ " loadfile " , 0 , POPT_ARG_STRING , NULL , OPT_LOADFILE , " loadfile " , NULL } ,
{ " unclist " , 0 , POPT_ARG_STRING , NULL , OPT_UNCLIST , " unclist " , NULL } ,
{ " timelimit " , ' t ' , POPT_ARG_STRING , NULL , OPT_TIMELIMIT , " timelimit " , NULL } ,
2004-09-10 07:39:11 +04:00
{ " failures " , ' f ' , POPT_ARG_INT , & torture_failures , 0 , " failures " , NULL } ,
2004-09-10 06:16:41 +04:00
{ " parse-dns " , ' D ' , POPT_ARG_STRING , NULL , OPT_DNS , " parse-dns " , NULL } ,
{ " dangerous " , ' X ' , POPT_ARG_NONE , NULL , OPT_DANGEROUS , " dangerous " , NULL } ,
2004-08-25 18:31:59 +04:00
POPT_COMMON_SAMBA
POPT_COMMON_CONNECTION
POPT_COMMON_CREDENTIALS
POPT_COMMON_VERSION
POPT_TABLEEND
} ;
2003-08-13 05:53:07 +04:00
setup_logging ( " smbtorture " , DEBUG_STDOUT ) ;
# ifdef HAVE_SETBUFFER
setbuffer ( stdout , NULL , 0 ) ;
# endif
2004-08-25 18:31:59 +04:00
pc = poptGetContext ( " smbtorture " , argc , ( const char * * ) argv , long_options ,
POPT_CONTEXT_KEEP_FIRST ) ;
poptSetOtherOptionHelp ( pc , " <binding>|<unc> TEST1 TEST2 ... " ) ;
while ( ( opt = poptGetNextOpt ( pc ) ) ! = - 1 ) {
switch ( opt ) {
2004-09-10 06:16:41 +04:00
case OPT_LOADFILE :
2004-08-25 18:31:59 +04:00
lp_set_cmdline ( " torture:loadfile " , poptGetOptArg ( pc ) ) ;
break ;
2004-09-10 06:16:41 +04:00
case OPT_UNCLIST :
2004-08-25 18:31:59 +04:00
lp_set_cmdline ( " torture:unclist " , poptGetOptArg ( pc ) ) ;
break ;
2004-09-10 06:16:41 +04:00
case OPT_TIMELIMIT :
2004-08-25 18:31:59 +04:00
lp_set_cmdline ( " torture:timelimit " , poptGetOptArg ( pc ) ) ;
break ;
2004-09-10 06:16:41 +04:00
case OPT_DNS :
2004-08-25 18:31:59 +04:00
parse_dns ( poptGetOptArg ( pc ) ) ;
break ;
2004-09-10 06:16:41 +04:00
case OPT_DANGEROUS :
2004-08-25 18:31:59 +04:00
lp_set_cmdline ( " torture:dangerous " , " 1 " ) ;
break ;
default :
d_printf ( " Invalid option %s: %s \n " ,
poptBadOption ( pc , 0 ) , poptStrerror ( opt ) ) ;
usage ( pc ) ;
exit ( 1 ) ;
}
}
2003-08-13 05:53:07 +04:00
lp_load ( dyn_CONFIGFILE , True , False , False ) ;
load_interfaces ( ) ;
2004-10-30 08:59:52 +04:00
if ( torture_seed = = 0 ) {
torture_seed = time ( NULL ) ;
}
printf ( " Using seed %d \n " , torture_seed ) ;
srandom ( torture_seed ) ;
2004-08-25 18:31:59 +04:00
2004-10-28 17:40:50 +04:00
argv_new = discard_const_p ( char * , poptGetArgs ( pc ) ) ;
2003-08-13 05:53:07 +04:00
2004-08-25 18:31:59 +04:00
argc_new = argc ;
for ( i = 0 ; i < argc ; i + + ) {
if ( argv_new [ i ] = = NULL ) {
argc_new = i ;
break ;
}
2003-08-13 05:53:07 +04:00
}
2004-08-25 18:31:59 +04:00
if ( argc_new < 3 ) {
usage ( pc ) ;
exit ( 1 ) ;
}
2003-08-13 05:53:07 +04:00
2004-08-25 18:31:59 +04:00
for ( p = argv_new [ 1 ] ; * p ; p + + ) {
if ( * p = = ' \\ ' )
* p = ' / ' ;
}
2003-11-24 16:19:00 +03:00
/* see if its a RPC transport specifier */
2004-10-18 19:33:34 +04:00
if ( is_binding_string ( argv_new [ 1 ] ) ) {
2004-08-25 18:31:59 +04:00
lp_set_cmdline ( " torture:binding " , argv_new [ 1 ] ) ;
2003-11-24 16:19:00 +03:00
} else {
2004-02-04 00:53:51 +03:00
char * binding = NULL ;
2004-08-25 18:31:59 +04:00
char * host = NULL , * share = NULL ;
2003-11-24 16:19:00 +03:00
2004-08-25 18:31:59 +04:00
if ( ! parse_unc ( argv_new [ 1 ] , & host , & share ) ) {
2004-10-31 02:10:14 +04:00
d_printf ( " Invalid option: %s is not a valid torture target (share or binding string) \n \n " , argv_new [ 1 ] ) ;
2004-08-25 18:31:59 +04:00
usage ( pc ) ;
2003-11-24 16:19:00 +03:00
}
2004-03-09 12:04:06 +03:00
2003-11-24 16:19:00 +03:00
lp_set_cmdline ( " torture:host " , host ) ;
lp_set_cmdline ( " torture:share " , share ) ;
2004-02-04 00:53:51 +03:00
asprintf ( & binding , " ncacn_np:%s " , host ) ;
lp_set_cmdline ( " torture:binding " , binding ) ;
2003-08-13 05:53:07 +04:00
}
2004-08-25 18:31:59 +04:00
if ( ! lp_parm_string ( - 1 , " torture " , " username " ) ) {
lp_set_cmdline ( " torture:username " , cmdline_get_username ( ) ) ;
2003-08-13 05:53:07 +04:00
}
2004-08-25 18:31:59 +04:00
if ( ! lp_parm_string ( - 1 , " torture " , " userdomain " ) ) {
/*
* backward compatibility
* maybe we should remove this to make this consistent
* for all cmdline tools
* - - metze
*/
if ( strequal ( lp_netbios_name ( ) , cmdline_get_userdomain ( ) ) ) {
cmdline_set_userdomain ( lp_workgroup ( ) ) ;
2003-08-13 05:53:07 +04:00
}
2004-08-25 18:31:59 +04:00
lp_set_cmdline ( " torture:userdomain " , cmdline_get_userdomain ( ) ) ;
2003-08-13 05:53:07 +04:00
}
2004-05-22 18:40:49 +04:00
if ( ! lp_parm_string ( - 1 , " torture " , " password " ) ) {
2004-08-25 18:31:59 +04:00
lp_set_cmdline ( " torture:password " , cmdline_get_userpassword ( ) ) ;
2004-05-22 18:40:49 +04:00
}
2004-08-25 18:31:59 +04:00
if ( argc_new = = 0 ) {
2003-08-13 05:53:07 +04:00
printf ( " You must specify a test to run, or 'ALL' \n " ) ;
} else {
2004-08-25 18:31:59 +04:00
for ( i = 2 ; i < argc_new ; i + + ) {
if ( ! run_test ( argv_new [ i ] ) ) {
2003-08-13 05:53:07 +04:00
correct = False ;
}
}
}
if ( correct ) {
return ( 0 ) ;
} else {
return ( 1 ) ;
}
}