2011-09-22 22:23:08 +02:00
/*
Unix SMB / CIFS implementation .
test suite for SMB2 ioctl operations
Copyright ( C ) David Disseldorp 2011
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 3 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program . If not , see < http : //www.gnu.org/licenses/>.
*/
# include "includes.h"
# include "librpc/gen_ndr/security.h"
# include "libcli/smb2/smb2.h"
# include "libcli/smb2/smb2_calls.h"
# include "torture/torture.h"
# include "torture/smb2/proto.h"
2011-09-27 16:40:20 +02:00
# include "librpc/gen_ndr/ndr_ioctl.h"
2011-09-22 22:23:08 +02:00
2011-09-27 16:40:20 +02:00
# define FNAME "testfsctl.dat"
# define FNAME2 "testfsctl2.dat"
2011-09-22 22:23:08 +02:00
/*
basic testing of SMB2 shadow copy calls
*/
static bool test_ioctl_get_shadow_copy ( struct torture_context * torture ,
struct smb2_tree * tree )
{
struct smb2_handle h ;
uint8_t buf [ 100 ] ;
NTSTATUS status ;
union smb_ioctl ioctl ;
TALLOC_CTX * tmp_ctx = talloc_new ( tree ) ;
smb2_util_unlink ( tree , FNAME ) ;
status = torture_smb2_testfile ( tree , FNAME , & h ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ntstatus_ok ( torture , status , " create write " ) ;
2011-09-22 22:23:08 +02:00
ZERO_ARRAY ( buf ) ;
status = smb2_util_write ( tree , h , buf , 0 , ARRAY_SIZE ( buf ) ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ntstatus_ok ( torture , status , " write " ) ;
2011-09-22 22:23:08 +02:00
ZERO_STRUCT ( ioctl ) ;
ioctl . smb2 . level = RAW_IOCTL_SMB2 ;
ioctl . smb2 . in . file . handle = h ;
2011-10-18 13:55:22 +02:00
ioctl . smb2 . in . function = FSCTL_SRV_ENUM_SNAPS ;
2011-09-22 22:23:08 +02:00
ioctl . smb2 . in . max_response_size = 16 ;
2011-09-26 13:38:16 +02:00
ioctl . smb2 . in . flags = SMB2_IOCTL_FLAG_IS_FSCTL ;
2011-09-22 22:23:08 +02:00
status = smb2_ioctl ( tree , tmp_ctx , & ioctl . smb2 ) ;
2013-01-15 17:23:03 +01:00
if ( NT_STATUS_EQUAL ( status , NT_STATUS_NOT_SUPPORTED )
| | NT_STATUS_EQUAL ( status , NT_STATUS_INVALID_DEVICE_REQUEST ) ) {
2013-01-15 17:23:07 +01:00
torture_skip ( torture , " FSCTL_SRV_ENUM_SNAPS not supported \n " ) ;
2013-01-15 17:23:03 +01:00
}
2012-02-29 11:02:20 +01:00
torture_assert_ntstatus_ok ( torture , status , " FSCTL_SRV_ENUM_SNAPS " ) ;
2011-09-22 22:23:08 +02:00
return true ;
}
2011-09-27 16:40:20 +02:00
/*
basic testing of the SMB2 server side copy ioctls
*/
static bool test_ioctl_req_resume_key ( struct torture_context * torture ,
struct smb2_tree * tree )
{
struct smb2_handle h ;
uint8_t buf [ 100 ] ;
NTSTATUS status ;
union smb_ioctl ioctl ;
TALLOC_CTX * tmp_ctx = talloc_new ( tree ) ;
struct req_resume_key_rsp res_key ;
enum ndr_err_code ndr_ret ;
smb2_util_unlink ( tree , FNAME ) ;
status = torture_smb2_testfile ( tree , FNAME , & h ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ntstatus_ok ( torture , status , " create write " ) ;
2011-09-27 16:40:20 +02:00
ZERO_ARRAY ( buf ) ;
status = smb2_util_write ( tree , h , buf , 0 , ARRAY_SIZE ( buf ) ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ntstatus_ok ( torture , status , " write " ) ;
2011-09-27 16:40:20 +02:00
ZERO_STRUCT ( ioctl ) ;
ioctl . smb2 . level = RAW_IOCTL_SMB2 ;
ioctl . smb2 . in . file . handle = h ;
ioctl . smb2 . in . function = FSCTL_SRV_REQUEST_RESUME_KEY ;
ioctl . smb2 . in . max_response_size = 32 ;
ioctl . smb2 . in . flags = SMB2_IOCTL_FLAG_IS_FSCTL ;
status = smb2_ioctl ( tree , tmp_ctx , & ioctl . smb2 ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ntstatus_ok ( torture , status , " FSCTL_SRV_REQUEST_RESUME_KEY " ) ;
2011-09-27 16:40:20 +02:00
ndr_ret = ndr_pull_struct_blob ( & ioctl . smb2 . out . out , tmp_ctx , & res_key ,
( ndr_pull_flags_fn_t ) ndr_pull_req_resume_key_rsp ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ndr_success ( torture , ndr_ret ,
" ndr_pull_req_resume_key_rsp " ) ;
2011-09-27 16:40:20 +02:00
ndr_print_debug ( ( ndr_print_fn_t ) ndr_print_req_resume_key_rsp , " yo " , & res_key ) ;
talloc_free ( tmp_ctx ) ;
return true ;
}
2011-10-11 11:44:03 +02:00
static uint64_t patt_hash ( uint64_t off )
{
return off ;
}
2012-02-29 11:02:20 +01:00
static bool check_pattern ( struct torture_context * torture ,
struct smb2_tree * tree , TALLOC_CTX * mem_ctx ,
2011-10-11 11:44:03 +02:00
struct smb2_handle h , uint64_t off , uint64_t len ,
uint64_t patt_off )
{
uint64_t i ;
struct smb2_read r ;
NTSTATUS status ;
ZERO_STRUCT ( r ) ;
r . in . file . handle = h ;
r . in . length = len ;
r . in . offset = off ;
status = smb2_read ( tree , mem_ctx , & r ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ntstatus_ok ( torture , status , " read " ) ;
torture_assert_u64_equal ( torture , r . out . data . length , len ,
" read data len mismatch " ) ;
2011-10-11 11:44:03 +02:00
for ( i = 0 ; i < = len - 8 ; i + = 8 , patt_off + = 8 ) {
2012-02-29 11:02:20 +01:00
uint64_t data = BVAL ( r . out . data . data , i ) ;
torture_assert_u64_equal ( torture , data , patt_hash ( patt_off ) ,
talloc_asprintf ( torture , " read data "
" pattern bad at %llu \n " ,
( unsigned long long ) i ) ) ;
2011-10-11 11:44:03 +02:00
}
talloc_free ( r . out . data . data ) ;
return true ;
}
2012-02-29 11:02:20 +01:00
static bool test_setup_copy_chunk ( struct torture_context * torture ,
struct smb2_tree * tree , TALLOC_CTX * mem_ctx ,
2011-09-30 12:50:36 +02:00
uint32_t nchunks ,
struct smb2_handle * src_h ,
uint64_t src_size ,
struct smb2_handle * dest_h ,
uint64_t dest_size ,
struct srv_copychunk_copy * cc_copy ,
union smb_ioctl * ioctl )
2011-09-27 16:40:20 +02:00
{
struct req_resume_key_rsp res_key ;
2011-09-30 12:50:36 +02:00
NTSTATUS status ;
2011-09-27 16:40:20 +02:00
enum ndr_err_code ndr_ret ;
2011-10-11 11:44:03 +02:00
uint64_t i ;
2011-09-30 12:50:36 +02:00
uint8_t * buf = talloc_zero_size ( mem_ctx , MAX ( src_size , dest_size ) ) ;
2013-01-15 17:23:07 +01:00
torture_assert ( torture , ( buf ! = NULL ) , " no memory for file data buf " ) ;
2011-09-27 16:40:20 +02:00
smb2_util_unlink ( tree , FNAME ) ;
smb2_util_unlink ( tree , FNAME2 ) ;
2011-09-30 12:50:36 +02:00
status = torture_smb2_testfile ( tree , FNAME , src_h ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ntstatus_ok ( torture , status , " create write " ) ;
2011-09-27 16:40:20 +02:00
2011-09-30 12:50:36 +02:00
if ( src_size > 0 ) {
2011-10-11 11:44:03 +02:00
for ( i = 0 ; i < = src_size - 8 ; i + = 8 ) {
SBVAL ( buf , i , patt_hash ( i ) ) ;
}
2011-09-30 12:50:36 +02:00
status = smb2_util_write ( tree , * src_h , buf , 0 , src_size ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ntstatus_ok ( torture , status , " src write " ) ;
2011-09-30 12:50:36 +02:00
}
status = torture_smb2_testfile ( tree , FNAME2 , dest_h ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ntstatus_ok ( torture , status , " create write " ) ;
2011-09-27 16:40:20 +02:00
2011-09-30 12:50:36 +02:00
if ( dest_size > 0 ) {
2012-11-30 10:57:39 +01:00
for ( i = 0 ; i < = dest_size - 8 ; i + = 8 ) {
2011-10-11 11:44:03 +02:00
SBVAL ( buf , i , patt_hash ( i ) ) ;
}
2011-09-30 12:50:36 +02:00
status = smb2_util_write ( tree , * dest_h , buf , 0 , dest_size ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ntstatus_ok ( torture , status , " dest write " ) ;
2011-09-30 12:50:36 +02:00
}
ZERO_STRUCTPN ( ioctl ) ;
ioctl - > smb2 . level = RAW_IOCTL_SMB2 ;
ioctl - > smb2 . in . file . handle = * src_h ;
ioctl - > smb2 . in . function = FSCTL_SRV_REQUEST_RESUME_KEY ;
2011-09-27 16:40:20 +02:00
/* Allow for Key + ContextLength + Context */
2011-09-30 12:50:36 +02:00
ioctl - > smb2 . in . max_response_size = 32 ;
ioctl - > smb2 . in . flags = SMB2_IOCTL_FLAG_IS_FSCTL ;
2011-09-27 16:40:20 +02:00
2011-09-30 12:50:36 +02:00
status = smb2_ioctl ( tree , mem_ctx , & ioctl - > smb2 ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ntstatus_ok ( torture , status ,
" FSCTL_SRV_REQUEST_RESUME_KEY " ) ;
2011-09-27 16:40:20 +02:00
2011-09-30 12:50:36 +02:00
ndr_ret = ndr_pull_struct_blob ( & ioctl - > smb2 . out . out , mem_ctx , & res_key ,
2011-09-27 16:40:20 +02:00
( ndr_pull_flags_fn_t ) ndr_pull_req_resume_key_rsp ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ndr_success ( torture , ndr_ret ,
" ndr_pull_req_resume_key_rsp " ) ;
2011-09-27 16:40:20 +02:00
2011-09-30 12:50:36 +02:00
ZERO_STRUCTPN ( ioctl ) ;
ioctl - > smb2 . level = RAW_IOCTL_SMB2 ;
ioctl - > smb2 . in . file . handle = * dest_h ;
ioctl - > smb2 . in . function = FSCTL_SRV_COPYCHUNK ;
ioctl - > smb2 . in . max_response_size = sizeof ( struct srv_copychunk_rsp ) ;
ioctl - > smb2 . in . flags = SMB2_IOCTL_FLAG_IS_FSCTL ;
ZERO_STRUCTPN ( cc_copy ) ;
memcpy ( cc_copy - > source_key , res_key . resume_key , ARRAY_SIZE ( cc_copy - > source_key ) ) ;
cc_copy - > chunk_count = nchunks ;
cc_copy - > chunks = talloc_zero_array ( mem_ctx , struct srv_copychunk , nchunks ) ;
2013-01-15 17:23:07 +01:00
torture_assert ( torture , ( cc_copy - > chunks ! = NULL ) , " no memory for chunks " ) ;
2011-09-27 16:40:20 +02:00
2011-09-30 12:50:36 +02:00
return true ;
}
2011-09-27 16:40:20 +02:00
2012-02-29 11:02:20 +01:00
static bool check_copy_chunk_rsp ( struct torture_context * torture ,
struct srv_copychunk_rsp * cc_rsp ,
2011-09-30 12:50:36 +02:00
uint32_t ex_chunks_written ,
uint32_t ex_chunk_bytes_written ,
uint32_t ex_total_bytes_written )
{
2012-02-29 11:02:20 +01:00
torture_assert_int_equal ( torture , cc_rsp - > chunks_written ,
ex_chunks_written , " num chunks " ) ;
torture_assert_int_equal ( torture , cc_rsp - > chunk_bytes_written ,
ex_chunk_bytes_written , " chunk bytes written " ) ;
torture_assert_int_equal ( torture , cc_rsp - > total_bytes_written ,
ex_total_bytes_written , " chunk total bytes " ) ;
2011-09-30 12:50:36 +02:00
return true ;
}
2011-09-30 14:33:36 +02:00
static bool test_ioctl_copy_chunk_simple ( struct torture_context * torture ,
struct smb2_tree * tree )
2011-09-30 12:50:36 +02:00
{
struct smb2_handle src_h ;
struct smb2_handle dest_h ;
NTSTATUS status ;
union smb_ioctl ioctl ;
TALLOC_CTX * tmp_ctx = talloc_new ( tree ) ;
struct srv_copychunk_copy cc_copy ;
struct srv_copychunk_rsp cc_rsp ;
enum ndr_err_code ndr_ret ;
bool ok ;
2012-02-29 11:02:20 +01:00
ok = test_setup_copy_chunk ( torture , tree , tmp_ctx ,
2011-09-30 12:50:36 +02:00
1 , /* 1 chunk */
2011-09-30 14:33:36 +02:00
& src_h , 4096 , /* fill 4096 byte src file */
2011-09-30 12:50:36 +02:00
& dest_h , 0 , /* 0 byte dest file */
& cc_copy ,
& ioctl ) ;
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " setup copy chunk error " ) ;
2011-09-30 12:50:36 +02:00
}
/* copy all src file data (via a single chunk desc) */
cc_copy . chunks [ 0 ] . source_off = 0 ;
cc_copy . chunks [ 0 ] . target_off = 0 ;
2011-09-30 14:33:36 +02:00
cc_copy . chunks [ 0 ] . length = 4096 ;
2011-09-27 16:40:20 +02:00
ndr_ret = ndr_push_struct_blob ( & ioctl . smb2 . in . out , tmp_ctx ,
& cc_copy ,
( ndr_push_flags_fn_t ) ndr_push_srv_copychunk_copy ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ndr_success ( torture , ndr_ret ,
" ndr_push_srv_copychunk_copy " ) ;
2011-09-27 16:40:20 +02:00
status = smb2_ioctl ( tree , tmp_ctx , & ioctl . smb2 ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ntstatus_ok ( torture , status , " FSCTL_SRV_COPYCHUNK " ) ;
2011-09-27 16:40:20 +02:00
ndr_ret = ndr_pull_struct_blob ( & ioctl . smb2 . out . out , tmp_ctx ,
& cc_rsp ,
( ndr_pull_flags_fn_t ) ndr_pull_srv_copychunk_rsp ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ndr_success ( torture , ndr_ret ,
" ndr_pull_srv_copychunk_rsp " ) ;
2011-09-30 12:50:36 +02:00
2012-02-29 11:02:20 +01:00
ok = check_copy_chunk_rsp ( torture , & cc_rsp ,
2011-09-30 12:50:36 +02:00
1 , /* chunks written */
0 , /* chunk bytes unsuccessfully written */
2011-09-30 14:33:36 +02:00
4096 ) ; /* total bytes written */
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " bad copy chunk response data " ) ;
2011-09-30 14:33:36 +02:00
}
2012-02-29 11:02:20 +01:00
ok = check_pattern ( torture , tree , tmp_ctx , dest_h , 0 , 4096 , 0 ) ;
2011-10-11 11:44:03 +02:00
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " inconsistent file data " ) ;
2011-10-11 11:44:03 +02:00
}
2011-09-30 14:33:36 +02:00
smb2_util_close ( tree , src_h ) ;
smb2_util_close ( tree , dest_h ) ;
talloc_free ( tmp_ctx ) ;
return true ;
}
static bool test_ioctl_copy_chunk_multi ( struct torture_context * torture ,
struct smb2_tree * tree )
{
struct smb2_handle src_h ;
struct smb2_handle dest_h ;
NTSTATUS status ;
union smb_ioctl ioctl ;
TALLOC_CTX * tmp_ctx = talloc_new ( tree ) ;
struct srv_copychunk_copy cc_copy ;
struct srv_copychunk_rsp cc_rsp ;
enum ndr_err_code ndr_ret ;
bool ok ;
2012-02-29 11:02:20 +01:00
ok = test_setup_copy_chunk ( torture , tree , tmp_ctx ,
2011-09-30 14:33:36 +02:00
2 , /* chunks */
& src_h , 8192 , /* src file */
& dest_h , 0 , /* dest file */
& cc_copy ,
& ioctl ) ;
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " setup copy chunk error " ) ;
2011-09-30 14:33:36 +02:00
}
/* copy all src file data via two chunks */
cc_copy . chunks [ 0 ] . source_off = 0 ;
cc_copy . chunks [ 0 ] . target_off = 0 ;
cc_copy . chunks [ 0 ] . length = 4096 ;
cc_copy . chunks [ 1 ] . source_off = 4096 ;
cc_copy . chunks [ 1 ] . target_off = 4096 ;
cc_copy . chunks [ 1 ] . length = 4096 ;
ndr_ret = ndr_push_struct_blob ( & ioctl . smb2 . in . out , tmp_ctx ,
& cc_copy ,
( ndr_push_flags_fn_t ) ndr_push_srv_copychunk_copy ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ndr_success ( torture , ndr_ret ,
" ndr_push_srv_copychunk_copy " ) ;
2011-09-30 14:33:36 +02:00
status = smb2_ioctl ( tree , tmp_ctx , & ioctl . smb2 ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ntstatus_ok ( torture , status , " FSCTL_SRV_COPYCHUNK " ) ;
2011-09-30 14:33:36 +02:00
ndr_ret = ndr_pull_struct_blob ( & ioctl . smb2 . out . out , tmp_ctx ,
& cc_rsp ,
( ndr_pull_flags_fn_t ) ndr_pull_srv_copychunk_rsp ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ndr_success ( torture , ndr_ret ,
" ndr_pull_srv_copychunk_rsp " ) ;
2011-09-30 14:33:36 +02:00
2012-02-29 11:02:20 +01:00
ok = check_copy_chunk_rsp ( torture , & cc_rsp ,
2011-09-30 14:33:36 +02:00
2 , /* chunks written */
0 , /* chunk bytes unsuccessfully written */
8192 ) ; /* total bytes written */
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " bad copy chunk response data " ) ;
2011-09-30 14:33:36 +02:00
}
smb2_util_close ( tree , src_h ) ;
smb2_util_close ( tree , dest_h ) ;
talloc_free ( tmp_ctx ) ;
return true ;
}
static bool test_ioctl_copy_chunk_tiny ( struct torture_context * torture ,
struct smb2_tree * tree )
{
struct smb2_handle src_h ;
struct smb2_handle dest_h ;
NTSTATUS status ;
union smb_ioctl ioctl ;
TALLOC_CTX * tmp_ctx = talloc_new ( tree ) ;
struct srv_copychunk_copy cc_copy ;
struct srv_copychunk_rsp cc_rsp ;
enum ndr_err_code ndr_ret ;
bool ok ;
2012-02-29 11:02:20 +01:00
ok = test_setup_copy_chunk ( torture , tree , tmp_ctx ,
2011-09-30 14:33:36 +02:00
2 , /* chunks */
& src_h , 100 , /* src file */
& dest_h , 0 , /* dest file */
& cc_copy ,
& ioctl ) ;
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " setup copy chunk error " ) ;
2011-09-30 14:33:36 +02:00
}
/* copy all src file data via two chunks, sub block size chunks */
cc_copy . chunks [ 0 ] . source_off = 0 ;
cc_copy . chunks [ 0 ] . target_off = 0 ;
cc_copy . chunks [ 0 ] . length = 50 ;
cc_copy . chunks [ 1 ] . source_off = 50 ;
cc_copy . chunks [ 1 ] . target_off = 50 ;
cc_copy . chunks [ 1 ] . length = 50 ;
ndr_ret = ndr_push_struct_blob ( & ioctl . smb2 . in . out , tmp_ctx ,
& cc_copy ,
( ndr_push_flags_fn_t ) ndr_push_srv_copychunk_copy ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ndr_success ( torture , ndr_ret ,
" ndr_push_srv_copychunk_copy " ) ;
2011-09-30 14:33:36 +02:00
status = smb2_ioctl ( tree , tmp_ctx , & ioctl . smb2 ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ntstatus_ok ( torture , status , " FSCTL_SRV_COPYCHUNK " ) ;
2011-09-30 14:33:36 +02:00
ndr_ret = ndr_pull_struct_blob ( & ioctl . smb2 . out . out , tmp_ctx ,
& cc_rsp ,
( ndr_pull_flags_fn_t ) ndr_pull_srv_copychunk_rsp ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ndr_success ( torture , ndr_ret ,
" ndr_pull_srv_copychunk_rsp " ) ;
2011-09-30 14:33:36 +02:00
2012-02-29 11:02:20 +01:00
ok = check_copy_chunk_rsp ( torture , & cc_rsp ,
2011-09-30 14:33:36 +02:00
2 , /* chunks written */
0 , /* chunk bytes unsuccessfully written */
2011-09-30 12:50:36 +02:00
100 ) ; /* total bytes written */
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " bad copy chunk response data " ) ;
2011-09-27 16:40:20 +02:00
}
2012-02-29 11:02:20 +01:00
ok = check_pattern ( torture , tree , tmp_ctx , dest_h , 0 , 100 , 0 ) ;
2011-10-11 11:44:03 +02:00
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " inconsistent file data " ) ;
2011-10-11 11:44:03 +02:00
}
2011-09-30 14:33:36 +02:00
smb2_util_close ( tree , src_h ) ;
smb2_util_close ( tree , dest_h ) ;
talloc_free ( tmp_ctx ) ;
return true ;
}
static bool test_ioctl_copy_chunk_over ( struct torture_context * torture ,
struct smb2_tree * tree )
{
struct smb2_handle src_h ;
struct smb2_handle dest_h ;
NTSTATUS status ;
union smb_ioctl ioctl ;
TALLOC_CTX * tmp_ctx = talloc_new ( tree ) ;
struct srv_copychunk_copy cc_copy ;
struct srv_copychunk_rsp cc_rsp ;
enum ndr_err_code ndr_ret ;
bool ok ;
2012-02-29 11:02:20 +01:00
ok = test_setup_copy_chunk ( torture , tree , tmp_ctx ,
2011-09-30 14:33:36 +02:00
2 , /* chunks */
& src_h , 8192 , /* src file */
& dest_h , 4096 , /* dest file */
& cc_copy ,
& ioctl ) ;
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " setup copy chunk error " ) ;
2011-09-30 14:33:36 +02:00
}
/* first chunk overwrites existing dest data */
cc_copy . chunks [ 0 ] . source_off = 0 ;
cc_copy . chunks [ 0 ] . target_off = 0 ;
cc_copy . chunks [ 0 ] . length = 4096 ;
/* second chunk overwrites the first */
cc_copy . chunks [ 1 ] . source_off = 4096 ;
cc_copy . chunks [ 1 ] . target_off = 0 ;
cc_copy . chunks [ 1 ] . length = 4096 ;
ndr_ret = ndr_push_struct_blob ( & ioctl . smb2 . in . out , tmp_ctx ,
& cc_copy ,
( ndr_push_flags_fn_t ) ndr_push_srv_copychunk_copy ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ndr_success ( torture , ndr_ret ,
" ndr_push_srv_copychunk_copy " ) ;
2011-09-30 14:33:36 +02:00
status = smb2_ioctl ( tree , tmp_ctx , & ioctl . smb2 ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ntstatus_ok ( torture , status , " FSCTL_SRV_COPYCHUNK " ) ;
2011-09-30 14:33:36 +02:00
ndr_ret = ndr_pull_struct_blob ( & ioctl . smb2 . out . out , tmp_ctx ,
& cc_rsp ,
( ndr_pull_flags_fn_t ) ndr_pull_srv_copychunk_rsp ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ndr_success ( torture , ndr_ret ,
" ndr_pull_srv_copychunk_rsp " ) ;
2011-09-30 14:33:36 +02:00
2012-02-29 11:02:20 +01:00
ok = check_copy_chunk_rsp ( torture , & cc_rsp ,
2011-09-30 14:33:36 +02:00
2 , /* chunks written */
0 , /* chunk bytes unsuccessfully written */
8192 ) ; /* total bytes written */
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " bad copy chunk response data " ) ;
2011-09-30 14:33:36 +02:00
}
2012-02-29 11:02:20 +01:00
ok = check_pattern ( torture , tree , tmp_ctx , dest_h , 0 , 4096 , 4096 ) ;
2011-10-11 11:44:03 +02:00
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " inconsistent file data " ) ;
2011-10-11 11:44:03 +02:00
}
2011-09-30 14:33:36 +02:00
smb2_util_close ( tree , src_h ) ;
smb2_util_close ( tree , dest_h ) ;
talloc_free ( tmp_ctx ) ;
return true ;
}
static bool test_ioctl_copy_chunk_append ( struct torture_context * torture ,
struct smb2_tree * tree )
{
struct smb2_handle src_h ;
struct smb2_handle dest_h ;
NTSTATUS status ;
union smb_ioctl ioctl ;
TALLOC_CTX * tmp_ctx = talloc_new ( tree ) ;
struct srv_copychunk_copy cc_copy ;
struct srv_copychunk_rsp cc_rsp ;
enum ndr_err_code ndr_ret ;
bool ok ;
2012-02-29 11:02:20 +01:00
ok = test_setup_copy_chunk ( torture , tree , tmp_ctx ,
2011-09-30 14:33:36 +02:00
2 , /* chunks */
& src_h , 4096 , /* src file */
& dest_h , 0 , /* dest file */
& cc_copy ,
& ioctl ) ;
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " setup copy chunk error " ) ;
2011-09-30 14:33:36 +02:00
}
cc_copy . chunks [ 0 ] . source_off = 0 ;
cc_copy . chunks [ 0 ] . target_off = 0 ;
cc_copy . chunks [ 0 ] . length = 4096 ;
/* second chunk appends the same data to the first */
cc_copy . chunks [ 1 ] . source_off = 0 ;
cc_copy . chunks [ 1 ] . target_off = 4096 ;
cc_copy . chunks [ 1 ] . length = 4096 ;
ndr_ret = ndr_push_struct_blob ( & ioctl . smb2 . in . out , tmp_ctx ,
& cc_copy ,
( ndr_push_flags_fn_t ) ndr_push_srv_copychunk_copy ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ndr_success ( torture , ndr_ret ,
" ndr_push_srv_copychunk_copy " ) ;
2011-09-30 14:33:36 +02:00
status = smb2_ioctl ( tree , tmp_ctx , & ioctl . smb2 ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ntstatus_ok ( torture , status , " FSCTL_SRV_COPYCHUNK " ) ;
2011-09-30 14:33:36 +02:00
ndr_ret = ndr_pull_struct_blob ( & ioctl . smb2 . out . out , tmp_ctx ,
& cc_rsp ,
( ndr_pull_flags_fn_t ) ndr_pull_srv_copychunk_rsp ) ;
2012-02-29 11:02:20 +01:00
torture_assert_ndr_success ( torture , ndr_ret ,
" ndr_pull_srv_copychunk_rsp " ) ;
2011-09-30 14:33:36 +02:00
2012-02-29 11:02:20 +01:00
ok = check_copy_chunk_rsp ( torture , & cc_rsp ,
2011-09-30 14:33:36 +02:00
2 , /* chunks written */
0 , /* chunk bytes unsuccessfully written */
8192 ) ; /* total bytes written */
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " bad copy chunk response data " ) ;
2011-09-30 14:33:36 +02:00
}
2012-02-29 11:02:20 +01:00
ok = check_pattern ( torture , tree , tmp_ctx , dest_h , 0 , 4096 , 0 ) ;
2011-10-11 11:44:03 +02:00
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " inconsistent file data " ) ;
2011-10-11 11:44:03 +02:00
}
2012-02-29 11:02:20 +01:00
ok = check_pattern ( torture , tree , tmp_ctx , dest_h , 4096 , 4096 , 0 ) ;
2011-10-11 11:44:03 +02:00
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " inconsistent file data " ) ;
2011-10-11 11:44:03 +02:00
}
2011-09-30 14:33:36 +02:00
smb2_util_close ( tree , src_h ) ;
smb2_util_close ( tree , dest_h ) ;
2011-09-27 16:40:20 +02:00
talloc_free ( tmp_ctx ) ;
return true ;
}
2013-01-15 17:23:01 +01:00
static bool test_ioctl_copy_chunk_limits ( struct torture_context * torture ,
struct smb2_tree * tree )
{
struct smb2_handle src_h ;
struct smb2_handle dest_h ;
NTSTATUS status ;
union smb_ioctl ioctl ;
TALLOC_CTX * tmp_ctx = talloc_new ( tree ) ;
struct srv_copychunk_copy cc_copy ;
struct srv_copychunk_rsp cc_rsp ;
enum ndr_err_code ndr_ret ;
bool ok ;
ok = test_setup_copy_chunk ( torture , tree , tmp_ctx ,
1 , /* chunks */
& src_h , 4096 , /* src file */
& dest_h , 0 , /* dest file */
& cc_copy ,
& ioctl ) ;
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " setup copy chunk error " ) ;
2013-01-15 17:23:01 +01:00
}
/* send huge chunk length request */
cc_copy . chunks [ 0 ] . source_off = 0 ;
cc_copy . chunks [ 0 ] . target_off = 0 ;
cc_copy . chunks [ 0 ] . length = UINT_MAX ;
ndr_ret = ndr_push_struct_blob ( & ioctl . smb2 . in . out , tmp_ctx ,
& cc_copy ,
( ndr_push_flags_fn_t ) ndr_push_srv_copychunk_copy ) ;
torture_assert_ndr_success ( torture , ndr_ret , " marshalling request " ) ;
status = smb2_ioctl ( tree , tmp_ctx , & ioctl . smb2 ) ;
torture_assert_ntstatus_equal ( torture , status ,
NT_STATUS_INVALID_PARAMETER ,
" bad oversize chunk response " ) ;
ndr_ret = ndr_pull_struct_blob ( & ioctl . smb2 . out . out , tmp_ctx ,
& cc_rsp ,
( ndr_pull_flags_fn_t ) ndr_pull_srv_copychunk_rsp ) ;
torture_assert_ndr_success ( torture , ndr_ret , " unmarshalling response " ) ;
torture_comment ( torture , " limit max chunks, got %u \n " ,
cc_rsp . chunks_written ) ;
torture_comment ( torture , " limit max chunk len, got %u \n " ,
cc_rsp . chunk_bytes_written ) ;
torture_comment ( torture , " limit max total bytes, got %u \n " ,
cc_rsp . total_bytes_written ) ;
smb2_util_close ( tree , src_h ) ;
smb2_util_close ( tree , dest_h ) ;
talloc_free ( tmp_ctx ) ;
return true ;
}
2013-01-15 17:23:06 +01:00
static bool test_ioctl_copy_chunk_src_lck ( struct torture_context * torture ,
struct smb2_tree * tree )
{
struct smb2_handle src_h ;
struct smb2_handle src_h2 ;
struct smb2_handle dest_h ;
NTSTATUS status ;
union smb_ioctl ioctl ;
TALLOC_CTX * tmp_ctx = talloc_new ( tree ) ;
struct srv_copychunk_copy cc_copy ;
struct srv_copychunk_rsp cc_rsp ;
enum ndr_err_code ndr_ret ;
bool ok ;
struct smb2_lock lck ;
struct smb2_lock_element el [ 1 ] ;
ok = test_setup_copy_chunk ( torture , tree , tmp_ctx ,
1 , /* chunks */
& src_h , 4096 , /* src file */
& dest_h , 0 , /* dest file */
& cc_copy ,
& ioctl ) ;
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " setup copy chunk error " ) ;
2013-01-15 17:23:06 +01:00
}
cc_copy . chunks [ 0 ] . source_off = 0 ;
cc_copy . chunks [ 0 ] . target_off = 0 ;
cc_copy . chunks [ 0 ] . length = 4096 ;
/* open and lock the copychunk src file */
status = torture_smb2_testfile ( tree , FNAME , & src_h2 ) ;
torture_assert_ntstatus_ok ( torture , status , " 2nd src open " ) ;
lck . in . lock_count = 0x0001 ;
lck . in . lock_sequence = 0x00000000 ;
lck . in . file . handle = src_h2 ;
lck . in . locks = el ;
el [ 0 ] . offset = cc_copy . chunks [ 0 ] . source_off ;
el [ 0 ] . length = cc_copy . chunks [ 0 ] . length ;
el [ 0 ] . reserved = 0 ;
el [ 0 ] . flags = SMB2_LOCK_FLAG_EXCLUSIVE ;
status = smb2_lock ( tree , & lck ) ;
torture_assert_ntstatus_ok ( torture , status , " lock " ) ;
ndr_ret = ndr_push_struct_blob ( & ioctl . smb2 . in . out , tmp_ctx ,
& cc_copy ,
( ndr_push_flags_fn_t ) ndr_push_srv_copychunk_copy ) ;
torture_assert_ndr_success ( torture , ndr_ret ,
" ndr_push_srv_copychunk_copy " ) ;
status = smb2_ioctl ( tree , tmp_ctx , & ioctl . smb2 ) ;
/*
* 2 k12 & Samba return lock_conflict , Windows 7 & 2 k8 return success . . .
*
* Edgar Olougouna @ MS wrote :
* Regarding the FSCTL_SRV_COPYCHUNK and STATUS_FILE_LOCK_CONFLICT
* discrepancy observed between Windows versions , we confirm that the
* behavior change is expected .
*
* CopyChunk in Windows Server 2012 use regular Readfile / Writefile APIs
* to move the chunks from the source to the destination .
* These ReadFile / WriteFile APIs go through the byte - range lock checks ,
* and this explains the observed STATUS_FILE_LOCK_CONFLICT error .
*
* Prior to Windows Server 2012 , CopyChunk used mapped sections to move
* the data . And byte range locks are not enforced on mapped I / O , and
* this explains the STATUS_SUCCESS observed on Windows Server 2008 R2 .
*/
torture_assert_ntstatus_equal ( torture , status ,
NT_STATUS_FILE_LOCK_CONFLICT ,
" FSCTL_SRV_COPYCHUNK locked " ) ;
/* should get cc response data with the lock conflict status */
ndr_ret = ndr_pull_struct_blob ( & ioctl . smb2 . out . out , tmp_ctx ,
& cc_rsp ,
( ndr_pull_flags_fn_t ) ndr_pull_srv_copychunk_rsp ) ;
torture_assert_ndr_success ( torture , ndr_ret ,
" ndr_pull_srv_copychunk_rsp " ) ;
ok = check_copy_chunk_rsp ( torture , & cc_rsp ,
0 , /* chunks written */
0 , /* chunk bytes unsuccessfully written */
0 ) ; /* total bytes written */
lck . in . lock_count = 0x0001 ;
lck . in . lock_sequence = 0x00000001 ;
lck . in . file . handle = src_h2 ;
lck . in . locks = el ;
el [ 0 ] . offset = cc_copy . chunks [ 0 ] . source_off ;
el [ 0 ] . length = cc_copy . chunks [ 0 ] . length ;
el [ 0 ] . reserved = 0 ;
el [ 0 ] . flags = SMB2_LOCK_FLAG_UNLOCK ;
status = smb2_lock ( tree , & lck ) ;
torture_assert_ntstatus_ok ( torture , status , " unlock " ) ;
status = smb2_ioctl ( tree , tmp_ctx , & ioctl . smb2 ) ;
torture_assert_ntstatus_ok ( torture , status ,
" FSCTL_SRV_COPYCHUNK unlocked " ) ;
ndr_ret = ndr_pull_struct_blob ( & ioctl . smb2 . out . out , tmp_ctx ,
& cc_rsp ,
( ndr_pull_flags_fn_t ) ndr_pull_srv_copychunk_rsp ) ;
torture_assert_ndr_success ( torture , ndr_ret ,
" ndr_pull_srv_copychunk_rsp " ) ;
ok = check_copy_chunk_rsp ( torture , & cc_rsp ,
1 , /* chunks written */
0 , /* chunk bytes unsuccessfully written */
4096 ) ; /* total bytes written */
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " bad copy chunk response data " ) ;
2013-01-15 17:23:06 +01:00
}
ok = check_pattern ( torture , tree , tmp_ctx , dest_h , 0 , 4096 , 0 ) ;
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " inconsistent file data " ) ;
2013-01-15 17:23:06 +01:00
}
smb2_util_close ( tree , src_h2 ) ;
smb2_util_close ( tree , src_h ) ;
smb2_util_close ( tree , dest_h ) ;
talloc_free ( tmp_ctx ) ;
return true ;
}
static bool test_ioctl_copy_chunk_dest_lck ( struct torture_context * torture ,
struct smb2_tree * tree )
{
struct smb2_handle src_h ;
struct smb2_handle dest_h ;
struct smb2_handle dest_h2 ;
NTSTATUS status ;
union smb_ioctl ioctl ;
TALLOC_CTX * tmp_ctx = talloc_new ( tree ) ;
struct srv_copychunk_copy cc_copy ;
struct srv_copychunk_rsp cc_rsp ;
enum ndr_err_code ndr_ret ;
bool ok ;
struct smb2_lock lck ;
struct smb2_lock_element el [ 1 ] ;
ok = test_setup_copy_chunk ( torture , tree , tmp_ctx ,
1 , /* chunks */
& src_h , 4096 , /* src file */
& dest_h , 4096 , /* dest file */
& cc_copy ,
& ioctl ) ;
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " setup copy chunk error " ) ;
2013-01-15 17:23:06 +01:00
}
cc_copy . chunks [ 0 ] . source_off = 0 ;
cc_copy . chunks [ 0 ] . target_off = 0 ;
cc_copy . chunks [ 0 ] . length = 4096 ;
/* open and lock the copychunk dest file */
status = torture_smb2_testfile ( tree , FNAME2 , & dest_h2 ) ;
torture_assert_ntstatus_ok ( torture , status , " 2nd src open " ) ;
lck . in . lock_count = 0x0001 ;
lck . in . lock_sequence = 0x00000000 ;
lck . in . file . handle = dest_h2 ;
lck . in . locks = el ;
el [ 0 ] . offset = cc_copy . chunks [ 0 ] . target_off ;
el [ 0 ] . length = cc_copy . chunks [ 0 ] . length ;
el [ 0 ] . reserved = 0 ;
el [ 0 ] . flags = SMB2_LOCK_FLAG_EXCLUSIVE ;
status = smb2_lock ( tree , & lck ) ;
torture_assert_ntstatus_ok ( torture , status , " lock " ) ;
ndr_ret = ndr_push_struct_blob ( & ioctl . smb2 . in . out , tmp_ctx ,
& cc_copy ,
( ndr_push_flags_fn_t ) ndr_push_srv_copychunk_copy ) ;
torture_assert_ndr_success ( torture , ndr_ret ,
" ndr_push_srv_copychunk_copy " ) ;
status = smb2_ioctl ( tree , tmp_ctx , & ioctl . smb2 ) ;
torture_assert_ntstatus_equal ( torture , status ,
NT_STATUS_FILE_LOCK_CONFLICT ,
" FSCTL_SRV_COPYCHUNK locked " ) ;
lck . in . lock_count = 0x0001 ;
lck . in . lock_sequence = 0x00000001 ;
lck . in . file . handle = dest_h2 ;
lck . in . locks = el ;
el [ 0 ] . offset = cc_copy . chunks [ 0 ] . target_off ;
el [ 0 ] . length = cc_copy . chunks [ 0 ] . length ;
el [ 0 ] . reserved = 0 ;
el [ 0 ] . flags = SMB2_LOCK_FLAG_UNLOCK ;
status = smb2_lock ( tree , & lck ) ;
torture_assert_ntstatus_ok ( torture , status , " unlock " ) ;
status = smb2_ioctl ( tree , tmp_ctx , & ioctl . smb2 ) ;
torture_assert_ntstatus_ok ( torture , status ,
" FSCTL_SRV_COPYCHUNK unlocked " ) ;
ndr_ret = ndr_pull_struct_blob ( & ioctl . smb2 . out . out , tmp_ctx ,
& cc_rsp ,
( ndr_pull_flags_fn_t ) ndr_pull_srv_copychunk_rsp ) ;
torture_assert_ndr_success ( torture , ndr_ret ,
" ndr_pull_srv_copychunk_rsp " ) ;
ok = check_copy_chunk_rsp ( torture , & cc_rsp ,
1 , /* chunks written */
0 , /* chunk bytes unsuccessfully written */
4096 ) ; /* total bytes written */
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " bad copy chunk response data " ) ;
2013-01-15 17:23:06 +01:00
}
ok = check_pattern ( torture , tree , tmp_ctx , dest_h , 0 , 4096 , 0 ) ;
if ( ! ok ) {
2013-01-15 17:23:07 +01:00
torture_fail ( torture , " inconsistent file data " ) ;
2013-01-15 17:23:06 +01:00
}
smb2_util_close ( tree , dest_h2 ) ;
smb2_util_close ( tree , src_h ) ;
smb2_util_close ( tree , dest_h ) ;
talloc_free ( tmp_ctx ) ;
return true ;
}
2011-09-22 22:23:08 +02:00
/*
basic testing of SMB2 ioctls
*/
struct torture_suite * torture_smb2_ioctl_init ( void )
{
struct torture_suite * suite = torture_suite_create ( talloc_autofree_context ( ) , " ioctl " ) ;
2011-09-30 14:33:36 +02:00
torture_suite_add_1smb2_test ( suite , " shadow_copy " ,
test_ioctl_get_shadow_copy ) ;
torture_suite_add_1smb2_test ( suite , " req_resume_key " ,
test_ioctl_req_resume_key ) ;
torture_suite_add_1smb2_test ( suite , " copy_chunk_simple " ,
test_ioctl_copy_chunk_simple ) ;
torture_suite_add_1smb2_test ( suite , " copy_chunk_multi " ,
test_ioctl_copy_chunk_multi ) ;
torture_suite_add_1smb2_test ( suite , " copy_chunk_tiny " ,
test_ioctl_copy_chunk_tiny ) ;
torture_suite_add_1smb2_test ( suite , " copy_chunk_overwrite " ,
test_ioctl_copy_chunk_over ) ;
torture_suite_add_1smb2_test ( suite , " copy_chunk_append " ,
test_ioctl_copy_chunk_append ) ;
2013-01-15 17:23:01 +01:00
torture_suite_add_1smb2_test ( suite , " copy_chunk_limits " ,
test_ioctl_copy_chunk_limits ) ;
2013-01-15 17:23:06 +01:00
torture_suite_add_1smb2_test ( suite , " copy_chunk_src_lock " ,
test_ioctl_copy_chunk_src_lck ) ;
torture_suite_add_1smb2_test ( suite , " copy_chunk_dest_lock " ,
test_ioctl_copy_chunk_dest_lck ) ;
2011-09-22 22:23:08 +02:00
suite - > description = talloc_strdup ( suite , " SMB2-IOCTL tests " ) ;
return suite ;
}