2005-11-18 06:31:33 +00:00
/*
Unix SMB / CIFS implementation .
SMB2 setinfo individual test suite
Copyright ( C ) Andrew Tridgell 2005
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
2005-11-18 06:31:33 +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/>.
2005-11-18 06:31:33 +00:00
*/
# include "includes.h"
# include "system/time.h"
2005-12-28 15:38:36 +00:00
# include "libcli/smb2/smb2.h"
2006-01-03 17:27:33 +00:00
# include "libcli/smb2/smb2_calls.h"
2005-11-18 06:31:33 +00:00
2006-03-25 16:01:28 +00:00
# include "torture/torture.h"
2006-02-04 14:08:24 +00:00
# include "torture/smb2/proto.h"
2006-06-29 07:03:09 +00:00
# include "libcli/security/security.h"
# include "librpc/gen_ndr/ndr_security.h"
2005-11-18 06:31:33 +00:00
# define BASEDIR ""
2009-07-04 16:16:23 -07:00
# define FAIL_UNLESS(__cond) \
do { \
if ( __cond ) { } else { \
torture_result ( tctx , TORTURE_FAIL , " %s) condition violated: %s \n " , \
__location__ , # __cond ) ; \
ret = false ; goto done ; \
} \
} while ( 0 )
2005-11-18 06:31:33 +00:00
/* basic testing of all SMB2 setinfo calls
for each call we test that it succeeds , and where possible test
for consistency between the calls .
*/
2009-07-04 16:16:23 -07:00
bool torture_smb2_setinfo ( struct torture_context * tctx )
2005-11-18 06:31:33 +00:00
{
struct smb2_tree * tree ;
2007-10-06 22:28:14 +00:00
bool ret = true ;
2005-11-18 06:31:33 +00:00
struct smb2_handle handle ;
char * fname ;
2005-11-19 05:55:08 +00:00
union smb_fileinfo finfo2 ;
2005-11-18 06:31:33 +00:00
union smb_setfileinfo sfinfo ;
2006-06-29 07:03:09 +00:00
struct security_ace ace ;
struct security_descriptor * sd ;
struct dom_sid * test_sid ;
2006-09-10 10:43:31 +00:00
NTSTATUS status , status2 = NT_STATUS_OK ;
2005-11-18 06:31:33 +00:00
const char * call_name ;
time_t basetime = ( time ( NULL ) - 86400 ) & ~ 1 ;
int n = time ( NULL ) % 100 ;
2005-12-02 03:18:34 +00:00
ZERO_STRUCT ( handle ) ;
2009-07-04 16:16:23 -07:00
fname = talloc_asprintf ( tctx , BASEDIR " fnum_test_%d.txt " , n ) ;
2005-11-18 06:31:33 +00:00
2009-07-04 16:16:23 -07:00
if ( ! torture_smb2_connection ( tctx , & tree ) ) {
2007-10-06 22:28:14 +00:00
return false ;
2005-11-18 06:31:33 +00:00
}
# define RECREATE_FILE(fname) do { \
smb2_util_close ( tree , handle ) ; \
status = smb2_create_complex_file ( tree , fname , & handle ) ; \
if ( ! NT_STATUS_IS_OK ( status ) ) { \
2012-02-28 12:53:29 +01:00
torture_result ( tctx , TORTURE_FAIL , " (%s) ERROR: open of %s failed (%s) \n " , \
2005-11-18 06:31:33 +00:00
__location__ , fname , nt_errstr ( status ) ) ; \
2007-10-06 22:28:14 +00:00
ret = false ; \
2005-11-18 06:31:33 +00:00
goto done ; \
} } while ( 0 )
# define RECREATE_BOTH do { \
RECREATE_FILE ( fname ) ; \
} while ( 0 )
RECREATE_BOTH ;
# define CHECK_CALL(call, rightstatus) do { \
call_name = # call ; \
sfinfo . generic . level = RAW_SFILEINFO_ # # call ; \
2006-03-12 22:48:25 +00:00
sfinfo . generic . in . file . handle = handle ; \
2005-11-18 06:31:33 +00:00
status = smb2_setinfo_file ( tree , & sfinfo ) ; \
if ( ! NT_STATUS_EQUAL ( status , rightstatus ) ) { \
2009-07-04 16:16:23 -07:00
torture_result ( tctx , TORTURE_FAIL , " (%s) %s - %s (should be %s) \n " , __location__ , # call , \
2005-11-18 06:31:33 +00:00
nt_errstr ( status ) , nt_errstr ( rightstatus ) ) ; \
2007-10-06 22:28:14 +00:00
ret = false ; \
2006-07-01 14:21:21 +00:00
goto done ; \
2005-11-18 06:31:33 +00:00
} \
2005-11-19 05:55:08 +00:00
} while ( 0 )
2005-11-18 06:31:33 +00:00
# define CHECK1(call) \
do { if ( NT_STATUS_IS_OK ( status ) ) { \
finfo2 . generic . level = RAW_FILEINFO_ # # call ; \
2006-03-12 22:48:25 +00:00
finfo2 . generic . in . file . handle = handle ; \
2009-07-04 16:16:23 -07:00
status2 = smb2_getinfo_file ( tree , tctx , & finfo2 ) ; \
2005-11-18 06:31:33 +00:00
if ( ! NT_STATUS_IS_OK ( status2 ) ) { \
2009-07-04 16:16:23 -07:00
torture_result ( tctx , TORTURE_FAIL , " (%s) %s - %s \n " , __location__ , # call , nt_errstr ( status2 ) ) ; \
2007-10-06 22:28:14 +00:00
ret = false ; \
2006-07-01 14:21:21 +00:00
goto done ; \
2005-11-18 06:31:33 +00:00
} \
} } while ( 0 )
# define CHECK_VALUE(call, stype, field, value) do { \
CHECK1 ( call ) ; \
if ( NT_STATUS_IS_OK ( status ) & & NT_STATUS_IS_OK ( status2 ) & & finfo2 . stype . out . field ! = value ) { \
2009-07-04 16:16:23 -07:00
torture_result ( tctx , TORTURE_FAIL , " (%s) %s - %s/%s should be 0x%x - 0x%x \n " , __location__ , \
2005-11-18 06:31:33 +00:00
call_name , # stype , # field , \
2010-01-05 09:42:54 -08:00
( unsigned int ) value , ( unsigned int ) finfo2 . stype . out . field ) ; \
2005-11-19 05:55:08 +00:00
torture_smb2_all_info ( tree , handle ) ; \
2007-10-06 22:28:14 +00:00
ret = false ; \
2006-07-01 14:21:21 +00:00
goto done ; \
2005-11-18 06:31:33 +00:00
} } while ( 0 )
# define CHECK_TIME(call, stype, field, value) do { \
CHECK1 ( call ) ; \
if ( NT_STATUS_IS_OK ( status ) & & NT_STATUS_IS_OK ( status2 ) & & nt_time_to_unix ( finfo2 . stype . out . field ) ! = value ) { \
2009-07-04 16:16:23 -07:00
torture_result ( tctx , TORTURE_FAIL , " (%s) %s - %s/%s should be 0x%x - 0x%x \n " , __location__ , \
2005-11-18 06:31:33 +00:00
call_name , # stype , # field , \
2010-01-05 09:42:54 -08:00
( unsigned int ) value , \
( unsigned int ) nt_time_to_unix ( finfo2 . stype . out . field ) ) ; \
2009-07-04 16:16:23 -07:00
torture_warning ( tctx , " \t %s " , timestring ( tctx , value ) ) ; \
torture_warning ( tctx , " \t %s \n " , nt_time_string ( tctx , finfo2 . stype . out . field ) ) ; \
2005-11-19 05:55:08 +00:00
torture_smb2_all_info ( tree , handle ) ; \
2007-10-06 22:28:14 +00:00
ret = false ; \
2006-07-01 14:21:21 +00:00
goto done ; \
2005-11-18 06:31:33 +00:00
} } while ( 0 )
# define CHECK_STATUS(status, correct) do { \
if ( ! NT_STATUS_EQUAL ( status , correct ) ) { \
2009-07-04 16:16:23 -07:00
torture_result ( tctx , TORTURE_FAIL , " (%s) Incorrect status %s - should be %s \n " , \
2005-11-18 06:31:33 +00:00
__location__ , nt_errstr ( status ) , nt_errstr ( correct ) ) ; \
2007-10-06 22:28:14 +00:00
ret = false ; \
2005-11-18 06:31:33 +00:00
goto done ; \
} } while ( 0 )
2005-11-18 09:25:25 +00:00
torture_smb2_all_info ( tree , handle ) ;
2005-11-18 06:31:33 +00:00
2010-04-11 20:56:48 +02:00
torture_comment ( tctx , " Test basic_information level \n " ) ;
2005-11-18 06:31:33 +00:00
basetime + = 86400 ;
unix_to_nt_time ( & sfinfo . basic_info . in . create_time , basetime + 100 ) ;
unix_to_nt_time ( & sfinfo . basic_info . in . access_time , basetime + 200 ) ;
unix_to_nt_time ( & sfinfo . basic_info . in . write_time , basetime + 300 ) ;
unix_to_nt_time ( & sfinfo . basic_info . in . change_time , basetime + 400 ) ;
sfinfo . basic_info . in . attrib = FILE_ATTRIBUTE_READONLY ;
CHECK_CALL ( BASIC_INFORMATION , NT_STATUS_OK ) ;
CHECK_TIME ( SMB2_ALL_INFORMATION , all_info2 , create_time , basetime + 100 ) ;
CHECK_TIME ( SMB2_ALL_INFORMATION , all_info2 , access_time , basetime + 200 ) ;
CHECK_TIME ( SMB2_ALL_INFORMATION , all_info2 , write_time , basetime + 300 ) ;
CHECK_TIME ( SMB2_ALL_INFORMATION , all_info2 , change_time , basetime + 400 ) ;
CHECK_VALUE ( SMB2_ALL_INFORMATION , all_info2 , attrib , FILE_ATTRIBUTE_READONLY ) ;
2009-07-04 16:16:23 -07:00
torture_comment ( tctx , " a zero time means don't change \n " ) ;
2005-11-18 06:31:33 +00:00
unix_to_nt_time ( & sfinfo . basic_info . in . create_time , 0 ) ;
unix_to_nt_time ( & sfinfo . basic_info . in . access_time , 0 ) ;
unix_to_nt_time ( & sfinfo . basic_info . in . write_time , 0 ) ;
unix_to_nt_time ( & sfinfo . basic_info . in . change_time , 0 ) ;
sfinfo . basic_info . in . attrib = FILE_ATTRIBUTE_NORMAL ;
CHECK_CALL ( BASIC_INFORMATION , NT_STATUS_OK ) ;
CHECK_TIME ( SMB2_ALL_INFORMATION , all_info2 , create_time , basetime + 100 ) ;
CHECK_TIME ( SMB2_ALL_INFORMATION , all_info2 , access_time , basetime + 200 ) ;
CHECK_TIME ( SMB2_ALL_INFORMATION , all_info2 , write_time , basetime + 300 ) ;
CHECK_TIME ( SMB2_ALL_INFORMATION , all_info2 , change_time , basetime + 400 ) ;
CHECK_VALUE ( SMB2_ALL_INFORMATION , all_info2 , attrib , FILE_ATTRIBUTE_NORMAL ) ;
2009-07-04 16:16:23 -07:00
torture_comment ( tctx , " change the attribute \n " ) ;
2005-11-19 05:55:08 +00:00
sfinfo . basic_info . in . attrib = FILE_ATTRIBUTE_HIDDEN ;
CHECK_CALL ( BASIC_INFORMATION , NT_STATUS_OK ) ;
CHECK_VALUE ( SMB2_ALL_INFORMATION , all_info2 , attrib , FILE_ATTRIBUTE_HIDDEN ) ;
2009-07-04 16:16:23 -07:00
torture_comment ( tctx , " zero attrib means don't change \n " ) ;
2005-11-19 05:55:08 +00:00
sfinfo . basic_info . in . attrib = 0 ;
CHECK_CALL ( BASIC_INFORMATION , NT_STATUS_OK ) ;
CHECK_VALUE ( SMB2_ALL_INFORMATION , all_info2 , attrib , FILE_ATTRIBUTE_HIDDEN ) ;
2009-07-04 16:16:23 -07:00
torture_comment ( tctx , " can't change a file to a directory \n " ) ;
2008-05-28 20:06:48 +10:00
sfinfo . basic_info . in . attrib = FILE_ATTRIBUTE_DIRECTORY ;
CHECK_CALL ( BASIC_INFORMATION , NT_STATUS_INVALID_PARAMETER ) ;
2009-07-04 16:16:23 -07:00
torture_comment ( tctx , " restore attribute \n " ) ;
2005-11-19 05:55:08 +00:00
sfinfo . basic_info . in . attrib = FILE_ATTRIBUTE_NORMAL ;
CHECK_CALL ( BASIC_INFORMATION , NT_STATUS_OK ) ;
CHECK_VALUE ( SMB2_ALL_INFORMATION , all_info2 , attrib , FILE_ATTRIBUTE_NORMAL ) ;
2010-04-11 20:56:48 +02:00
torture_comment ( tctx , " Test disposition_information level \n " ) ;
2005-11-18 06:31:33 +00:00
sfinfo . disposition_info . in . delete_on_close = 1 ;
CHECK_CALL ( DISPOSITION_INFORMATION , NT_STATUS_OK ) ;
CHECK_VALUE ( SMB2_ALL_INFORMATION , all_info2 , delete_pending , 1 ) ;
CHECK_VALUE ( SMB2_ALL_INFORMATION , all_info2 , nlink , 0 ) ;
sfinfo . disposition_info . in . delete_on_close = 0 ;
CHECK_CALL ( DISPOSITION_INFORMATION , NT_STATUS_OK ) ;
CHECK_VALUE ( SMB2_ALL_INFORMATION , all_info2 , delete_pending , 0 ) ;
CHECK_VALUE ( SMB2_ALL_INFORMATION , all_info2 , nlink , 1 ) ;
2010-04-11 20:56:48 +02:00
torture_comment ( tctx , " Test allocation_information level \n " ) ;
2005-11-18 06:31:33 +00:00
sfinfo . allocation_info . in . alloc_size = 0 ;
CHECK_CALL ( ALLOCATION_INFORMATION , NT_STATUS_OK ) ;
CHECK_VALUE ( SMB2_ALL_INFORMATION , all_info2 , size , 0 ) ;
CHECK_VALUE ( SMB2_ALL_INFORMATION , all_info2 , alloc_size , 0 ) ;
sfinfo . allocation_info . in . alloc_size = 4096 ;
CHECK_CALL ( ALLOCATION_INFORMATION , NT_STATUS_OK ) ;
CHECK_VALUE ( SMB2_ALL_INFORMATION , all_info2 , alloc_size , 4096 ) ;
CHECK_VALUE ( SMB2_ALL_INFORMATION , all_info2 , size , 0 ) ;
2010-04-11 20:56:48 +02:00
torture_comment ( tctx , " Test end_of_file_info level \n " ) ;
2005-11-18 06:31:33 +00:00
sfinfo . end_of_file_info . in . size = 37 ;
CHECK_CALL ( END_OF_FILE_INFORMATION , NT_STATUS_OK ) ;
CHECK_VALUE ( SMB2_ALL_INFORMATION , all_info2 , size , 37 ) ;
sfinfo . end_of_file_info . in . size = 7 ;
CHECK_CALL ( END_OF_FILE_INFORMATION , NT_STATUS_OK ) ;
CHECK_VALUE ( SMB2_ALL_INFORMATION , all_info2 , size , 7 ) ;
2010-04-11 20:56:48 +02:00
torture_comment ( tctx , " Test position_information level \n " ) ;
2005-11-18 06:31:33 +00:00
sfinfo . position_information . in . position = 123456 ;
CHECK_CALL ( POSITION_INFORMATION , NT_STATUS_OK ) ;
CHECK_VALUE ( POSITION_INFORMATION , position_information , position , 123456 ) ;
2005-11-19 05:55:08 +00:00
CHECK_VALUE ( SMB2_ALL_INFORMATION , all_info2 , position , 123456 ) ;
2005-11-18 06:31:33 +00:00
2010-04-11 20:56:48 +02:00
torture_comment ( tctx , " Test mode_information level \n " ) ;
2005-11-18 06:31:33 +00:00
sfinfo . mode_information . in . mode = 2 ;
CHECK_CALL ( MODE_INFORMATION , NT_STATUS_OK ) ;
CHECK_VALUE ( MODE_INFORMATION , mode_information , mode , 2 ) ;
2005-11-19 05:55:08 +00:00
CHECK_VALUE ( SMB2_ALL_INFORMATION , all_info2 , mode , 2 ) ;
2005-11-18 06:31:33 +00:00
sfinfo . mode_information . in . mode = 1 ;
CHECK_CALL ( MODE_INFORMATION , NT_STATUS_INVALID_PARAMETER ) ;
sfinfo . mode_information . in . mode = 0 ;
CHECK_CALL ( MODE_INFORMATION , NT_STATUS_OK ) ;
CHECK_VALUE ( MODE_INFORMATION , mode_information , mode , 0 ) ;
2010-04-11 20:56:48 +02:00
torture_comment ( tctx , " Test sec_desc level \n " ) ;
2006-06-29 07:03:09 +00:00
ZERO_STRUCT ( finfo2 ) ;
finfo2 . query_secdesc . in . secinfo_flags =
SECINFO_OWNER |
SECINFO_GROUP |
SECINFO_DACL ;
CHECK1 ( SEC_DESC ) ;
sd = finfo2 . query_secdesc . out . sd ;
2009-07-04 16:16:23 -07:00
test_sid = dom_sid_parse_talloc ( tctx , SID_NT_AUTHENTICATED_USERS ) ;
2006-06-29 14:00:57 +00:00
ZERO_STRUCT ( ace ) ;
2006-06-29 07:03:09 +00:00
ace . type = SEC_ACE_TYPE_ACCESS_ALLOWED ;
ace . flags = 0 ;
ace . access_mask = SEC_STD_ALL ;
ace . trustee = * test_sid ;
status = security_descriptor_dacl_add ( sd , & ace ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
2009-07-04 16:16:23 -07:00
torture_comment ( tctx , " add a new ACE to the DACL \n " ) ;
2006-06-29 07:03:09 +00:00
sfinfo . set_secdesc . in . secinfo_flags = finfo2 . query_secdesc . in . secinfo_flags ;
sfinfo . set_secdesc . in . sd = sd ;
CHECK_CALL ( SEC_DESC , NT_STATUS_OK ) ;
2009-07-04 16:16:23 -07:00
FAIL_UNLESS ( smb2_util_verify_sd ( tctx , tree , handle , sd ) ) ;
2006-06-29 07:03:09 +00:00
2009-07-04 16:16:23 -07:00
torture_comment ( tctx , " remove it again \n " ) ;
2006-06-29 07:03:09 +00:00
status = security_descriptor_dacl_del ( sd , test_sid ) ;
CHECK_STATUS ( status , NT_STATUS_OK ) ;
sfinfo . set_secdesc . in . secinfo_flags = finfo2 . query_secdesc . in . secinfo_flags ;
sfinfo . set_secdesc . in . sd = sd ;
CHECK_CALL ( SEC_DESC , NT_STATUS_OK ) ;
2009-07-04 16:16:23 -07:00
FAIL_UNLESS ( smb2_util_verify_sd ( tctx , tree , handle , sd ) ) ;
2006-06-29 07:03:09 +00:00
2005-11-18 06:31:33 +00:00
done :
status = smb2_util_close ( tree , handle ) ;
if ( NT_STATUS_IS_ERR ( status ) ) {
2009-07-04 16:16:23 -07:00
torture_warning ( tctx , " Failed to delete %s - %s \n " , fname , nt_errstr ( status ) ) ;
2005-11-18 06:31:33 +00:00
}
smb2_util_unlink ( tree , fname ) ;
return ret ;
}