2006-12-21 09:48:42 +03:00
/*
TODO : add splitting of writes for servers with signing
*/
2004-03-09 05:13:13 +03:00
/*
Unix SMB / CIFS implementation .
SMB torture tester
Copyright ( C ) Andrew Tridgell 1997 - 1998
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
2007-07-10 06:07:03 +04:00
the Free Software Foundation ; either version 3 of the License , or
2004-03-09 05:13:13 +03:00
( 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
2007-07-10 06:07:03 +04:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2004-03-09 05:13:13 +03:00
*/
# include "includes.h"
2004-11-02 03:24:21 +03:00
# include "system/time.h"
2005-02-10 08:09:35 +03:00
# include "system/filesys.h"
2008-10-11 23:31:42 +04:00
# include "../lib/util/dlinklist.h"
2006-01-03 18:40:05 +03:00
# include "libcli/libcli.h"
2006-03-17 20:59:58 +03:00
# include "torture/util.h"
2007-08-27 15:17:48 +04:00
# include "torture/nbench/proto.h"
2004-03-09 05:13:13 +03:00
extern int nbench_line_count ;
2004-11-26 02:05:43 +03:00
static int nbio_id = - 1 ;
2004-03-09 05:13:13 +03:00
static int nprocs ;
2007-10-07 02:28:14 +04:00
static bool bypass_io ;
2004-11-26 02:05:43 +03:00
static struct timeval tv_start , tv_end ;
static int warmup , timelimit ;
static int in_cleanup ;
2004-03-09 05:13:13 +03:00
2007-08-27 15:17:48 +04:00
struct lock_info {
struct lock_info * next , * prev ;
off_t offset ;
int size ;
} ;
struct createx_params {
char * fname ;
2010-01-05 20:42:54 +03:00
unsigned int create_options ;
unsigned int create_disposition ;
2007-08-27 15:17:48 +04:00
int handle ;
} ;
2004-03-26 05:11:31 +03:00
struct ftable {
struct ftable * next , * prev ;
int fd ; /* the fd that we got back from the server */
int handle ; /* the handle in the load file */
2007-08-27 15:17:48 +04:00
struct createx_params cp ;
struct lock_info * locks ;
2004-03-26 05:11:31 +03:00
} ;
static struct ftable * ftable ;
2004-03-09 05:13:13 +03:00
static struct {
2004-11-26 02:05:43 +03:00
double bytes , warmup_bytes ;
2004-03-09 05:13:13 +03:00
int line ;
int done ;
2007-08-27 15:17:48 +04:00
bool connected ;
2007-06-17 23:35:00 +04:00
double max_latency ;
struct timeval starttime ;
2004-03-09 05:13:13 +03:00
} * children ;
2007-08-27 15:17:48 +04:00
static bool nb_do_createx ( struct ftable * f ,
const char * fname ,
2010-01-05 20:42:54 +03:00
unsigned int create_options ,
unsigned int create_disposition ,
2007-08-27 15:17:48 +04:00
int handle ,
NTSTATUS status ,
bool retry ) ;
static bool nb_do_lockx ( bool relock , int handle , off_t offset , int size , NTSTATUS status ) ;
static void nb_set_createx_params ( struct ftable * f ,
const char * fname ,
2010-01-05 20:42:54 +03:00
unsigned int create_options ,
unsigned int create_disposition ,
2007-08-27 15:17:48 +04:00
int handle )
{
struct createx_params * cp = & f - > cp ;
if ( fname ! = NULL ) {
cp - > fname = talloc_strdup ( f , fname ) ;
if ( cp - > fname = = NULL ) {
perror ( " nb_set_createx_params: strdup " ) ;
nb_exit ( 1 ) ;
}
} else {
cp - > fname = NULL ;
}
cp - > create_options = create_options ;
cp - > create_disposition = create_disposition ;
cp - > handle = handle ;
}
static bool nb_reestablish_locks ( struct ftable * f )
{
struct lock_info * linfo = f - > locks ;
while ( linfo ! = NULL ) {
DEBUG ( 1 , ( " nb_reestablish_locks: lock for file %d at %lu \n " ,
f - > handle , ( unsigned long ) linfo - > offset ) ) ;
if ( ! nb_do_lockx ( true , f - > handle , linfo - > offset , linfo - > size , NT_STATUS_OK ) ) {
printf ( " nb_reestablish_locks: failed to get lock for file %s at %lu \n " ,
f - > cp . fname , ( unsigned long ) linfo - > offset ) ;
2007-10-07 02:28:14 +04:00
return false ;
2007-08-27 15:17:48 +04:00
}
linfo = linfo - > next ;
}
2007-10-07 02:28:14 +04:00
return true ;
2007-08-27 15:17:48 +04:00
}
static bool nb_reopen_all_files ( void )
{
struct ftable * f = ftable ;
while ( f ! = NULL ) {
DEBUG ( 1 , ( " -- nb_reopen_all_files: opening %s (handle %d) \n " ,
f - > cp . fname , f - > cp . handle ) ) ;
if ( ! nb_do_createx ( f ,
f - > cp . fname ,
f - > cp . create_options ,
f - > cp . create_disposition ,
f - > cp . handle ,
NT_STATUS_OK ,
2007-10-07 02:28:14 +04:00
true ) )
2007-08-27 15:17:48 +04:00
{
printf ( " -- nb_reopen_all_files: failed to open file %s \n " , f - > cp . fname ) ;
2007-10-07 02:28:14 +04:00
return false ;
2007-08-27 15:17:48 +04:00
}
if ( ! nb_reestablish_locks ( f ) ) {
printf ( " --nb_reopen_all_files: failed to reestablish locks \n " ) ;
2007-10-07 02:28:14 +04:00
return false ;
2007-08-27 15:17:48 +04:00
}
f = f - > next ;
}
2007-10-07 02:28:14 +04:00
return true ;
2007-08-27 15:17:48 +04:00
}
2007-12-03 17:53:07 +03:00
bool nb_reconnect ( struct smbcli_state * * cli , struct torture_context * tctx , int client )
2007-08-27 15:17:48 +04:00
{
children [ client ] . connected = false ;
if ( * cli ! = NULL ) {
talloc_free ( * cli ) ;
}
2007-12-03 17:53:07 +03:00
if ( ! torture_open_connection ( cli , tctx , client ) ) {
2007-08-27 15:17:48 +04:00
printf ( " nb_reconnect: failed to connect \n " ) ;
* cli = NULL ;
2007-10-07 02:28:14 +04:00
return false ;
2007-08-27 15:17:48 +04:00
}
nb_setup ( * cli , client ) ;
if ( ! nb_reopen_all_files ( ) ) {
printf ( " nb_reconnect: failed to reopen files in client %d \n " , client ) ;
2007-10-07 02:28:14 +04:00
return false ;
2007-08-27 15:17:48 +04:00
}
2007-10-07 02:28:14 +04:00
return true ;
2007-08-27 15:17:48 +04:00
}
2007-06-18 00:03:31 +04:00
void nbio_target_rate ( double rate )
{
static double last_bytes ;
static struct timeval last_time ;
double tdelay ;
if ( last_bytes = = 0 ) {
last_bytes = children [ nbio_id ] . bytes ;
last_time = timeval_current ( ) ;
return ;
}
tdelay = ( children [ nbio_id ] . bytes - last_bytes ) / ( 1.0e6 * rate ) - timeval_elapsed ( & last_time ) ;
if ( tdelay > 0 ) {
2010-09-16 23:36:37 +04:00
smb_msleep ( tdelay * 1000 ) ;
2007-06-18 00:03:31 +04:00
} else {
children [ nbio_id ] . max_latency = MAX ( children [ nbio_id ] . max_latency , - tdelay ) ;
}
last_time = timeval_current ( ) ;
last_bytes = children [ nbio_id ] . bytes ;
}
2007-06-17 23:35:00 +04:00
void nbio_time_reset ( void )
{
children [ nbio_id ] . starttime = timeval_current ( ) ;
}
void nbio_time_delay ( double targett )
{
double elapsed = timeval_elapsed ( & children [ nbio_id ] . starttime ) ;
if ( targett > elapsed ) {
2010-09-16 23:36:37 +04:00
smb_msleep ( 1000 * ( targett - elapsed ) ) ;
2007-06-17 23:35:00 +04:00
} else if ( elapsed - targett > children [ nbio_id ] . max_latency ) {
2007-06-18 00:03:31 +04:00
children [ nbio_id ] . max_latency = MAX ( elapsed - targett , children [ nbio_id ] . max_latency ) ;
2007-06-17 23:35:00 +04:00
}
}
2004-11-26 02:05:43 +03:00
double nbio_result ( void )
2004-03-09 05:13:13 +03:00
{
int i ;
double total = 0 ;
for ( i = 0 ; i < nprocs ; i + + ) {
2004-11-26 02:05:43 +03:00
total + = children [ i ] . bytes - children [ i ] . warmup_bytes ;
2004-03-09 05:13:13 +03:00
}
2004-11-26 02:05:43 +03:00
return 1.0e-6 * total / timeval_elapsed2 ( & tv_start , & tv_end ) ;
2004-03-09 05:13:13 +03:00
}
2007-06-17 23:35:00 +04:00
double nbio_latency ( void )
{
int i ;
double max_latency = 0 ;
for ( i = 0 ; i < nprocs ; i + + ) {
if ( children [ i ] . max_latency > max_latency ) {
max_latency = children [ i ] . max_latency ;
children [ i ] . max_latency = 0 ;
}
}
return max_latency ;
}
2007-10-07 02:28:14 +04:00
bool nb_tick ( void )
2004-03-10 06:24:03 +03:00
{
2004-11-26 02:05:43 +03:00
return children [ nbio_id ] . done ;
2004-03-10 06:24:03 +03:00
}
2004-10-28 17:41:35 +04:00
void nb_alarm ( int sig )
2004-03-09 05:13:13 +03:00
{
int i ;
2004-11-26 02:05:43 +03:00
int lines = 0 ;
2004-03-09 05:13:13 +03:00
double t ;
2004-11-26 02:05:43 +03:00
int in_warmup = 0 ;
2007-08-27 15:17:48 +04:00
int num_connected = 0 ;
2004-03-09 05:13:13 +03:00
if ( nbio_id ! = - 1 ) return ;
for ( i = 0 ; i < nprocs ; i + + ) {
2007-08-27 15:17:48 +04:00
if ( children [ i ] . connected ) {
num_connected + + ;
}
2004-11-26 02:05:43 +03:00
if ( children [ i ] . bytes = = 0 ) {
in_warmup = 1 ;
}
2004-03-09 05:13:13 +03:00
lines + = children [ i ] . line ;
}
2004-11-26 02:05:43 +03:00
t = timeval_elapsed ( & tv_start ) ;
2004-03-09 05:13:13 +03:00
2004-11-26 02:05:43 +03:00
if ( ! in_warmup & & warmup > 0 & & t > warmup ) {
tv_start = timeval_current ( ) ;
warmup = 0 ;
for ( i = 0 ; i < nprocs ; i + + ) {
children [ i ] . warmup_bytes = children [ i ] . bytes ;
}
goto next ;
}
if ( t < warmup ) {
in_warmup = 1 ;
} else if ( ! in_warmup & & ! in_cleanup & & t > timelimit ) {
for ( i = 0 ; i < nprocs ; i + + ) {
children [ i ] . done = 1 ;
}
tv_end = timeval_current ( ) ;
in_cleanup = 1 ;
}
if ( t < 1 ) {
goto next ;
}
if ( ! in_cleanup ) {
tv_end = timeval_current ( ) ;
}
if ( in_warmup ) {
2004-03-26 05:11:31 +03:00
printf ( " %4d %8d %.2f MB/sec warmup %.0f sec \n " ,
2007-08-27 15:17:48 +04:00
num_connected , lines / nprocs ,
2004-11-26 02:05:43 +03:00
nbio_result ( ) , t ) ;
} else if ( in_cleanup ) {
printf ( " %4d %8d %.2f MB/sec cleanup %.0f sec \n " ,
2007-08-27 15:17:48 +04:00
num_connected , lines / nprocs ,
2004-11-26 02:05:43 +03:00
nbio_result ( ) , t ) ;
2004-03-10 08:05:56 +03:00
} else {
2007-06-17 23:35:00 +04:00
printf ( " %4d %8d %.2f MB/sec execute %.0f sec latency %.2f msec \n " ,
2007-08-27 15:17:48 +04:00
num_connected , lines / nprocs ,
2007-06-17 23:35:00 +04:00
nbio_result ( ) , t , nbio_latency ( ) * 1.0e3 ) ;
2004-03-10 06:24:03 +03:00
}
2004-03-09 05:13:13 +03:00
2004-03-10 08:05:56 +03:00
fflush ( stdout ) ;
2004-11-26 02:05:43 +03:00
next :
2004-03-09 05:13:13 +03:00
signal ( SIGALRM , nb_alarm ) ;
alarm ( 1 ) ;
}
2004-11-26 02:05:43 +03:00
void nbio_shmem ( int n , int t_timelimit , int t_warmup )
2004-03-09 05:13:13 +03:00
{
nprocs = n ;
2012-07-25 10:35:44 +04:00
children = anonymous_shared_allocate ( sizeof ( * children ) * nprocs ) ;
2004-03-09 05:13:13 +03:00
if ( ! children ) {
printf ( " Failed to setup shared memory! \n " ) ;
2007-08-27 15:17:48 +04:00
nb_exit ( 1 ) ;
2004-03-09 05:13:13 +03:00
}
2004-11-26 02:05:43 +03:00
memset ( children , 0 , sizeof ( * children ) * nprocs ) ;
timelimit = t_timelimit ;
warmup = t_warmup ;
in_cleanup = 0 ;
tv_start = timeval_current ( ) ;
2004-03-09 05:13:13 +03:00
}
2007-08-27 15:17:48 +04:00
static struct lock_info * find_lock ( struct lock_info * linfo , off_t offset , int size )
{
while ( linfo ! = NULL ) {
if ( linfo - > offset = = offset & &
linfo - > size = = size )
{
return linfo ;
}
linfo = linfo - > next ;
}
return NULL ;
}
2004-03-26 05:11:31 +03:00
static struct ftable * find_ftable ( int handle )
{
struct ftable * f ;
for ( f = ftable ; f ; f = f - > next ) {
if ( f - > handle = = handle ) return f ;
}
return NULL ;
}
2007-08-27 15:17:48 +04:00
static int find_handle ( int handle , struct ftable * * f_ret )
2004-03-09 05:13:13 +03:00
{
2004-03-26 05:11:31 +03:00
struct ftable * f ;
2007-08-27 15:17:48 +04:00
if ( f_ret ! = NULL )
* f_ret = NULL ;
2004-03-09 05:13:13 +03:00
children [ nbio_id ] . line = nbench_line_count ;
2004-03-26 05:11:31 +03:00
f = find_ftable ( handle ) ;
if ( f ) {
2007-08-27 15:17:48 +04:00
if ( f_ret ! = NULL )
* f_ret = f ;
2004-03-26 05:11:31 +03:00
return f - > fd ;
2004-03-09 05:13:13 +03:00
}
printf ( " (%d) ERROR: handle %d was not found \n " ,
nbench_line_count , handle ) ;
2007-08-27 15:17:48 +04:00
nb_exit ( 1 ) ;
2004-03-09 05:13:13 +03:00
return - 1 ; /* Not reached */
}
2004-03-26 05:11:31 +03:00
2004-08-04 17:23:35 +04:00
static struct smbcli_state * c ;
2004-03-09 05:13:13 +03:00
2004-03-26 05:11:31 +03:00
/*
a handler function for oplock break requests
*/
2007-10-07 02:28:14 +04:00
static bool oplock_handler ( struct smbcli_transport * transport , uint16_t tid ,
2009-02-02 10:44:44 +03:00
uint16_t fnum , uint8_t level , void * private_data )
2004-03-26 05:11:31 +03:00
{
2009-02-02 10:44:44 +03:00
struct smbcli_tree * tree = ( struct smbcli_tree * ) private_data ;
2006-04-11 08:36:09 +04:00
return smbcli_oplock_ack ( tree , fnum , OPLOCK_BREAK_TO_NONE ) ;
2004-03-26 05:11:31 +03:00
}
2004-11-26 02:05:43 +03:00
void nb_setup ( struct smbcli_state * cli , int id )
2004-03-09 05:13:13 +03:00
{
nbio_id = id ;
c = cli ;
if ( bypass_io )
printf ( " skipping I/O \n " ) ;
2004-03-26 05:11:31 +03:00
if ( cli ) {
2004-08-04 17:23:35 +04:00
smbcli_oplock_handler ( cli - > transport , oplock_handler , cli - > tree ) ;
2004-03-26 05:11:31 +03:00
}
2007-08-27 15:17:48 +04:00
children [ id ] . connected = true ;
2004-03-09 05:13:13 +03:00
}
2007-08-27 15:17:48 +04:00
static bool check_status ( const char * op , NTSTATUS status , NTSTATUS ret )
2004-03-09 05:13:13 +03:00
{
2007-08-27 15:17:48 +04:00
if ( ( NT_STATUS_EQUAL ( ret , NT_STATUS_END_OF_FILE ) | |
NT_STATUS_EQUAL ( ret , NT_STATUS_NET_WRITE_FAULT ) | |
NT_STATUS_EQUAL ( ret , NT_STATUS_CONNECTION_RESET ) )
& & ! NT_STATUS_EQUAL ( status , ret ) )
{
2007-10-07 02:28:14 +04:00
return false ;
2007-08-27 15:17:48 +04:00
}
2004-03-09 05:13:13 +03:00
if ( ! NT_STATUS_IS_OK ( status ) & & NT_STATUS_IS_OK ( ret ) ) {
printf ( " [%d] Error: %s should have failed with %s \n " ,
nbench_line_count , op , nt_errstr ( status ) ) ;
2007-08-27 15:17:48 +04:00
nb_exit ( 1 ) ;
2004-03-09 05:13:13 +03:00
}
if ( NT_STATUS_IS_OK ( status ) & & ! NT_STATUS_IS_OK ( ret ) ) {
printf ( " [%d] Error: %s should have succeeded - %s \n " ,
nbench_line_count , op , nt_errstr ( ret ) ) ;
2007-08-27 15:17:48 +04:00
nb_exit ( 1 ) ;
2004-03-09 05:13:13 +03:00
}
if ( ! NT_STATUS_EQUAL ( status , ret ) ) {
printf ( " [%d] Warning: got status %s but expected %s \n " ,
nbench_line_count , nt_errstr ( ret ) , nt_errstr ( status ) ) ;
}
2007-08-27 15:17:48 +04:00
2007-10-07 02:28:14 +04:00
return true ;
2004-03-09 05:13:13 +03:00
}
2007-08-27 15:17:48 +04:00
bool nb_unlink ( const char * fname , int attr , NTSTATUS status , bool retry )
2004-03-09 05:13:13 +03:00
{
2006-03-10 23:49:20 +03:00
union smb_unlink io ;
2004-03-09 05:13:13 +03:00
NTSTATUS ret ;
2006-03-10 23:49:20 +03:00
io . unlink . in . pattern = fname ;
2004-03-09 05:13:13 +03:00
2006-03-10 23:49:20 +03:00
io . unlink . in . attrib = FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN ;
2004-03-09 05:13:13 +03:00
if ( strchr ( fname , ' * ' ) = = 0 ) {
2006-03-10 23:49:20 +03:00
io . unlink . in . attrib | = FILE_ATTRIBUTE_DIRECTORY ;
2004-03-09 05:13:13 +03:00
}
ret = smb_raw_unlink ( c - > tree , & io ) ;
2007-08-27 15:17:48 +04:00
if ( ! retry )
return check_status ( " Unlink " , status , ret ) ;
2004-03-09 05:13:13 +03:00
2007-10-07 02:28:14 +04:00
return true ;
2007-08-27 15:17:48 +04:00
}
2004-03-09 05:13:13 +03:00
2007-08-27 15:17:48 +04:00
static bool nb_do_createx ( struct ftable * f ,
const char * fname ,
2010-01-05 20:42:54 +03:00
unsigned int create_options ,
unsigned int create_disposition ,
2007-08-27 15:17:48 +04:00
int handle ,
NTSTATUS status ,
bool retry )
2004-03-09 05:13:13 +03:00
{
union smb_open io ;
2004-05-25 20:24:13 +04:00
uint32_t desired_access ;
2004-03-09 05:13:13 +03:00
NTSTATUS ret ;
TALLOC_CTX * mem_ctx ;
2010-01-05 20:42:54 +03:00
unsigned int flags = 0 ;
2004-03-09 05:13:13 +03:00
mem_ctx = talloc_init ( " raw_open " ) ;
if ( create_options & NTCREATEX_OPTIONS_DIRECTORY ) {
2004-11-30 07:33:27 +03:00
desired_access = SEC_FILE_READ_DATA ;
2004-03-09 05:13:13 +03:00
} else {
desired_access =
2004-11-30 07:33:27 +03:00
SEC_FILE_READ_DATA |
SEC_FILE_WRITE_DATA |
SEC_FILE_READ_ATTRIBUTE |
SEC_FILE_WRITE_ATTRIBUTE ;
2004-03-26 05:39:48 +03:00
flags = NTCREATEX_FLAGS_EXTENDED |
2004-03-26 05:11:31 +03:00
NTCREATEX_FLAGS_REQUEST_OPLOCK |
NTCREATEX_FLAGS_REQUEST_BATCH_OPLOCK ;
2004-03-09 05:13:13 +03:00
}
io . ntcreatex . level = RAW_OPEN_NTCREATEX ;
2004-03-26 05:11:31 +03:00
io . ntcreatex . in . flags = flags ;
2009-10-15 11:26:19 +04:00
io . ntcreatex . in . root_fid . fnum = 0 ;
2004-03-09 05:13:13 +03:00
io . ntcreatex . in . access_mask = desired_access ;
io . ntcreatex . in . file_attr = 0 ;
io . ntcreatex . in . alloc_size = 0 ;
io . ntcreatex . in . share_access = NTCREATEX_SHARE_ACCESS_READ | NTCREATEX_SHARE_ACCESS_WRITE ;
io . ntcreatex . in . open_disposition = create_disposition ;
io . ntcreatex . in . create_options = create_options ;
io . ntcreatex . in . impersonation = 0 ;
io . ntcreatex . in . security_flags = 0 ;
io . ntcreatex . in . fname = fname ;
2007-08-27 15:17:48 +04:00
if ( retry ) {
/* Reopening after a disconnect. */
io . ntcreatex . in . open_disposition = NTCREATEX_DISP_OPEN_IF ;
} else
if ( f ! = NULL & &
f - > cp . create_disposition = = NTCREATEX_DISP_CREATE & &
NT_STATUS_IS_OK ( status ) )
{
/* Reopening after nb_createx() error. */
io . ntcreatex . in . open_disposition = NTCREATEX_DISP_OPEN_IF ;
}
2004-03-09 05:13:13 +03:00
ret = smb_raw_open ( c - > tree , mem_ctx , & io ) ;
2005-01-27 10:08:20 +03:00
talloc_free ( mem_ctx ) ;
2004-03-09 05:13:13 +03:00
2007-08-27 15:17:48 +04:00
if ( ! check_status ( " NTCreateX " , status , ret ) )
2007-10-07 02:28:14 +04:00
return false ;
2004-03-09 05:13:13 +03:00
2007-08-27 15:17:48 +04:00
if ( ! NT_STATUS_IS_OK ( ret ) )
2007-10-07 02:28:14 +04:00
return true ;
2007-08-27 15:17:48 +04:00
if ( f = = NULL ) {
f = talloc ( NULL , struct ftable ) ;
f - > locks = NULL ;
nb_set_createx_params ( f , fname , create_options , create_disposition , handle ) ;
2016-02-05 13:32:18 +03:00
DLIST_ADD_END ( ftable , f ) ;
2007-08-27 15:17:48 +04:00
}
2004-03-09 05:13:13 +03:00
2004-03-26 05:11:31 +03:00
f - > handle = handle ;
2006-03-13 01:48:25 +03:00
f - > fd = io . ntcreatex . out . file . fnum ;
2004-03-26 05:11:31 +03:00
2007-10-07 02:28:14 +04:00
return true ;
2007-08-27 15:17:48 +04:00
}
bool nb_createx ( const char * fname ,
2010-01-05 20:42:54 +03:00
unsigned int create_options , unsigned int create_disposition , int handle ,
2007-08-27 15:17:48 +04:00
NTSTATUS status )
{
2007-10-07 02:28:14 +04:00
return nb_do_createx ( NULL , fname , create_options , create_disposition , handle , status , false ) ;
2004-03-09 05:13:13 +03:00
}
2007-08-27 15:17:48 +04:00
bool nb_writex ( int handle , off_t offset , int size , int ret_size , NTSTATUS status )
2004-03-09 05:13:13 +03:00
{
union smb_write io ;
int i ;
NTSTATUS ret ;
2004-12-04 16:56:25 +03:00
uint8_t * buf ;
2004-03-09 05:13:13 +03:00
2007-08-27 15:17:48 +04:00
i = find_handle ( handle , NULL ) ;
2004-03-09 05:13:13 +03:00
2007-08-27 15:17:48 +04:00
if ( bypass_io )
2007-10-07 02:28:14 +04:00
return true ;
2004-03-09 05:13:13 +03:00
buf = malloc ( size ) ;
memset ( buf , 0xab , size ) ;
io . writex . level = RAW_WRITE_WRITEX ;
2006-03-13 01:48:25 +03:00
io . writex . in . file . fnum = i ;
2004-03-09 05:13:13 +03:00
io . writex . in . wmode = 0 ;
io . writex . in . remaining = 0 ;
io . writex . in . offset = offset ;
io . writex . in . count = size ;
io . writex . in . data = buf ;
ret = smb_raw_write ( c - > tree , & io ) ;
free ( buf ) ;
2007-08-27 15:17:48 +04:00
if ( ! check_status ( " WriteX " , status , ret ) )
2007-10-07 02:28:14 +04:00
return false ;
2004-03-09 05:13:13 +03:00
2004-03-09 06:28:21 +03:00
if ( NT_STATUS_IS_OK ( ret ) & & io . writex . out . nwritten ! = ret_size ) {
2004-03-09 05:13:13 +03:00
printf ( " [%d] Warning: WriteX got count %d expected %d \n " ,
nbench_line_count ,
io . writex . out . nwritten , ret_size ) ;
}
2004-11-26 02:05:43 +03:00
children [ nbio_id ] . bytes + = ret_size ;
2007-08-27 15:17:48 +04:00
2007-10-07 02:28:14 +04:00
return true ;
2004-03-09 05:13:13 +03:00
}
2007-08-27 15:17:48 +04:00
bool nb_write ( int handle , off_t offset , int size , int ret_size , NTSTATUS status )
2004-03-09 05:13:13 +03:00
{
union smb_write io ;
int i ;
NTSTATUS ret ;
2004-12-04 16:56:25 +03:00
uint8_t * buf ;
2004-03-09 05:13:13 +03:00
2007-08-27 15:17:48 +04:00
i = find_handle ( handle , NULL ) ;
2004-03-09 05:13:13 +03:00
2007-08-27 15:17:48 +04:00
if ( bypass_io )
2007-10-07 02:28:14 +04:00
return true ;
2004-03-09 05:13:13 +03:00
buf = malloc ( size ) ;
memset ( buf , 0x12 , size ) ;
io . write . level = RAW_WRITE_WRITE ;
2006-03-13 01:48:25 +03:00
io . write . in . file . fnum = i ;
2004-03-09 05:13:13 +03:00
io . write . in . remaining = 0 ;
io . write . in . offset = offset ;
io . write . in . count = size ;
io . write . in . data = buf ;
ret = smb_raw_write ( c - > tree , & io ) ;
free ( buf ) ;
2007-08-27 15:17:48 +04:00
if ( ! check_status ( " Write " , status , ret ) )
2007-10-07 02:28:14 +04:00
return false ;
2004-03-09 05:13:13 +03:00
2004-03-09 06:28:21 +03:00
if ( NT_STATUS_IS_OK ( ret ) & & io . write . out . nwritten ! = ret_size ) {
2004-03-09 05:13:13 +03:00
printf ( " [%d] Warning: Write got count %d expected %d \n " ,
nbench_line_count ,
io . write . out . nwritten , ret_size ) ;
}
2004-11-26 02:05:43 +03:00
children [ nbio_id ] . bytes + = ret_size ;
2004-03-09 05:13:13 +03:00
2007-10-07 02:28:14 +04:00
return true ;
2007-08-27 15:17:48 +04:00
}
2004-03-09 05:13:13 +03:00
2007-08-27 15:17:48 +04:00
static bool nb_do_lockx ( bool relock , int handle , off_t offset , int size , NTSTATUS status )
2004-03-09 05:13:13 +03:00
{
union smb_lock io ;
int i ;
NTSTATUS ret ;
struct smb_lock_entry lck ;
2007-08-27 15:17:48 +04:00
struct ftable * f ;
2004-03-09 05:13:13 +03:00
2007-08-27 15:17:48 +04:00
i = find_handle ( handle , & f ) ;
2004-03-09 05:13:13 +03:00
lck . pid = getpid ( ) ;
lck . offset = offset ;
lck . count = size ;
io . lockx . level = RAW_LOCK_LOCKX ;
2006-03-13 01:48:25 +03:00
io . lockx . in . file . fnum = i ;
2004-03-09 05:13:13 +03:00
io . lockx . in . mode = 0 ;
io . lockx . in . timeout = 0 ;
io . lockx . in . ulock_cnt = 0 ;
io . lockx . in . lock_cnt = 1 ;
io . lockx . in . locks = & lck ;
ret = smb_raw_lock ( c - > tree , & io ) ;
2007-08-27 15:17:48 +04:00
if ( ! check_status ( " LockX " , status , ret ) )
2007-10-07 02:28:14 +04:00
return false ;
2007-08-27 15:17:48 +04:00
if ( f ! = NULL & &
! relock )
{
struct lock_info * linfo ;
linfo = talloc ( f , struct lock_info ) ;
linfo - > offset = offset ;
linfo - > size = size ;
2016-02-05 13:32:18 +03:00
DLIST_ADD_END ( f - > locks , linfo ) ;
2007-08-27 15:17:48 +04:00
}
2007-10-07 02:28:14 +04:00
return true ;
2004-03-09 05:13:13 +03:00
}
2007-08-27 15:17:48 +04:00
bool nb_lockx ( int handle , off_t offset , int size , NTSTATUS status )
{
return nb_do_lockx ( false , handle , offset , size , status ) ;
}
2010-01-05 20:42:54 +03:00
bool nb_unlockx ( int handle , unsigned int offset , int size , NTSTATUS status )
2004-03-09 05:13:13 +03:00
{
union smb_lock io ;
int i ;
NTSTATUS ret ;
struct smb_lock_entry lck ;
2007-08-27 15:17:48 +04:00
struct ftable * f ;
2004-03-09 05:13:13 +03:00
2007-08-27 15:17:48 +04:00
i = find_handle ( handle , & f ) ;
2004-03-09 05:13:13 +03:00
lck . pid = getpid ( ) ;
lck . offset = offset ;
lck . count = size ;
io . lockx . level = RAW_LOCK_LOCKX ;
2006-03-13 01:48:25 +03:00
io . lockx . in . file . fnum = i ;
2004-03-09 05:13:13 +03:00
io . lockx . in . mode = 0 ;
io . lockx . in . timeout = 0 ;
io . lockx . in . ulock_cnt = 1 ;
io . lockx . in . lock_cnt = 0 ;
io . lockx . in . locks = & lck ;
ret = smb_raw_lock ( c - > tree , & io ) ;
2007-08-27 15:17:48 +04:00
if ( ! check_status ( " UnlockX " , status , ret ) )
2007-10-07 02:28:14 +04:00
return false ;
2007-08-27 15:17:48 +04:00
if ( f ! = NULL ) {
struct lock_info * linfo ;
linfo = find_lock ( f - > locks , offset , size ) ;
if ( linfo ! = NULL )
DLIST_REMOVE ( f - > locks , linfo ) ;
else
printf ( " nb_unlockx: unknown lock (%d) \n " , handle ) ;
}
2007-10-07 02:28:14 +04:00
return true ;
2004-03-09 05:13:13 +03:00
}
2007-08-27 15:17:48 +04:00
bool nb_readx ( int handle , off_t offset , int size , int ret_size , NTSTATUS status )
2004-03-09 05:13:13 +03:00
{
union smb_read io ;
int i ;
NTSTATUS ret ;
2004-12-04 16:56:25 +03:00
uint8_t * buf ;
2004-03-09 05:13:13 +03:00
2007-08-27 15:17:48 +04:00
i = find_handle ( handle , NULL ) ;
2004-03-09 05:13:13 +03:00
2007-08-27 15:17:48 +04:00
if ( bypass_io )
2007-10-07 02:28:14 +04:00
return true ;
2004-03-09 05:13:13 +03:00
buf = malloc ( size ) ;
io . readx . level = RAW_READ_READX ;
2006-03-13 01:48:25 +03:00
io . readx . in . file . fnum = i ;
2004-03-09 05:13:13 +03:00
io . readx . in . offset = offset ;
io . readx . in . mincnt = size ;
io . readx . in . maxcnt = size ;
io . readx . in . remaining = 0 ;
2007-10-07 02:28:14 +04:00
io . readx . in . read_for_execute = false ;
2004-03-09 05:13:13 +03:00
io . readx . out . data = buf ;
2007-08-27 15:17:48 +04:00
2004-03-09 05:13:13 +03:00
ret = smb_raw_read ( c - > tree , & io ) ;
free ( buf ) ;
2007-08-27 15:17:48 +04:00
if ( ! check_status ( " ReadX " , status , ret ) )
2007-10-07 02:28:14 +04:00
return false ;
2004-03-09 05:13:13 +03:00
2004-03-09 06:28:21 +03:00
if ( NT_STATUS_IS_OK ( ret ) & & io . readx . out . nread ! = ret_size ) {
2004-03-26 05:11:31 +03:00
printf ( " [%d] ERROR: ReadX got count %d expected %d \n " ,
2004-03-09 05:13:13 +03:00
nbench_line_count ,
io . readx . out . nread , ret_size ) ;
2007-08-27 15:17:48 +04:00
nb_exit ( 1 ) ;
2004-03-09 05:13:13 +03:00
}
2004-11-26 02:05:43 +03:00
children [ nbio_id ] . bytes + = ret_size ;
2007-08-27 15:17:48 +04:00
2007-10-07 02:28:14 +04:00
return true ;
2004-03-09 05:13:13 +03:00
}
2007-08-27 15:17:48 +04:00
bool nb_close ( int handle , NTSTATUS status )
2004-03-09 05:13:13 +03:00
{
NTSTATUS ret ;
union smb_close io ;
2004-03-26 05:11:31 +03:00
int i ;
2007-08-27 15:17:48 +04:00
i = find_handle ( handle , NULL ) ;
2004-03-09 05:13:13 +03:00
io . close . level = RAW_CLOSE_CLOSE ;
2006-03-13 01:48:25 +03:00
io . close . in . file . fnum = i ;
2004-03-09 05:13:13 +03:00
io . close . in . write_time = 0 ;
ret = smb_raw_close ( c - > tree , & io ) ;
2007-08-27 15:17:48 +04:00
if ( ! check_status ( " Close " , status , ret ) )
2007-10-07 02:28:14 +04:00
return false ;
2004-03-09 05:13:13 +03:00
if ( NT_STATUS_IS_OK ( ret ) ) {
2004-03-26 05:11:31 +03:00
struct ftable * f = find_ftable ( handle ) ;
DLIST_REMOVE ( ftable , f ) ;
2007-08-27 15:17:48 +04:00
talloc_free ( f ) ;
2004-03-09 05:13:13 +03:00
}
2007-08-27 15:17:48 +04:00
2007-10-07 02:28:14 +04:00
return true ;
2004-03-09 05:13:13 +03:00
}
2007-08-27 15:17:48 +04:00
bool nb_rmdir ( const char * dname , NTSTATUS status , bool retry )
2004-03-09 05:13:13 +03:00
{
NTSTATUS ret ;
struct smb_rmdir io ;
io . in . path = dname ;
ret = smb_raw_rmdir ( c - > tree , & io ) ;
2007-08-27 15:17:48 +04:00
if ( ! retry )
return check_status ( " Rmdir " , status , ret ) ;
2007-10-07 02:28:14 +04:00
return true ;
2004-03-09 05:13:13 +03:00
}
2007-08-27 15:17:48 +04:00
bool nb_mkdir ( const char * dname , NTSTATUS status , bool retry )
2004-03-10 12:00:24 +03:00
{
union smb_mkdir io ;
io . mkdir . level = RAW_MKDIR_MKDIR ;
io . mkdir . in . path = dname ;
/* NOTE! no error checking. Used for base fileset creation */
smb_raw_mkdir ( c - > tree , & io ) ;
2007-08-27 15:17:48 +04:00
2007-10-07 02:28:14 +04:00
return true ;
2004-03-10 12:00:24 +03:00
}
2009-02-02 15:01:45 +03:00
bool nb_rename ( const char * o , const char * n , NTSTATUS status , bool retry )
2004-03-09 05:13:13 +03:00
{
NTSTATUS ret ;
union smb_rename io ;
io . generic . level = RAW_RENAME_RENAME ;
io . rename . in . attrib = FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_DIRECTORY ;
2009-02-02 15:01:45 +03:00
io . rename . in . pattern1 = o ;
io . rename . in . pattern2 = n ;
2004-03-09 05:13:13 +03:00
ret = smb_raw_rename ( c - > tree , & io ) ;
2007-08-27 15:17:48 +04:00
if ( ! retry )
return check_status ( " Rename " , status , ret ) ;
2007-10-07 02:28:14 +04:00
return true ;
2004-03-09 05:13:13 +03:00
}
2007-08-27 15:17:48 +04:00
bool nb_qpathinfo ( const char * fname , int level , NTSTATUS status )
2004-03-09 05:13:13 +03:00
{
union smb_fileinfo io ;
TALLOC_CTX * mem_ctx ;
NTSTATUS ret ;
mem_ctx = talloc_init ( " nb_qpathinfo " ) ;
io . generic . level = level ;
2006-03-13 01:48:25 +03:00
io . generic . in . file . path = fname ;
2004-03-09 05:13:13 +03:00
ret = smb_raw_pathinfo ( c - > tree , mem_ctx , & io ) ;
2005-01-27 10:08:20 +03:00
talloc_free ( mem_ctx ) ;
2004-03-09 05:13:13 +03:00
2007-08-27 15:17:48 +04:00
return check_status ( " Pathinfo " , status , ret ) ;
2004-03-09 05:13:13 +03:00
}
2007-08-27 15:17:48 +04:00
bool nb_qfileinfo ( int fnum , int level , NTSTATUS status )
2004-03-09 05:13:13 +03:00
{
union smb_fileinfo io ;
TALLOC_CTX * mem_ctx ;
NTSTATUS ret ;
int i ;
2007-08-27 15:17:48 +04:00
i = find_handle ( fnum , NULL ) ;
2004-03-09 05:13:13 +03:00
mem_ctx = talloc_init ( " nb_qfileinfo " ) ;
io . generic . level = level ;
2006-03-13 01:48:25 +03:00
io . generic . in . file . fnum = i ;
2004-03-09 05:13:13 +03:00
ret = smb_raw_fileinfo ( c - > tree , mem_ctx , & io ) ;
2005-01-27 10:08:20 +03:00
talloc_free ( mem_ctx ) ;
2004-03-09 05:13:13 +03:00
2007-08-27 15:17:48 +04:00
return check_status ( " Fileinfo " , status , ret ) ;
2004-03-09 05:13:13 +03:00
}
2007-08-27 15:17:48 +04:00
bool nb_sfileinfo ( int fnum , int level , NTSTATUS status )
2004-03-09 05:13:13 +03:00
{
union smb_setfileinfo io ;
NTSTATUS ret ;
int i ;
if ( level ! = RAW_SFILEINFO_BASIC_INFORMATION ) {
printf ( " [%d] Warning: setfileinfo level %d not handled \n " , nbench_line_count , level ) ;
2007-10-07 02:28:14 +04:00
return true ;
2004-03-09 05:13:13 +03:00
}
ZERO_STRUCT ( io ) ;
2007-08-27 15:17:48 +04:00
i = find_handle ( fnum , NULL ) ;
2004-03-09 05:13:13 +03:00
io . generic . level = level ;
2006-03-13 01:48:25 +03:00
io . generic . in . file . fnum = i ;
2004-03-09 05:13:13 +03:00
unix_to_nt_time ( & io . basic_info . in . create_time , time ( NULL ) ) ;
unix_to_nt_time ( & io . basic_info . in . access_time , 0 ) ;
unix_to_nt_time ( & io . basic_info . in . write_time , 0 ) ;
unix_to_nt_time ( & io . basic_info . in . change_time , 0 ) ;
io . basic_info . in . attrib = 0 ;
ret = smb_raw_setfileinfo ( c - > tree , & io ) ;
2007-08-27 15:17:48 +04:00
return check_status ( " Setfileinfo " , status , ret ) ;
2004-03-09 05:13:13 +03:00
}
2007-08-27 15:17:48 +04:00
bool nb_qfsinfo ( int level , NTSTATUS status )
2004-03-09 05:13:13 +03:00
{
union smb_fsinfo io ;
TALLOC_CTX * mem_ctx ;
NTSTATUS ret ;
2004-08-04 17:23:35 +04:00
mem_ctx = talloc_init ( " smbcli_dskattr " ) ;
2004-03-09 05:13:13 +03:00
io . generic . level = level ;
ret = smb_raw_fsinfo ( c - > tree , mem_ctx , & io ) ;
2005-01-27 10:08:20 +03:00
talloc_free ( mem_ctx ) ;
2004-03-09 05:13:13 +03:00
2007-08-27 15:17:48 +04:00
return check_status ( " Fsinfo " , status , ret ) ;
2004-03-09 05:13:13 +03:00
}
/* callback function used for trans2 search */
2009-02-02 10:44:44 +03:00
static bool findfirst_callback ( void * private_data , const union smb_search_data * file )
2004-03-09 05:13:13 +03:00
{
2007-10-07 02:28:14 +04:00
return true ;
2004-03-09 05:13:13 +03:00
}
2007-08-27 15:17:48 +04:00
bool nb_findfirst ( const char * mask , int level , int maxcnt , int count , NTSTATUS status )
2004-03-09 05:13:13 +03:00
{
union smb_search_first io ;
TALLOC_CTX * mem_ctx ;
NTSTATUS ret ;
2004-08-04 17:23:35 +04:00
mem_ctx = talloc_init ( " smbcli_dskattr " ) ;
2004-03-09 05:13:13 +03:00
2006-07-26 15:41:08 +04:00
io . t2ffirst . level = RAW_SEARCH_TRANS2 ;
io . t2ffirst . data_level = level ;
2004-03-09 05:13:13 +03:00
io . t2ffirst . in . max_count = maxcnt ;
io . t2ffirst . in . search_attrib = FILE_ATTRIBUTE_DIRECTORY ;
io . t2ffirst . in . pattern = mask ;
io . t2ffirst . in . flags = FLAG_TRANS2_FIND_CLOSE ;
io . t2ffirst . in . storage_type = 0 ;
ret = smb_raw_search_first ( c - > tree , mem_ctx , & io , NULL , findfirst_callback ) ;
2005-01-27 10:08:20 +03:00
talloc_free ( mem_ctx ) ;
2004-03-09 05:13:13 +03:00
2007-08-27 15:17:48 +04:00
if ( ! check_status ( " Search " , status , ret ) )
2007-10-07 02:28:14 +04:00
return false ;
2004-03-09 05:13:13 +03:00
2004-03-09 06:28:21 +03:00
if ( NT_STATUS_IS_OK ( ret ) & & io . t2ffirst . out . count ! = count ) {
2004-03-09 05:13:13 +03:00
printf ( " [%d] Warning: got count %d expected %d \n " ,
nbench_line_count ,
io . t2ffirst . out . count , count ) ;
}
2007-08-27 15:17:48 +04:00
2007-10-07 02:28:14 +04:00
return true ;
2004-03-09 05:13:13 +03:00
}
2007-08-27 15:17:48 +04:00
bool nb_flush ( int fnum , NTSTATUS status )
2004-03-09 05:13:13 +03:00
{
2006-03-10 23:49:20 +03:00
union smb_flush io ;
2004-03-09 05:13:13 +03:00
NTSTATUS ret ;
int i ;
2007-08-27 15:17:48 +04:00
i = find_handle ( fnum , NULL ) ;
2004-03-09 05:13:13 +03:00
2006-05-19 19:10:39 +04:00
io . flush . level = RAW_FLUSH_FLUSH ;
io . flush . in . file . fnum = i ;
2004-03-09 05:13:13 +03:00
ret = smb_raw_flush ( c - > tree , & io ) ;
2007-08-27 15:17:48 +04:00
return check_status ( " Flush " , status , ret ) ;
2004-03-09 05:13:13 +03:00
}
2004-10-18 17:58:41 +04:00
void nb_sleep ( int usec , NTSTATUS status )
{
2005-12-31 12:08:46 +03:00
usleep ( usec ) ;
2004-10-18 17:58:41 +04:00
}
2007-08-27 15:17:48 +04:00
bool nb_deltree ( const char * dname , bool retry )
2004-03-09 05:13:13 +03:00
{
int total_deleted ;
2004-03-09 05:24:14 +03:00
smb_raw_exit ( c - > session ) ;
2004-03-26 05:11:31 +03:00
while ( ftable ) {
struct ftable * f = ftable ;
DLIST_REMOVE ( ftable , f ) ;
2007-08-27 15:17:48 +04:00
talloc_free ( f ) ;
2004-03-26 05:11:31 +03:00
}
2004-03-09 05:24:14 +03:00
2004-08-04 17:23:35 +04:00
total_deleted = smbcli_deltree ( c - > tree , dname ) ;
2004-03-09 05:13:13 +03:00
if ( total_deleted = = - 1 ) {
printf ( " Failed to cleanup tree %s - exiting \n " , dname ) ;
2007-08-27 15:17:48 +04:00
nb_exit ( 1 ) ;
2004-03-09 05:13:13 +03:00
}
2004-08-04 17:23:35 +04:00
smbcli_rmdir ( c - > tree , dname ) ;
2007-08-27 15:17:48 +04:00
2007-10-07 02:28:14 +04:00
return true ;
2004-03-09 05:13:13 +03:00
}
2007-08-27 15:17:48 +04:00
void nb_exit ( int status )
{
children [ nbio_id ] . connected = false ;
printf ( " [%d] client %d exiting with status %d \n " ,
nbench_line_count , nbio_id , status ) ;
exit ( status ) ;
}