2003-11-23 16:44:19 +03:00
/*
Unix SMB / CIFS implementation .
SMB torture tester
Copyright ( C ) Andrew Tridgell 2003
2007-02-27 21:00:15 +03:00
Copyright ( C ) Jelmer Vernooij 2006
2003-11-23 16:44:19 +03: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 06:07:03 +04:00
the Free Software Foundation ; either version 3 of the License , or
2003-11-23 16:44:19 +03:00
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
2007-07-10 06:07:03 +04:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2003-11-23 16:44:19 +03:00
*/
# include "includes.h"
2007-03-02 15:47:51 +03:00
# if (_SAMBA_BUILD_ >= 4)
2004-11-02 05:57:18 +03:00
# include "lib/cmdline/popt_common.h"
2005-03-13 06:16:07 +03:00
# include "system/filesys.h"
2006-05-13 23:14:12 +04:00
# include "system/locale.h"
2007-08-21 23:35:43 +04:00
# include "librpc/ndr/libndr.h"
# include "librpc/ndr/ndr_table.h"
2007-03-02 15:47:51 +03:00
# endif
2003-11-23 16:44:19 +03:00
2007-08-18 14:30:40 +04:00
static const struct ndr_interface_call * find_function (
2007-08-20 00:46:45 +04:00
const struct ndr_interface_table * p ,
2003-11-23 16:44:19 +03:00
const char * function )
{
int i ;
2004-06-14 12:07:26 +04:00
if ( isdigit ( function [ 0 ] ) ) {
i = strtol ( function , NULL , 0 ) ;
return & p - > calls [ i ] ;
}
2003-11-23 16:44:19 +03:00
for ( i = 0 ; i < p - > num_calls ; i + + ) {
if ( strcmp ( p - > calls [ i ] . name , function ) = = 0 ) {
break ;
}
}
if ( i = = p - > num_calls ) {
printf ( " Function '%s' not found \n " , function ) ;
exit ( 1 ) ;
}
return & p - > calls [ i ] ;
}
2007-03-02 15:47:51 +03:00
# if (_SAMBA_BUILD_ >= 4)
2003-11-23 16:44:19 +03:00
2007-09-07 17:31:15 +04:00
_NORETURN_ static void show_pipes ( void )
2003-11-23 16:44:19 +03:00
{
2007-08-20 00:46:45 +04:00
const struct ndr_interface_list * l ;
2003-11-23 16:44:19 +03:00
printf ( " \n You must specify a pipe \n " ) ;
printf ( " known pipes are: \n " ) ;
2007-08-21 23:35:43 +04:00
for ( l = ndr_table_list ( ) ; l ; l = l - > next ) {
2005-01-09 11:34:05 +03:00
if ( l - > table - > helpstring ) {
printf ( " \t %s - %s \n " , l - > table - > name , l - > table - > helpstring ) ;
2004-08-12 23:32:16 +04:00
} else {
2005-01-09 11:34:05 +03:00
printf ( " \t %s \n " , l - > table - > name ) ;
2004-08-12 23:32:16 +04:00
}
2003-11-23 16:44:19 +03:00
}
exit ( 1 ) ;
}
2007-03-02 15:47:51 +03:00
# endif
2007-09-07 17:31:15 +04:00
_NORETURN_ static void show_functions ( const struct ndr_interface_table * p )
2003-11-23 16:44:19 +03:00
{
int i ;
printf ( " \n You must specify a function \n " ) ;
printf ( " known functions on '%s' are: \n " , p - > name ) ;
for ( i = 0 ; i < p - > num_calls ; i + + ) {
printf ( " \t 0x%02x (%2d) %s \n " , i , i , p - > calls [ i ] . name ) ;
}
exit ( 1 ) ;
}
2005-03-13 06:16:07 +03:00
static char * stdin_load ( TALLOC_CTX * mem_ctx , size_t * size )
{
int num_read , total_len = 0 ;
char buf [ 255 ] ;
char * result = NULL ;
while ( ( num_read = read ( STDIN_FILENO , buf , 255 ) ) > 0 ) {
if ( result ) {
2007-09-08 17:34:42 +04:00
result = talloc_realloc (
mem_ctx , result , char , total_len + num_read ) ;
2005-03-13 06:16:07 +03:00
} else {
2007-09-08 17:34:42 +04:00
result = talloc_array ( mem_ctx , char , num_read ) ;
2005-03-13 06:16:07 +03:00
}
memcpy ( result + total_len , buf , num_read ) ;
total_len + = num_read ;
}
if ( size )
* size = total_len ;
return result ;
}
2007-09-08 17:34:42 +04:00
static const struct ndr_interface_table * load_iface_from_plugin ( const char * plugin , const char * pipe_name )
2003-11-23 16:44:19 +03:00
{
2007-08-20 00:46:45 +04:00
const struct ndr_interface_table * p ;
2006-04-08 17:43:57 +04:00
void * handle ;
char * symbol ;
handle = dlopen ( plugin , RTLD_NOW ) ;
if ( handle = = NULL ) {
printf ( " %s: Unable to open: %s \n " , plugin , dlerror ( ) ) ;
return NULL ;
}
2007-08-20 01:23:03 +04:00
symbol = talloc_asprintf ( NULL , " ndr_table_%s " , pipe_name ) ;
2007-09-08 17:34:42 +04:00
p = ( const struct ndr_interface_table * ) dlsym ( handle , symbol ) ;
2006-04-08 17:43:57 +04:00
if ( ! p ) {
printf ( " %s: Unable to find DCE/RPC interface table for '%s': %s \n " , plugin , pipe_name , dlerror ( ) ) ;
talloc_free ( symbol ) ;
return NULL ;
}
talloc_free ( symbol ) ;
return p ;
}
2007-11-02 13:36:06 +03:00
static void ndrdump_data ( uint8_t * d , uint32_t l , bool force )
{
if ( force ) {
dump_data ( 0 , d , l ) ;
} else {
dump_data_skip_zeros ( 0 , d , l ) ;
}
}
2006-04-08 17:43:57 +04:00
int main ( int argc , const char * argv [ ] )
{
2007-08-20 00:46:45 +04:00
const struct ndr_interface_table * p = NULL ;
2007-08-18 14:30:40 +04:00
const struct ndr_interface_call * f ;
2003-11-23 16:44:19 +03:00
const char * pipe_name , * function , * inout , * filename ;
2004-11-25 22:01:35 +03:00
uint8_t * data ;
2003-11-23 16:44:19 +03:00
size_t size ;
DATA_BLOB blob ;
2005-08-22 20:21:29 +04:00
struct ndr_pull * ndr_pull ;
struct ndr_print * ndr_print ;
2003-11-23 16:44:19 +03:00
TALLOC_CTX * mem_ctx ;
int flags ;
2004-10-16 23:00:27 +04:00
poptContext pc ;
2003-11-23 16:44:19 +03:00
NTSTATUS status ;
2007-11-09 21:24:25 +03:00
enum ndr_err_code ndr_err ;
2003-11-23 16:44:19 +03:00
void * st ;
2005-08-22 20:21:29 +04:00
void * v_st ;
2004-10-31 22:07:28 +03:00
const char * ctx_filename = NULL ;
2006-04-08 17:43:57 +04:00
const char * plugin = NULL ;
2007-02-14 16:22:09 +03:00
bool validate = false ;
bool dumpdata = false ;
2004-10-16 23:00:27 +04:00
int opt ;
2007-02-14 16:22:09 +03:00
enum { OPT_CONTEXT_FILE = 1000 , OPT_VALIDATE , OPT_DUMP_DATA , OPT_LOAD_DSO } ;
2004-10-16 23:00:27 +04:00
struct poptOption long_options [ ] = {
POPT_AUTOHELP
2007-02-14 16:22:09 +03:00
{ " context-file " , ' c ' , POPT_ARG_STRING , NULL , OPT_CONTEXT_FILE , " In-filename to parse first " , " CTX-FILE " } ,
{ " validate " , 0 , POPT_ARG_NONE , NULL , OPT_VALIDATE , " try to validate the data " , NULL } ,
{ " dump-data " , 0 , POPT_ARG_NONE , NULL , OPT_DUMP_DATA , " dump the hex data " , NULL } ,
{ " load-dso " , ' l ' , POPT_ARG_STRING , NULL , OPT_LOAD_DSO , " load from shared object file " , NULL } ,
POPT_COMMON_SAMBA
POPT_COMMON_VERSION
2006-09-06 16:28:01 +04:00
{ NULL }
2004-10-16 23:00:27 +04:00
} ;
2003-11-23 16:44:19 +03:00
2007-03-02 15:47:51 +03:00
# if (_SAMBA_BUILD_ >= 4)
2007-08-20 01:23:03 +04:00
ndr_table_init ( ) ;
2007-03-02 16:44:56 +03:00
# else
/* Initialise samba stuff */
load_case_tables ( ) ;
setlinebuf ( stdout ) ;
dbf = x_stderr ;
setup_logging ( argv [ 0 ] , True ) ;
2007-03-02 15:47:51 +03:00
# endif
2004-11-17 00:07:08 +03:00
2004-10-16 23:00:27 +04:00
pc = poptGetContext ( " ndrdump " , argc , argv , long_options , 0 ) ;
2005-03-16 09:18:20 +03:00
poptSetOtherOptionHelp (
pc , " <pipe|uuid> <function> <inout> [<filename>] " ) ;
2004-10-16 23:00:27 +04:00
while ( ( opt = poptGetNextOpt ( pc ) ) ! = - 1 ) {
2007-02-14 16:22:09 +03:00
switch ( opt ) {
case OPT_CONTEXT_FILE :
ctx_filename = poptGetOptArg ( pc ) ;
break ;
case OPT_VALIDATE :
validate = true ;
break ;
case OPT_DUMP_DATA :
dumpdata = true ;
break ;
case OPT_LOAD_DSO :
plugin = poptGetOptArg ( pc ) ;
break ;
}
2004-10-16 23:00:27 +04:00
}
pipe_name = poptGetArg ( pc ) ;
if ( ! pipe_name ) {
poptPrintUsage ( pc , stderr , 0 ) ;
2007-03-02 15:47:51 +03:00
# if (_SAMBA_BUILD_ >= 4)
2003-11-23 16:44:19 +03:00
show_pipes ( ) ;
2007-03-02 15:47:51 +03:00
# endif
2003-11-23 16:44:19 +03:00
exit ( 1 ) ;
}
2006-04-08 17:43:57 +04:00
if ( plugin ! = NULL ) {
p = load_iface_from_plugin ( plugin , pipe_name ) ;
2007-03-02 15:47:51 +03:00
}
# if (_SAMBA_BUILD_ <= 3)
else {
fprintf ( stderr , " Only loading from DSO's supported in Samba 3 \n " ) ;
exit ( 1 ) ;
2006-04-08 17:43:57 +04:00
}
2007-03-02 15:47:51 +03:00
# else
2006-04-08 17:43:57 +04:00
if ( ! p ) {
2007-08-21 23:35:43 +04:00
p = ndr_table_by_name ( pipe_name ) ;
2006-04-08 17:43:57 +04:00
}
2003-11-23 16:44:19 +03:00
2004-11-20 21:51:58 +03:00
if ( ! p ) {
2005-12-27 19:22:35 +03:00
struct GUID uuid ;
2005-03-16 09:18:20 +03:00
2005-12-27 19:22:35 +03:00
status = GUID_from_string ( pipe_name , & uuid ) ;
if ( NT_STATUS_IS_OK ( status ) ) {
2007-08-21 23:35:43 +04:00
p = ndr_table_by_uuid ( & uuid ) ;
2005-12-27 19:22:35 +03:00
}
2006-04-08 17:43:57 +04:00
}
2007-03-02 15:47:51 +03:00
# endif
2005-03-16 09:18:20 +03:00
2006-04-08 17:43:57 +04:00
if ( ! p ) {
printf ( " Unknown pipe or UUID '%s' \n " , pipe_name ) ;
exit ( 1 ) ;
2004-11-20 21:51:58 +03:00
}
2004-10-16 23:00:27 +04:00
function = poptGetArg ( pc ) ;
inout = poptGetArg ( pc ) ;
filename = poptGetArg ( pc ) ;
2005-03-13 06:16:07 +03:00
if ( ! function | | ! inout ) {
2004-10-16 23:00:27 +04:00
poptPrintUsage ( pc , stderr , 0 ) ;
2003-11-23 16:44:19 +03:00
show_functions ( p ) ;
exit ( 1 ) ;
}
if ( strcmp ( inout , " in " ) = = 0 | |
strcmp ( inout , " request " ) = = 0 ) {
flags = NDR_IN ;
} else if ( strcmp ( inout , " out " ) = = 0 | |
strcmp ( inout , " response " ) = = 0 ) {
flags = NDR_OUT ;
} else {
printf ( " Bad inout value '%s' \n " , inout ) ;
exit ( 1 ) ;
}
f = find_function ( p , function ) ;
2004-10-31 22:07:28 +03:00
mem_ctx = talloc_init ( " ndrdump " ) ;
2005-01-07 07:39:16 +03:00
st = talloc_zero_size ( mem_ctx , f - > struct_size ) ;
2004-10-31 22:07:28 +03:00
if ( ! st ) {
2005-07-17 13:20:52 +04:00
printf ( " Unable to allocate %d bytes \n " , ( int ) f - > struct_size ) ;
2004-10-31 22:07:28 +03:00
exit ( 1 ) ;
}
2005-08-22 20:21:29 +04:00
v_st = talloc_zero_size ( mem_ctx , f - > struct_size ) ;
if ( ! v_st ) {
printf ( " Unable to allocate %d bytes \n " , ( int ) f - > struct_size ) ;
exit ( 1 ) ;
}
2004-10-31 22:07:28 +03:00
if ( ctx_filename ) {
if ( flags = = NDR_IN ) {
printf ( " Context file can only be used for \" out \" packages \n " ) ;
exit ( 1 ) ;
}
2007-03-02 15:47:51 +03:00
# if (_SAMBA_BUILD_ >= 4)
2005-07-10 05:10:09 +04:00
data = ( uint8_t * ) file_load ( ctx_filename , & size , mem_ctx ) ;
2007-03-02 15:47:51 +03:00
# else
data = ( uint8_t * ) file_load ( ctx_filename , & size , 0 ) ;
# endif
2004-10-31 22:07:28 +03:00
if ( ! data ) {
perror ( ctx_filename ) ;
exit ( 1 ) ;
}
blob . data = data ;
blob . length = size ;
2005-08-22 20:21:29 +04:00
ndr_pull = ndr_pull_init_blob ( & blob , mem_ctx ) ;
ndr_pull - > flags | = LIBNDR_FLAG_REF_ALLOC ;
2004-10-31 22:07:28 +03:00
2007-11-09 21:24:25 +03:00
ndr_err = f - > ndr_pull ( ndr_pull , NDR_IN , st ) ;
2004-10-31 22:07:28 +03:00
2005-08-22 20:21:29 +04:00
if ( ndr_pull - > offset ! = ndr_pull - > data_size ) {
printf ( " WARNING! %d unread bytes while parsing context file \n " , ndr_pull - > data_size - ndr_pull - > offset ) ;
2004-10-31 22:07:28 +03:00
}
2007-11-09 21:24:25 +03:00
if ( ! NDR_ERR_CODE_IS_SUCCESS ( ndr_err ) ) {
status = ndr_map_error2ntstatus ( ndr_err ) ;
2004-10-31 22:07:28 +03:00
printf ( " pull for context file returned %s \n " , nt_errstr ( status ) ) ;
exit ( 1 ) ;
}
2005-08-22 20:21:29 +04:00
memcpy ( v_st , st , f - > struct_size ) ;
2004-10-31 22:07:28 +03:00
}
2005-03-13 06:16:07 +03:00
if ( filename )
2007-03-02 15:47:51 +03:00
# if (_SAMBA_BUILD_ >= 4)
2005-07-10 05:10:09 +04:00
data = ( uint8_t * ) file_load ( filename , & size , mem_ctx ) ;
2007-03-02 15:47:51 +03:00
# else
data = ( uint8_t * ) file_load ( filename , & size , 0 ) ;
# endif
2005-03-13 06:16:07 +03:00
else
data = ( uint8_t * ) stdin_load ( mem_ctx , & size ) ;
2003-11-23 16:44:19 +03:00
if ( ! data ) {
2005-03-13 06:16:07 +03:00
if ( filename )
perror ( filename ) ;
else
perror ( " stdin " ) ;
2003-11-23 16:44:19 +03:00
exit ( 1 ) ;
}
blob . data = data ;
blob . length = size ;
2005-08-22 20:21:29 +04:00
ndr_pull = ndr_pull_init_blob ( & blob , mem_ctx ) ;
ndr_pull - > flags | = LIBNDR_FLAG_REF_ALLOC ;
2003-11-23 16:44:19 +03:00
2007-11-09 21:24:25 +03:00
ndr_err = f - > ndr_pull ( ndr_pull , flags , st ) ;
status = ndr_map_error2ntstatus ( ndr_err ) ;
2003-11-23 16:44:19 +03:00
printf ( " pull returned %s \n " , nt_errstr ( status ) ) ;
2005-08-22 20:21:29 +04:00
if ( ndr_pull - > offset ! = ndr_pull - > data_size ) {
printf ( " WARNING! %d unread bytes \n " , ndr_pull - > data_size - ndr_pull - > offset ) ;
2007-11-02 13:36:06 +03:00
ndrdump_data ( ndr_pull - > data + ndr_pull - > offset ,
ndr_pull - > data_size - ndr_pull - > offset ,
dumpdata ) ;
2003-11-23 16:44:19 +03:00
}
2005-08-22 20:21:29 +04:00
if ( dumpdata ) {
printf ( " %d bytes consumed \n " , ndr_pull - > offset ) ;
2007-11-02 13:36:06 +03:00
ndrdump_data ( blob . data , blob . length , dumpdata ) ;
2005-08-22 20:21:29 +04:00
}
ndr_print = talloc_zero ( mem_ctx , struct ndr_print ) ;
ndr_print - > print = ndr_print_debug_helper ;
ndr_print - > depth = 1 ;
f - > ndr_print ( ndr_print , function , flags , st ) ;
2003-11-23 16:44:19 +03:00
2005-09-10 13:16:29 +04:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2003-11-24 00:51:24 +03:00
printf ( " dump FAILED \n " ) ;
exit ( 1 ) ;
}
2005-08-22 20:21:29 +04:00
if ( validate ) {
DATA_BLOB v_blob ;
struct ndr_push * ndr_v_push ;
struct ndr_pull * ndr_v_pull ;
struct ndr_print * ndr_v_print ;
2007-02-14 16:24:37 +03:00
uint32_t i ;
uint8_t byte_a , byte_b ;
bool differ ;
2005-08-22 20:21:29 +04:00
ndr_v_push = ndr_push_init_ctx ( mem_ctx ) ;
2007-11-09 21:24:25 +03:00
ndr_err = f - > ndr_push ( ndr_v_push , flags , st ) ;
status = ndr_map_error2ntstatus ( ndr_err ) ;
printf ( " push returned %s \n " , nt_errstr ( status ) ) ;
if ( ! NDR_ERR_CODE_IS_SUCCESS ( ndr_err ) ) {
2005-08-22 20:21:29 +04:00
printf ( " validate push FAILED \n " ) ;
exit ( 1 ) ;
}
v_blob = ndr_push_blob ( ndr_v_push ) ;
if ( dumpdata ) {
2005-09-03 15:52:42 +04:00
printf ( " %ld bytes generated (validate) \n " , ( long ) v_blob . length ) ;
2007-11-02 13:36:06 +03:00
ndrdump_data ( v_blob . data , v_blob . length , dumpdata ) ;
2005-08-22 20:21:29 +04:00
}
ndr_v_pull = ndr_pull_init_blob ( & v_blob , mem_ctx ) ;
ndr_v_pull - > flags | = LIBNDR_FLAG_REF_ALLOC ;
2007-11-09 21:24:25 +03:00
ndr_err = f - > ndr_pull ( ndr_v_pull , flags , v_st ) ;
status = ndr_map_error2ntstatus ( ndr_err ) ;
printf ( " pull returned %s \n " , nt_errstr ( status ) ) ;
if ( ! NDR_ERR_CODE_IS_SUCCESS ( ndr_err ) ) {
2005-08-22 20:21:29 +04:00
printf ( " validate pull FAILED \n " ) ;
exit ( 1 ) ;
}
if ( ndr_v_pull - > offset ! = ndr_v_pull - > data_size ) {
2005-09-10 13:16:29 +04:00
printf ( " WARNING! %d unread bytes in validation \n " , ndr_v_pull - > data_size - ndr_v_pull - > offset ) ;
2007-11-02 13:36:06 +03:00
ndrdump_data ( ndr_v_pull - > data + ndr_v_pull - > offset ,
ndr_v_pull - > data_size - ndr_v_pull - > offset ,
dumpdata ) ;
2005-08-22 20:21:29 +04:00
}
ndr_v_print = talloc_zero ( mem_ctx , struct ndr_print ) ;
ndr_v_print - > print = ndr_print_debug_helper ;
ndr_v_print - > depth = 1 ;
f - > ndr_print ( ndr_v_print , function , flags , v_st ) ;
if ( blob . length ! = v_blob . length ) {
2007-08-23 06:10:17 +04:00
printf ( " WARNING! orig bytes:%llu validated pushed bytes:%llu \n " ,
( unsigned long long ) blob . length , ( unsigned long long ) v_blob . length ) ;
2005-08-22 20:21:29 +04:00
}
if ( ndr_pull - > offset ! = ndr_v_pull - > offset ) {
2007-08-23 06:10:17 +04:00
printf ( " WARNING! orig pulled bytes:%llu validated pulled bytes:%llu \n " ,
( unsigned long long ) ndr_pull - > offset , ( unsigned long long ) ndr_v_pull - > offset ) ;
2007-02-14 16:24:37 +03:00
}
differ = false ;
byte_a = 0x00 ;
byte_b = 0x00 ;
for ( i = 0 ; i < blob . length ; i + + ) {
byte_a = blob . data [ i ] ;
if ( i = = v_blob . length ) {
byte_b = 0x00 ;
differ = true ;
break ;
}
byte_b = v_blob . data [ i ] ;
if ( byte_a ! = byte_b ) {
differ = true ;
break ;
}
}
if ( differ ) {
printf ( " WARNING! orig and validated differ at byte 0x%02X (%u) \n " , i , i ) ;
printf ( " WARNING! orig byte[0x%02X] = 0x%02X validated byte[0x%02X] = 0x%02X \n " ,
i , byte_a , i , byte_b ) ;
2005-08-22 20:21:29 +04:00
}
}
2003-11-24 00:51:24 +03:00
printf ( " dump OK \n " ) ;
2004-08-21 11:43:29 +04:00
2005-08-22 20:21:29 +04:00
talloc_free ( mem_ctx ) ;
2004-10-16 23:00:27 +04:00
poptFreeContext ( pc ) ;
2003-11-23 16:44:19 +03:00
return 0 ;
}