2003-08-13 05:53:07 +04:00
/*
Unix SMB / CIFS implementation .
basic raw test suite for multiplexing
Copyright ( C ) Andrew Tridgell 2003
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
2007-07-10 06:07:03 +04:00
the Free Software Foundation ; either version 3 of the License , or
2003-08-13 05:53:07 +04:00
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
2007-07-10 06:07:03 +04:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2003-08-13 05:53:07 +04:00
*/
# include "includes.h"
2005-02-10 08:09:35 +03:00
# include "system/filesys.h"
2004-11-01 04:03:22 +03:00
# include "libcli/raw/libcliraw.h"
2008-04-02 06:53:27 +04:00
# include "libcli/raw/raw_proto.h"
2006-01-03 18:40:05 +03:00
# include "libcli/libcli.h"
2006-03-17 20:59:58 +03:00
# include "torture/util.h"
2011-03-19 02:42:42 +03:00
# include "torture/raw/proto.h"
2003-08-13 05:53:07 +04:00
# define BASEDIR "\\test_mux"
/*
test the delayed reply to a open that leads to a sharing violation
*/
2014-06-20 14:28:45 +04:00
static bool test_mux_open ( struct torture_context * tctx , struct smbcli_state * cli , TALLOC_CTX * mem_ctx )
2003-08-13 05:53:07 +04:00
{
union smb_open io ;
NTSTATUS status ;
2004-11-05 04:14:06 +03:00
int fnum1 , fnum2 ;
2007-10-07 02:28:14 +04:00
bool ret = true ;
2004-11-05 05:22:07 +03:00
struct smbcli_request * req1 , * req2 ;
2004-11-03 13:09:48 +03:00
struct timeval tv ;
double d ;
2003-08-13 05:53:07 +04:00
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " Testing multiplexed open/open/close \n " ) ;
2003-08-13 05:53:07 +04:00
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " send first open \n " ) ;
2003-08-13 05:53:07 +04:00
io . generic . level = RAW_OPEN_NTCREATEX ;
2009-10-15 11:26:19 +04:00
io . ntcreatex . in . root_fid . fnum = 0 ;
2003-08-13 05:53:07 +04:00
io . ntcreatex . in . flags = 0 ;
2004-11-30 07:33:27 +03:00
io . ntcreatex . in . access_mask = SEC_FILE_READ_DATA ;
2003-08-13 05:53:07 +04:00
io . ntcreatex . in . create_options = 0 ;
io . ntcreatex . in . file_attr = FILE_ATTRIBUTE_NORMAL ;
2004-11-05 04:14:06 +03:00
io . ntcreatex . in . share_access = NTCREATEX_SHARE_ACCESS_READ ;
2003-08-13 05:53:07 +04:00
io . ntcreatex . in . alloc_size = 0 ;
io . ntcreatex . in . open_disposition = NTCREATEX_DISP_CREATE ;
io . ntcreatex . in . impersonation = NTCREATEX_IMPERSONATION_ANONYMOUS ;
io . ntcreatex . in . security_flags = 0 ;
io . ntcreatex . in . fname = BASEDIR " \\ open.dat " ;
status = smb_raw_open ( cli - > tree , mem_ctx , & io ) ;
2014-06-20 14:28:45 +04:00
torture_assert_ntstatus_equal ( tctx , status , NT_STATUS_OK , " send first open " ) ;
2006-03-13 01:48:25 +03:00
fnum1 = io . ntcreatex . out . file . fnum ;
2004-11-05 04:14:06 +03:00
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " send 2nd open, non-conflicting \n " ) ;
2004-11-05 04:14:06 +03:00
io . ntcreatex . in . open_disposition = NTCREATEX_DISP_OPEN ;
status = smb_raw_open ( cli - > tree , mem_ctx , & io ) ;
2014-06-20 14:28:45 +04:00
torture_assert_ntstatus_equal ( tctx , status , NT_STATUS_OK , " send 2nd open, non-conflicting " ) ;
2006-03-13 01:48:25 +03:00
fnum2 = io . ntcreatex . out . file . fnum ;
2003-08-13 05:53:07 +04:00
2004-11-03 13:09:48 +03:00
tv = timeval_current ( ) ;
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " send 3rd open, conflicting \n " ) ;
2004-11-05 04:14:06 +03:00
io . ntcreatex . in . share_access = 0 ;
2003-08-13 05:53:07 +04:00
status = smb_raw_open ( cli - > tree , mem_ctx , & io ) ;
2014-06-20 14:28:45 +04:00
torture_assert_ntstatus_equal ( tctx , status , NT_STATUS_SHARING_VIOLATION , " send 3rd open, conflicting " ) ;
2003-08-13 05:53:07 +04:00
2004-11-03 13:09:48 +03:00
d = timeval_elapsed ( & tv ) ;
if ( d < 0.5 | | d > 1.5 ) {
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " bad timeout for conflict - %.2f should be 1.0 \n " , d ) ;
2004-11-03 13:09:48 +03:00
} else {
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " open delay %.2f \n " , d ) ;
2004-11-03 13:09:48 +03:00
}
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " send async open, conflicting \n " ) ;
2004-11-05 05:22:07 +03:00
tv = timeval_current ( ) ;
req1 = smb_raw_open_send ( cli - > tree , & io ) ;
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " send 2nd async open, conflicting \n " ) ;
2004-11-03 13:09:48 +03:00
tv = timeval_current ( ) ;
2004-11-05 05:22:07 +03:00
req2 = smb_raw_open_send ( cli - > tree , & io ) ;
2003-08-13 05:53:07 +04:00
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " close first sync open \n " ) ;
2004-11-05 04:14:06 +03:00
smbcli_close ( cli - > tree , fnum1 ) ;
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " cancel 2nd async open (should be ignored) \n " ) ;
2004-11-05 05:22:07 +03:00
smb_raw_ntcancel ( req2 ) ;
d = timeval_elapsed ( & tv ) ;
if ( d > 0.25 ) {
2014-06-20 14:28:46 +04:00
torture_comment ( tctx , " bad timeout after cancel - %.2f should be <0.25 \n " , d ) ;
torture_assert ( tctx , d < = 0.25 , " bad timeout after cancel " ) ;
2004-11-05 05:22:07 +03:00
}
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " close the 2nd sync open \n " ) ;
2004-11-05 04:14:06 +03:00
smbcli_close ( cli - > tree , fnum2 ) ;
2003-08-13 05:53:07 +04:00
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " see if the 1st async open now succeeded \n " ) ;
2004-11-05 05:22:07 +03:00
status = smb_raw_open_recv ( req1 , mem_ctx , & io ) ;
2014-06-20 14:28:45 +04:00
torture_assert_ntstatus_equal ( tctx , status , NT_STATUS_OK , " see if the 1st async open now succeeded " ) ;
2003-08-13 05:53:07 +04:00
2004-11-03 13:09:48 +03:00
d = timeval_elapsed ( & tv ) ;
if ( d > 0.25 ) {
2014-06-20 14:28:46 +04:00
torture_comment ( tctx , " bad timeout for async conflict - %.2f should be <0.25 \n " , d ) ;
torture_assert ( tctx , d < = 0.25 , " bad timeout for async conflict " ) ;
2004-11-03 13:09:48 +03:00
} else {
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " async open delay %.2f \n " , d ) ;
2004-11-03 13:09:48 +03:00
}
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " 2nd async open should have timed out \n " ) ;
2004-11-05 05:22:07 +03:00
status = smb_raw_open_recv ( req2 , mem_ctx , & io ) ;
2014-06-20 14:28:45 +04:00
torture_assert_ntstatus_equal ( tctx , status , NT_STATUS_SHARING_VIOLATION , " 2nd async open should have timed out " ) ;
2004-11-05 05:22:07 +03:00
d = timeval_elapsed ( & tv ) ;
if ( d < 0.8 ) {
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " bad timeout for async conflict - %.2f should be 1.0 \n " , d ) ;
2004-11-05 05:22:07 +03:00
}
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " close the 1st async open \n " ) ;
2006-03-13 01:48:25 +03:00
smbcli_close ( cli - > tree , io . ntcreatex . out . file . fnum ) ;
2003-08-13 05:53:07 +04:00
return ret ;
}
/*
test a write that hits a byte range lock and send the close after the write
*/
2014-06-20 14:28:45 +04:00
static bool test_mux_write ( struct torture_context * tctx , struct smbcli_state * cli , TALLOC_CTX * mem_ctx )
2003-08-13 05:53:07 +04:00
{
union smb_write io ;
NTSTATUS status ;
int fnum ;
2007-10-07 02:28:14 +04:00
bool ret = true ;
2004-08-04 17:23:35 +04:00
struct smbcli_request * req ;
2003-08-13 05:53:07 +04:00
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " Testing multiplexed lock/write/close \n " ) ;
2003-08-13 05:53:07 +04:00
2004-08-04 17:23:35 +04:00
fnum = smbcli_open ( cli - > tree , BASEDIR " \\ write.dat " , O_RDWR | O_CREAT , DENY_NONE ) ;
2003-08-13 05:53:07 +04:00
if ( fnum = = - 1 ) {
2014-06-20 14:28:46 +04:00
torture_comment ( tctx , " open failed in mux_write - %s \n " , smbcli_errstr ( cli - > tree ) ) ;
torture_assert ( tctx , fnum ! = - 1 , " open failed in mux_write " ) ;
2003-08-13 05:53:07 +04:00
}
cli - > session - > pid = 1 ;
2014-06-20 14:28:46 +04:00
status = smbcli_lock ( cli - > tree , fnum , 0 , 4 , 0 , WRITE_LOCK ) ;
2003-08-13 05:53:07 +04:00
/* lock a range */
2014-06-20 14:28:46 +04:00
if ( NT_STATUS_IS_ERR ( status ) ) {
torture_assert_ntstatus_ok ( tctx , status , " lock failed in mux_write " ) ;
2003-08-13 05:53:07 +04:00
}
cli - > session - > pid = 2 ;
/* send an async write */
io . generic . level = RAW_WRITE_WRITEX ;
2006-03-13 01:48:25 +03:00
io . writex . in . file . fnum = fnum ;
2003-08-13 05:53:07 +04:00
io . writex . in . offset = 0 ;
io . writex . in . wmode = 0 ;
io . writex . in . remaining = 0 ;
io . writex . in . count = 4 ;
2007-09-08 20:46:30 +04:00
io . writex . in . data = ( const uint8_t * ) & fnum ;
2003-08-13 05:53:07 +04:00
req = smb_raw_write_send ( cli - > tree , & io ) ;
/* unlock the range */
cli - > session - > pid = 1 ;
2004-08-04 17:23:35 +04:00
smbcli_unlock ( cli - > tree , fnum , 0 , 4 ) ;
2003-08-13 05:53:07 +04:00
/* and recv the async write reply */
status = smb_raw_write_recv ( req , & io ) ;
2014-06-20 14:28:45 +04:00
torture_assert_ntstatus_equal ( tctx , status , NT_STATUS_FILE_LOCK_CONFLICT , " recv the async write reply " ) ;
2003-08-13 05:53:07 +04:00
2004-08-04 17:23:35 +04:00
smbcli_close ( cli - > tree , fnum ) ;
2003-08-13 05:53:07 +04:00
return ret ;
}
/*
test a lock that conflicts with an existing lock
*/
2014-06-20 14:28:45 +04:00
static bool test_mux_lock ( struct torture_context * tctx , struct smbcli_state * cli , TALLOC_CTX * mem_ctx )
2003-08-13 05:53:07 +04:00
{
union smb_lock io ;
NTSTATUS status ;
int fnum ;
2007-10-07 02:28:14 +04:00
bool ret = true ;
2004-08-04 17:23:35 +04:00
struct smbcli_request * req ;
2003-08-13 05:53:07 +04:00
struct smb_lock_entry lock [ 1 ] ;
2007-02-05 06:04:01 +03:00
struct timeval t ;
2003-08-13 05:53:07 +04:00
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " TESTING MULTIPLEXED LOCK/LOCK/UNLOCK \n " ) ;
2003-08-13 05:53:07 +04:00
2004-08-04 17:23:35 +04:00
fnum = smbcli_open ( cli - > tree , BASEDIR " \\ write.dat " , O_RDWR | O_CREAT , DENY_NONE ) ;
2003-08-13 05:53:07 +04:00
if ( fnum = = - 1 ) {
2014-06-20 14:28:46 +04:00
torture_comment ( tctx , " open failed in mux_lock - %s \n " , smbcli_errstr ( cli - > tree ) ) ;
torture_assert ( tctx , fnum ! = - 1 , " open failed in mux_lock " ) ;
2003-08-13 05:53:07 +04:00
}
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " establishing a lock \n " ) ;
2003-08-13 05:53:07 +04:00
io . lockx . level = RAW_LOCK_LOCKX ;
2006-03-13 01:48:25 +03:00
io . lockx . in . file . fnum = fnum ;
2003-08-13 05:53:07 +04:00
io . lockx . in . mode = 0 ;
io . lockx . in . timeout = 0 ;
io . lockx . in . lock_cnt = 1 ;
io . lockx . in . ulock_cnt = 0 ;
lock [ 0 ] . pid = 1 ;
lock [ 0 ] . offset = 0 ;
lock [ 0 ] . count = 4 ;
io . lockx . in . locks = & lock [ 0 ] ;
status = smb_raw_lock ( cli - > tree , & io ) ;
2014-06-20 14:28:45 +04:00
torture_assert_ntstatus_equal ( tctx , status , NT_STATUS_OK , " establishing a lock " ) ;
2003-08-13 05:53:07 +04:00
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " the second lock will conflict with the first \n " ) ;
2003-08-13 05:53:07 +04:00
lock [ 0 ] . pid = 2 ;
io . lockx . in . timeout = 1000 ;
status = smb_raw_lock ( cli - > tree , & io ) ;
2014-06-20 14:28:45 +04:00
torture_assert_ntstatus_equal ( tctx , status , NT_STATUS_FILE_LOCK_CONFLICT , " the second lock will conflict with the first " ) ;
2003-08-13 05:53:07 +04:00
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " this will too, but we'll unlock while waiting \n " ) ;
2007-02-05 06:04:01 +03:00
t = timeval_current ( ) ;
2003-08-13 05:53:07 +04:00
req = smb_raw_lock_send ( cli - > tree , & io ) ;
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " unlock the first range \n " ) ;
2003-08-13 05:53:07 +04:00
lock [ 0 ] . pid = 1 ;
io . lockx . in . ulock_cnt = 1 ;
io . lockx . in . lock_cnt = 0 ;
io . lockx . in . timeout = 0 ;
status = smb_raw_lock ( cli - > tree , & io ) ;
2014-06-20 14:28:45 +04:00
torture_assert_ntstatus_equal ( tctx , status , NT_STATUS_OK , " unlock the first range " ) ;
2003-08-13 05:53:07 +04:00
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " recv the async reply \n " ) ;
2004-08-04 17:23:35 +04:00
status = smbcli_request_simple_recv ( req ) ;
2014-06-20 14:28:45 +04:00
torture_assert_ntstatus_equal ( tctx , status , NT_STATUS_OK , " recv the async reply " ) ;
2003-08-13 05:53:07 +04:00
2014-06-20 14:28:46 +04:00
torture_comment ( tctx , " async lock took %.2f msec \n " , timeval_elapsed ( & t ) * 1000 ) ;
torture_assert ( tctx , timeval_elapsed ( & t ) < = 0.1 , " failed to trigger early lock retry \n " ) ;
2007-02-05 06:04:01 +03:00
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " reopening with an exit \n " ) ;
2003-08-13 05:53:07 +04:00
smb_raw_exit ( cli - > session ) ;
2004-08-04 17:23:35 +04:00
fnum = smbcli_open ( cli - > tree , BASEDIR " \\ write.dat " , O_RDWR | O_CREAT , DENY_NONE ) ;
2003-08-13 05:53:07 +04:00
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " Now trying with a cancel \n " ) ;
2003-08-13 05:53:07 +04:00
io . lockx . level = RAW_LOCK_LOCKX ;
2006-03-13 01:48:25 +03:00
io . lockx . in . file . fnum = fnum ;
2003-08-13 05:53:07 +04:00
io . lockx . in . mode = 0 ;
io . lockx . in . timeout = 0 ;
io . lockx . in . lock_cnt = 1 ;
io . lockx . in . ulock_cnt = 0 ;
lock [ 0 ] . pid = 1 ;
lock [ 0 ] . offset = 0 ;
lock [ 0 ] . count = 4 ;
io . lockx . in . locks = & lock [ 0 ] ;
status = smb_raw_lock ( cli - > tree , & io ) ;
2014-06-20 14:28:45 +04:00
torture_assert_ntstatus_equal ( tctx , status , NT_STATUS_OK , " Now trying with a cancel " ) ;
2003-08-13 05:53:07 +04:00
lock [ 0 ] . pid = 2 ;
io . lockx . in . timeout = 1000 ;
status = smb_raw_lock ( cli - > tree , & io ) ;
2014-06-20 14:28:45 +04:00
torture_assert_ntstatus_equal ( tctx , status , NT_STATUS_FILE_LOCK_CONFLICT , " Now trying with a cancel pid 2 " ) ;
2003-08-13 05:53:07 +04:00
req = smb_raw_lock_send ( cli - > tree , & io ) ;
/* cancel the blocking lock */
smb_raw_ntcancel ( req ) ;
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " sending 2nd cancel \n " ) ;
2004-11-04 14:28:38 +03:00
/* the 2nd cancel is totally harmless, but tests the server trying to
cancel an already cancelled request */
smb_raw_ntcancel ( req ) ;
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " sent 2nd cancel \n " ) ;
2007-02-07 10:10:23 +03:00
2003-08-13 05:53:07 +04:00
lock [ 0 ] . pid = 1 ;
io . lockx . in . ulock_cnt = 1 ;
io . lockx . in . lock_cnt = 0 ;
io . lockx . in . timeout = 0 ;
status = smb_raw_lock ( cli - > tree , & io ) ;
2014-06-20 14:28:45 +04:00
torture_assert_ntstatus_equal ( tctx , status , NT_STATUS_OK , " clear lock " ) ;
2003-08-13 05:53:07 +04:00
2004-08-04 17:23:35 +04:00
status = smbcli_request_simple_recv ( req ) ;
2014-06-20 14:28:45 +04:00
torture_assert_ntstatus_equal ( tctx , status , NT_STATUS_FILE_LOCK_CONFLICT , " recv 2nd cancel " ) ;
2003-08-13 05:53:07 +04:00
2014-06-20 14:28:47 +04:00
torture_comment ( tctx , " cancel a lock using exit to close file \n " ) ;
2007-02-07 10:10:23 +03:00
lock [ 0 ] . pid = 1 ;
io . lockx . in . ulock_cnt = 0 ;
io . lockx . in . lock_cnt = 1 ;
io . lockx . in . timeout = 1000 ;
status = smb_raw_lock ( cli - > tree , & io ) ;
2014-06-20 14:28:45 +04:00
torture_assert_ntstatus_equal ( tctx , status , NT_STATUS_OK , " cancel a lock using exit to close file " ) ;
2007-02-07 10:10:23 +03:00
t = timeval_current ( ) ;
lock [ 0 ] . pid = 2 ;
req = smb_raw_lock_send ( cli - > tree , & io ) ;
smb_raw_exit ( cli - > session ) ;
smb_raw_exit ( cli - > session ) ;
smb_raw_exit ( cli - > session ) ;
smb_raw_exit ( cli - > session ) ;
2014-06-20 14:28:46 +04:00
torture_comment ( tctx , " recv the async reply \n " ) ;
2007-02-07 10:10:23 +03:00
status = smbcli_request_simple_recv ( req ) ;
2014-06-20 14:28:45 +04:00
torture_assert_ntstatus_equal ( tctx , status , NT_STATUS_RANGE_NOT_LOCKED , " recv the async reply " ) ;
2014-06-20 14:28:46 +04:00
torture_comment ( tctx , " async lock exit took %.2f msec \n " , timeval_elapsed ( & t ) * 1000 ) ;
torture_assert ( tctx , timeval_elapsed ( & t ) < = 0.1 , " failed to trigger early lock failure \n " ) ;
2003-08-13 05:53:07 +04:00
return ret ;
}
/*
basic testing of multiplexing notify
*/
2007-08-28 16:54:27 +04:00
bool torture_raw_mux ( struct torture_context * torture , struct smbcli_state * cli )
2003-08-13 05:53:07 +04:00
{
2007-08-28 16:54:27 +04:00
bool ret = true ;
2014-06-20 14:28:45 +04:00
TALLOC_CTX * frame ;
2012-05-18 09:43:31 +04:00
torture_assert ( torture , torture_setup_dir ( cli , BASEDIR ) , " Failed to setup up test directory: " BASEDIR ) ;
2014-06-20 14:28:45 +04:00
frame = talloc_stackframe ( ) ;
2003-08-13 05:53:07 +04:00
2014-06-20 14:28:45 +04:00
ret & = test_mux_open ( torture , cli , frame ) ;
ret & = test_mux_write ( torture , cli , frame ) ;
ret & = test_mux_lock ( torture , cli , frame ) ;
2003-08-13 05:53:07 +04:00
smb_raw_exit ( cli - > session ) ;
2004-08-04 17:23:35 +04:00
smbcli_deltree ( cli - > tree , BASEDIR ) ;
2014-06-20 14:28:45 +04:00
TALLOC_FREE ( frame ) ;
2003-08-13 05:53:07 +04:00
return ret ;
}