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-30 08:37:57 +03:00
# include "clilist.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"
2005-02-10 08:09:35 +03:00
# include "system/filesys.h"
2004-11-02 09:42:15 +03:00
# include "ioctl.h"
2004-11-30 07:33:27 +03:00
# include "librpc/gen_ndr/ndr_security.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 ;
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
{
2005-01-21 14:18:56 +03:00
struct nbt_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
2005-05-22 14:23:01 +04:00
make_nbt_name_client ( & calling , lp_netbios_name ( ) ) ;
2005-01-21 14:18:56 +03:00
nbt_choose_called_name ( NULL , & called , host , NBT_NAME_SERVER ) ;
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 ;
}
2004-08-04 17:23:35 +04:00
if ( ! smbcli_transport_establish ( cli , & calling , & called ) ) {
2005-01-21 14:18:56 +03:00
printf ( " %s rejected the session \n " , host ) ;
smbcli_shutdown ( cli ) ;
return NULL ;
2003-08-13 05:53:07 +04:00
}
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
{
NTSTATUS status ;
2004-09-28 09:44:59 +04:00
status = smbcli_full_connection ( NULL ,
r6028: A MAJOR update to intergrate the new credentails system fully with
GENSEC, and to pull SCHANNEL into GENSEC, by making it less 'special'.
GENSEC now no longer has it's own handling of 'set username' etc,
instead it uses cli_credentials calls.
In order to link the credentails code right though Samba, a lot of
interfaces have changed to remove 'username, domain, password'
arguments, and these have been replaced with a single 'struct
cli_credentials'.
In the session setup code, a new parameter 'workgroup' contains the
client/server current workgroup, which seems unrelated to the
authentication exchange (it was being filled in from the auth info).
This allows in particular kerberos to only call back for passwords
when it actually needs to perform the kinit.
The kerberos code has been modified not to use the SPNEGO provided
'principal name' (in the mechListMIC), but to instead use the name the
host was connected to as. This better matches Microsoft behaviour,
is more secure and allows better use of standard kerberos functions.
To achieve this, I made changes to our socket code so that the
hostname (before name resolution) is now recorded on the socket.
In schannel, most of the code from librpc/rpc/dcerpc_schannel.c is now
in libcli/auth/schannel.c, and it looks much more like a standard
GENSEC module. The actual sign/seal code moved to
libcli/auth/schannel_sign.c in a previous commit.
The schannel credentails structure is now merged with the rest of the
credentails, as many of the values (username, workstation, domain)
where already present there. This makes handling this in a generic
manner much easier, as there is no longer a custom entry-point.
The auth_domain module continues to be developed, but is now just as
functional as auth_winbind. The changes here are consequential to the
schannel changes.
The only removed function at this point is the RPC-LOGIN test
(simulating the load of a WinXP login), which needs much more work to
clean it up (it contains copies of too much code from all over the
torture suite, and I havn't been able to penetrate its 'structure').
Andrew Bartlett
(This used to be commit 2301a4b38a21aa60917973451687063d83d18d66)
2005-03-24 07:14:06 +03:00
c , hostname ,
2005-01-15 14:58:52 +03:00
sharename , NULL ,
2005-03-22 02:35:58 +03:00
cmdline_credentials ) ;
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 ;
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 ;
}
2004-09-02 15:26:58 +04:00
/* open a rpc connection to the chosen binding string */
2005-03-22 11:00:45 +03:00
NTSTATUS torture_rpc_connection ( TALLOC_CTX * parent_ctx ,
struct dcerpc_pipe * * p ,
2003-11-24 16:19:00 +03:00
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
}
2005-03-22 11:00:45 +03:00
status = dcerpc_pipe_connect ( parent_ctx ,
p , binding , pipe_uuid , pipe_version ,
cmdline_credentials ) ;
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 */
2005-03-22 11:00:45 +03:00
NTSTATUS torture_rpc_connection_transport ( TALLOC_CTX * parent_ctx ,
struct dcerpc_pipe * * p ,
2004-09-03 12:28:24 +04:00
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 " ) ;
r5902: A rather large change...
I wanted to add a simple 'workstation' argument to the DCERPC
authenticated binding calls, but this patch kind of grew from there.
With SCHANNEL, the 'workstation' name (the netbios name of the client)
matters, as this is what ties the session between the NETLOGON ops and
the SCHANNEL bind. This changes a lot of files, and these will again
be changed when jelmer does the credentials work.
I also correct some schannel IDL to distinguish between workstation
names and account names. The distinction matters for domain trust
accounts.
Issues in handling this (issues with lifetime of talloc pointers)
caused me to change the 'creds_CredentialsState' and 'struct
dcerpc_binding' pointers to always be talloc()ed pointers.
In the schannel DB, we now store both the domain and computername, and
query on both. This should ensure we fault correctly when the domain
is specified incorrectly in the SCHANNEL bind.
In the RPC-SCHANNEL test, I finally fixed a bug that vl pointed out,
where the comment claimed we re-used a connection, but in fact we made
a new connection.
This was achived by breaking apart some of the
dcerpc_secondary_connection() logic.
The addition of workstation handling was also propogated to NTLMSSP
and GENSEC, for completeness.
The RPC-SAMSYNC test has been cleaned up a little, using a loop over
usernames/passwords rather than manually expanded tests. This will be
expanded further (the code in #if 0 in this patch) to use a newly
created user account for testing.
In making this test pass test_rpc.sh, I found a bug in the RPC-ECHO
server, caused by the removal of [ref] and the assoicated pointer from
the IDL. This has been re-added, until the underlying pidl issues are
solved.
(This used to be commit 824289dcc20908ddec957a4a892a103eec2da9b9)
2005-03-19 11:34:43 +03:00
struct dcerpc_binding * b ;
2005-03-22 11:00:45 +03:00
TALLOC_CTX * mem_ctx = talloc_named ( parent_ctx , 0 , " torture_rpc_connection_smb " ) ;
2004-09-02 15:26:58 +04:00
if ( ! binding ) {
printf ( " You must specify a ncacn binding string \n " ) ;
2005-03-22 11:00:45 +03:00
talloc_free ( mem_ctx ) ;
2004-09-02 15:26:58 +04:00
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 ) ) ;
2005-01-27 10:08:20 +03:00
talloc_free ( mem_ctx ) ;
2004-09-02 15:26:58 +04:00
return status ;
}
r5902: A rather large change...
I wanted to add a simple 'workstation' argument to the DCERPC
authenticated binding calls, but this patch kind of grew from there.
With SCHANNEL, the 'workstation' name (the netbios name of the client)
matters, as this is what ties the session between the NETLOGON ops and
the SCHANNEL bind. This changes a lot of files, and these will again
be changed when jelmer does the credentials work.
I also correct some schannel IDL to distinguish between workstation
names and account names. The distinction matters for domain trust
accounts.
Issues in handling this (issues with lifetime of talloc pointers)
caused me to change the 'creds_CredentialsState' and 'struct
dcerpc_binding' pointers to always be talloc()ed pointers.
In the schannel DB, we now store both the domain and computername, and
query on both. This should ensure we fault correctly when the domain
is specified incorrectly in the SCHANNEL bind.
In the RPC-SCHANNEL test, I finally fixed a bug that vl pointed out,
where the comment claimed we re-used a connection, but in fact we made
a new connection.
This was achived by breaking apart some of the
dcerpc_secondary_connection() logic.
The addition of workstation handling was also propogated to NTLMSSP
and GENSEC, for completeness.
The RPC-SAMSYNC test has been cleaned up a little, using a loop over
usernames/passwords rather than manually expanded tests. This will be
expanded further (the code in #if 0 in this patch) to use a newly
created user account for testing.
In making this test pass test_rpc.sh, I found a bug in the RPC-ECHO
server, caused by the removal of [ref] and the assoicated pointer from
the IDL. This has been re-added, until the underlying pidl issues are
solved.
(This used to be commit 824289dcc20908ddec957a4a892a103eec2da9b9)
2005-03-19 11:34:43 +03:00
b - > transport = transport ;
2004-09-02 15:26:58 +04:00
2005-03-22 11:00:45 +03:00
status = dcerpc_pipe_connect_b ( mem_ctx , p , b , pipe_uuid , pipe_version ,
2005-03-22 00:22:07 +03:00
cmdline_credentials ) ;
2005-03-22 11:00:45 +03:00
if ( NT_STATUS_IS_OK ( status ) ) {
* p = talloc_reference ( parent_ctx , * p ) ;
} else {
* p = NULL ;
}
talloc_free ( mem_ctx ) ;
2004-09-02 15:26:58 +04:00
return status ;
}
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 ;
2004-12-04 16:56:25 +03:00
uint8_t buf [ 1024 ] ;
2003-08-13 05:53:07 +04:00
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 + + ) {
2005-01-21 16:16:33 +03:00
uint_t n = ( uint_t ) 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-12-04 16:56:25 +03:00
if ( smbcli_write ( c - > tree , fnum , 0 , & pid , 0 , sizeof ( pid ) ) ! = sizeof ( pid ) ) {
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 ;
}
for ( j = 0 ; j < 50 ; j + + ) {
2004-12-04 16:56:25 +03:00
if ( smbcli_write ( c - > tree , fnum , 0 , 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-12-04 16:56:25 +03:00
if ( smbcli_read ( c - > tree , fnum , & pid2 , 0 , sizeof ( pid ) ) ! = sizeof ( pid ) ) {
2004-08-04 17:23:35 +04:00
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_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 + + )
{
2005-01-21 16:16:33 +03:00
size_t buf_size = ( ( uint_t ) 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 ;
}
2005-02-10 10:39:14 +03:00
# define BOOLSTR(b) ((b) ? "Yes" : "No")
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 ) ;
}
/*
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 ;
2004-12-04 16:56:25 +03:00
uint8_t buf [ 4 ] ;
2003-08-13 05:53:07 +04:00
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
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 " ) ;
2003-08-13 05:53:07 +04:00
2004-09-28 09:44:59 +04:00
status = smbcli_full_connection ( NULL ,
r6028: A MAJOR update to intergrate the new credentails system fully with
GENSEC, and to pull SCHANNEL into GENSEC, by making it less 'special'.
GENSEC now no longer has it's own handling of 'set username' etc,
instead it uses cli_credentials calls.
In order to link the credentails code right though Samba, a lot of
interfaces have changed to remove 'username, domain, password'
arguments, and these have been replaced with a single 'struct
cli_credentials'.
In the session setup code, a new parameter 'workgroup' contains the
client/server current workgroup, which seems unrelated to the
authentication exchange (it was being filled in from the auth info).
This allows in particular kerberos to only call back for passwords
when it actually needs to perform the kinit.
The kerberos code has been modified not to use the SPNEGO provided
'principal name' (in the mechListMIC), but to instead use the name the
host was connected to as. This better matches Microsoft behaviour,
is more secure and allows better use of standard kerberos functions.
To achieve this, I made changes to our socket code so that the
hostname (before name resolution) is now recorded on the socket.
In schannel, most of the code from librpc/rpc/dcerpc_schannel.c is now
in libcli/auth/schannel.c, and it looks much more like a standard
GENSEC module. The actual sign/seal code moved to
libcli/auth/schannel_sign.c in a previous commit.
The schannel credentails structure is now merged with the rest of the
credentails, as many of the values (username, workstation, domain)
where already present there. This makes handling this in a generic
manner much easier, as there is no longer a custom entry-point.
The auth_domain module continues to be developed, but is now just as
functional as auth_winbind. The changes here are consequential to the
schannel changes.
The only removed function at this point is the RPC-LOGIN test
(simulating the load of a WinXP login), which needs much more work to
clean it up (it contains copies of too much code from all over the
torture suite, and I havn't been able to penetrate its 'structure').
Andrew Bartlett
(This used to be commit 2301a4b38a21aa60917973451687063d83d18d66)
2005-03-24 07:14:06 +03:00
& cli1 , host ,
2005-01-15 14:58:52 +03:00
share , NULL ,
2005-03-22 02:35:58 +03:00
cmdline_credentials ) ;
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 ;
2004-12-04 16:56:25 +03:00
uint8_t buf [ 1024 ] ;
2003-08-13 05:53:07 +04:00
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-11-30 07:33:27 +03:00
fnum = smbcli_nt_create_full ( cli - > tree , fname , 0 ,
2004-12-02 07:37:36 +03:00
SEC_RIGHTS_FILE_ALL ,
2004-11-30 07:33:27 +03:00
FILE_ATTRIBUTE_NORMAL ,
NTCREATEX_SHARE_ACCESS_NONE ,
NTCREATEX_DISP_OPEN_IF , 0 , 0 ) ;
2004-06-08 23:25:26 +04:00
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 ) ;
2005-01-15 14:58:52 +03:00
req = smb_raw_negotiate_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-12-04 16:56:25 +03:00
smbcli_write ( cli - > tree , fnum , 0 , & fnum , 0 , sizeof ( fnum ) ) ;
2004-08-04 17:23:35 +04:00
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 ;
}
/* FIRST_DESIRED_ACCESS 0xf019f */
2004-11-30 07:33:27 +03:00
# define FIRST_DESIRED_ACCESS SEC_FILE_READ_DATA|SEC_FILE_WRITE_DATA|SEC_FILE_APPEND_DATA|\
SEC_FILE_READ_EA | /* 0xf */ \
SEC_FILE_WRITE_EA | SEC_FILE_READ_ATTRIBUTE | /* 0x90 */ \
SEC_FILE_WRITE_ATTRIBUTE | /* 0x100 */ \
SEC_STD_DELETE | SEC_STD_READ_CONTROL | \
SEC_STD_WRITE_DAC | SEC_STD_WRITE_OWNER /* 0xf0000 */
2003-08-13 05:53:07 +04:00
/* SECOND_DESIRED_ACCESS 0xe0080 */
2004-11-30 07:33:27 +03:00
# define SECOND_DESIRED_ACCESS SEC_FILE_READ_ATTRIBUTE| /* 0x80 */ \
SEC_STD_READ_CONTROL | SEC_STD_WRITE_DAC | \
SEC_STD_WRITE_OWNER /* 0xe0000 */
2003-08-13 05:53:07 +04:00
#if 0
2004-11-30 07:33:27 +03:00
# define THIRD_DESIRED_ACCESS FILE_READ_ATTRIBUTE| /* 0x80 */ \
READ_CONTROL | WRITE_DAC | \
SEC_FILE_READ_DATA | \
WRITE_OWNER /* */
2003-08-13 05:53:07 +04:00
# 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 ,
2004-11-30 07:33:27 +03:00
FIRST_DESIRED_ACCESS ,
FILE_ATTRIBUTE_ARCHIVE ,
NTCREATEX_SHARE_ACCESS_NONE ,
NTCREATEX_DISP_OVERWRITE_IF ,
0x4044 , 0 ) ;
2003-08-13 05:53:07 +04:00
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-11-30 07:33:27 +03:00
fnum = smbcli_nt_create_full ( cli1 - > tree , pipe_name , 0 , SEC_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-12-03 09:24:38 +03:00
cli = malloc_array_p ( 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 " ;
2005-05-25 23:16:18 +04:00
char * control_char_fname ;
2003-08-13 05:53:07 +04:00
int fnum1 , fnum2 ;
2004-12-04 16:56:25 +03:00
uint8_t buf [ 20 ] ;
2003-08-13 05:53:07 +04:00
size_t fsize ;
BOOL correct = True ;
char * tmp_path ;
int failures = 0 ;
2005-05-25 23:16:18 +04:00
int i ;
2003-08-13 05:53:07 +04:00
printf ( " starting open test \n " ) ;
if ( ! torture_open_connection ( & cli1 ) ) {
return False ;
}
2005-05-25 23:16:18 +04:00
asprintf ( & control_char_fname , " \\ readonly.afile " ) ;
for ( i = 1 ; i < = 0x1f ; i + + ) {
control_char_fname [ 10 ] = i ;
fnum1 = smbcli_nt_create_full ( cli1 - > tree , control_char_fname , 0 , SEC_FILE_WRITE_DATA , FILE_ATTRIBUTE_NORMAL ,
NTCREATEX_SHARE_ACCESS_NONE , NTCREATEX_DISP_OVERWRITE_IF , 0 , 0 ) ;
if ( ! check_error ( __location__ , cli1 , ERRDOS , ERRinvalidname ,
NT_STATUS_OBJECT_NAME_INVALID ) ) {
printf ( " Error code should be NT_STATUS_OBJECT_NAME_INVALID, was %s for file with %d char \n " ,
smbcli_errstr ( cli1 - > tree ) , i ) ;
failures + + ;
}
if ( fnum1 ! = - 1 ) {
smbcli_close ( cli1 - > tree , fnum1 ) ;
}
smbcli_setatr ( cli1 - > tree , control_char_fname , 0 , 0 ) ;
smbcli_unlink ( cli1 - > tree , control_char_fname ) ;
}
free ( control_char_fname ) ;
if ( ! failures )
printf ( " Create file with control char names passed. \n " ) ;
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-11-30 07:33:27 +03:00
fnum1 = smbcli_nt_create_full ( cli1 - > tree , fname , 0 , SEC_FILE_READ_ATTRIBUTE , 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-11-30 07:33:27 +03:00
fnum2 = smbcli_nt_create_full ( cli2 - > tree , fname , 0 , SEC_FILE_READ_ATTRIBUTE , 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-11-30 07:33:27 +03:00
fnum1 = smbcli_nt_create_full ( cli1 - > tree , fname , 0 , SEC_STD_DELETE | SEC_FILE_READ_ATTRIBUTE , 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-11-30 07:33:27 +03:00
fnum2 = smbcli_nt_create_full ( cli2 - > tree , fname , 0 , SEC_FILE_READ_ATTRIBUTE , 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-11-30 07:33:27 +03:00
fnum1 = smbcli_nt_create_full ( cli1 - > tree , fname , 0 , SEC_FILE_READ_ATTRIBUTE , 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-11-30 07:33:27 +03:00
fnum2 = smbcli_nt_create_full ( cli2 - > tree , fname , 0 , SEC_STD_DELETE | SEC_FILE_READ_ATTRIBUTE , 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-11-30 07:33:27 +03:00
fnum1 = smbcli_nt_create_full ( cli1 - > tree , fname , 0 , SEC_STD_DELETE | SEC_FILE_READ_ATTRIBUTE , 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-11-30 07:33:27 +03:00
fnum2 = smbcli_nt_create_full ( cli2 - > tree , fname , 0 , SEC_STD_DELETE | SEC_FILE_READ_ATTRIBUTE , 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-11-30 07:33:27 +03:00
fnum1 = smbcli_nt_create_full ( cli1 - > tree , fname , 0 , SEC_STD_DELETE | SEC_FILE_READ_ATTRIBUTE , 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-11-30 07:33:27 +03:00
fnum2 = smbcli_nt_create_full ( cli2 - > tree , fname , 0 , SEC_STD_DELETE | SEC_FILE_READ_ATTRIBUTE , 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-11-30 07:33:27 +03:00
fnum1 = smbcli_nt_create_full ( cli1 - > tree , fname , 0 , SEC_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-11-30 07:33:27 +03:00
fnum2 = smbcli_nt_create_full ( cli2 - > tree , fname , 0 , SEC_FILE_READ_ATTRIBUTE , 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-11-30 07:33:27 +03:00
fnum1 = smbcli_nt_create_full ( cli1 - > tree , fname , 0 , SEC_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-11-30 07:33:27 +03:00
fnum2 = smbcli_nt_create_full ( cli2 - > tree , fname , 0 , SEC_STD_DELETE | SEC_FILE_READ_ATTRIBUTE , 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-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
2005-01-02 10:51:13 +03:00
* result = True ;
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 ;
2005-01-15 14:58:52 +03:00
const char * hostname = NULL , * sharename ;
2004-03-09 12:04:06 +03:00
2003-08-13 05:53:07 +04:00
pid_t mypid = getpid ( ) ;
2005-01-21 16:16:33 +03:00
srandom ( ( ( int ) mypid ) ^ ( ( int ) time ( NULL ) ) ) ;
2003-08-13 05:53:07 +04:00
asprintf ( & myname , " CLIENT%d " , i ) ;
lp_set_cmdline ( " netbios name " , myname ) ;
free ( myname ) ;
2004-03-09 12:04:06 +03:00
if ( unc_list ) {
2005-01-15 14:58:52 +03:00
if ( ! smbcli_parse_unc ( unc_list [ i % num_unc_names ] ,
NULL , & hostname , & sharename ) ) {
2004-03-09 12:04:06 +03:00
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 ;
}
2004-03-09 05:13:13 +03:00
printf ( " %d clients started \n " , torture_nprocs ) ;
2003-08-13 05:53:07 +04:00
2004-11-25 01:26:19 +03:00
kill ( 0 , SIGCONT ) ;
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-11-08 04:21:45 +03:00
{ " BASE-DENYDOS " , torture_denydos_sharing , 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 } ,
{ " 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-11-25 01:26:19 +03:00
{ " BASE-PROPERTIES " , torture_test_properties , 0 } ,
2004-08-03 08:44:27 +04:00
{ " 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-11-09 12:26:47 +03:00
{ " BASE-DISCONNECT " , torture_disconnect , 0 } ,
2004-11-12 12:37:59 +03:00
{ " BASE-DELAYWRITE " , torture_delay_write , 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 } ,
2005-01-31 12:54:39 +03:00
{ " BENCH-NBT " , torture_bench_nbt , 0 } ,
2005-02-15 07:55:56 +03:00
{ " BENCH-WINS " , torture_bench_wins , 0 } ,
2005-05-01 14:18:19 +04:00
{ " BENCH-RPC " , torture_bench_rpc , 0 } ,
2005-05-18 08:17:24 +04:00
{ " BENCH-CLDAP " , torture_bench_cldap , 0 } ,
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-11-15 09:55:27 +03:00
{ " RAW-EAS " , torture_raw_eas , 0 } ,
{ " RAW-STREAMS " , torture_raw_streams , 0 } ,
2004-11-18 04:02:27 +03:00
{ " RAW-ACLS " , torture_raw_acls , 0 } ,
2004-07-12 20:35:48 +04:00
{ " RAW-RAP " , torture_raw_rap , 0 } ,
2005-01-12 10:03:29 +03:00
{ " RAW-COMPOSITE " , torture_raw_composite , 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 } ,
2005-01-25 04:21:59 +03:00
{ " SCAN-RAP " , torture_rap_scan , 0 } ,
2004-08-03 08:44:27 +04:00
/* 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 } ,
2005-05-19 19:45:25 +04:00
{ " RPC-UNIXINFO " , torture_rpc_unixinfo , 0 } ,
2003-12-01 04:41:38 +03:00
{ " RPC-NETLOGON " , torture_rpc_netlogon , 0 } ,
2004-11-11 07:32:01 +03:00
{ " RPC-SAMLOGON " , torture_rpc_samlogon , 0 } ,
2004-11-12 02:24:30 +03:00
{ " RPC-SAMSYNC " , torture_rpc_samsync , 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 } ,
2005-03-17 23:28:01 +03:00
{ " RPC-INITSHUTDOWN " , torture_rpc_initshutdown , 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-11-10 19:50:15 +03:00
{ " RPC-LOGIN " , torture_rpc_login , 0 } ,
2004-11-20 21:51:58 +03:00
{ " RPC-ROT " , torture_rpc_rot , 0 } ,
2004-12-31 13:49:35 +03:00
{ " RPC-DSSETUP " , torture_rpc_dssetup , 0 } ,
2005-01-09 12:38:16 +03:00
{ " RPC-ALTERCONTEXT " , torture_rpc_alter_context , 0 } ,
2004-08-03 08:44:27 +04:00
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 } ,
2005-01-19 06:20:20 +03:00
{ " LOCAL-SOCKET " , torture_local_socket , 0 } ,
2004-08-03 08:44:27 +04:00
2005-02-21 17:30:49 +03:00
/* COM (Component Object Model) testers */
{ " COM-SIMPLE " , torture_com_simple , 0 } ,
2004-08-12 12:00:45 +04:00
/* ldap testers */
{ " LDAP-BASIC " , torture_ldap_basic , 0 } ,
2005-05-10 06:03:47 +04:00
{ " LDAP-CLDAP " , torture_cldap , 0 } ,
2004-08-12 12:00:45 +04:00
2005-02-01 09:16:12 +03:00
/* nbt tests */
{ " NBT-REGISTER " , torture_nbt_register , 0 } ,
2005-02-10 06:22:47 +03:00
{ " NBT-WINS " , torture_nbt_wins , 0 } ,
2005-02-16 13:04:52 +03:00
{ " NBT-WINSREPLICATION " , torture_nbt_winsreplication , 0 } ,
2005-04-06 15:17:08 +04:00
{ " NBT-DGRAM " , torture_nbt_dgram , 0 } ,
2005-03-04 03:24:21 +03:00
/* libnet tests */
{ " NET-USERINFO " , torture_userinfo , 0 } ,
2005-04-21 05:01:26 +04:00
{ " NET-USERADD " , torture_useradd , 0 } ,
2005-04-21 11:25:16 +04:00
{ " NET-USERDEL " , torture_userdel , 0 } ,
2005-02-01 09:16:12 +03:00
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 ) {
2005-01-02 10:51:13 +03:00
BOOL result = False ;
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
2004-12-13 12:26:13 +03:00
printf ( " The binding format is: \n \n " ) ;
printf ( " TRANSPORT:host[flags] \n \n " ) ;
printf ( " where TRANSPORT is either ncacn_np for SMB or ncacn_ip_tcp for RPC/TCP \n \n " ) ;
printf ( " 'host' is an IP or hostname or netbios name. If the binding string \n " ) ;
printf ( " identifies the server side of an endpoint, 'host' may be an empty \n " ) ;
printf ( " string. \n \n " ) ;
printf ( " 'flags' can include a SMB pipe name if using the ncacn_np transport or \n " ) ;
printf ( " a TCP port number if using the ncacn_ip_tcp transport, otherwise they \n " ) ;
printf ( " will be auto-determined. \n \n " ) ;
printf ( " other recognised flags are: \n \n " ) ;
printf ( " sign : enable ntlmssp signing \n " ) ;
printf ( " seal : enable ntlmssp sealing \n " ) ;
printf ( " connect : enable rpc connect level auth (auth, but no sign or seal) \n " ) ;
printf ( " validate: enable the NDR validator \n " ) ;
printf ( " print: enable debugging of the packets \n " ) ;
printf ( " bigendian: use bigendian RPC \n " ) ;
printf ( " padcheck: check reply data for non-zero pad bytes \n \n " ) ;
printf ( " For example, these all connect to the samr pipe: \n \n " ) ;
printf ( " ncacn_np:myserver \n " ) ;
printf ( " ncacn_np:myserver[samr] \n " ) ;
printf ( " ncacn_np:myserver[ \\ pipe \\ samr] \n " ) ;
printf ( " ncacn_np:myserver[/pipe/samr] \n " ) ;
printf ( " ncacn_np:myserver[samr,sign,print] \n " ) ;
printf ( " ncacn_np:myserver[ \\ pipe \\ samr,sign,seal,bigendian] \n " ) ;
printf ( " ncacn_np:myserver[/pipe/samr,seal,validate] \n " ) ;
printf ( " ncacn_np: \n " ) ;
printf ( " ncacn_np:[/pipe/samr] \n \n " ) ;
printf ( " ncacn_ip_tcp:myserver \n " ) ;
printf ( " ncacn_ip_tcp:myserver[1024] \n " ) ;
printf ( " ncacn_ip_tcp:myserver[1024,sign,seal] \n \n " ) ;
printf ( " The unc format is: \n \n " ) ;
printf ( " //server/share \n \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 " ) ;
2004-12-13 12:26:13 +03:00
2003-08-13 05:53:07 +04:00
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 " ) ;
r5902: A rather large change...
I wanted to add a simple 'workstation' argument to the DCERPC
authenticated binding calls, but this patch kind of grew from there.
With SCHANNEL, the 'workstation' name (the netbios name of the client)
matters, as this is what ties the session between the NETLOGON ops and
the SCHANNEL bind. This changes a lot of files, and these will again
be changed when jelmer does the credentials work.
I also correct some schannel IDL to distinguish between workstation
names and account names. The distinction matters for domain trust
accounts.
Issues in handling this (issues with lifetime of talloc pointers)
caused me to change the 'creds_CredentialsState' and 'struct
dcerpc_binding' pointers to always be talloc()ed pointers.
In the schannel DB, we now store both the domain and computername, and
query on both. This should ensure we fault correctly when the domain
is specified incorrectly in the SCHANNEL bind.
In the RPC-SCHANNEL test, I finally fixed a bug that vl pointed out,
where the comment claimed we re-used a connection, but in fact we made
a new connection.
This was achived by breaking apart some of the
dcerpc_secondary_connection() logic.
The addition of workstation handling was also propogated to NTLMSSP
and GENSEC, for completeness.
The RPC-SAMSYNC test has been cleaned up a little, using a loop over
usernames/passwords rather than manually expanded tests. This will be
expanded further (the code in #if 0 in this patch) to use a newly
created user account for testing.
In making this test pass test_rpc.sh, I found a bug in the RPC-ECHO
server, caused by the removal of [ref] and the assoicated pointer from
the IDL. This has been re-added, until the underlying pidl issues are
solved.
(This used to be commit 824289dcc20908ddec957a4a892a103eec2da9b9)
2005-03-19 11:34:43 +03:00
struct dcerpc_binding * binding_struct ;
2004-10-18 19:33:34 +04:00
NTSTATUS status ;
status = dcerpc_parse_binding ( mem_ctx , binding_string , & binding_struct ) ;
2005-01-27 10:08:20 +03:00
talloc_free ( mem_ctx ) ;
2004-10-18 19:33:34 +04:00
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 ,
2005-03-22 11:00:45 +03:00
POPT_CONTEXT_KEEP_FIRST ) ;
2004-08-25 18:31:59 +04:00
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-12-31 04:02:22 +03:00
lp_set_cmdline ( " torture:dangerous " , " Yes " ) ;
2004-08-25 18:31:59 +04:00
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-11-09 12:26:47 +03:00
smbtorture_init_subsystems ;
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 ;
2005-01-15 14:58:52 +03:00
const char * host = NULL , * share = NULL ;
2003-11-24 16:19:00 +03:00
2005-01-15 14:58:52 +03:00
if ( ! smbcli_parse_unc ( argv_new [ 1 ] , NULL , & 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 ( 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 ) ;
}
}