2003-11-03 07:26:30 +00:00
/*
Unix SMB / CIFS implementation .
test suite for echo rpc operations
Copyright ( C ) Andrew Tridgell 2003
2005-01-11 16:53:02 +00:00
Copyright ( C ) Stefan ( metze ) Metzmacher 2005
2005-02-18 23:43:40 +00:00
Copyright ( C ) Jelmer Vernooij 2005
2003-11-03 07:26:30 +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
the Free Software Foundation ; either version 2 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 , write to the Free Software
Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
# include "includes.h"
2005-02-03 11:56:03 +00:00
# include "lib/events/events.h"
2004-11-01 10:30:34 +00:00
# include "librpc/gen_ndr/ndr_echo.h"
2003-11-03 07:26:30 +00:00
2003-11-03 08:37:48 +00:00
/*
test the AddOne interface
*/
static BOOL test_addone ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx )
{
int i ;
NTSTATUS status ;
printf ( " \n Testing AddOne \n " ) ;
for ( i = 0 ; i < 10 ; i + + ) {
2004-05-25 16:24:13 +00:00
uint32_t n = i ;
2003-11-09 07:24:06 +00:00
struct echo_AddOne r ;
r . in . v = & n ;
r . out . v = & n ;
status = dcerpc_echo_AddOne ( p , mem_ctx , & r ) ;
2003-11-03 08:37:48 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
printf ( " AddOne(%d) failed - %s \n " , i , nt_errstr ( status ) ) ;
return False ;
}
2003-11-09 07:24:06 +00:00
printf ( " %d + 1 = %u \n " , i , n ) ;
2003-11-03 08:37:48 +00:00
}
return True ;
}
2003-11-03 09:18:38 +00:00
/*
test the EchoData interface
*/
static BOOL test_echodata ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx )
{
int i ;
NTSTATUS status ;
2004-11-25 18:01:40 +00:00
uint8_t * data_in , * data_out ;
2003-11-04 09:10:31 +00:00
int len = 1 + ( random ( ) % 5000 ) ;
2003-11-09 07:24:06 +00:00
struct echo_EchoData r ;
2003-11-03 09:18:38 +00:00
printf ( " \n Testing EchoData \n " ) ;
2005-01-06 03:06:58 +00:00
data_in = talloc_size ( mem_ctx , len ) ;
data_out = talloc_size ( mem_ctx , len ) ;
2003-11-03 09:18:38 +00:00
for ( i = 0 ; i < len ; i + + ) {
2003-11-09 07:24:06 +00:00
data_in [ i ] = i ;
2003-11-03 09:18:38 +00:00
}
2003-11-09 07:24:06 +00:00
r . in . len = len ;
2003-11-13 09:26:53 +00:00
r . in . in_data = data_in ;
2003-11-03 09:18:38 +00:00
2003-11-09 07:24:06 +00:00
status = dcerpc_echo_EchoData ( p , mem_ctx , & r ) ;
2003-11-03 09:18:38 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2003-11-04 02:28:08 +00:00
printf ( " EchoData(%d) failed - %s \n " , len , nt_errstr ( status ) ) ;
2003-11-03 09:18:38 +00:00
return False ;
}
2003-11-13 09:26:53 +00:00
data_out = r . out . out_data ;
for ( i = 0 ; i < len ; i + + ) {
if ( data_in [ i ] ! = data_out [ i ] ) {
printf ( " Bad data returned for len %d at offset %d \n " ,
len , i ) ;
printf ( " in: \n " ) ;
2003-11-13 10:29:58 +00:00
dump_data ( 0 , data_in + i , MIN ( len - i , 16 ) ) ;
2003-11-13 09:26:53 +00:00
printf ( " out: \n " ) ;
2003-11-13 10:29:58 +00:00
dump_data ( 0 , data_out + i , MIN ( len - 1 , 16 ) ) ;
2003-11-13 09:26:53 +00:00
return False ;
}
2003-11-03 09:18:38 +00:00
}
return True ;
}
2003-11-03 10:01:20 +00:00
/*
test the SourceData interface
*/
static BOOL test_sourcedata ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx )
{
int i ;
NTSTATUS status ;
2003-11-04 09:10:31 +00:00
int len = 200000 + ( random ( ) % 5000 ) ;
2004-11-25 18:01:40 +00:00
uint8_t * data_out ;
2003-11-09 07:24:06 +00:00
struct echo_SourceData r ;
2003-11-03 10:01:20 +00:00
printf ( " \n Testing SourceData \n " ) ;
2005-01-06 03:06:58 +00:00
data_out = talloc_size ( mem_ctx , len ) ;
2003-11-09 07:24:06 +00:00
r . in . len = len ;
r . out . data = data_out ;
status = dcerpc_echo_SourceData ( p , mem_ctx , & r ) ;
2003-11-03 10:01:20 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2003-11-04 02:28:08 +00:00
printf ( " SourceData(%d) failed - %s \n " , len , nt_errstr ( status ) ) ;
2003-11-03 10:01:20 +00:00
return False ;
}
for ( i = 0 ; i < len ; i + + ) {
2004-05-29 08:11:46 +00:00
uint8_t * v = ( uint8_t * ) data_out ;
2003-11-04 02:28:08 +00:00
if ( v [ i ] ! = ( i & 0xFF ) ) {
2004-05-29 08:11:46 +00:00
printf ( " bad data 0x%x at %d \n " , ( uint8_t ) data_out [ i ] , i ) ;
2003-11-03 10:01:20 +00:00
return False ;
}
}
return True ;
}
/*
test the SinkData interface
*/
static BOOL test_sinkdata ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx )
{
int i ;
NTSTATUS status ;
2004-11-25 18:01:40 +00:00
uint8_t * data_in ;
2003-11-04 09:10:31 +00:00
int len = 200000 + ( random ( ) % 5000 ) ;
2003-11-09 07:24:06 +00:00
struct echo_SinkData r ;
2003-11-03 10:01:20 +00:00
printf ( " \n Testing SinkData \n " ) ;
2005-01-06 03:06:58 +00:00
data_in = talloc_size ( mem_ctx , len ) ;
2003-11-03 10:01:20 +00:00
for ( i = 0 ; i < len ; i + + ) {
data_in [ i ] = i + 1 ;
}
2003-11-09 07:24:06 +00:00
r . in . len = len ;
r . in . data = data_in ;
status = dcerpc_echo_SinkData ( p , mem_ctx , & r ) ;
2003-11-03 10:01:20 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2003-11-04 02:28:08 +00:00
printf ( " SinkData(%d) failed - %s \n " , len , nt_errstr ( status ) ) ;
2003-11-03 10:01:20 +00:00
return False ;
}
2003-11-04 09:10:31 +00:00
printf ( " sunk %d bytes \n " , len ) ;
2003-11-03 10:01:20 +00:00
return True ;
}
2003-11-13 09:26:53 +00:00
/*
test the testcall interface
*/
static BOOL test_testcall ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx )
{
NTSTATUS status ;
2003-12-14 13:22:12 +00:00
struct echo_TestCall r ;
2003-11-17 09:34:19 +00:00
2003-11-19 12:03:11 +00:00
r . in . s1 = " input string " ;
2003-11-13 09:26:53 +00:00
printf ( " \n Testing TestCall \n " ) ;
2003-12-14 13:22:12 +00:00
status = dcerpc_echo_TestCall ( p , mem_ctx , & r ) ;
2003-11-13 09:26:53 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
printf ( " TestCall failed - %s \n " , nt_errstr ( status ) ) ;
return False ;
}
return True ;
}
2003-11-19 12:03:11 +00:00
/*
test the testcall interface
*/
static BOOL test_testcall2 ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx )
{
NTSTATUS status ;
2003-12-14 13:22:12 +00:00
struct echo_TestCall2 r ;
2003-11-19 12:03:11 +00:00
int i ;
BOOL ret = True ;
for ( i = 1 ; i < = 7 ; i + + ) {
r . in . level = i ;
printf ( " \n Testing TestCall2 level %d \n " , i ) ;
2003-12-14 13:22:12 +00:00
status = dcerpc_echo_TestCall2 ( p , mem_ctx , & r ) ;
2003-11-19 12:03:11 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
printf ( " TestCall2 failed - %s \n " , nt_errstr ( status ) ) ;
ret = False ;
}
}
return ret ;
}
2004-08-30 07:36:16 +00:00
/*
test the TestSleep interface
*/
static BOOL test_sleep ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx )
{
int i ;
NTSTATUS status ;
2005-01-11 16:53:02 +00:00
# define ASYNC_COUNT 3
2004-08-30 07:36:16 +00:00
struct rpc_request * req [ ASYNC_COUNT ] ;
struct echo_TestSleep r [ ASYNC_COUNT ] ;
2005-01-11 16:53:02 +00:00
BOOL done [ ASYNC_COUNT ] ;
struct timeval snd [ ASYNC_COUNT ] ;
struct timeval rcv [ ASYNC_COUNT ] ;
struct timeval diff [ ASYNC_COUNT ] ;
2004-08-30 07:36:16 +00:00
struct event_context * ctx ;
int total_done = 0 ;
BOOL ret = True ;
printf ( " \n Testing TestSleep \n " ) ;
for ( i = 0 ; i < ASYNC_COUNT ; i + + ) {
2005-01-11 16:53:02 +00:00
done [ i ] = False ;
snd [ i ] = timeval_current ( ) ;
rcv [ i ] = timeval_zero ( ) ;
2004-08-30 07:36:16 +00:00
r [ i ] . in . seconds = ASYNC_COUNT - i ;
req [ i ] = dcerpc_echo_TestSleep_send ( p , mem_ctx , & r [ i ] ) ;
if ( ! req [ i ] ) {
printf ( " Failed to send async sleep request \n " ) ;
return False ;
}
}
ctx = dcerpc_event_context ( p ) ;
while ( total_done < ASYNC_COUNT ) {
2004-09-20 10:40:11 +00:00
if ( event_loop_once ( ctx ) ! = 0 ) {
return False ;
}
2004-08-30 07:36:16 +00:00
for ( i = 0 ; i < ASYNC_COUNT ; i + + ) {
2005-01-11 16:53:02 +00:00
if ( done [ i ] = = False & & req [ i ] - > state = = RPC_REQUEST_DONE ) {
2004-08-30 07:36:16 +00:00
total_done + + ;
2005-01-11 16:53:02 +00:00
done [ i ] = True ;
rcv [ i ] = timeval_current ( ) ;
diff [ i ] = timeval_diff ( & rcv [ i ] , & snd [ i ] ) ;
status = dcerpc_ndr_request_recv ( req [ i ] ) ;
2004-08-30 07:36:16 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
printf ( " TestSleep(%d) failed - %s \n " ,
i , nt_errstr ( status ) ) ;
ret = False ;
2005-01-11 16:53:02 +00:00
} else if ( r [ i ] . out . result ! = r [ i ] . in . seconds ) {
printf ( " Failed - Sleeped for %u seconds (but we said %u seconds and the reply takes only %u seconds) \n " ,
r [ i ] . out . result , r [ i ] . in . seconds , ( uint_t ) diff [ i ] . tv_sec ) ;
ret = False ;
2004-08-30 07:36:16 +00:00
} else {
2005-02-14 09:44:58 +00:00
if ( r [ i ] . out . result > diff [ i ] . tv_sec ) {
printf ( " Failed - Sleeped for %u seconds (but reply takes only %u.%06u seconds) \n " ,
r [ i ] . out . result , ( uint_t ) diff [ i ] . tv_sec , ( uint_t ) diff [ i ] . tv_usec ) ;
2005-01-11 16:53:02 +00:00
} else if ( r [ i ] . out . result + 1 = = diff [ i ] . tv_sec ) {
2005-02-14 09:44:58 +00:00
printf ( " Sleeped for %u seconds (but reply takes %u.%06u seconds - busy server?) \n " ,
r [ i ] . out . result , ( uint_t ) diff [ i ] . tv_sec , ( uint_t ) diff [ i ] . tv_usec ) ;
2005-01-11 16:53:02 +00:00
} else if ( r [ i ] . out . result = = diff [ i ] . tv_sec ) {
2005-02-14 09:44:58 +00:00
printf ( " Sleeped for %u seconds (reply takes %u.%06u seconds - ok) \n " ,
r [ i ] . out . result , ( uint_t ) diff [ i ] . tv_sec , ( uint_t ) diff [ i ] . tv_usec ) ;
2005-01-11 16:53:02 +00:00
} else {
2005-02-14 09:44:58 +00:00
printf ( " (Failed) - Not async - Sleeped for %u seconds (but reply takes %u.%06u seconds) \n " ,
r [ i ] . out . result , ( uint_t ) diff [ i ] . tv_sec , ( uint_t ) diff [ i ] . tv_usec ) ;
2005-01-11 16:53:02 +00:00
/* TODO: let the test fail here, when we support async rpc on ncacn_np
ret = False ; */
}
2004-08-30 07:36:16 +00:00
}
}
}
}
return ret ;
}
2005-01-04 23:27:12 +00:00
/*
test enum handling
*/
static BOOL test_enum ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx )
{
NTSTATUS status ;
struct echo_TestEnum r ;
BOOL ret = True ;
enum echo_Enum1 v = ECHO_ENUM1 ;
struct echo_Enum2 e2 ;
union echo_Enum3 e3 ;
r . in . foo1 = & v ;
r . in . foo2 = & e2 ;
r . in . foo3 = & e3 ;
r . out . foo1 = & v ;
r . out . foo2 = & e2 ;
r . out . foo3 = & e3 ;
e2 . e1 = 76 ;
e2 . e2 = ECHO_ENUM1_32 ;
e3 . e1 = ECHO_ENUM2 ;
printf ( " \n Testing TestEnum \n " ) ;
status = dcerpc_echo_TestEnum ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
printf ( " TestEnum failed - %s \n " , nt_errstr ( status ) ) ;
ret = False ;
}
return ret ;
}
2005-02-18 23:30:26 +00:00
/*
test surrounding conformant array handling
*/
static BOOL test_surrounding ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx )
{
NTSTATUS status ;
struct echo_TestSurrounding r ;
BOOL ret = True ;
ZERO_STRUCT ( r ) ;
r . in . data = talloc ( mem_ctx , struct echo_Surrounding ) ;
r . in . data - > x = 20 ;
r . in . data - > surrounding = talloc_zero_array ( mem_ctx , uint16_t , r . in . data - > x ) ;
r . out . data = talloc ( mem_ctx , struct echo_Surrounding ) ;
printf ( " \n Testing TestSurrounding \n " ) ;
status = dcerpc_echo_TestSurrounding ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
printf ( " TestSurrounding failed - %s \n " , nt_errstr ( status ) ) ;
ret = False ;
}
if ( r . out . data - > x ! = 2 * r . in . data - > x ) {
printf ( " TestSurrounding did not make the array twice as large \n " ) ;
ret = False ;
}
return ret ;
}
2005-02-20 02:57:38 +00:00
/*
test multiple levels of pointers
*/
static BOOL test_doublepointer ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx )
{
NTSTATUS status ;
struct echo_TestDoublePointer r ;
BOOL ret = True ;
uint16_t value = 12 ;
uint16_t * pvalue = & value ;
uint16_t * * ppvalue = & pvalue ;
ZERO_STRUCT ( r ) ;
r . in . data = & ppvalue ;
printf ( " \n Testing TestDoublePointer \n " ) ;
status = dcerpc_echo_TestDoublePointer ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
printf ( " TestDoublePointer failed - %s \n " , nt_errstr ( status ) ) ;
ret = False ;
}
if ( value ! = r . out . result ) {
printf ( " TestSurrounding did not return original value \n " ) ;
ret = False ;
}
pvalue = NULL ;
status = dcerpc_echo_TestDoublePointer ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
printf ( " TestDoublePointer failed - %s \n " , nt_errstr ( status ) ) ;
ret = False ;
}
if ( r . out . result ! = 0 ) {
printf ( " TestSurrounding did not return 0 when passed a NULL pointer \n " ) ;
ret = False ;
}
return ret ;
}
2004-10-28 13:40:50 +00:00
BOOL torture_rpc_echo ( void )
2003-11-03 07:26:30 +00:00
{
2005-02-18 23:43:40 +00:00
NTSTATUS status ;
struct dcerpc_pipe * p ;
2003-11-03 07:26:30 +00:00
TALLOC_CTX * mem_ctx ;
2003-11-03 08:37:48 +00:00
BOOL ret = True ;
2003-11-03 07:26:30 +00:00
mem_ctx = talloc_init ( " torture_rpc_echo " ) ;
2003-11-18 05:01:10 +00:00
status = torture_rpc_connection ( & p ,
DCERPC_RPCECHO_NAME ,
DCERPC_RPCECHO_UUID ,
DCERPC_RPCECHO_VERSION ) ;
2003-11-03 07:26:30 +00:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
return False ;
}
2003-11-03 08:37:48 +00:00
2005-01-21 06:56:57 +00:00
ret & = test_addone ( p , mem_ctx ) ;
ret & = test_sinkdata ( p , mem_ctx ) ;
ret & = test_echodata ( p , mem_ctx ) ;
ret & = test_sourcedata ( p , mem_ctx ) ;
ret & = test_testcall ( p , mem_ctx ) ;
ret & = test_testcall2 ( p , mem_ctx ) ;
ret & = test_enum ( p , mem_ctx ) ;
2005-02-18 23:30:26 +00:00
ret & = test_surrounding ( p , mem_ctx ) ;
2005-02-20 02:57:38 +00:00
ret & = test_doublepointer ( p , mem_ctx ) ;
2005-02-18 23:43:40 +00:00
ret & = test_sleep ( p , mem_ctx ) ;
2005-01-11 16:53:02 +00:00
2003-11-03 08:37:48 +00:00
printf ( " \n " ) ;
2004-09-27 08:41:39 +00:00
talloc_free ( mem_ctx ) ;
2003-11-22 08:11:32 +00:00
2005-02-18 23:43:40 +00:00
torture_rpc_close ( p ) ;
2003-11-03 08:37:48 +00:00
return ret ;
2003-11-03 07:26:30 +00:00
}