2004-11-12 09:37:59 +00:00
/*
Unix SMB / CIFS implementation .
test suite for delayed write update
Copyright ( C ) Volker Lendecke 2004
Copyright ( C ) Andrew Tridgell 2004
2004-11-20 01:51:32 +00:00
Copyright ( C ) Jeremy Allison 2004
2004-11-12 09:37:59 +00:00
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 02:07:03 +00:00
the Free Software Foundation ; either version 3 of the License , or
2004-11-12 09:37:59 +00: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 02:07:03 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2004-11-12 09:37:59 +00:00
*/
# include "includes.h"
2006-01-03 13:41:17 +00:00
# include "torture/torture.h"
2004-11-12 09:37:59 +00:00
# include "libcli/raw/libcliraw.h"
2008-04-02 04:53:27 +02:00
# include "libcli/raw/raw_proto.h"
2004-11-12 09:37:59 +00:00
# include "system/time.h"
2005-02-10 05:09:35 +00:00
# include "system/filesys.h"
2006-01-03 15:40:05 +00:00
# include "libcli/libcli.h"
2006-03-17 17:59:58 +00:00
# include "torture/util.h"
2004-11-12 09:37:59 +00:00
2009-06-03 14:12:18 -07:00
# define W2K8R2_TIMEDELAY_SECS 1
# define W2K3_TIMEDELAY_SECS 2
# define TIMEDELAY_SECS W2K3_TIMEDELAY_SECS
2004-11-12 09:37:59 +00:00
# define BASEDIR "\\delaywrite"
2007-10-06 22:28:14 +00:00
static bool test_delayed_write_update ( struct torture_context * tctx , struct smbcli_state * cli )
2004-11-12 09:37:59 +00:00
{
union smb_fileinfo finfo1 , finfo2 ;
const char * fname = BASEDIR " \\ torture_file.txt " ;
NTSTATUS status ;
int fnum1 = - 1 ;
2007-10-06 22:28:14 +00:00
bool ret = true ;
2004-11-12 09:37:59 +00:00
ssize_t written ;
2008-04-08 10:25:51 +02:00
struct timeval start ;
struct timeval end ;
int used_delay = torture_setting_int ( tctx , " writetimeupdatedelay " , 2000000 ) ;
int normal_delay = 2000000 ;
double sec = ( ( double ) used_delay ) / ( ( double ) normal_delay ) ;
int msec = 1000 * sec ;
2004-11-12 09:37:59 +00:00
2008-09-08 14:58:58 -07:00
torture_comment ( tctx , " \n Running test_delayed_write_update \n " ) ;
2004-11-12 09:37:59 +00:00
if ( ! torture_setup_dir ( cli , BASEDIR ) ) {
2007-10-06 22:28:14 +00:00
return false ;
2004-11-12 09:37:59 +00:00
}
fnum1 = smbcli_open ( cli - > tree , fname , O_RDWR | O_CREAT , DENY_NONE ) ;
if ( fnum1 = = - 1 ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL , " Failed to open %s " , fname ) ;
2007-10-06 22:28:14 +00:00
return false ;
2004-11-12 09:37:59 +00:00
}
finfo1 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
2006-03-12 22:48:25 +00:00
finfo1 . basic_info . in . file . fnum = fnum1 ;
2004-11-12 09:37:59 +00:00
finfo2 = finfo1 ;
2006-10-16 13:06:41 +00:00
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo1 ) ;
2004-11-12 09:37:59 +00:00
2008-09-18 16:05:46 +02:00
torture_assert_ntstatus_ok ( tctx , status , " fileinfo failed " ) ;
2004-11-12 09:37:59 +00:00
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " Initial write time %s \n " ,
nt_time_string ( tctx , finfo1 . basic_info . out . write_time ) ) ;
2004-11-12 09:37:59 +00:00
/* 3 second delay to ensure we get past any 2 second time
granularity ( older systems may have that ) */
2008-04-08 10:25:51 +02:00
msleep ( 3 * msec ) ;
2004-11-12 09:37:59 +00:00
written = smbcli_write ( cli - > tree , fnum1 , 0 , " x " , 0 , 1 ) ;
if ( written ! = 1 ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL ,
" write failed - wrote %d bytes (%s) \n " ,
( int ) written , __location__ ) ;
2007-10-06 22:28:14 +00:00
return false ;
2004-11-12 09:37:59 +00:00
}
2008-04-08 10:25:51 +02:00
start = timeval_current ( ) ;
end = timeval_add ( & start , ( 120 * sec ) , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
2006-10-16 13:06:41 +00:00
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo2 ) ;
2004-11-12 09:37:59 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " fileinfo failed: %s \n " , nt_errstr ( status ) ) ) ;
2007-10-06 22:28:14 +00:00
ret = false ;
2004-11-12 09:37:59 +00:00
break ;
}
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " write time %s \n " ,
nt_time_string ( tctx , finfo2 . basic_info . out . write_time ) ) ;
2004-11-12 09:37:59 +00:00
if ( finfo1 . basic_info . out . write_time ! = finfo2 . basic_info . out . write_time ) {
2008-04-08 10:25:51 +02:00
double diff = timeval_elapsed ( & start ) ;
2009-06-03 14:12:18 -07:00
if ( diff < ( TIMEDELAY_SECS * sec * 0.75 ) ) { /* 0.75 to cope with vmware timing */
2008-04-08 10:25:51 +02:00
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1 sec == %.2f)(wrong!) \n " ,
diff , sec ) ;
2007-12-01 11:39:39 +01:00
ret = false ;
break ;
}
2008-04-08 10:25:51 +02:00
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1 sec == %.2f)(correct) \n " ,
diff , sec ) ;
2004-11-12 09:37:59 +00:00
break ;
}
fflush ( stdout ) ;
2008-04-08 10:25:51 +02:00
msleep ( 1 * msec ) ;
2004-11-12 09:37:59 +00:00
}
if ( finfo1 . basic_info . out . write_time = = finfo2 . basic_info . out . write_time ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL ,
" Server did not update write time (wrong!) " ) ;
2007-10-06 22:28:14 +00:00
ret = false ;
2004-11-12 09:37:59 +00:00
}
if ( fnum1 ! = - 1 )
smbcli_close ( cli - > tree , fnum1 ) ;
smbcli_unlink ( cli - > tree , fname ) ;
smbcli_deltree ( cli - > tree , BASEDIR ) ;
return ret ;
}
2008-09-08 08:31:34 +02:00
static bool test_delayed_write_update1 ( struct torture_context * tctx , struct smbcli_state * cli )
{
union smb_fileinfo finfo1 , finfo2 , finfo3 , pinfo4 ;
const char * fname = BASEDIR " \\ torture_file1.txt " ;
NTSTATUS status ;
int fnum1 = - 1 ;
bool ret = true ;
ssize_t written ;
struct timeval start ;
struct timeval end ;
int used_delay = torture_setting_int ( tctx , " writetimeupdatedelay " , 2000000 ) ;
int normal_delay = 2000000 ;
double sec = ( ( double ) used_delay ) / ( ( double ) normal_delay ) ;
int msec = 1000 * sec ;
char buf [ 2048 ] ;
2008-09-08 14:58:58 -07:00
torture_comment ( tctx , " \n Running test_delayed_write_update1 \n " ) ;
2008-09-08 08:31:34 +02:00
if ( ! torture_setup_dir ( cli , BASEDIR ) ) {
return false ;
}
fnum1 = smbcli_open ( cli - > tree , fname , O_RDWR | O_CREAT , DENY_NONE ) ;
if ( fnum1 = = - 1 ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL , " Failed to open %s " , fname ) ;
2008-09-08 08:31:34 +02:00
return false ;
}
memset ( buf , ' x ' , 2048 ) ;
written = smbcli_write ( cli - > tree , fnum1 , 0 , buf , 0 , 2048 ) ;
/* 3 second delay to ensure we get past any 2 second time
granularity ( older systems may have that ) */
msleep ( 3 * msec ) ;
finfo1 . all_info . level = RAW_FILEINFO_ALL_INFO ;
finfo1 . all_info . in . file . fnum = fnum1 ;
finfo2 = finfo1 ;
finfo3 = finfo1 ;
pinfo4 . all_info . level = RAW_FILEINFO_ALL_INFO ;
pinfo4 . all_info . in . file . path = fname ;
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo1 ) ;
2008-09-18 16:05:46 +02:00
torture_assert_ntstatus_ok ( tctx , status , " fileinfo failed " ) ;
2008-09-08 08:31:34 +02:00
torture_comment ( tctx , " Initial write time %s \n " ,
nt_time_string ( tctx , finfo1 . all_info . out . write_time ) ) ;
/* Do a zero length SMBwrite call to truncate. */
written = smbcli_smbwrite ( cli - > tree , fnum1 , " x " , 1024 , 0 ) ;
if ( written ! = 0 ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL ,
" write failed - wrote %d bytes (%s) \n " ,
( int ) written , __location__ ) ;
2008-09-08 08:31:34 +02:00
return false ;
}
start = timeval_current ( ) ;
end = timeval_add ( & start , ( 120 * sec ) , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo2 ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " fileinfo failed: %s \n " , nt_errstr ( status ) ) ) ;
ret = false ;
break ;
}
if ( finfo2 . all_info . out . size ! = 1024 ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL ,
" file not truncated, size = %u (should be 1024) " ,
( unsigned int ) finfo2 . all_info . out . size ) ;
2008-09-08 08:31:34 +02:00
ret = false ;
break ;
}
torture_comment ( tctx , " write time %s \n " ,
nt_time_string ( tctx , finfo2 . all_info . out . write_time ) ) ;
if ( finfo1 . all_info . out . write_time ! = finfo2 . all_info . out . write_time ) {
double diff = timeval_elapsed ( & start ) ;
if ( diff > ( 0.25 * sec * 0.75 ) ) { /* 0.75 to cope with vmware timing */
torture_comment ( tctx , " After SMBwrite truncate "
" server updated write_time after %.2f seconds "
" (1 sec == %.2f)(wrong!) \n " ,
diff , sec ) ;
ret = false ;
break ;
}
torture_comment ( tctx , " After SMBwrite truncate "
" server updated write_time after %.2f seconds "
" (1 sec == %.2f)(correct) \n " ,
diff , sec ) ;
break ;
}
fflush ( stdout ) ;
msleep ( 1 * msec ) ;
}
if ( finfo1 . all_info . out . write_time = = finfo2 . all_info . out . write_time ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL ,
" Server did not update write time (wrong!) " ) ;
2008-09-08 08:31:34 +02:00
ret = false ;
}
2009-08-24 18:21:23 -07:00
fflush ( stdout ) ;
msleep ( 2 * msec ) ;
2008-09-08 08:31:34 +02:00
/* Do a non-zero length SMBwrite and make sure it doesn't update the write time. */
written = smbcli_smbwrite ( cli - > tree , fnum1 , " x " , 0 , 1 ) ;
if ( written ! = 1 ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL ,
" write failed - wrote %d bytes (%s) " ,
( int ) written , __location__ ) ;
2008-09-08 08:31:34 +02:00
return false ;
}
start = timeval_current ( ) ;
end = timeval_add ( & start , ( 10 * sec ) , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo3 ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " fileinfo failed: %s \n " , nt_errstr ( status ) ) ) ;
ret = false ;
break ;
}
if ( finfo3 . all_info . out . size ! = 1024 ) {
2008-09-08 14:58:58 -07:00
DEBUG ( 0 , ( " file not truncated, size = %u (should be 1024) \n " ,
( unsigned int ) finfo3 . all_info . out . size ) ) ;
2008-09-08 08:31:34 +02:00
ret = false ;
break ;
}
torture_comment ( tctx , " write time %s \n " ,
nt_time_string ( tctx , finfo3 . all_info . out . write_time ) ) ;
if ( finfo2 . all_info . out . write_time ! = finfo3 . all_info . out . write_time ) {
double diff = timeval_elapsed ( & start ) ;
torture_comment ( tctx , " server updated write_time after %.2f seconds "
" (1 sec == %.2f)(correct) \n " ,
diff , sec ) ;
break ;
}
fflush ( stdout ) ;
msleep ( 1 * msec ) ;
}
if ( finfo2 . all_info . out . write_time ! = finfo3 . all_info . out . write_time ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL ,
" Server updated write time (wrong!) " ) ;
2008-09-08 08:31:34 +02:00
ret = false ;
}
/* the close should trigger an write time update */
smbcli_close ( cli - > tree , fnum1 ) ;
fnum1 = - 1 ;
status = smb_raw_pathinfo ( cli - > tree , tctx , & pinfo4 ) ;
2008-09-18 16:05:46 +02:00
torture_assert_ntstatus_ok ( tctx , status , " pathinfo failed " ) ;
2008-09-08 08:31:34 +02:00
if ( finfo3 . all_info . out . write_time = = pinfo4 . all_info . out . write_time ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL ,
" Server did not update write time on close (wrong!) " ) ;
2008-09-08 08:31:34 +02:00
ret = false ;
} else if ( finfo3 . all_info . out . write_time < pinfo4 . all_info . out . write_time ) {
torture_comment ( tctx , " Server updated write time on close (correct) \n " ) ;
}
if ( fnum1 ! = - 1 )
smbcli_close ( cli - > tree , fnum1 ) ;
smbcli_unlink ( cli - > tree , fname ) ;
smbcli_deltree ( cli - > tree , BASEDIR ) ;
return ret ;
}
2008-09-05 14:24:36 -07:00
/* Updating with a SMBwrite of zero length
* changes the write time immediately - even on expand . */
static bool test_delayed_write_update1a ( struct torture_context * tctx , struct smbcli_state * cli )
{
2008-09-08 08:31:34 +02:00
union smb_fileinfo finfo1 , finfo2 , finfo3 , pinfo4 ;
2008-09-05 14:24:36 -07:00
const char * fname = BASEDIR " \\ torture_file1a.txt " ;
NTSTATUS status ;
int fnum1 = - 1 ;
bool ret = true ;
ssize_t written ;
struct timeval start ;
struct timeval end ;
int used_delay = torture_setting_int ( tctx , " writetimeupdatedelay " , 2000000 ) ;
int normal_delay = 2000000 ;
double sec = ( ( double ) used_delay ) / ( ( double ) normal_delay ) ;
int msec = 1000 * sec ;
char buf [ 2048 ] ;
2008-09-08 14:58:58 -07:00
torture_comment ( tctx , " \n Running test_delayed_write_update1a \n " ) ;
2008-09-05 14:24:36 -07:00
if ( ! torture_setup_dir ( cli , BASEDIR ) ) {
return false ;
}
fnum1 = smbcli_open ( cli - > tree , fname , O_RDWR | O_CREAT , DENY_NONE ) ;
if ( fnum1 = = - 1 ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL , " Failed to open %s " , fname ) ;
2008-09-05 14:24:36 -07:00
return false ;
}
memset ( buf , ' x ' , 2048 ) ;
written = smbcli_write ( cli - > tree , fnum1 , 0 , buf , 0 , 2048 ) ;
/* 3 second delay to ensure we get past any 2 second time
granularity ( older systems may have that ) */
msleep ( 3 * msec ) ;
finfo1 . all_info . level = RAW_FILEINFO_ALL_INFO ;
finfo1 . all_info . in . file . fnum = fnum1 ;
finfo2 = finfo1 ;
2008-09-08 08:31:34 +02:00
finfo3 = finfo1 ;
pinfo4 . all_info . level = RAW_FILEINFO_ALL_INFO ;
pinfo4 . all_info . in . file . path = fname ;
2008-09-05 14:24:36 -07:00
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo1 ) ;
2008-09-18 16:05:46 +02:00
torture_assert_ntstatus_ok ( tctx , status , " fileinfo failed " ) ;
2008-09-05 14:24:36 -07:00
torture_comment ( tctx , " Initial write time %s \n " ,
nt_time_string ( tctx , finfo1 . all_info . out . write_time ) ) ;
/* Do a zero length SMBwrite call to truncate. */
written = smbcli_smbwrite ( cli - > tree , fnum1 , " x " , 10240 , 0 ) ;
if ( written ! = 0 ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL , " write failed - wrote %d bytes (%s) " ,
2008-09-05 14:24:36 -07:00
( int ) written , __location__ ) ;
return false ;
}
start = timeval_current ( ) ;
end = timeval_add ( & start , ( 120 * sec ) , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo2 ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL , " fileinfo failed: %s " ,
nt_errstr ( status ) ) ;
2008-09-05 14:24:36 -07:00
ret = false ;
break ;
}
if ( finfo2 . all_info . out . size ! = 10240 ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL ,
" file not truncated, size = %u (should be 10240) " ,
( unsigned int ) finfo2 . all_info . out . size ) ;
2008-09-05 14:24:36 -07:00
ret = false ;
break ;
}
torture_comment ( tctx , " write time %s \n " ,
nt_time_string ( tctx , finfo2 . all_info . out . write_time ) ) ;
if ( finfo1 . all_info . out . write_time ! = finfo2 . all_info . out . write_time ) {
double diff = timeval_elapsed ( & start ) ;
if ( diff > ( 0.25 * sec * 0.75 ) ) { /* 0.75 to cope with vmware timing */
torture_comment ( tctx , " After SMBwrite truncate "
" server updated write_time after %.2f seconds "
" (1 sec == %.2f)(wrong!) \n " ,
diff , sec ) ;
ret = false ;
break ;
}
torture_comment ( tctx , " After SMBwrite truncate "
" server updated write_time after %.2f seconds "
" (1 sec == %.2f)(correct) \n " ,
diff , sec ) ;
break ;
}
fflush ( stdout ) ;
msleep ( 1 * msec ) ;
}
if ( finfo1 . all_info . out . write_time = = finfo2 . all_info . out . write_time ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL ,
" Server did not update write time (wrong!) " ) ;
2008-09-05 14:24:36 -07:00
ret = false ;
}
2009-08-24 18:21:23 -07:00
fflush ( stdout ) ;
msleep ( 2 * msec ) ;
2008-09-08 08:31:34 +02:00
/* Do a non-zero length SMBwrite and make sure it doesn't update the write time. */
written = smbcli_smbwrite ( cli - > tree , fnum1 , " x " , 0 , 1 ) ;
2008-09-18 16:05:46 +02:00
torture_assert_int_equal ( tctx , written , 1 ,
" unexpected number of bytes written " ) ;
2008-09-08 08:31:34 +02:00
start = timeval_current ( ) ;
end = timeval_add ( & start , ( 10 * sec ) , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo3 ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL , " fileinfo failed: %s \n " ,
nt_errstr ( status ) ) ;
2008-09-08 08:31:34 +02:00
ret = false ;
break ;
}
if ( finfo3 . all_info . out . size ! = 10240 ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL ,
" file not truncated, size = %u (should be 10240) " ,
( unsigned int ) finfo3 . all_info . out . size ) ;
2008-09-08 08:31:34 +02:00
ret = false ;
break ;
}
torture_comment ( tctx , " write time %s \n " ,
nt_time_string ( tctx , finfo3 . all_info . out . write_time ) ) ;
if ( finfo2 . all_info . out . write_time ! = finfo3 . all_info . out . write_time ) {
double diff = timeval_elapsed ( & start ) ;
torture_comment ( tctx , " server updated write_time after %.2f seconds "
" (1 sec == %.2f)(correct) \n " ,
diff , sec ) ;
break ;
}
fflush ( stdout ) ;
msleep ( 1 * msec ) ;
}
if ( finfo2 . all_info . out . write_time ! = finfo3 . all_info . out . write_time ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL ,
" Server updated write time (wrong!) " ) ;
2008-09-08 08:31:34 +02:00
ret = false ;
}
/* the close should trigger an write time update */
smbcli_close ( cli - > tree , fnum1 ) ;
fnum1 = - 1 ;
status = smb_raw_pathinfo ( cli - > tree , tctx , & pinfo4 ) ;
2008-09-18 16:05:46 +02:00
torture_assert_ntstatus_ok ( tctx , status , " pathinfo failed " ) ;
2008-09-08 08:31:34 +02:00
if ( finfo3 . all_info . out . write_time = = pinfo4 . all_info . out . write_time ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL ,
" Server did not update write time on close (wrong!) " ) ;
2008-09-08 08:31:34 +02:00
ret = false ;
} else if ( finfo3 . all_info . out . write_time < pinfo4 . all_info . out . write_time ) {
torture_comment ( tctx , " Server updated write time on close (correct) \n " ) ;
}
2008-09-05 14:24:36 -07:00
if ( fnum1 ! = - 1 )
smbcli_close ( cli - > tree , fnum1 ) ;
smbcli_unlink ( cli - > tree , fname ) ;
smbcli_deltree ( cli - > tree , BASEDIR ) ;
return ret ;
}
/* Updating with a SET_FILE_END_OF_FILE_INFO
* changes the write time immediately - even on expand . */
static bool test_delayed_write_update1b ( struct torture_context * tctx , struct smbcli_state * cli )
{
2008-09-08 08:31:34 +02:00
union smb_fileinfo finfo1 , finfo2 , finfo3 , pinfo4 ;
2008-09-05 14:24:36 -07:00
const char * fname = BASEDIR " \\ torture_file1b.txt " ;
NTSTATUS status ;
int fnum1 = - 1 ;
bool ret = true ;
ssize_t written ;
struct timeval start ;
struct timeval end ;
int used_delay = torture_setting_int ( tctx , " writetimeupdatedelay " , 2000000 ) ;
int normal_delay = 2000000 ;
double sec = ( ( double ) used_delay ) / ( ( double ) normal_delay ) ;
int msec = 1000 * sec ;
char buf [ 2048 ] ;
2008-09-08 14:58:58 -07:00
torture_comment ( tctx , " \n Running test_delayed_write_update1b \n " ) ;
2008-09-05 14:24:36 -07:00
if ( ! torture_setup_dir ( cli , BASEDIR ) ) {
return false ;
}
fnum1 = smbcli_open ( cli - > tree , fname , O_RDWR | O_CREAT , DENY_NONE ) ;
if ( fnum1 = = - 1 ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL , " Failed to open %s " , fname ) ;
2008-09-05 14:24:36 -07:00
return false ;
}
memset ( buf , ' x ' , 2048 ) ;
written = smbcli_write ( cli - > tree , fnum1 , 0 , buf , 0 , 2048 ) ;
/* 3 second delay to ensure we get past any 2 second time
granularity ( older systems may have that ) */
msleep ( 3 * msec ) ;
finfo1 . all_info . level = RAW_FILEINFO_ALL_INFO ;
finfo1 . all_info . in . file . fnum = fnum1 ;
finfo2 = finfo1 ;
2008-09-08 08:31:34 +02:00
finfo3 = finfo1 ;
pinfo4 . all_info . level = RAW_FILEINFO_ALL_INFO ;
pinfo4 . all_info . in . file . path = fname ;
2008-09-05 14:24:36 -07:00
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo1 ) ;
2008-09-18 16:05:46 +02:00
torture_assert_ntstatus_ok ( tctx , status , " fileinfo failed " ) ;
2008-09-05 14:24:36 -07:00
torture_comment ( tctx , " Initial write time %s \n " ,
nt_time_string ( tctx , finfo1 . all_info . out . write_time ) ) ;
/* Do a SET_END_OF_FILE_INFO call to truncate. */
status = smbcli_ftruncate ( cli - > tree , fnum1 , ( uint64_t ) 10240 ) ;
2008-09-18 16:05:46 +02:00
torture_assert_ntstatus_ok ( tctx , status , " SET_END_OF_FILE failed " ) ;
2008-09-05 14:24:36 -07:00
start = timeval_current ( ) ;
end = timeval_add ( & start , ( 120 * sec ) , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo2 ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " fileinfo failed: %s \n " , nt_errstr ( status ) ) ) ;
ret = false ;
break ;
}
if ( finfo2 . all_info . out . size ! = 10240 ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL ,
" file not truncated (size = %u, should be 10240) " ,
( unsigned int ) finfo2 . all_info . out . size ) ;
2008-09-05 14:24:36 -07:00
ret = false ;
break ;
}
torture_comment ( tctx , " write time %s \n " ,
nt_time_string ( tctx , finfo2 . all_info . out . write_time ) ) ;
if ( finfo1 . all_info . out . write_time ! = finfo2 . all_info . out . write_time ) {
double diff = timeval_elapsed ( & start ) ;
if ( diff > ( 0.25 * sec * 0.75 ) ) { /* 0.75 to cope with vmware timing */
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL ,
" After SET_END_OF_FILE truncate "
2008-09-05 14:24:36 -07:00
" server updated write_time after %.2f seconds "
2008-09-18 16:05:46 +02:00
" (1 sec == %.2f)(wrong!) " ,
2008-09-05 14:24:36 -07:00
diff , sec ) ;
ret = false ;
break ;
}
torture_comment ( tctx , " After SET_END_OF_FILE truncate "
" server updated write_time after %.2f seconds "
" (1 sec == %.2f)(correct) \n " ,
diff , sec ) ;
break ;
}
fflush ( stdout ) ;
msleep ( 1 * msec ) ;
}
if ( finfo1 . all_info . out . write_time = = finfo2 . all_info . out . write_time ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL ,
" Server did not update write time (wrong!) " ) ;
2008-09-05 14:24:36 -07:00
ret = false ;
}
2009-08-24 18:21:23 -07:00
fflush ( stdout ) ;
msleep ( 2 * msec ) ;
2008-09-08 08:31:34 +02:00
/* Do a non-zero length SMBwrite and make sure it doesn't update the write time. */
written = smbcli_smbwrite ( cli - > tree , fnum1 , " x " , 0 , 1 ) ;
2008-09-18 16:05:46 +02:00
torture_assert_int_equal ( tctx , written , 1 ,
" unexpected number of bytes written " ) ;
2008-09-08 08:31:34 +02:00
start = timeval_current ( ) ;
end = timeval_add ( & start , ( 10 * sec ) , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo3 ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL ,
" fileinfo failed: %s " , nt_errstr ( status ) ) ;
2008-09-08 08:31:34 +02:00
ret = false ;
break ;
}
if ( finfo3 . all_info . out . size ! = 10240 ) {
2008-09-08 14:58:58 -07:00
DEBUG ( 0 , ( " file not truncated (size = %u, should be 10240) \n " ,
( unsigned int ) finfo3 . all_info . out . size ) ) ;
2008-09-08 08:31:34 +02:00
ret = false ;
break ;
}
torture_comment ( tctx , " write time %s \n " ,
nt_time_string ( tctx , finfo3 . all_info . out . write_time ) ) ;
if ( finfo2 . all_info . out . write_time ! = finfo3 . all_info . out . write_time ) {
double diff = timeval_elapsed ( & start ) ;
torture_comment ( tctx , " server updated write_time after %.2f seconds "
" (1 sec == %.2f)(correct) \n " ,
diff , sec ) ;
break ;
}
fflush ( stdout ) ;
msleep ( 1 * msec ) ;
}
if ( finfo2 . all_info . out . write_time ! = finfo3 . all_info . out . write_time ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL , " Server updated write time (wrong!) \n " ) ;
2008-09-08 08:31:34 +02:00
ret = false ;
}
/* the close should trigger an write time update */
smbcli_close ( cli - > tree , fnum1 ) ;
fnum1 = - 1 ;
status = smb_raw_pathinfo ( cli - > tree , tctx , & pinfo4 ) ;
2008-09-18 16:05:46 +02:00
torture_assert_ntstatus_ok ( tctx , status , " pathinfo failed " ) ;
2008-09-08 08:31:34 +02:00
if ( finfo3 . all_info . out . write_time = = pinfo4 . all_info . out . write_time ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL , " Server did not update write time on close (wrong!) \n " ) ;
2008-09-08 08:31:34 +02:00
ret = false ;
} else if ( finfo3 . all_info . out . write_time < pinfo4 . all_info . out . write_time ) {
torture_comment ( tctx , " Server updated write time on close (correct) \n " ) ;
}
2008-09-05 14:24:36 -07:00
if ( fnum1 ! = - 1 )
smbcli_close ( cli - > tree , fnum1 ) ;
smbcli_unlink ( cli - > tree , fname ) ;
smbcli_deltree ( cli - > tree , BASEDIR ) ;
return ret ;
}
/* Updating with a SET_ALLOCATION_INFO (truncate) does so immediately. */
static bool test_delayed_write_update1c ( struct torture_context * tctx , struct smbcli_state * cli )
{
union smb_setfileinfo parms ;
2008-09-08 08:31:34 +02:00
union smb_fileinfo finfo1 , finfo2 , finfo3 , pinfo4 ;
2008-09-05 14:24:36 -07:00
const char * fname = BASEDIR " \\ torture_file1c.txt " ;
NTSTATUS status ;
int fnum1 = - 1 ;
bool ret = true ;
ssize_t written ;
struct timeval start ;
struct timeval end ;
int used_delay = torture_setting_int ( tctx , " writetimeupdatedelay " , 2000000 ) ;
int normal_delay = 2000000 ;
double sec = ( ( double ) used_delay ) / ( ( double ) normal_delay ) ;
int msec = 1000 * sec ;
char buf [ 2048 ] ;
2008-09-08 14:58:58 -07:00
torture_comment ( tctx , " \n Running test_delayed_write_update1c \n " ) ;
2008-09-05 14:24:36 -07:00
if ( ! torture_setup_dir ( cli , BASEDIR ) ) {
return false ;
}
fnum1 = smbcli_open ( cli - > tree , fname , O_RDWR | O_CREAT , DENY_NONE ) ;
if ( fnum1 = = - 1 ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL , " Failed to open %s " , fname ) ;
2008-09-05 14:24:36 -07:00
return false ;
}
memset ( buf , ' x ' , 2048 ) ;
written = smbcli_write ( cli - > tree , fnum1 , 0 , buf , 0 , 2048 ) ;
/* 3 second delay to ensure we get past any 2 second time
granularity ( older systems may have that ) */
msleep ( 3 * msec ) ;
finfo1 . all_info . level = RAW_FILEINFO_ALL_INFO ;
finfo1 . all_info . in . file . fnum = fnum1 ;
finfo2 = finfo1 ;
2008-09-08 08:31:34 +02:00
finfo3 = finfo1 ;
pinfo4 . all_info . level = RAW_FILEINFO_ALL_INFO ;
pinfo4 . all_info . in . file . path = fname ;
2008-09-05 14:24:36 -07:00
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo1 ) ;
2008-09-18 16:05:46 +02:00
torture_assert_ntstatus_ok ( tctx , status , " fileinfo failed " ) ;
2008-09-05 14:24:36 -07:00
torture_comment ( tctx , " Initial write time %s \n " ,
nt_time_string ( tctx , finfo1 . all_info . out . write_time ) ) ;
/* Do a SET_ALLOCATION_SIZE call to truncate. */
parms . allocation_info . level = RAW_SFILEINFO_ALLOCATION_INFO ;
parms . allocation_info . in . file . fnum = fnum1 ;
parms . allocation_info . in . alloc_size = 0 ;
2008-09-18 16:05:46 +02:00
status = smb_raw_setfileinfo ( cli - > tree , & parms ) ;
2008-09-05 14:24:36 -07:00
2008-09-18 16:05:46 +02:00
torture_assert_ntstatus_ok ( tctx , status ,
" RAW_SFILEINFO_ALLOCATION_INFO failed " ) ;
2008-09-05 14:24:36 -07:00
start = timeval_current ( ) ;
end = timeval_add ( & start , ( 120 * sec ) , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo2 ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL , " fileinfo failed: %s " ,
nt_errstr ( status ) ) ;
2008-09-05 14:24:36 -07:00
ret = false ;
break ;
}
if ( finfo2 . all_info . out . size ! = 0 ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL ,
" file not truncated (size = %u, should be 10240) " ,
( unsigned int ) finfo2 . all_info . out . size ) ;
2008-09-05 14:24:36 -07:00
ret = false ;
break ;
}
torture_comment ( tctx , " write time %s \n " ,
nt_time_string ( tctx , finfo2 . all_info . out . write_time ) ) ;
if ( finfo1 . all_info . out . write_time ! = finfo2 . all_info . out . write_time ) {
double diff = timeval_elapsed ( & start ) ;
if ( diff > ( 0.25 * sec * 0.75 ) ) { /* 0.75 to cope with vmware timing */
torture_comment ( tctx , " After SET_ALLOCATION_INFO truncate "
" server updated write_time after %.2f seconds "
" (1 sec == %.2f)(wrong!) \n " ,
diff , sec ) ;
ret = false ;
break ;
}
torture_comment ( tctx , " After SET_ALLOCATION_INFO truncate "
" server updated write_time after %.2f seconds "
" (1 sec == %.2f)(correct) \n " ,
diff , sec ) ;
break ;
}
fflush ( stdout ) ;
msleep ( 1 * msec ) ;
}
if ( finfo1 . all_info . out . write_time = = finfo2 . all_info . out . write_time ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL ,
" Server did not update write time (wrong!) " ) ;
2008-09-05 14:24:36 -07:00
ret = false ;
}
2009-08-24 18:21:23 -07:00
fflush ( stdout ) ;
msleep ( 2 * msec ) ;
2008-09-08 08:31:34 +02:00
/* Do a non-zero length SMBwrite and make sure it doesn't update the write time. */
written = smbcli_smbwrite ( cli - > tree , fnum1 , " x " , 0 , 1 ) ;
2008-09-18 16:05:46 +02:00
torture_assert_int_equal ( tctx , written , 1 ,
" Unexpected number of bytes written " ) ;
2008-09-08 08:31:34 +02:00
start = timeval_current ( ) ;
end = timeval_add ( & start , ( 10 * sec ) , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo3 ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL , " fileinfo failed: %s " ,
nt_errstr ( status ) ) ;
2008-09-08 08:31:34 +02:00
ret = false ;
break ;
}
if ( finfo3 . all_info . out . size ! = 1 ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL , " file not expanded " ) ;
2008-09-08 08:31:34 +02:00
ret = false ;
break ;
}
torture_comment ( tctx , " write time %s \n " ,
nt_time_string ( tctx , finfo3 . all_info . out . write_time ) ) ;
if ( finfo2 . all_info . out . write_time ! = finfo3 . all_info . out . write_time ) {
double diff = timeval_elapsed ( & start ) ;
torture_comment ( tctx , " server updated write_time after %.2f seconds "
" (1 sec == %.2f)(correct) \n " ,
diff , sec ) ;
break ;
}
fflush ( stdout ) ;
msleep ( 1 * msec ) ;
}
if ( finfo2 . all_info . out . write_time ! = finfo3 . all_info . out . write_time ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL ,
" Server updated write time (wrong!) " ) ;
2008-09-08 08:31:34 +02:00
ret = false ;
}
/* the close should trigger an write time update */
smbcli_close ( cli - > tree , fnum1 ) ;
fnum1 = - 1 ;
status = smb_raw_pathinfo ( cli - > tree , tctx , & pinfo4 ) ;
2008-09-18 16:05:46 +02:00
torture_assert_ntstatus_ok ( tctx , status , " pathinfo failed " ) ;
2008-09-08 08:31:34 +02:00
if ( finfo3 . all_info . out . write_time = = pinfo4 . all_info . out . write_time ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL , " Server did not update write time on close (wrong!) \n " ) ;
2008-09-08 08:31:34 +02:00
ret = false ;
} else if ( finfo3 . all_info . out . write_time < pinfo4 . all_info . out . write_time ) {
torture_comment ( tctx , " Server updated write time on close (correct) \n " ) ;
}
2008-09-05 14:24:36 -07:00
if ( fnum1 ! = - 1 )
smbcli_close ( cli - > tree , fnum1 ) ;
smbcli_unlink ( cli - > tree , fname ) ;
smbcli_deltree ( cli - > tree , BASEDIR ) ;
return ret ;
}
/*
2004-11-20 01:09:25 +00:00
* Do as above , but using 2 connections .
*/
2007-10-06 22:28:14 +00:00
static bool test_delayed_write_update2 ( struct torture_context * tctx , struct smbcli_state * cli ,
2006-10-16 13:06:41 +00:00
struct smbcli_state * cli2 )
2004-11-20 01:09:25 +00:00
{
union smb_fileinfo finfo1 , finfo2 ;
const char * fname = BASEDIR " \\ torture_file.txt " ;
NTSTATUS status ;
int fnum1 = - 1 ;
2005-03-10 21:42:20 +00:00
int fnum2 = - 1 ;
2007-10-06 22:28:14 +00:00
bool ret = true ;
2004-11-20 01:09:25 +00:00
ssize_t written ;
2008-04-08 10:25:51 +02:00
struct timeval start ;
struct timeval end ;
int used_delay = torture_setting_int ( tctx , " writetimeupdatedelay " , 2000000 ) ;
int normal_delay = 2000000 ;
double sec = ( ( double ) used_delay ) / ( ( double ) normal_delay ) ;
int msec = 1000 * sec ;
2006-03-10 20:49:20 +00:00
union smb_flush flsh ;
2004-11-20 01:09:25 +00:00
2008-09-08 14:58:58 -07:00
torture_comment ( tctx , " \n Running test_delayed_write_update2 \n " ) ;
2004-11-20 01:09:25 +00:00
if ( ! torture_setup_dir ( cli , BASEDIR ) ) {
2007-10-06 22:28:14 +00:00
return false ;
2004-11-20 01:09:25 +00:00
}
fnum1 = smbcli_open ( cli - > tree , fname , O_RDWR | O_CREAT , DENY_NONE ) ;
if ( fnum1 = = - 1 ) {
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " Failed to open %s \n " , fname ) ;
2007-10-06 22:28:14 +00:00
return false ;
2004-11-20 01:09:25 +00:00
}
finfo1 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
2006-03-12 22:48:25 +00:00
finfo1 . basic_info . in . file . fnum = fnum1 ;
2004-11-20 01:09:25 +00:00
finfo2 = finfo1 ;
2006-10-16 13:06:41 +00:00
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo1 ) ;
2004-11-20 01:09:25 +00:00
2008-09-18 16:05:46 +02:00
torture_assert_ntstatus_ok ( tctx , status , " fileinfo failed " ) ;
2004-11-20 01:09:25 +00:00
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " Initial write time %s \n " ,
nt_time_string ( tctx , finfo1 . basic_info . out . write_time ) ) ;
2004-11-20 01:09:25 +00:00
/* 3 second delay to ensure we get past any 2 second time
granularity ( older systems may have that ) */
2008-04-08 10:25:51 +02:00
msleep ( 3 * msec ) ;
2004-11-20 01:09:25 +00:00
2004-11-20 01:51:32 +00:00
{
/* Try using setfileinfo instead of write to update write time. */
union smb_setfileinfo sfinfo ;
time_t t_set = time ( NULL ) ;
sfinfo . basic_info . level = RAW_SFILEINFO_BASIC_INFO ;
2006-03-12 22:48:25 +00:00
sfinfo . basic_info . in . file . fnum = fnum1 ;
2004-11-20 01:51:32 +00:00
sfinfo . basic_info . in . create_time = finfo1 . basic_info . out . create_time ;
sfinfo . basic_info . in . access_time = finfo1 . basic_info . out . access_time ;
/* I tried this with both + and - ve to see if it makes a different.
It doesn ' t - once the filetime is set via setfileinfo it stays that way . */
# if 1
unix_to_nt_time ( & sfinfo . basic_info . in . write_time , t_set - 30000 ) ;
# else
unix_to_nt_time ( & sfinfo . basic_info . in . write_time , t_set + 30000 ) ;
# endif
sfinfo . basic_info . in . change_time = finfo1 . basic_info . out . change_time ;
sfinfo . basic_info . in . attrib = finfo1 . basic_info . out . attrib ;
status = smb_raw_setfileinfo ( cli - > tree , & sfinfo ) ;
2004-11-20 01:09:25 +00:00
2008-09-18 16:05:46 +02:00
torture_assert_ntstatus_ok ( tctx , status , " sfileinfo failed " ) ;
2004-11-20 01:09:25 +00:00
}
2007-12-01 11:39:39 +01:00
finfo2 . basic_info . in . file . path = fname ;
2004-11-20 01:09:25 +00:00
2007-12-01 11:39:39 +01:00
status = smb_raw_pathinfo ( cli2 - > tree , tctx , & finfo2 ) ;
2004-11-20 01:09:25 +00:00
2007-12-01 11:39:39 +01:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " fileinfo failed: %s \n " , nt_errstr ( status ) ) ) ;
return false ;
2004-11-20 01:09:25 +00:00
}
2007-12-01 11:39:39 +01:00
torture_comment ( tctx , " write time %s \n " ,
nt_time_string ( tctx , finfo2 . basic_info . out . write_time ) ) ;
if ( finfo1 . basic_info . out . write_time ! = finfo2 . basic_info . out . write_time ) {
torture_comment ( tctx , " Server updated write_time (correct) \n " ) ;
} else {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL , " Server did not update write time (wrong!) \n " ) ;
2007-10-06 22:28:14 +00:00
ret = false ;
2004-11-20 01:09:25 +00:00
}
2004-11-20 01:51:32 +00:00
/* Now try a write to see if the write time gets reset. */
finfo1 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
2006-03-12 22:48:25 +00:00
finfo1 . basic_info . in . file . fnum = fnum1 ;
2004-11-20 01:51:32 +00:00
finfo2 = finfo1 ;
2006-10-16 13:06:41 +00:00
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo1 ) ;
2004-11-20 01:51:32 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " fileinfo failed: %s \n " , nt_errstr ( status ) ) ) ;
2007-10-06 22:28:14 +00:00
return false ;
2004-11-20 01:51:32 +00:00
}
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " Modified write time %s \n " ,
nt_time_string ( tctx , finfo1 . basic_info . out . write_time ) ) ;
2004-11-20 01:51:32 +00:00
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " Doing a 10 byte write to extend the file and see if this changes the last write time. \n " ) ;
2004-11-20 01:51:32 +00:00
2004-11-20 02:33:12 +00:00
written = smbcli_write ( cli - > tree , fnum1 , 0 , " 0123456789 " , 1 , 10 ) ;
if ( written ! = 10 ) {
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " write failed - wrote %d bytes (%s) \n " ,
2005-07-17 09:20:52 +00:00
( int ) written , __location__ ) ;
2007-10-06 22:28:14 +00:00
return false ;
2004-11-20 01:51:32 +00:00
}
2004-11-20 02:33:12 +00:00
/* Just to prove to tridge that the an smbflush has no effect on
the write time : - ) . The setfileinfo IS STICKY . JRA . */
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " Doing flush after write \n " ) ;
2004-11-20 02:33:12 +00:00
2006-05-19 15:10:39 +00:00
flsh . flush . level = RAW_FLUSH_FLUSH ;
2006-03-12 22:48:25 +00:00
flsh . flush . in . file . fnum = fnum1 ;
2004-11-20 02:33:12 +00:00
status = smb_raw_flush ( cli - > tree , & flsh ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " smbflush failed: %s \n " , nt_errstr ( status ) ) ) ;
2007-10-06 22:28:14 +00:00
return false ;
2004-11-20 02:33:12 +00:00
}
2004-11-20 01:51:32 +00:00
/* Once the time was set using setfileinfo then it stays set - writes
don ' t have any effect . But make sure . */
2008-04-08 10:25:51 +02:00
start = timeval_current ( ) ;
end = timeval_add ( & start , ( 15 * sec ) , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
2006-10-16 13:06:41 +00:00
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo2 ) ;
2004-11-20 01:51:32 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " fileinfo failed: %s \n " , nt_errstr ( status ) ) ) ;
2007-10-06 22:28:14 +00:00
ret = false ;
2004-11-20 01:51:32 +00:00
break ;
}
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " write time %s \n " ,
nt_time_string ( tctx , finfo2 . basic_info . out . write_time ) ) ;
2004-11-20 01:51:32 +00:00
if ( finfo1 . basic_info . out . write_time ! = finfo2 . basic_info . out . write_time ) {
2008-04-08 10:25:51 +02:00
double diff = timeval_elapsed ( & start ) ;
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (wrong!) \n " ,
diff , sec ) ;
2007-12-01 11:39:39 +01:00
ret = false ;
2004-11-20 01:51:32 +00:00
break ;
}
fflush ( stdout ) ;
2008-04-08 10:25:51 +02:00
msleep ( 1 * msec ) ;
2004-11-20 01:51:32 +00:00
}
if ( finfo1 . basic_info . out . write_time = = finfo2 . basic_info . out . write_time ) {
2007-12-01 11:39:39 +01:00
torture_comment ( tctx , " Server did not update write time (correct) \n " ) ;
2005-03-10 21:42:20 +00:00
}
2009-08-24 18:21:23 -07:00
fflush ( stdout ) ;
msleep ( 2 * msec ) ;
2005-03-10 21:42:20 +00:00
fnum2 = smbcli_open ( cli - > tree , fname , O_RDWR , DENY_NONE ) ;
if ( fnum2 = = - 1 ) {
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " Failed to open %s \n " , fname ) ;
2007-10-06 22:28:14 +00:00
return false ;
2005-03-10 21:42:20 +00:00
}
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " Doing a 10 byte write to extend the file via second fd and see if this changes the last write time. \n " ) ;
2005-03-10 21:42:20 +00:00
written = smbcli_write ( cli - > tree , fnum2 , 0 , " 0123456789 " , 11 , 10 ) ;
if ( written ! = 10 ) {
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " write failed - wrote %d bytes (%s) \n " ,
2005-07-17 09:20:52 +00:00
( int ) written , __location__ ) ;
2007-10-06 22:28:14 +00:00
return false ;
2005-03-10 21:42:20 +00:00
}
2006-10-16 13:06:41 +00:00
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo2 ) ;
2005-03-10 21:42:20 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " fileinfo failed: %s \n " , nt_errstr ( status ) ) ) ;
2007-10-06 22:28:14 +00:00
return false ;
2005-03-10 21:42:20 +00:00
}
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " write time %s \n " ,
nt_time_string ( tctx , finfo2 . basic_info . out . write_time ) ) ;
2005-03-10 21:42:20 +00:00
if ( finfo1 . basic_info . out . write_time ! = finfo2 . basic_info . out . write_time ) {
2007-12-01 11:39:39 +01:00
torture_comment ( tctx , " Server updated write_time (wrong!) \n " ) ;
ret = false ;
2005-03-10 21:42:20 +00:00
}
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " Closing the first fd to see if write time updated. \n " ) ;
2005-03-10 21:42:20 +00:00
smbcli_close ( cli - > tree , fnum1 ) ;
fnum1 = - 1 ;
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " Doing a 10 byte write to extend the file via second fd and see if this changes the last write time. \n " ) ;
2005-03-10 21:42:20 +00:00
written = smbcli_write ( cli - > tree , fnum2 , 0 , " 0123456789 " , 21 , 10 ) ;
if ( written ! = 10 ) {
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " write failed - wrote %d bytes (%s) \n " ,
2005-07-17 09:20:52 +00:00
( int ) written , __location__ ) ;
2007-10-06 22:28:14 +00:00
return false ;
2005-03-10 21:42:20 +00:00
}
finfo1 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
2006-03-12 22:48:25 +00:00
finfo1 . basic_info . in . file . fnum = fnum2 ;
2005-03-10 21:42:20 +00:00
finfo2 = finfo1 ;
2006-10-16 13:06:41 +00:00
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo2 ) ;
2005-03-10 21:42:20 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " fileinfo failed: %s \n " , nt_errstr ( status ) ) ) ;
2007-10-06 22:28:14 +00:00
return false ;
2005-03-10 21:42:20 +00:00
}
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " write time %s \n " ,
nt_time_string ( tctx , finfo2 . basic_info . out . write_time ) ) ;
2005-03-10 21:42:20 +00:00
if ( finfo1 . basic_info . out . write_time ! = finfo2 . basic_info . out . write_time ) {
2007-12-01 11:39:39 +01:00
torture_comment ( tctx , " Server updated write_time (wrong!) \n " ) ;
ret = false ;
2005-03-10 21:42:20 +00:00
}
/* Once the time was set using setfileinfo then it stays set - writes
don ' t have any effect . But make sure . */
2008-04-08 10:25:51 +02:00
start = timeval_current ( ) ;
end = timeval_add ( & start , ( 15 * sec ) , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
2006-10-16 13:06:41 +00:00
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo2 ) ;
2005-03-10 21:42:20 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " fileinfo failed: %s \n " , nt_errstr ( status ) ) ) ;
2007-10-06 22:28:14 +00:00
ret = false ;
2005-03-10 21:42:20 +00:00
break ;
}
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " write time %s \n " ,
nt_time_string ( tctx , finfo2 . basic_info . out . write_time ) ) ;
2005-03-10 21:42:20 +00:00
if ( finfo1 . basic_info . out . write_time ! = finfo2 . basic_info . out . write_time ) {
2008-04-08 10:25:51 +02:00
double diff = timeval_elapsed ( & start ) ;
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (wrong!) \n " ,
diff , sec ) ;
2007-12-01 11:39:39 +01:00
ret = false ;
2005-03-10 21:42:20 +00:00
break ;
}
fflush ( stdout ) ;
2008-04-08 10:25:51 +02:00
msleep ( 1 * msec ) ;
2005-03-10 21:42:20 +00:00
}
if ( finfo1 . basic_info . out . write_time = = finfo2 . basic_info . out . write_time ) {
2007-12-01 11:39:39 +01:00
torture_comment ( tctx , " Server did not update write time (correct) \n " ) ;
2005-03-10 21:42:20 +00:00
}
2007-12-01 11:39:39 +01:00
torture_comment ( tctx , " Closing second fd to see if write time updated. \n " ) ;
2005-03-10 21:42:20 +00:00
smbcli_close ( cli - > tree , fnum2 ) ;
fnum2 = - 1 ;
fnum1 = smbcli_open ( cli - > tree , fname , O_RDWR , DENY_NONE ) ;
if ( fnum1 = = - 1 ) {
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " Failed to open %s \n " , fname ) ;
2007-10-06 22:28:14 +00:00
return false ;
2005-03-10 21:42:20 +00:00
}
finfo1 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
2006-03-12 22:48:25 +00:00
finfo1 . basic_info . in . file . fnum = fnum1 ;
2005-03-10 21:42:20 +00:00
finfo2 = finfo1 ;
2006-10-16 13:06:41 +00:00
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo1 ) ;
2005-03-10 21:42:20 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " fileinfo failed: %s \n " , nt_errstr ( status ) ) ) ;
2007-10-06 22:28:14 +00:00
return false ;
2005-03-10 21:42:20 +00:00
}
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " Second open initial write time %s \n " ,
nt_time_string ( tctx , finfo1 . basic_info . out . write_time ) ) ;
2005-03-10 21:42:20 +00:00
2008-04-08 10:25:51 +02:00
msleep ( 10 * msec ) ;
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " Doing a 10 byte write to extend the file to see if this changes the last write time. \n " ) ;
2005-03-10 21:42:20 +00:00
written = smbcli_write ( cli - > tree , fnum1 , 0 , " 0123456789 " , 31 , 10 ) ;
if ( written ! = 10 ) {
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " write failed - wrote %d bytes (%s) \n " ,
2005-07-17 09:20:52 +00:00
( int ) written , __location__ ) ;
2007-10-06 22:28:14 +00:00
return false ;
2005-03-10 21:42:20 +00:00
}
finfo1 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
2006-03-12 22:48:25 +00:00
finfo1 . basic_info . in . file . fnum = fnum1 ;
2005-03-10 21:42:20 +00:00
finfo2 = finfo1 ;
2006-10-16 13:06:41 +00:00
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo2 ) ;
2005-03-10 21:42:20 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " fileinfo failed: %s \n " , nt_errstr ( status ) ) ) ;
2007-10-06 22:28:14 +00:00
return false ;
2005-03-10 21:42:20 +00:00
}
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " write time %s \n " ,
nt_time_string ( tctx , finfo2 . basic_info . out . write_time ) ) ;
2005-03-10 21:42:20 +00:00
if ( finfo1 . basic_info . out . write_time ! = finfo2 . basic_info . out . write_time ) {
2007-12-01 11:39:39 +01:00
torture_comment ( tctx , " Server updated write_time (wrong!) \n " ) ;
ret = false ;
2005-03-10 21:42:20 +00:00
}
2007-12-01 11:39:39 +01:00
/* Now the write time should be updated again */
2008-04-08 10:25:51 +02:00
start = timeval_current ( ) ;
end = timeval_add ( & start , ( 15 * sec ) , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
2006-10-16 13:06:41 +00:00
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo2 ) ;
2005-03-10 21:42:20 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " fileinfo failed: %s \n " , nt_errstr ( status ) ) ) ;
2007-10-06 22:28:14 +00:00
ret = false ;
2005-03-10 21:42:20 +00:00
break ;
}
2006-10-16 13:06:41 +00:00
torture_comment ( tctx , " write time %s \n " ,
nt_time_string ( tctx , finfo2 . basic_info . out . write_time ) ) ;
2005-03-10 21:42:20 +00:00
if ( finfo1 . basic_info . out . write_time ! = finfo2 . basic_info . out . write_time ) {
2008-04-08 10:25:51 +02:00
double diff = timeval_elapsed ( & start ) ;
2009-06-03 14:12:18 -07:00
if ( diff < ( TIMEDELAY_SECS * sec * 0.75 ) ) { /* 0.75 to cope with vmware timing */
2008-04-08 10:25:51 +02:00
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (wrong!) \n " ,
diff , sec ) ;
2007-12-01 11:39:39 +01:00
ret = false ;
break ;
}
2008-04-08 10:25:51 +02:00
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (correct) \n " ,
diff , sec ) ;
2005-03-10 21:42:20 +00:00
break ;
}
fflush ( stdout ) ;
2008-04-08 10:25:51 +02:00
msleep ( 1 * msec ) ;
2005-03-10 21:42:20 +00:00
}
if ( finfo1 . basic_info . out . write_time = = finfo2 . basic_info . out . write_time ) {
2008-09-18 16:05:46 +02:00
torture_result ( tctx , TORTURE_FAIL , " Server did not update write time (wrong!) \n " ) ;
2007-12-01 11:39:39 +01:00
ret = false ;
2004-11-20 01:51:32 +00:00
}
2005-03-10 21:42:20 +00:00
2004-11-20 01:51:32 +00:00
/* One more test to do. We should read the filetime via findfirst on the
second connection to ensure it ' s the same . This is very easy for a Windows
server but a bastard to get right on a POSIX server . JRA . */
2004-11-20 01:09:25 +00:00
if ( fnum1 ! = - 1 )
smbcli_close ( cli - > tree , fnum1 ) ;
smbcli_unlink ( cli - > tree , fname ) ;
smbcli_deltree ( cli - > tree , BASEDIR ) ;
return ret ;
}
2004-11-12 09:37:59 +00:00
/* Windows does obviously not update the stat info during a write call. I
* * think * this is the problem causing a spurious Excel 2003 on XP error
* message when saving a file . Excel does a setfileinfo , writes , and then does
* a getpath ( ! ) info . Or so . . . For Samba sometimes it displays an error message
* that the file might have been changed in between . What i ' ve been able to
* trace down is that this happens if the getpathinfo after the write shows a
* different last write time than the setfileinfo showed . This is really
* nasty . . . .
*/
2007-10-06 22:28:14 +00:00
static bool test_finfo_after_write ( struct torture_context * tctx , struct smbcli_state * cli ,
2006-10-16 13:06:41 +00:00
struct smbcli_state * cli2 )
2004-11-12 09:37:59 +00:00
{
union smb_fileinfo finfo1 , finfo2 ;
const char * fname = BASEDIR " \\ torture_file.txt " ;
NTSTATUS status ;
int fnum1 = - 1 ;
int fnum2 ;
2007-10-06 22:28:14 +00:00
bool ret = true ;
2004-11-12 09:37:59 +00:00
ssize_t written ;
2008-04-08 10:25:51 +02:00
int used_delay = torture_setting_int ( tctx , " writetimeupdatedelay " , 2000000 ) ;
int normal_delay = 2000000 ;
double sec = ( ( double ) used_delay ) / ( ( double ) normal_delay ) ;
int msec = 1000 * sec ;
2004-11-12 09:37:59 +00:00
2008-09-08 14:58:58 -07:00
torture_comment ( tctx , " \n Running test_finfo_after_write \n " ) ;
2004-11-12 09:37:59 +00:00
if ( ! torture_setup_dir ( cli , BASEDIR ) ) {
2007-10-06 22:28:14 +00:00
return false ;
2004-11-12 09:37:59 +00:00
}
fnum1 = smbcli_open ( cli - > tree , fname , O_RDWR | O_CREAT , DENY_NONE ) ;
if ( fnum1 = = - 1 ) {
2007-10-06 22:28:14 +00:00
ret = false ;
2007-03-05 21:28:55 +00:00
torture_result ( tctx , TORTURE_FAIL , __location__ " : unable to open %s " , fname ) ;
2004-11-12 09:37:59 +00:00
goto done ;
}
finfo1 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
2006-03-12 22:48:25 +00:00
finfo1 . basic_info . in . file . fnum = fnum1 ;
2004-11-12 09:37:59 +00:00
2006-10-16 13:06:41 +00:00
status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo1 ) ;
2004-11-12 09:37:59 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2007-10-06 22:28:14 +00:00
ret = false ;
2007-03-05 21:28:55 +00:00
torture_result ( tctx , TORTURE_FAIL , __location__ " : fileinfo failed: %s " , nt_errstr ( status ) ) ;
2004-11-12 09:37:59 +00:00
goto done ;
}
2008-04-08 10:25:51 +02:00
msleep ( 1 * msec ) ;
2004-11-12 09:37:59 +00:00
written = smbcli_write ( cli - > tree , fnum1 , 0 , " x " , 0 , 1 ) ;
if ( written ! = 1 ) {
2007-03-05 21:28:55 +00:00
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 1 " , ( int ) written ) ;
2007-10-06 22:28:14 +00:00
ret = false ;
2004-11-12 09:37:59 +00:00
goto done ;
}
fnum2 = smbcli_open ( cli2 - > tree , fname , O_RDWR , DENY_NONE ) ;
if ( fnum2 = = - 1 ) {
2007-03-05 21:28:55 +00:00
torture_result ( tctx , TORTURE_FAIL , __location__ " : failed to open 2nd time - %s " ,
smbcli_errstr ( cli2 - > tree ) ) ;
2007-10-06 22:28:14 +00:00
ret = false ;
2004-11-12 09:37:59 +00:00
goto done ;
}
written = smbcli_write ( cli2 - > tree , fnum2 , 0 , " x " , 0 , 1 ) ;
if ( written ! = 1 ) {
2007-03-05 21:28:55 +00:00
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 1 " ,
( int ) written ) ;
2007-10-06 22:28:14 +00:00
ret = false ;
2004-11-12 09:37:59 +00:00
goto done ;
}
finfo2 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
2006-03-12 22:48:25 +00:00
finfo2 . basic_info . in . file . path = fname ;
2004-11-12 09:37:59 +00:00
2006-10-16 13:06:41 +00:00
status = smb_raw_pathinfo ( cli2 - > tree , tctx , & finfo2 ) ;
2004-11-12 09:37:59 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2007-03-05 21:28:55 +00:00
torture_result ( tctx , TORTURE_FAIL , __location__ " : fileinfo failed: %s " ,
nt_errstr ( status ) ) ;
2007-10-06 22:28:14 +00:00
ret = false ;
2004-11-12 09:37:59 +00:00
goto done ;
}
if ( finfo1 . basic_info . out . create_time ! =
finfo2 . basic_info . out . create_time ) {
2007-03-05 21:28:55 +00:00
torture_result ( tctx , TORTURE_FAIL , __location__ " : create_time changed " ) ;
2007-10-06 22:28:14 +00:00
ret = false ;
2004-11-12 09:37:59 +00:00
goto done ;
}
if ( finfo1 . basic_info . out . access_time ! =
finfo2 . basic_info . out . access_time ) {
2007-03-05 21:28:55 +00:00
torture_result ( tctx , TORTURE_FAIL , __location__ " : access_time changed " ) ;
2007-10-06 22:28:14 +00:00
ret = false ;
2004-11-12 09:37:59 +00:00
goto done ;
}
if ( finfo1 . basic_info . out . write_time ! =
finfo2 . basic_info . out . write_time ) {
2007-03-05 21:28:55 +00:00
torture_result ( tctx , TORTURE_FAIL , __location__ " : write_time changed: \n "
" write time conn 1 = %s, conn 2 = %s " ,
2006-10-16 13:06:41 +00:00
nt_time_string ( tctx , finfo1 . basic_info . out . write_time ) ,
nt_time_string ( tctx , finfo2 . basic_info . out . write_time ) ) ;
2007-10-06 22:28:14 +00:00
ret = false ;
2004-11-12 09:37:59 +00:00
goto done ;
}
if ( finfo1 . basic_info . out . change_time ! =
finfo2 . basic_info . out . change_time ) {
2007-03-05 21:28:55 +00:00
torture_result ( tctx , TORTURE_FAIL , __location__ " : change_time changed " ) ;
2007-10-06 22:28:14 +00:00
ret = false ;
2004-11-12 09:37:59 +00:00
goto done ;
}
/* One of the two following calls updates the qpathinfo. */
/* If you had skipped the smbcli_write on fnum2, it would
* * not * have updated the stat on disk */
smbcli_close ( cli2 - > tree , fnum2 ) ;
cli2 = NULL ;
/* This call is only for the people looking at ethereal :-) */
finfo2 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
2006-03-12 22:48:25 +00:00
finfo2 . basic_info . in . file . path = fname ;
2004-11-12 09:37:59 +00:00
2006-10-16 13:06:41 +00:00
status = smb_raw_pathinfo ( cli - > tree , tctx , & finfo2 ) ;
2004-11-12 09:37:59 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2007-03-05 21:28:55 +00:00
torture_result ( tctx , TORTURE_FAIL , __location__ " : fileinfo failed: %s " , nt_errstr ( status ) ) ;
2007-10-06 22:28:14 +00:00
ret = false ;
2004-11-12 09:37:59 +00:00
goto done ;
}
done :
if ( fnum1 ! = - 1 )
smbcli_close ( cli - > tree , fnum1 ) ;
smbcli_unlink ( cli - > tree , fname ) ;
smbcli_deltree ( cli - > tree , BASEDIR ) ;
return ret ;
}
2007-12-01 11:43:38 +01:00
# define COMPARE_WRITE_TIME_CMP(given, correct, cmp) do { \
2008-04-08 19:04:44 +02:00
uint64_t r = 10 * 1000 * 1000 ; \
2007-12-01 11:43:38 +01:00
NTTIME g = ( given ) . basic_info . out . write_time ; \
2008-04-08 19:04:44 +02:00
NTTIME gr = ( g / r ) * r ; \
2007-12-01 11:43:38 +01:00
NTTIME c = ( correct ) . basic_info . out . write_time ; \
2008-04-08 19:04:44 +02:00
NTTIME cr = ( c / r ) * r ; \
bool strict = torture_setting_bool ( tctx , " strict mode " , false ) ; \
bool err = false ; \
if ( strict & & ( g cmp c ) ) { \
err = true ; \
2008-06-03 14:01:02 +02:00
} else if ( ( g cmp c ) & & ( gr cmp cr ) ) { \
2008-04-08 19:04:44 +02:00
/* handle filesystem without high resolution timestamps */ \
err = true ; \
} \
if ( err ) { \
2008-04-08 10:25:51 +02:00
torture_result ( tctx , TORTURE_FAIL , __location__ " : wrong write_time (%s)%s(%llu) %s (%s)%s(%llu) " , \
# given, nt_time_string(tctx, g), (unsigned long long)g, \
# cmp, #correct, nt_time_string(tctx, c), (unsigned long long)c); \
2007-12-01 11:43:38 +01:00
ret = false ; \
goto done ; \
} \
} while ( 0 )
# define COMPARE_WRITE_TIME_EQUAL(given,correct) \
COMPARE_WRITE_TIME_CMP ( given , correct , ! = )
# define COMPARE_WRITE_TIME_GREATER(given,correct) \
COMPARE_WRITE_TIME_CMP ( given , correct , < = )
# define COMPARE_WRITE_TIME_LESS(given,correct) \
COMPARE_WRITE_TIME_CMP ( given , correct , > = )
2007-12-06 13:17:34 +01:00
# define COMPARE_ACCESS_TIME_CMP(given, correct, cmp) do { \
NTTIME g = ( given ) . basic_info . out . access_time ; \
NTTIME c = ( correct ) . basic_info . out . access_time ; \
if ( g cmp c ) { \
torture_result ( tctx , TORTURE_FAIL , __location__ " : wrong access_time (%s)%s %s (%s)%s " , \
# given, nt_time_string(tctx, g), \
# cmp, #correct, nt_time_string(tctx, c)); \
ret = false ; \
goto done ; \
} \
} while ( 0 )
# define COMPARE_ACCESS_TIME_EQUAL(given,correct) \
COMPARE_ACCESS_TIME_CMP ( given , correct , ! = )
# define COMPARE_BOTH_TIMES_EQUAL(given,correct) do { \
COMPARE_ACCESS_TIME_EQUAL ( given , correct ) ; \
COMPARE_WRITE_TIME_EQUAL ( given , correct ) ; \
} while ( 0 )
2007-12-01 11:43:38 +01:00
# define GET_INFO_FILE(finfo) do { \
NTSTATUS _status ; \
_status = smb_raw_fileinfo ( cli - > tree , tctx , & finfo ) ; \
if ( ! NT_STATUS_IS_OK ( _status ) ) { \
ret = false ; \
torture_result ( tctx , TORTURE_FAIL , __location__ " : fileinfo failed: %s " , \
nt_errstr ( _status ) ) ; \
goto done ; \
} \
2007-12-06 13:17:34 +01:00
torture_comment ( tctx , " fileinfo: Access(%s) Write(%s) \n " , \
nt_time_string ( tctx , finfo . basic_info . out . access_time ) , \
2007-12-01 11:43:38 +01:00
nt_time_string ( tctx , finfo . basic_info . out . write_time ) ) ; \
} while ( 0 )
# define GET_INFO_PATH(pinfo) do { \
NTSTATUS _status ; \
_status = smb_raw_pathinfo ( cli2 - > tree , tctx , & pinfo ) ; \
if ( ! NT_STATUS_IS_OK ( _status ) ) { \
torture_result ( tctx , TORTURE_FAIL , __location__ " : pathinfo failed: %s " , \
nt_errstr ( _status ) ) ; \
ret = false ; \
goto done ; \
} \
2007-12-06 13:17:34 +01:00
torture_comment ( tctx , " pathinfo: Access(%s) Write(%s) \n " , \
nt_time_string ( tctx , pinfo . basic_info . out . access_time ) , \
2007-12-01 11:43:38 +01:00
nt_time_string ( tctx , pinfo . basic_info . out . write_time ) ) ; \
} while ( 0 )
# define GET_INFO_BOTH(finfo,pinfo) do { \
GET_INFO_FILE ( finfo ) ; \
GET_INFO_PATH ( pinfo ) ; \
2007-12-06 13:17:34 +01:00
COMPARE_BOTH_TIMES_EQUAL ( finfo , pinfo ) ; \
2007-12-01 11:43:38 +01:00
} while ( 0 )
# define SET_INFO_FILE_EX(finfo, wrtime, tree, tfnum) do { \
NTSTATUS _status ; \
union smb_setfileinfo sfinfo ; \
sfinfo . basic_info . level = RAW_SFILEINFO_BASIC_INFO ; \
sfinfo . basic_info . in . file . fnum = tfnum ; \
2007-12-06 15:06:36 +01:00
sfinfo . basic_info . in . create_time = 0 ; \
sfinfo . basic_info . in . access_time = 0 ; \
2007-12-01 11:43:38 +01:00
unix_to_nt_time ( & sfinfo . basic_info . in . write_time , ( wrtime ) ) ; \
2007-12-06 15:06:36 +01:00
sfinfo . basic_info . in . change_time = 0 ; \
2007-12-01 11:43:38 +01:00
sfinfo . basic_info . in . attrib = finfo1 . basic_info . out . attrib ; \
_status = smb_raw_setfileinfo ( tree , & sfinfo ) ; \
if ( ! NT_STATUS_IS_OK ( _status ) ) { \
torture_result ( tctx , TORTURE_FAIL , __location__ " : setfileinfo failed: %s " , \
nt_errstr ( _status ) ) ; \
ret = false ; \
goto done ; \
} \
} while ( 0 )
# define SET_INFO_FILE(finfo, wrtime) \
SET_INFO_FILE_EX ( finfo , wrtime , cli - > tree , fnum1 )
static bool test_delayed_write_update3 ( struct torture_context * tctx ,
struct smbcli_state * cli ,
struct smbcli_state * cli2 )
{
union smb_fileinfo finfo0 , finfo1 , finfo2 , finfo3 , finfo4 ;
union smb_fileinfo pinfo0 , pinfo1 , pinfo2 , pinfo3 , pinfo4 , pinfo5 ;
2008-09-08 14:58:58 -07:00
const char * fname = BASEDIR " \\ torture_file3.txt " ;
2007-12-01 11:43:38 +01:00
int fnum1 = - 1 ;
bool ret = true ;
ssize_t written ;
2008-04-08 10:25:51 +02:00
struct timeval start ;
struct timeval end ;
int used_delay = torture_setting_int ( tctx , " writetimeupdatedelay " , 2000000 ) ;
int normal_delay = 2000000 ;
double sec = ( ( double ) used_delay ) / ( ( double ) normal_delay ) ;
int msec = 1000 * sec ;
2007-12-01 11:43:38 +01:00
2008-09-08 14:58:58 -07:00
torture_comment ( tctx , " \n Running test_delayed_write_update3 \n " ) ;
2007-12-01 11:43:38 +01:00
if ( ! torture_setup_dir ( cli , BASEDIR ) ) {
return false ;
}
torture_comment ( tctx , " Open the file handle \n " ) ;
fnum1 = smbcli_open ( cli - > tree , fname , O_RDWR | O_CREAT , DENY_NONE ) ;
if ( fnum1 = = - 1 ) {
ret = false ;
torture_result ( tctx , TORTURE_FAIL , __location__ " : unable to open %s " , fname ) ;
goto done ;
}
finfo0 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
finfo0 . basic_info . in . file . fnum = fnum1 ;
finfo1 = finfo0 ;
finfo2 = finfo0 ;
finfo3 = finfo0 ;
finfo4 = finfo0 ;
pinfo0 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
pinfo0 . basic_info . in . file . path = fname ;
pinfo1 = pinfo0 ;
pinfo2 = pinfo0 ;
pinfo3 = pinfo0 ;
pinfo4 = pinfo0 ;
pinfo5 = pinfo0 ;
/* get the initial times */
GET_INFO_BOTH ( finfo0 , pinfo0 ) ;
/*
* make sure the write time is updated 2 seconds later
* calcuated from the first write
* ( but expect upto 5 seconds extra time for a busy server )
*/
2008-04-08 10:25:51 +02:00
start = timeval_current ( ) ;
end = timeval_add ( & start , 7 * sec , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
2007-12-01 11:43:38 +01:00
/* do a write */
torture_comment ( tctx , " Do a write on the file handle \n " ) ;
written = smbcli_write ( cli - > tree , fnum1 , 0 , " x " , 0 , 1 ) ;
if ( written ! = 1 ) {
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 1 " , ( int ) written ) ;
ret = false ;
goto done ;
}
/* get the times after the write */
GET_INFO_FILE ( finfo1 ) ;
if ( finfo1 . basic_info . out . write_time > finfo0 . basic_info . out . write_time ) {
2008-04-08 10:25:51 +02:00
double diff = timeval_elapsed ( & start ) ;
2009-06-03 14:12:18 -07:00
if ( diff < ( TIMEDELAY_SECS * sec * 0.75 ) ) { /* 0.75 to cope with vmware timing */
2008-04-08 10:25:51 +02:00
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (wrong!) \n " ,
diff , sec ) ;
2007-12-01 11:43:38 +01:00
ret = false ;
break ;
}
2008-04-08 10:25:51 +02:00
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (correct) \n " ,
diff , sec ) ;
2007-12-01 11:43:38 +01:00
break ;
}
2008-04-08 10:25:51 +02:00
msleep ( 0.5 * msec ) ;
2007-12-01 11:43:38 +01:00
}
GET_INFO_BOTH ( finfo1 , pinfo1 ) ;
2008-05-05 15:05:41 +02:00
COMPARE_WRITE_TIME_GREATER ( pinfo1 , pinfo0 ) ;
2007-12-01 11:43:38 +01:00
/* sure any further write doesn't update the write time */
2008-04-08 10:25:51 +02:00
start = timeval_current ( ) ;
end = timeval_add ( & start , 15 * sec , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
2007-12-01 11:43:38 +01:00
/* do a write */
torture_comment ( tctx , " Do a write on the file handle \n " ) ;
written = smbcli_write ( cli - > tree , fnum1 , 0 , " x " , 0 , 1 ) ;
if ( written ! = 1 ) {
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 1 " , ( int ) written ) ;
ret = false ;
goto done ;
}
/* get the times after the write */
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
if ( finfo2 . basic_info . out . write_time > finfo1 . basic_info . out . write_time ) {
2008-04-08 10:25:51 +02:00
double diff = timeval_elapsed ( & start ) ;
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (wrong!) \n " ,
diff , sec ) ;
2007-12-01 11:43:38 +01:00
ret = false ;
break ;
}
2009-06-03 14:12:18 -07:00
msleep ( 1 * msec ) ;
2007-12-01 11:43:38 +01:00
}
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo2 , finfo1 ) ;
if ( finfo2 . basic_info . out . write_time = = finfo1 . basic_info . out . write_time ) {
torture_comment ( tctx , " Server did not update write_time (correct) \n " ) ;
}
/* sleep */
2008-04-08 10:25:51 +02:00
msleep ( 5 * msec ) ;
2007-12-01 11:43:38 +01:00
GET_INFO_BOTH ( finfo3 , pinfo3 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo3 , finfo2 ) ;
/*
* the close updates the write time to the time of the close
* and not to the time of the last write !
*/
torture_comment ( tctx , " Close the file handle \n " ) ;
smbcli_close ( cli - > tree , fnum1 ) ;
fnum1 = - 1 ;
GET_INFO_PATH ( pinfo4 ) ;
COMPARE_WRITE_TIME_GREATER ( pinfo4 , pinfo3 ) ;
if ( pinfo4 . basic_info . out . write_time > pinfo3 . basic_info . out . write_time ) {
torture_comment ( tctx , " Server updated the write_time on close (correct) \n " ) ;
}
done :
if ( fnum1 ! = - 1 )
smbcli_close ( cli - > tree , fnum1 ) ;
smbcli_unlink ( cli - > tree , fname ) ;
smbcli_deltree ( cli - > tree , BASEDIR ) ;
return ret ;
}
2008-09-12 14:59:32 -07:00
/*
* Show that a truncate write always updates the write time even
* if an initial write has already updated the write time .
*/
2008-09-08 13:59:51 +02:00
static bool test_delayed_write_update3a ( struct torture_context * tctx ,
struct smbcli_state * cli ,
struct smbcli_state * cli2 )
{
union smb_fileinfo finfo0 , finfo1 , finfo2 , finfo3 , finfo4 ;
union smb_fileinfo pinfo0 , pinfo1 , pinfo2 , pinfo3 , pinfo4 , pinfo5 ;
2008-09-08 14:58:58 -07:00
const char * fname = BASEDIR " \\ torture_file3a.txt " ;
2008-09-08 13:59:51 +02:00
int fnum1 = - 1 ;
bool ret = true ;
ssize_t written ;
int i ;
struct timeval start ;
struct timeval end ;
int used_delay = torture_setting_int ( tctx , " writetimeupdatedelay " , 2000000 ) ;
int normal_delay = 2000000 ;
double sec = ( ( double ) used_delay ) / ( ( double ) normal_delay ) ;
int msec = 1000 * sec ;
2008-09-08 14:58:58 -07:00
torture_comment ( tctx , " \n Running test_delayed_write_update3a \n " ) ;
2008-09-08 13:59:51 +02:00
if ( ! torture_setup_dir ( cli , BASEDIR ) ) {
return false ;
}
torture_comment ( tctx , " Open the file handle \n " ) ;
fnum1 = smbcli_open ( cli - > tree , fname , O_RDWR | O_CREAT , DENY_NONE ) ;
if ( fnum1 = = - 1 ) {
ret = false ;
torture_result ( tctx , TORTURE_FAIL , __location__ " : unable to open %s " , fname ) ;
goto done ;
}
finfo0 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
finfo0 . basic_info . in . file . fnum = fnum1 ;
finfo1 = finfo0 ;
finfo2 = finfo0 ;
finfo3 = finfo0 ;
finfo4 = finfo0 ;
pinfo0 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
pinfo0 . basic_info . in . file . path = fname ;
pinfo1 = pinfo0 ;
pinfo2 = pinfo0 ;
pinfo3 = pinfo0 ;
pinfo4 = pinfo0 ;
pinfo5 = pinfo0 ;
/* get the initial times */
GET_INFO_BOTH ( finfo0 , pinfo0 ) ;
/*
* sleep some time , to demonstrate the handling of write times
* doesn ' t depend on the time since the open
*/
msleep ( 5 * msec ) ;
/* get the initial times */
GET_INFO_BOTH ( finfo1 , pinfo1 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo1 , finfo0 ) ;
/*
* make sure the write time is updated 2 seconds later
* calcuated from the first write
* ( but expect upto 5 seconds extra time for a busy server )
*/
start = timeval_current ( ) ;
end = timeval_add ( & start , 7 * sec , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
/* do a write */
torture_comment ( tctx , " Do a write on the file handle \n " ) ;
written = smbcli_write ( cli - > tree , fnum1 , 0 , " x " , 0 , 1 ) ;
if ( written ! = 1 ) {
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 1 " , ( int ) written ) ;
ret = false ;
goto done ;
}
/* get the times after the write */
GET_INFO_FILE ( finfo1 ) ;
if ( finfo1 . basic_info . out . write_time > finfo0 . basic_info . out . write_time ) {
double diff = timeval_elapsed ( & start ) ;
2009-06-03 14:12:18 -07:00
if ( diff < ( TIMEDELAY_SECS * sec * 0.75 ) ) { /* 0.75 to cope with vmware timing */
2008-09-08 13:59:51 +02:00
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (wrong!) \n " ,
diff , sec ) ;
ret = false ;
break ;
}
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (correct) \n " ,
diff , sec ) ;
break ;
}
msleep ( 0.5 * msec ) ;
}
GET_INFO_BOTH ( finfo1 , pinfo1 ) ;
COMPARE_WRITE_TIME_GREATER ( pinfo1 , pinfo0 ) ;
/*
* demonstrate that a truncate write always
* updates the write time immediately
*/
for ( i = 0 ; i < 3 ; i + + ) {
msleep ( 1 * msec ) ;
/* do a write */
torture_comment ( tctx , " Do a truncate SMBwrite [%d] on the file handle \n " , i ) ;
written = smbcli_smbwrite ( cli - > tree , fnum1 , " x " , 10240 , 0 ) ;
if ( written ! = 0 ) {
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 0 " , ( int ) written ) ;
ret = false ;
goto done ;
}
/* get the times after the write */
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
COMPARE_WRITE_TIME_GREATER ( finfo2 , finfo1 ) ;
finfo1 = finfo2 ;
}
/* sure any further write doesn't update the write time */
start = timeval_current ( ) ;
end = timeval_add ( & start , 15 * sec , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
/* do a write */
torture_comment ( tctx , " Do a write on the file handle \n " ) ;
written = smbcli_write ( cli - > tree , fnum1 , 0 , " x " , 0 , 1 ) ;
if ( written ! = 1 ) {
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 1 " , ( int ) written ) ;
ret = false ;
goto done ;
}
/* get the times after the write */
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
if ( finfo2 . basic_info . out . write_time > finfo1 . basic_info . out . write_time ) {
double diff = timeval_elapsed ( & start ) ;
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (wrong!) \n " ,
diff , sec ) ;
ret = false ;
break ;
}
msleep ( 2 * msec ) ;
}
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo2 , finfo1 ) ;
if ( finfo2 . basic_info . out . write_time = = finfo1 . basic_info . out . write_time ) {
torture_comment ( tctx , " Server did not update write_time (correct) \n " ) ;
}
/* sleep */
msleep ( 5 * msec ) ;
/* get the initial times */
GET_INFO_BOTH ( finfo1 , pinfo1 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo1 , finfo2 ) ;
/*
* demonstrate that a truncate write always
* updates the write time immediately
*/
for ( i = 0 ; i < 3 ; i + + ) {
msleep ( 1 * msec ) ;
/* do a write */
torture_comment ( tctx , " Do a truncate SMBwrite [%d] on the file handle \n " , i ) ;
written = smbcli_smbwrite ( cli - > tree , fnum1 , " x " , 512 , 0 ) ;
if ( written ! = 0 ) {
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 0 " , ( int ) written ) ;
ret = false ;
goto done ;
}
/* get the times after the write */
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
COMPARE_WRITE_TIME_GREATER ( finfo2 , finfo1 ) ;
finfo1 = finfo2 ;
}
/* sleep */
msleep ( 5 * msec ) ;
GET_INFO_BOTH ( finfo3 , pinfo3 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo3 , finfo2 ) ;
/*
* the close doesn ' t update the write time
*/
torture_comment ( tctx , " Close the file handle \n " ) ;
smbcli_close ( cli - > tree , fnum1 ) ;
fnum1 = - 1 ;
GET_INFO_PATH ( pinfo4 ) ;
COMPARE_WRITE_TIME_EQUAL ( pinfo4 , pinfo3 ) ;
if ( pinfo4 . basic_info . out . write_time = = pinfo3 . basic_info . out . write_time ) {
torture_comment ( tctx , " Server did not update the write_time on close (correct) \n " ) ;
}
done :
if ( fnum1 ! = - 1 )
smbcli_close ( cli - > tree , fnum1 ) ;
smbcli_unlink ( cli - > tree , fname ) ;
smbcli_deltree ( cli - > tree , BASEDIR ) ;
return ret ;
}
2008-09-12 14:59:32 -07:00
/*
* Show a close after write updates the write timestamp to
* the close time , not the last write time .
*/
2008-09-08 12:12:25 +02:00
static bool test_delayed_write_update3b ( struct torture_context * tctx ,
struct smbcli_state * cli ,
struct smbcli_state * cli2 )
{
union smb_fileinfo finfo0 , finfo1 , finfo2 , finfo3 , finfo4 ;
union smb_fileinfo pinfo0 , pinfo1 , pinfo2 , pinfo3 , pinfo4 , pinfo5 ;
2008-09-08 14:58:58 -07:00
const char * fname = BASEDIR " \\ torture_file3b.txt " ;
2008-09-08 12:12:25 +02:00
int fnum1 = - 1 ;
bool ret = true ;
ssize_t written ;
struct timeval start ;
struct timeval end ;
int used_delay = torture_setting_int ( tctx , " writetimeupdatedelay " , 2000000 ) ;
int normal_delay = 2000000 ;
double sec = ( ( double ) used_delay ) / ( ( double ) normal_delay ) ;
int msec = 1000 * sec ;
2008-09-08 14:58:58 -07:00
torture_comment ( tctx , " \n Running test_delayed_write_update3b \n " ) ;
2008-09-08 12:12:25 +02:00
if ( ! torture_setup_dir ( cli , BASEDIR ) ) {
return false ;
}
torture_comment ( tctx , " Open the file handle \n " ) ;
fnum1 = smbcli_open ( cli - > tree , fname , O_RDWR | O_CREAT , DENY_NONE ) ;
if ( fnum1 = = - 1 ) {
ret = false ;
torture_result ( tctx , TORTURE_FAIL , __location__ " : unable to open %s " , fname ) ;
goto done ;
}
finfo0 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
finfo0 . basic_info . in . file . fnum = fnum1 ;
finfo1 = finfo0 ;
finfo2 = finfo0 ;
finfo3 = finfo0 ;
finfo4 = finfo0 ;
pinfo0 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
pinfo0 . basic_info . in . file . path = fname ;
pinfo1 = pinfo0 ;
pinfo2 = pinfo0 ;
pinfo3 = pinfo0 ;
pinfo4 = pinfo0 ;
pinfo5 = pinfo0 ;
/* get the initial times */
GET_INFO_BOTH ( finfo0 , pinfo0 ) ;
/*
* sleep some time , to demonstrate the handling of write times
* doesn ' t depend on the time since the open
*/
msleep ( 5 * msec ) ;
/* get the initial times */
GET_INFO_BOTH ( finfo1 , pinfo1 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo1 , finfo0 ) ;
/*
* make sure the write time is updated 2 seconds later
* calcuated from the first write
* ( but expect upto 5 seconds extra time for a busy server )
*/
start = timeval_current ( ) ;
end = timeval_add ( & start , 7 * sec , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
/* do a write */
torture_comment ( tctx , " Do a write on the file handle \n " ) ;
written = smbcli_write ( cli - > tree , fnum1 , 0 , " x " , 0 , 1 ) ;
if ( written ! = 1 ) {
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 1 " , ( int ) written ) ;
ret = false ;
goto done ;
}
/* get the times after the write */
GET_INFO_FILE ( finfo1 ) ;
if ( finfo1 . basic_info . out . write_time > finfo0 . basic_info . out . write_time ) {
double diff = timeval_elapsed ( & start ) ;
2009-06-03 14:12:18 -07:00
if ( diff < ( TIMEDELAY_SECS * sec * 0.75 ) ) { /* 0.75 to cope with vmware timing */
2008-09-08 12:12:25 +02:00
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (wrong!) \n " ,
diff , sec ) ;
ret = false ;
break ;
}
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (correct) \n " ,
diff , sec ) ;
break ;
}
msleep ( 0.5 * msec ) ;
}
GET_INFO_BOTH ( finfo1 , pinfo1 ) ;
COMPARE_WRITE_TIME_GREATER ( pinfo1 , pinfo0 ) ;
/* sure any further write doesn't update the write time */
start = timeval_current ( ) ;
end = timeval_add ( & start , 15 * sec , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
/* do a write */
torture_comment ( tctx , " Do a write on the file handle \n " ) ;
written = smbcli_write ( cli - > tree , fnum1 , 0 , " x " , 0 , 1 ) ;
if ( written ! = 1 ) {
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 1 " , ( int ) written ) ;
ret = false ;
goto done ;
}
/* get the times after the write */
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
if ( finfo2 . basic_info . out . write_time > finfo1 . basic_info . out . write_time ) {
double diff = timeval_elapsed ( & start ) ;
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (wrong!) \n " ,
diff , sec ) ;
ret = false ;
break ;
}
msleep ( 2 * msec ) ;
}
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo2 , finfo1 ) ;
if ( finfo2 . basic_info . out . write_time = = finfo1 . basic_info . out . write_time ) {
torture_comment ( tctx , " Server did not update write_time (correct) \n " ) ;
}
/* sleep */
msleep ( 5 * msec ) ;
GET_INFO_BOTH ( finfo3 , pinfo3 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo3 , finfo2 ) ;
/*
* the close updates the write time to the time of the close
* and not to the time of the last write !
*/
torture_comment ( tctx , " Close the file handle \n " ) ;
smbcli_close ( cli - > tree , fnum1 ) ;
fnum1 = - 1 ;
GET_INFO_PATH ( pinfo4 ) ;
COMPARE_WRITE_TIME_GREATER ( pinfo4 , pinfo3 ) ;
if ( pinfo4 . basic_info . out . write_time > pinfo3 . basic_info . out . write_time ) {
torture_comment ( tctx , " Server updated the write_time on close (correct) \n " ) ;
}
done :
if ( fnum1 ! = - 1 )
smbcli_close ( cli - > tree , fnum1 ) ;
smbcli_unlink ( cli - > tree , fname ) ;
smbcli_deltree ( cli - > tree , BASEDIR ) ;
return ret ;
}
2008-09-12 14:59:32 -07:00
/*
* Check that a write after a truncate write doesn ' t update
* the timestamp , but a truncate write after a write does .
* Also prove that a close after a truncate write updates the
* timestamp to current , not the time of last write .
*/
2008-09-08 14:11:17 +02:00
static bool test_delayed_write_update3c ( struct torture_context * tctx ,
struct smbcli_state * cli ,
struct smbcli_state * cli2 )
{
union smb_fileinfo finfo0 , finfo1 , finfo2 , finfo3 , finfo4 ;
union smb_fileinfo pinfo0 , pinfo1 , pinfo2 , pinfo3 , pinfo4 , pinfo5 ;
2008-09-08 14:58:58 -07:00
const char * fname = BASEDIR " \\ torture_file3c.txt " ;
2008-09-08 14:11:17 +02:00
int fnum1 = - 1 ;
bool ret = true ;
ssize_t written ;
int i ;
struct timeval start ;
struct timeval end ;
int used_delay = torture_setting_int ( tctx , " writetimeupdatedelay " , 2000000 ) ;
int normal_delay = 2000000 ;
double sec = ( ( double ) used_delay ) / ( ( double ) normal_delay ) ;
int msec = 1000 * sec ;
2008-09-08 14:58:58 -07:00
torture_comment ( tctx , " \n Running test_delayed_write_update3c \n " ) ;
2008-09-08 14:11:17 +02:00
if ( ! torture_setup_dir ( cli , BASEDIR ) ) {
return false ;
}
torture_comment ( tctx , " Open the file handle \n " ) ;
fnum1 = smbcli_open ( cli - > tree , fname , O_RDWR | O_CREAT , DENY_NONE ) ;
if ( fnum1 = = - 1 ) {
ret = false ;
torture_result ( tctx , TORTURE_FAIL , __location__ " : unable to open %s " , fname ) ;
goto done ;
}
finfo0 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
finfo0 . basic_info . in . file . fnum = fnum1 ;
finfo1 = finfo0 ;
finfo2 = finfo0 ;
finfo3 = finfo0 ;
finfo4 = finfo0 ;
pinfo0 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
pinfo0 . basic_info . in . file . path = fname ;
pinfo1 = pinfo0 ;
pinfo2 = pinfo0 ;
pinfo3 = pinfo0 ;
pinfo4 = pinfo0 ;
pinfo5 = pinfo0 ;
/* get the initial times */
GET_INFO_BOTH ( finfo0 , pinfo0 ) ;
/*
* sleep some time , to demonstrate the handling of write times
* doesn ' t depend on the time since the open
*/
msleep ( 5 * msec ) ;
/* get the initial times */
GET_INFO_BOTH ( finfo1 , pinfo1 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo1 , finfo0 ) ;
/*
* demonstrate that a truncate write always
* updates the write time immediately
*/
for ( i = 0 ; i < 3 ; i + + ) {
msleep ( 1 * msec ) ;
/* do a write */
torture_comment ( tctx , " Do a truncate SMBwrite [%d] on the file handle \n " , i ) ;
written = smbcli_smbwrite ( cli - > tree , fnum1 , " x " , 512 , 0 ) ;
if ( written ! = 0 ) {
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 0 " , ( int ) written ) ;
ret = false ;
goto done ;
}
/* get the times after the write */
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
COMPARE_WRITE_TIME_GREATER ( finfo2 , finfo1 ) ;
finfo1 = finfo2 ;
}
start = timeval_current ( ) ;
end = timeval_add ( & start , 7 * sec , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
/* do a write */
torture_comment ( tctx , " Do a write on the file handle \n " ) ;
written = smbcli_write ( cli - > tree , fnum1 , 0 , " x " , 0 , 1 ) ;
if ( written ! = 1 ) {
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 1 " , ( int ) written ) ;
ret = false ;
goto done ;
}
/* get the times after the write */
GET_INFO_FILE ( finfo2 ) ;
if ( finfo2 . basic_info . out . write_time > finfo1 . basic_info . out . write_time ) {
double diff = timeval_elapsed ( & start ) ;
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (wrong!) \n " ,
diff , sec ) ;
ret = false ;
break ;
}
msleep ( 2 * msec ) ;
}
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo2 , finfo1 ) ;
if ( finfo2 . basic_info . out . write_time = = finfo1 . basic_info . out . write_time ) {
torture_comment ( tctx , " Server did not update write_time (correct) \n " ) ;
}
/* sleep */
msleep ( 5 * msec ) ;
/* get the initial times */
GET_INFO_BOTH ( finfo1 , pinfo1 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo1 , finfo2 ) ;
/*
* demonstrate that a truncate write always
* updates the write time immediately
*/
for ( i = 0 ; i < 3 ; i + + ) {
msleep ( 1 * msec ) ;
/* do a write */
torture_comment ( tctx , " Do a truncate write [%d] on the file handle \n " , i ) ;
written = smbcli_smbwrite ( cli - > tree , fnum1 , " x " , 512 , 0 ) ;
if ( written ! = 0 ) {
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 0 " , ( int ) written ) ;
ret = false ;
goto done ;
}
/* get the times after the write */
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
COMPARE_WRITE_TIME_GREATER ( finfo2 , finfo1 ) ;
finfo1 = finfo2 ;
}
/* sleep */
msleep ( 5 * msec ) ;
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo2 , finfo1 ) ;
/* sure any further write doesn't update the write time */
start = timeval_current ( ) ;
end = timeval_add ( & start , 15 * sec , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
/* do a write */
torture_comment ( tctx , " Do a write on the file handle \n " ) ;
written = smbcli_write ( cli - > tree , fnum1 , 0 , " x " , 0 , 1 ) ;
if ( written ! = 1 ) {
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 1 " , ( int ) written ) ;
ret = false ;
goto done ;
}
/* get the times after the write */
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
if ( finfo2 . basic_info . out . write_time > finfo1 . basic_info . out . write_time ) {
double diff = timeval_elapsed ( & start ) ;
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (wrong!) \n " ,
diff , sec ) ;
ret = false ;
break ;
}
msleep ( 2 * msec ) ;
}
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo2 , finfo1 ) ;
if ( finfo2 . basic_info . out . write_time = = finfo1 . basic_info . out . write_time ) {
torture_comment ( tctx , " Server did not update write_time (correct) \n " ) ;
}
/* sleep */
msleep ( 5 * msec ) ;
GET_INFO_BOTH ( finfo3 , pinfo3 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo3 , finfo2 ) ;
/*
* the close updates the write time to the time of the close
* and not to the time of the last write !
*/
torture_comment ( tctx , " Close the file handle \n " ) ;
smbcli_close ( cli - > tree , fnum1 ) ;
fnum1 = - 1 ;
GET_INFO_PATH ( pinfo4 ) ;
COMPARE_WRITE_TIME_GREATER ( pinfo4 , pinfo3 ) ;
if ( pinfo4 . basic_info . out . write_time > pinfo3 . basic_info . out . write_time ) {
torture_comment ( tctx , " Server updated the write_time on close (correct) \n " ) ;
}
done :
if ( fnum1 ! = - 1 )
smbcli_close ( cli - > tree , fnum1 ) ;
smbcli_unlink ( cli - > tree , fname ) ;
smbcli_deltree ( cli - > tree , BASEDIR ) ;
return ret ;
}
2008-09-12 14:59:32 -07:00
/*
* Show only the first write updates the timestamp , and a close
* after writes updates to current ( I think this is the same
* as test 3 b . JRA ) .
*/
2007-12-01 11:43:38 +01:00
static bool test_delayed_write_update4 ( struct torture_context * tctx ,
struct smbcli_state * cli ,
struct smbcli_state * cli2 )
{
union smb_fileinfo finfo0 , finfo1 , finfo2 , finfo3 , finfo4 ;
union smb_fileinfo pinfo0 , pinfo1 , pinfo2 , pinfo3 , pinfo4 , pinfo5 ;
2008-09-08 14:58:58 -07:00
const char * fname = BASEDIR " \\ torture_file4.txt " ;
2007-12-01 11:43:38 +01:00
int fnum1 = - 1 ;
bool ret = true ;
ssize_t written ;
2008-04-08 10:25:51 +02:00
struct timeval start ;
struct timeval end ;
int used_delay = torture_setting_int ( tctx , " writetimeupdatedelay " , 2000000 ) ;
int normal_delay = 2000000 ;
double sec = ( ( double ) used_delay ) / ( ( double ) normal_delay ) ;
int msec = 1000 * sec ;
2007-12-01 11:43:38 +01:00
2008-09-08 14:58:58 -07:00
torture_comment ( tctx , " \n Running test_delayed_write_update4 \n " ) ;
2007-12-01 11:43:38 +01:00
if ( ! torture_setup_dir ( cli , BASEDIR ) ) {
return false ;
}
torture_comment ( tctx , " Open the file handle \n " ) ;
fnum1 = smbcli_open ( cli - > tree , fname , O_RDWR | O_CREAT , DENY_NONE ) ;
if ( fnum1 = = - 1 ) {
ret = false ;
torture_result ( tctx , TORTURE_FAIL , __location__ " : unable to open %s " , fname ) ;
goto done ;
}
finfo0 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
finfo0 . basic_info . in . file . fnum = fnum1 ;
finfo1 = finfo0 ;
finfo2 = finfo0 ;
finfo3 = finfo0 ;
finfo4 = finfo0 ;
pinfo0 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
pinfo0 . basic_info . in . file . path = fname ;
pinfo1 = pinfo0 ;
pinfo2 = pinfo0 ;
pinfo3 = pinfo0 ;
pinfo4 = pinfo0 ;
pinfo5 = pinfo0 ;
/* get the initial times */
GET_INFO_BOTH ( finfo0 , pinfo0 ) ;
/* sleep a bit */
2008-04-08 10:25:51 +02:00
msleep ( 5 * msec ) ;
2007-12-01 11:43:38 +01:00
/* do a write */
torture_comment ( tctx , " Do a write on the file handle \n " ) ;
written = smbcli_write ( cli - > tree , fnum1 , 0 , " x " , 0 , 1 ) ;
if ( written ! = 1 ) {
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 1 " , ( int ) written ) ;
ret = false ;
goto done ;
}
GET_INFO_BOTH ( finfo1 , pinfo1 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo1 , finfo0 ) ;
/*
* make sure the write time is updated 2 seconds later
* calcuated from the first write
* ( but expect upto 3 seconds extra time for a busy server )
*/
2008-04-08 10:25:51 +02:00
start = timeval_current ( ) ;
end = timeval_add ( & start , 5 * sec , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
2007-12-01 11:43:38 +01:00
/* get the times after the first write */
GET_INFO_FILE ( finfo1 ) ;
if ( finfo1 . basic_info . out . write_time > finfo0 . basic_info . out . write_time ) {
2008-04-08 10:25:51 +02:00
double diff = timeval_elapsed ( & start ) ;
2009-06-03 14:12:18 -07:00
if ( diff < ( TIMEDELAY_SECS * sec * 0.75 ) ) { /* 0.75 to cope with vmware timing */
2008-04-08 10:25:51 +02:00
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (wrong!) \n " ,
diff , sec ) ;
2007-12-01 11:43:38 +01:00
ret = false ;
break ;
}
2008-04-08 10:25:51 +02:00
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (correct) \n " ,
diff , sec ) ;
2007-12-01 11:43:38 +01:00
break ;
}
2008-04-08 10:25:51 +02:00
msleep ( 0.5 * msec ) ;
2007-12-01 11:43:38 +01:00
}
GET_INFO_BOTH ( finfo1 , pinfo1 ) ;
2008-05-05 15:05:41 +02:00
COMPARE_WRITE_TIME_GREATER ( pinfo1 , pinfo0 ) ;
2007-12-01 11:43:38 +01:00
/* sure any further write doesn't update the write time */
2008-04-08 10:25:51 +02:00
start = timeval_current ( ) ;
end = timeval_add ( & start , 15 * sec , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
2007-12-01 11:43:38 +01:00
/* do a write */
torture_comment ( tctx , " Do a write on the file handle \n " ) ;
written = smbcli_write ( cli - > tree , fnum1 , 0 , " x " , 0 , 1 ) ;
if ( written ! = 1 ) {
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 1 " , ( int ) written ) ;
ret = false ;
goto done ;
}
/* get the times after the write */
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
if ( finfo2 . basic_info . out . write_time > finfo1 . basic_info . out . write_time ) {
2008-04-08 10:25:51 +02:00
double diff = timeval_elapsed ( & start ) ;
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (wrong!) \n " ,
diff , sec ) ;
2007-12-01 11:43:38 +01:00
ret = false ;
break ;
}
2008-04-08 10:25:51 +02:00
msleep ( 2 * msec ) ;
2007-12-01 11:43:38 +01:00
}
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo2 , finfo1 ) ;
if ( finfo2 . basic_info . out . write_time = = finfo1 . basic_info . out . write_time ) {
torture_comment ( tctx , " Server did not updatewrite_time (correct) \n " ) ;
}
/* sleep */
2008-04-08 10:25:51 +02:00
msleep ( 5 * msec ) ;
2007-12-01 11:43:38 +01:00
GET_INFO_BOTH ( finfo3 , pinfo3 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo3 , finfo2 ) ;
/*
* the close updates the write time to the time of the close
* and not to the time of the last write !
*/
torture_comment ( tctx , " Close the file handle \n " ) ;
smbcli_close ( cli - > tree , fnum1 ) ;
fnum1 = - 1 ;
GET_INFO_PATH ( pinfo4 ) ;
COMPARE_WRITE_TIME_GREATER ( pinfo4 , pinfo3 ) ;
if ( pinfo4 . basic_info . out . write_time > pinfo3 . basic_info . out . write_time ) {
torture_comment ( tctx , " Server updated the write_time on close (correct) \n " ) ;
}
done :
if ( fnum1 ! = - 1 )
smbcli_close ( cli - > tree , fnum1 ) ;
smbcli_unlink ( cli - > tree , fname ) ;
smbcli_deltree ( cli - > tree , BASEDIR ) ;
return ret ;
}
2008-09-12 14:59:32 -07:00
/*
* Show writes and closes have no effect on updating times once a SETWRITETIME is done .
*/
2007-12-01 11:43:38 +01:00
static bool test_delayed_write_update5 ( struct torture_context * tctx ,
struct smbcli_state * cli ,
struct smbcli_state * cli2 )
{
union smb_fileinfo finfo0 , finfo1 , finfo2 , finfo3 , finfo4 , finfo5 ;
union smb_fileinfo pinfo0 , pinfo1 , pinfo2 , pinfo3 , pinfo4 , pinfo5 , pinfo6 ;
2008-09-08 14:58:58 -07:00
const char * fname = BASEDIR " \\ torture_file5.txt " ;
2007-12-01 11:43:38 +01:00
int fnum1 = - 1 ;
bool ret = true ;
ssize_t written ;
2008-04-08 10:25:51 +02:00
struct timeval start ;
struct timeval end ;
int used_delay = torture_setting_int ( tctx , " writetimeupdatedelay " , 2000000 ) ;
int normal_delay = 2000000 ;
double sec = ( ( double ) used_delay ) / ( ( double ) normal_delay ) ;
int msec = 1000 * sec ;
2007-12-01 11:43:38 +01:00
2008-09-08 14:58:58 -07:00
torture_comment ( tctx , " \n Running test_delayed_write_update5 \n " ) ;
2007-12-01 11:43:38 +01:00
if ( ! torture_setup_dir ( cli , BASEDIR ) ) {
return false ;
}
torture_comment ( tctx , " Open the file handle \n " ) ;
fnum1 = smbcli_open ( cli - > tree , fname , O_RDWR | O_CREAT , DENY_NONE ) ;
if ( fnum1 = = - 1 ) {
ret = false ;
torture_result ( tctx , TORTURE_FAIL , __location__ " : unable to open %s " , fname ) ;
goto done ;
}
finfo0 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
finfo0 . basic_info . in . file . fnum = fnum1 ;
finfo1 = finfo0 ;
finfo2 = finfo0 ;
finfo3 = finfo0 ;
finfo4 = finfo0 ;
finfo5 = finfo0 ;
pinfo0 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
pinfo0 . basic_info . in . file . path = fname ;
pinfo1 = pinfo0 ;
pinfo2 = pinfo0 ;
pinfo3 = pinfo0 ;
pinfo4 = pinfo0 ;
pinfo5 = pinfo0 ;
pinfo6 = pinfo0 ;
/* get the initial times */
GET_INFO_BOTH ( finfo0 , pinfo0 ) ;
/* do a write */
torture_comment ( tctx , " Do a write on the file handle \n " ) ;
written = smbcli_write ( cli - > tree , fnum1 , 0 , " x " , 0 , 1 ) ;
if ( written ! = 1 ) {
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 1 " , ( int ) written ) ;
ret = false ;
goto done ;
}
GET_INFO_BOTH ( finfo1 , pinfo1 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo1 , finfo0 ) ;
torture_comment ( tctx , " Set write time in the future on the file handle \n " ) ;
SET_INFO_FILE ( finfo0 , time ( NULL ) + 86400 ) ;
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
COMPARE_WRITE_TIME_GREATER ( finfo2 , finfo1 ) ;
torture_comment ( tctx , " Set write time in the past on the file handle \n " ) ;
SET_INFO_FILE ( finfo0 , time ( NULL ) - 86400 ) ;
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
COMPARE_WRITE_TIME_LESS ( finfo2 , finfo1 ) ;
/* make sure the 2 second delay from the first write are canceled */
2008-04-08 10:25:51 +02:00
start = timeval_current ( ) ;
end = timeval_add ( & start , 15 * sec , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
2007-12-01 11:43:38 +01:00
/* get the times after the first write */
GET_INFO_BOTH ( finfo3 , pinfo3 ) ;
if ( finfo3 . basic_info . out . write_time > finfo2 . basic_info . out . write_time ) {
2008-04-08 10:25:51 +02:00
double diff = timeval_elapsed ( & start ) ;
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (wrong!) \n " ,
diff , sec ) ;
2007-12-01 11:43:38 +01:00
ret = false ;
break ;
}
2008-04-08 10:25:51 +02:00
msleep ( 2 * msec ) ;
2007-12-01 11:43:38 +01:00
}
GET_INFO_BOTH ( finfo3 , pinfo3 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo3 , finfo2 ) ;
2008-09-05 21:47:06 -07:00
if ( finfo3 . basic_info . out . write_time = = finfo2 . basic_info . out . write_time ) {
2007-12-01 11:43:38 +01:00
torture_comment ( tctx , " Server did not update write_time (correct) \n " ) ;
}
/* sure any further write doesn't update the write time */
2008-04-08 10:25:51 +02:00
start = timeval_current ( ) ;
end = timeval_add ( & start , 15 * sec , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
2007-12-01 11:43:38 +01:00
/* do a write */
torture_comment ( tctx , " Do a write on the file handle \n " ) ;
written = smbcli_write ( cli - > tree , fnum1 , 0 , " x " , 0 , 1 ) ;
if ( written ! = 1 ) {
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 1 " , ( int ) written ) ;
ret = false ;
goto done ;
}
/* get the times after the write */
GET_INFO_BOTH ( finfo4 , pinfo4 ) ;
if ( finfo4 . basic_info . out . write_time > finfo3 . basic_info . out . write_time ) {
2008-04-08 10:25:51 +02:00
double diff = timeval_elapsed ( & start ) ;
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (wrong!) \n " ,
diff , sec ) ;
2007-12-01 11:43:38 +01:00
ret = false ;
break ;
}
2008-04-08 10:25:51 +02:00
msleep ( 2 * msec ) ;
2007-12-01 11:43:38 +01:00
}
GET_INFO_BOTH ( finfo4 , pinfo4 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo4 , finfo3 ) ;
if ( finfo4 . basic_info . out . write_time = = finfo3 . basic_info . out . write_time ) {
torture_comment ( tctx , " Server did not update write_time (correct) \n " ) ;
}
/* sleep */
2008-04-08 10:25:51 +02:00
msleep ( 5 * msec ) ;
2007-12-01 11:43:38 +01:00
GET_INFO_BOTH ( finfo5 , pinfo5 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo5 , finfo4 ) ;
/*
* the close doesn ' t update the write time
*/
torture_comment ( tctx , " Close the file handle \n " ) ;
smbcli_close ( cli - > tree , fnum1 ) ;
fnum1 = - 1 ;
GET_INFO_PATH ( pinfo6 ) ;
COMPARE_WRITE_TIME_EQUAL ( pinfo6 , pinfo5 ) ;
if ( pinfo6 . basic_info . out . write_time = = pinfo5 . basic_info . out . write_time ) {
torture_comment ( tctx , " Server did not update the write_time on close (correct) \n " ) ;
}
done :
if ( fnum1 ! = - 1 )
smbcli_close ( cli - > tree , fnum1 ) ;
smbcli_unlink ( cli - > tree , fname ) ;
smbcli_deltree ( cli - > tree , BASEDIR ) ;
return ret ;
}
2008-09-12 14:59:32 -07:00
/*
* Show truncate writes and closes have no effect on updating times once a SETWRITETIME is done .
*/
2008-09-08 12:27:43 +02:00
static bool test_delayed_write_update5b ( struct torture_context * tctx ,
struct smbcli_state * cli ,
struct smbcli_state * cli2 )
{
union smb_fileinfo finfo0 , finfo1 , finfo2 , finfo3 , finfo4 , finfo5 ;
union smb_fileinfo pinfo0 , pinfo1 , pinfo2 , pinfo3 , pinfo4 , pinfo5 , pinfo6 ;
2008-09-08 14:58:58 -07:00
const char * fname = BASEDIR " \\ torture_fileb.txt " ;
2008-09-08 12:27:43 +02:00
int fnum1 = - 1 ;
bool ret = true ;
ssize_t written ;
struct timeval start ;
struct timeval end ;
int used_delay = torture_setting_int ( tctx , " writetimeupdatedelay " , 2000000 ) ;
int normal_delay = 2000000 ;
double sec = ( ( double ) used_delay ) / ( ( double ) normal_delay ) ;
int msec = 1000 * sec ;
2008-09-08 14:58:58 -07:00
torture_comment ( tctx , " \n Running test_delayed_write_update5b \n " ) ;
2008-09-08 12:27:43 +02:00
if ( ! torture_setup_dir ( cli , BASEDIR ) ) {
return false ;
}
torture_comment ( tctx , " Open the file handle \n " ) ;
fnum1 = smbcli_open ( cli - > tree , fname , O_RDWR | O_CREAT , DENY_NONE ) ;
if ( fnum1 = = - 1 ) {
ret = false ;
torture_result ( tctx , TORTURE_FAIL , __location__ " : unable to open %s " , fname ) ;
goto done ;
}
finfo0 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
finfo0 . basic_info . in . file . fnum = fnum1 ;
finfo1 = finfo0 ;
finfo2 = finfo0 ;
finfo3 = finfo0 ;
finfo4 = finfo0 ;
finfo5 = finfo0 ;
pinfo0 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
pinfo0 . basic_info . in . file . path = fname ;
pinfo1 = pinfo0 ;
pinfo2 = pinfo0 ;
pinfo3 = pinfo0 ;
pinfo4 = pinfo0 ;
pinfo5 = pinfo0 ;
pinfo6 = pinfo0 ;
/* get the initial times */
GET_INFO_BOTH ( finfo0 , pinfo0 ) ;
/* do a write */
torture_comment ( tctx , " Do a write on the file handle \n " ) ;
written = smbcli_write ( cli - > tree , fnum1 , 0 , " x " , 0 , 1 ) ;
if ( written ! = 1 ) {
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 1 " , ( int ) written ) ;
ret = false ;
goto done ;
}
GET_INFO_BOTH ( finfo1 , pinfo1 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo1 , finfo0 ) ;
torture_comment ( tctx , " Set write time in the future on the file handle \n " ) ;
SET_INFO_FILE ( finfo0 , time ( NULL ) + 86400 ) ;
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
COMPARE_WRITE_TIME_GREATER ( finfo2 , finfo1 ) ;
torture_comment ( tctx , " Set write time in the past on the file handle \n " ) ;
SET_INFO_FILE ( finfo0 , time ( NULL ) - 86400 ) ;
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
COMPARE_WRITE_TIME_LESS ( finfo2 , finfo1 ) ;
/* make sure the 2 second delay from the first write are canceled */
start = timeval_current ( ) ;
end = timeval_add ( & start , 15 * sec , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
/* get the times after the first write */
GET_INFO_BOTH ( finfo3 , pinfo3 ) ;
if ( finfo3 . basic_info . out . write_time > finfo2 . basic_info . out . write_time ) {
double diff = timeval_elapsed ( & start ) ;
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (wrong!) \n " ,
diff , sec ) ;
ret = false ;
break ;
}
msleep ( 2 * msec ) ;
}
GET_INFO_BOTH ( finfo3 , pinfo3 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo3 , finfo2 ) ;
if ( finfo3 . basic_info . out . write_time = = finfo2 . basic_info . out . write_time ) {
torture_comment ( tctx , " Server did not update write_time (correct) \n " ) ;
}
2008-09-12 14:59:32 -07:00
/* Do any further write (truncates) update the write time ? */
2008-09-08 12:27:43 +02:00
start = timeval_current ( ) ;
end = timeval_add ( & start , 15 * sec , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
/* do a write */
torture_comment ( tctx , " Do a truncate write on the file handle \n " ) ;
2008-09-09 21:24:34 -07:00
written = smbcli_smbwrite ( cli - > tree , fnum1 , " x " , 1024 , 0 ) ;
2008-09-08 12:27:43 +02:00
if ( written ! = 0 ) {
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 1 " , ( int ) written ) ;
ret = false ;
goto done ;
}
/* get the times after the write */
GET_INFO_BOTH ( finfo4 , pinfo4 ) ;
if ( finfo4 . basic_info . out . write_time > finfo3 . basic_info . out . write_time ) {
double diff = timeval_elapsed ( & start ) ;
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (wrong!) \n " ,
diff , sec ) ;
ret = false ;
break ;
}
msleep ( 2 * msec ) ;
}
GET_INFO_BOTH ( finfo4 , pinfo4 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo4 , finfo3 ) ;
if ( finfo4 . basic_info . out . write_time = = finfo3 . basic_info . out . write_time ) {
torture_comment ( tctx , " Server did not update write_time (correct) \n " ) ;
}
/* sleep */
msleep ( 5 * msec ) ;
GET_INFO_BOTH ( finfo5 , pinfo5 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo5 , finfo4 ) ;
/*
* the close doesn ' t update the write time
*/
torture_comment ( tctx , " Close the file handle \n " ) ;
smbcli_close ( cli - > tree , fnum1 ) ;
fnum1 = - 1 ;
GET_INFO_PATH ( pinfo6 ) ;
COMPARE_WRITE_TIME_EQUAL ( pinfo6 , pinfo5 ) ;
if ( pinfo6 . basic_info . out . write_time = = pinfo5 . basic_info . out . write_time ) {
torture_comment ( tctx , " Server did not update the write_time on close (correct) \n " ) ;
}
done :
if ( fnum1 ! = - 1 )
smbcli_close ( cli - > tree , fnum1 ) ;
smbcli_unlink ( cli - > tree , fname ) ;
smbcli_deltree ( cli - > tree , BASEDIR ) ;
return ret ;
}
2008-09-12 14:59:32 -07:00
/*
* Open 2 handles on a file . Write one one and then set the
* WRITE TIME explicitly on the other . Ensure the write time
* update is cancelled . Ensure the write time is updated to
* the close time when the non - explicit set handle is closed .
*
*/
2007-12-01 11:43:38 +01:00
static bool test_delayed_write_update6 ( struct torture_context * tctx ,
struct smbcli_state * cli ,
struct smbcli_state * cli2 )
{
union smb_fileinfo finfo0 , finfo1 , finfo2 , finfo3 , finfo4 , finfo5 ;
union smb_fileinfo pinfo0 , pinfo1 , pinfo2 , pinfo3 , pinfo4 , pinfo5 , pinfo6 , pinfo7 ;
2008-09-08 14:58:58 -07:00
const char * fname = BASEDIR " \\ torture_file6.txt " ;
2007-12-01 11:43:38 +01:00
int fnum1 = - 1 ;
int fnum2 = - 1 ;
bool ret = true ;
ssize_t written ;
2008-04-08 10:25:51 +02:00
struct timeval start ;
struct timeval end ;
int used_delay = torture_setting_int ( tctx , " writetimeupdatedelay " , 2000000 ) ;
int normal_delay = 2000000 ;
double sec = ( ( double ) used_delay ) / ( ( double ) normal_delay ) ;
int msec = 1000 * sec ;
2007-12-01 11:43:38 +01:00
bool first = true ;
2008-09-08 14:58:58 -07:00
torture_comment ( tctx , " \n Running test_delayed_write_update6 \n " ) ;
2007-12-01 11:43:38 +01:00
if ( ! torture_setup_dir ( cli , BASEDIR ) ) {
return false ;
}
again :
torture_comment ( tctx , " Open the file handle \n " ) ;
fnum1 = smbcli_open ( cli - > tree , fname , O_RDWR | O_CREAT , DENY_NONE ) ;
if ( fnum1 = = - 1 ) {
ret = false ;
torture_result ( tctx , TORTURE_FAIL , __location__ " : unable to open %s " , fname ) ;
goto done ;
}
if ( fnum2 = = - 1 ) {
torture_comment ( tctx , " Open the 2nd file handle on 2nd connection \n " ) ;
fnum2 = smbcli_open ( cli2 - > tree , fname , O_RDWR | O_CREAT , DENY_NONE ) ;
if ( fnum2 = = - 1 ) {
ret = false ;
torture_result ( tctx , TORTURE_FAIL , __location__ " : unable to open %s " , fname ) ;
goto done ;
}
}
finfo0 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
finfo0 . basic_info . in . file . fnum = fnum1 ;
finfo1 = finfo0 ;
finfo2 = finfo0 ;
finfo3 = finfo0 ;
finfo4 = finfo0 ;
finfo5 = finfo0 ;
pinfo0 . basic_info . level = RAW_FILEINFO_BASIC_INFO ;
pinfo0 . basic_info . in . file . path = fname ;
pinfo1 = pinfo0 ;
pinfo2 = pinfo0 ;
pinfo3 = pinfo0 ;
pinfo4 = pinfo0 ;
pinfo5 = pinfo0 ;
pinfo6 = pinfo0 ;
pinfo7 = pinfo0 ;
/* get the initial times */
GET_INFO_BOTH ( finfo0 , pinfo0 ) ;
/* do a write */
torture_comment ( tctx , " Do a write on the file handle \n " ) ;
written = smbcli_write ( cli - > tree , fnum1 , 0 , " x " , 0 , 1 ) ;
if ( written ! = 1 ) {
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 1 " , ( int ) written ) ;
ret = false ;
goto done ;
}
GET_INFO_BOTH ( finfo1 , pinfo1 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo1 , finfo0 ) ;
torture_comment ( tctx , " Set write time in the future on the 2nd file handle \n " ) ;
SET_INFO_FILE_EX ( finfo0 , time ( NULL ) + 86400 , cli2 - > tree , fnum2 ) ;
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
COMPARE_WRITE_TIME_GREATER ( finfo2 , finfo1 ) ;
torture_comment ( tctx , " Set write time in the past on the 2nd file handle \n " ) ;
SET_INFO_FILE_EX ( finfo0 , time ( NULL ) - 86400 , cli2 - > tree , fnum2 ) ;
GET_INFO_BOTH ( finfo2 , pinfo2 ) ;
COMPARE_WRITE_TIME_LESS ( finfo2 , finfo1 ) ;
/* make sure the 2 second delay from the first write are canceled */
2008-04-08 10:25:51 +02:00
start = timeval_current ( ) ;
end = timeval_add ( & start , 15 * sec , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
2007-12-01 11:43:38 +01:00
/* get the times after the first write */
GET_INFO_BOTH ( finfo3 , pinfo3 ) ;
if ( finfo3 . basic_info . out . write_time > finfo2 . basic_info . out . write_time ) {
2008-04-08 10:25:51 +02:00
double diff = timeval_elapsed ( & start ) ;
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (wrong!) \n " ,
diff , sec ) ;
2007-12-01 11:43:38 +01:00
ret = false ;
break ;
}
2008-04-08 10:25:51 +02:00
msleep ( 2 * msec ) ;
2007-12-01 11:43:38 +01:00
}
GET_INFO_BOTH ( finfo3 , pinfo3 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo3 , finfo2 ) ;
2008-09-05 21:47:06 -07:00
if ( finfo3 . basic_info . out . write_time = = finfo2 . basic_info . out . write_time ) {
2007-12-01 11:43:38 +01:00
torture_comment ( tctx , " Server did not update write_time (correct) \n " ) ;
}
/* sure any further write doesn't update the write time */
2008-04-08 10:25:51 +02:00
start = timeval_current ( ) ;
end = timeval_add ( & start , 15 * sec , 0 ) ;
while ( ! timeval_expired ( & end ) ) {
2007-12-01 11:43:38 +01:00
/* do a write */
torture_comment ( tctx , " Do a write on the file handle \n " ) ;
written = smbcli_write ( cli - > tree , fnum1 , 0 , " x " , 0 , 1 ) ;
if ( written ! = 1 ) {
torture_result ( tctx , TORTURE_FAIL , __location__ " : written gave %d - should have been 1 " , ( int ) written ) ;
ret = false ;
goto done ;
}
/* get the times after the write */
GET_INFO_BOTH ( finfo4 , pinfo4 ) ;
if ( finfo4 . basic_info . out . write_time > finfo3 . basic_info . out . write_time ) {
2008-04-08 10:25:51 +02:00
double diff = timeval_elapsed ( & start ) ;
torture_comment ( tctx , " Server updated write_time after %.2f seconds "
" (1sec == %.2f) (wrong!) \n " ,
diff , sec ) ;
2007-12-01 11:43:38 +01:00
ret = false ;
break ;
}
2008-04-08 10:25:51 +02:00
msleep ( 2 * msec ) ;
2007-12-01 11:43:38 +01:00
}
GET_INFO_BOTH ( finfo4 , pinfo4 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo4 , finfo3 ) ;
if ( finfo4 . basic_info . out . write_time = = finfo3 . basic_info . out . write_time ) {
torture_comment ( tctx , " Server did not update write_time (correct) \n " ) ;
}
/* sleep */
2008-04-08 10:25:51 +02:00
msleep ( 5 * msec ) ;
2007-12-01 11:43:38 +01:00
GET_INFO_BOTH ( finfo5 , pinfo5 ) ;
COMPARE_WRITE_TIME_EQUAL ( finfo5 , finfo4 ) ;
/*
* the close updates the write time to the time of the close
* as the write time was set on the 2 nd handle
*/
torture_comment ( tctx , " Close the file handle \n " ) ;
smbcli_close ( cli - > tree , fnum1 ) ;
fnum1 = - 1 ;
GET_INFO_PATH ( pinfo6 ) ;
COMPARE_WRITE_TIME_GREATER ( pinfo6 , pinfo5 ) ;
if ( pinfo6 . basic_info . out . write_time > pinfo5 . basic_info . out . write_time ) {
torture_comment ( tctx , " Server updated the write_time on close (correct) \n " ) ;
}
/* keep the 2nd handle open and rerun tests */
if ( first ) {
first = false ;
goto again ;
}
/*
* closing the 2 nd handle will cause no write time update
* as the write time was explicit set on this handle
*/
torture_comment ( tctx , " Close the 2nd file handle \n " ) ;
smbcli_close ( cli2 - > tree , fnum2 ) ;
fnum2 = - 1 ;
GET_INFO_PATH ( pinfo7 ) ;
COMPARE_WRITE_TIME_EQUAL ( pinfo7 , pinfo6 ) ;
if ( pinfo7 . basic_info . out . write_time = = pinfo6 . basic_info . out . write_time ) {
torture_comment ( tctx , " Server did not update the write_time on close (correct) \n " ) ;
}
done :
if ( fnum1 ! = - 1 )
smbcli_close ( cli - > tree , fnum1 ) ;
if ( fnum2 ! = - 1 )
smbcli_close ( cli2 - > tree , fnum2 ) ;
smbcli_unlink ( cli - > tree , fname ) ;
smbcli_deltree ( cli - > tree , BASEDIR ) ;
return ret ;
}
2008-09-12 14:59:32 -07:00
/*
2004-11-12 09:37:59 +00:00
testing of delayed update of write_time
*/
2006-10-16 13:06:41 +00:00
struct torture_suite * torture_delay_write ( void )
2004-11-12 09:37:59 +00:00
{
2006-10-16 13:06:41 +00:00
struct torture_suite * suite = torture_suite_create ( talloc_autofree_context ( ) , " DELAYWRITE " ) ;
2004-11-12 09:37:59 +00:00
2006-10-16 13:06:41 +00:00
torture_suite_add_2smb_test ( suite , " finfo update on close " , test_finfo_after_write ) ;
torture_suite_add_1smb_test ( suite , " delayed update of write time " , test_delayed_write_update ) ;
2008-09-18 16:05:46 +02:00
torture_suite_add_1smb_test ( suite , " update of write time and SMBwrite truncate " , test_delayed_write_update1 ) ;
2008-09-08 08:31:34 +02:00
torture_suite_add_1smb_test ( suite , " update of write time and SMBwrite truncate expand " , test_delayed_write_update1a ) ;
2008-09-05 14:24:36 -07:00
torture_suite_add_1smb_test ( suite , " update of write time using SET_END_OF_FILE " , test_delayed_write_update1b ) ;
torture_suite_add_1smb_test ( suite , " update of write time using SET_ALLOCATION_SIZE " , test_delayed_write_update1c ) ;
2006-10-16 13:06:41 +00:00
torture_suite_add_2smb_test ( suite , " delayed update of write time using 2 connections " , test_delayed_write_update2 ) ;
2007-12-01 11:43:38 +01:00
torture_suite_add_2smb_test ( suite , " delayed update of write time 3 " , test_delayed_write_update3 ) ;
2008-09-08 13:59:51 +02:00
torture_suite_add_2smb_test ( suite , " delayed update of write time 3a " , test_delayed_write_update3a ) ;
2008-09-08 12:12:25 +02:00
torture_suite_add_2smb_test ( suite , " delayed update of write time 3b " , test_delayed_write_update3b ) ;
2008-09-08 14:11:17 +02:00
torture_suite_add_2smb_test ( suite , " delayed update of write time 3c " , test_delayed_write_update3c ) ;
2007-12-01 11:43:38 +01:00
torture_suite_add_2smb_test ( suite , " delayed update of write time 4 " , test_delayed_write_update4 ) ;
torture_suite_add_2smb_test ( suite , " delayed update of write time 5 " , test_delayed_write_update5 ) ;
2008-09-08 12:27:43 +02:00
torture_suite_add_2smb_test ( suite , " delayed update of write time 5b " , test_delayed_write_update5b ) ;
2007-12-01 11:43:38 +01:00
torture_suite_add_2smb_test ( suite , " delayed update of write time 6 " , test_delayed_write_update6 ) ;
2004-11-12 09:37:59 +00:00
2006-10-16 13:06:41 +00:00
return suite ;
2004-11-12 09:37:59 +00:00
}