2003-11-19 12:15:46 +03:00
/*
Unix SMB / CIFS implementation .
test suite for srvsvc rpc operations
Copyright ( C ) Stefan ( metze ) Metzmacher 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
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"
2006-01-03 16:41:17 +03:00
# include "torture/torture.h"
2004-11-01 13:30:34 +03:00
# include "librpc/gen_ndr/ndr_srvsvc.h"
2006-03-15 02:35:30 +03:00
# include "librpc/gen_ndr/ndr_srvsvc_c.h"
2006-03-14 18:02:05 +03:00
# include "torture/rpc/rpc.h"
2003-11-19 12:15:46 +03:00
2004-07-16 13:10:36 +04:00
/**************************/
/* srvsvc_NetCharDev */
/**************************/
static BOOL test_NetCharDevGetInfo ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx ,
const char * devname )
{
NTSTATUS status ;
struct srvsvc_NetCharDevGetInfo r ;
uint32_t levels [ ] = { 0 , 1 } ;
int i ;
BOOL ret = True ;
r . in . server_unc = talloc_asprintf ( mem_ctx , " \\ \\ %s " , dcerpc_server_name ( p ) ) ;
r . in . device_name = devname ;
for ( i = 0 ; i < ARRAY_SIZE ( levels ) ; i + + ) {
ZERO_STRUCT ( r . out ) ;
r . in . level = levels [ i ] ;
2006-09-11 05:22:28 +04:00
d_printf ( " testing NetCharDevGetInfo level %u on device '%s' \n " ,
2004-07-16 13:10:36 +04:00
r . in . level , r . in . device_name ) ;
status = dcerpc_srvsvc_NetCharDevGetInfo ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetCharDevGetInfo level %u on device '%s' failed - %s \n " ,
2004-07-16 13:10:36 +04:00
r . in . level , r . in . device_name , nt_errstr ( status ) ) ;
ret = False ;
continue ;
}
if ( ! W_ERROR_IS_OK ( r . out . result ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetCharDevGetInfo level %u on device '%s' failed - %s \n " ,
2004-07-16 13:10:36 +04:00
r . in . level , r . in . device_name , win_errstr ( r . out . result ) ) ;
continue ;
}
}
return ret ;
}
static BOOL test_NetCharDevControl ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx ,
const char * devname )
{
NTSTATUS status ;
struct srvsvc_NetCharDevControl r ;
uint32_t opcodes [ ] = { 0 , 1 } ;
int i ;
BOOL ret = True ;
r . in . server_unc = talloc_asprintf ( mem_ctx , " \\ \\ %s " , dcerpc_server_name ( p ) ) ;
r . in . device_name = devname ;
for ( i = 0 ; i < ARRAY_SIZE ( opcodes ) ; i + + ) {
ZERO_STRUCT ( r . out ) ;
r . in . opcode = opcodes [ i ] ;
2006-09-11 05:22:28 +04:00
d_printf ( " testing NetCharDevControl opcode %u on device '%s' \n " ,
2004-07-16 13:10:36 +04:00
r . in . opcode , r . in . device_name ) ;
status = dcerpc_srvsvc_NetCharDevControl ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetCharDevControl opcode %u failed - %s \n " , r . in . opcode , nt_errstr ( status ) ) ;
2004-07-16 13:10:36 +04:00
ret = False ;
continue ;
}
if ( ! W_ERROR_IS_OK ( r . out . result ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetCharDevControl opcode %u failed - %s \n " , r . in . opcode , win_errstr ( r . out . result ) ) ;
2004-07-16 13:10:36 +04:00
continue ;
}
}
return ret ;
}
static BOOL test_NetCharDevEnum ( struct dcerpc_pipe * p ,
TALLOC_CTX * mem_ctx )
{
NTSTATUS status ;
struct srvsvc_NetCharDevEnum r ;
struct srvsvc_NetCharDevCtr0 c0 ;
2004-07-16 13:25:52 +04:00
uint32_t levels [ ] = { 0 , 1 } ;
2004-07-16 13:10:36 +04:00
int i ;
BOOL ret = True ;
r . in . server_unc = talloc_asprintf ( mem_ctx , " \\ \\ %s " , dcerpc_server_name ( p ) ) ;
r . in . ctr . ctr0 = & c0 ;
r . in . ctr . ctr0 - > count = 0 ;
r . in . ctr . ctr0 - > array = NULL ;
2005-02-10 08:09:35 +03:00
r . in . max_buffer = ( uint32_t ) - 1 ;
2004-07-16 13:10:36 +04:00
r . in . resume_handle = NULL ;
for ( i = 0 ; i < ARRAY_SIZE ( levels ) ; i + + ) {
int j ;
ZERO_STRUCT ( r . out ) ;
r . in . level = levels [ i ] ;
printf ( " testing NetCharDevEnum level %u \n " , r . in . level ) ;
status = dcerpc_srvsvc_NetCharDevEnum ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetCharDevEnum level %u failed - %s \n " , r . in . level , nt_errstr ( status ) ) ;
2004-07-16 13:10:36 +04:00
ret = False ;
continue ;
}
if ( ! W_ERROR_IS_OK ( r . out . result ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetCharDevEnum level %u failed - %s \n " , r . in . level , win_errstr ( r . out . result ) ) ;
2004-07-16 13:10:36 +04:00
continue ;
}
/* call test_NetCharDevGetInfo and test_NetCharDevControl for each returned share */
if ( r . in . level = = 1 ) {
for ( j = 0 ; j < r . out . ctr . ctr1 - > count ; j + + ) {
const char * device ;
device = r . out . ctr . ctr1 - > array [ j ] . device ;
if ( ! test_NetCharDevGetInfo ( p , mem_ctx , device ) ) {
ret = False ;
}
if ( ! test_NetCharDevControl ( p , mem_ctx , device ) ) {
ret = False ;
}
}
}
}
return ret ;
}
2003-11-19 12:15:46 +03:00
2004-07-16 13:25:52 +04:00
/**************************/
/* srvsvc_NetCharDevQ */
/**************************/
static BOOL test_NetCharDevQGetInfo ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx ,
const char * devicequeue )
{
NTSTATUS status ;
struct srvsvc_NetCharDevQGetInfo r ;
uint32_t levels [ ] = { 0 , 1 } ;
int i ;
BOOL ret = True ;
r . in . server_unc = talloc_asprintf ( mem_ctx , " \\ \\ %s " , dcerpc_server_name ( p ) ) ;
r . in . queue_name = devicequeue ;
r . in . user = talloc_asprintf ( mem_ctx , " Administrator " ) ;
for ( i = 0 ; i < ARRAY_SIZE ( levels ) ; i + + ) {
ZERO_STRUCT ( r . out ) ;
r . in . level = levels [ i ] ;
2006-09-11 05:22:28 +04:00
d_printf ( " testing NetCharDevQGetInfo level %u on devicequeue '%s' \n " ,
2004-07-16 13:25:52 +04:00
r . in . level , r . in . queue_name ) ;
status = dcerpc_srvsvc_NetCharDevQGetInfo ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetCharDevQGetInfo level %u on devicequeue '%s' failed - %s \n " ,
2004-07-16 13:25:52 +04:00
r . in . level , r . in . queue_name , nt_errstr ( status ) ) ;
ret = False ;
continue ;
}
if ( ! W_ERROR_IS_OK ( r . out . result ) ) {
printf ( " NetCharDevQGetInfo level %u on devicequeue '%s' failed - %s \n " ,
r . in . level , r . in . queue_name , win_errstr ( r . out . result ) ) ;
continue ;
}
}
return ret ;
}
#if 0
static BOOL test_NetCharDevQSetInfo ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx ,
const char * devicequeue )
{
NTSTATUS status ;
struct srvsvc_NetCharDevQSetInfo r ;
uint32_t parm_error ;
uint32_t levels [ ] = { 0 , 1 } ;
int i ;
BOOL ret = True ;
r . in . server_unc = talloc_asprintf ( mem_ctx , " \\ \\ %s " , dcerpc_server_name ( p ) ) ;
r . in . queue_name = devicequeue ;
for ( i = 0 ; i < ARRAY_SIZE ( levels ) ; i + + ) {
ZERO_STRUCT ( r . out ) ;
parm_error = 0 ;
r . in . level = levels [ i ] ;
2006-09-11 05:22:28 +04:00
d_printf ( " testing NetCharDevQSetInfo level %u on devicequeue '%s' \n " ,
2004-07-16 13:25:52 +04:00
r . in . level , devicequeue ) ;
switch ( r . in . level ) {
case 0 :
2005-01-27 10:08:20 +03:00
r . in . info . info0 = talloc ( mem_ctx , struct srvsvc_NetCharDevQInfo0 ) ;
2004-07-16 13:25:52 +04:00
r . in . info . info0 - > device = r . in . queue_name ;
break ;
case 1 :
2005-01-27 10:08:20 +03:00
r . in . info . info1 = talloc ( mem_ctx , struct srvsvc_NetCharDevQInfo1 ) ;
2004-07-16 13:25:52 +04:00
r . in . info . info1 - > device = r . in . queue_name ;
r . in . info . info1 - > priority = 0x000 ;
r . in . info . info1 - > devices = r . in . queue_name ;
r . in . info . info1 - > users = 0x000 ;
r . in . info . info1 - > num_ahead = 0x000 ;
break ;
default :
break ;
}
r . in . parm_error = & parm_error ;
status = dcerpc_srvsvc_NetCharDevQSetInfo ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetCharDevQSetInfo level %u on devicequeue '%s' failed - %s \n " ,
2004-07-16 13:25:52 +04:00
r . in . level , r . in . queue_name , nt_errstr ( status ) ) ;
ret = False ;
continue ;
}
if ( ! W_ERROR_IS_OK ( r . out . result ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetCharDevQSetInfo level %u on devicequeue '%s' failed - %s \n " ,
2004-07-16 13:25:52 +04:00
r . in . level , r . in . queue_name , win_errstr ( r . out . result ) ) ;
continue ;
}
}
return ret ;
}
# endif
static BOOL test_NetCharDevQEnum ( struct dcerpc_pipe * p ,
TALLOC_CTX * mem_ctx )
{
NTSTATUS status ;
struct srvsvc_NetCharDevQEnum r ;
struct srvsvc_NetCharDevQCtr0 c0 ;
uint32_t levels [ ] = { 0 , 1 } ;
int i ;
BOOL ret = True ;
r . in . server_unc = talloc_asprintf ( mem_ctx , " \\ \\ %s " , dcerpc_server_name ( p ) ) ;
r . in . user = talloc_asprintf ( mem_ctx , " %s " , " Administrator " ) ;
r . in . ctr . ctr0 = & c0 ;
r . in . ctr . ctr0 - > count = 0 ;
r . in . ctr . ctr0 - > array = NULL ;
2005-02-10 08:09:35 +03:00
r . in . max_buffer = ( uint32_t ) - 1 ;
2004-07-16 13:25:52 +04:00
r . in . resume_handle = NULL ;
for ( i = 0 ; i < ARRAY_SIZE ( levels ) ; i + + ) {
int j ;
ZERO_STRUCT ( r . out ) ;
r . in . level = levels [ i ] ;
printf ( " testing NetCharDevQEnum level %u \n " , r . in . level ) ;
status = dcerpc_srvsvc_NetCharDevQEnum ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetCharDevQEnum level %u failed - %s \n " , r . in . level , nt_errstr ( status ) ) ;
2004-07-16 13:25:52 +04:00
ret = False ;
continue ;
}
if ( ! W_ERROR_IS_OK ( r . out . result ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetCharDevQEnum level %u failed - %s \n " , r . in . level , win_errstr ( r . out . result ) ) ;
2004-07-16 13:25:52 +04:00
continue ;
}
/* call test_NetCharDevGetInfo and test_NetCharDevControl for each returned share */
if ( r . in . level = = 1 ) {
for ( j = 0 ; j < r . out . ctr . ctr1 - > count ; j + + ) {
const char * device ;
device = r . out . ctr . ctr1 - > array [ j ] . device ;
if ( ! test_NetCharDevQGetInfo ( p , mem_ctx , device ) ) {
ret = False ;
}
}
}
}
return ret ;
}
2004-07-19 10:41:34 +04:00
/**************************/
/* srvsvc_NetConn */
/**************************/
2003-11-19 12:15:46 +03:00
static BOOL test_NetConnEnum ( struct dcerpc_pipe * p ,
TALLOC_CTX * mem_ctx )
{
NTSTATUS status ;
struct srvsvc_NetConnEnum r ;
struct srvsvc_NetConnCtr0 c0 ;
2004-05-25 20:24:13 +04:00
uint32_t levels [ ] = { 0 , 1 } ;
2003-11-19 12:15:46 +03:00
int i ;
BOOL ret = True ;
r . in . server_unc = talloc_asprintf ( mem_ctx , " \\ \\ %s " , dcerpc_server_name ( p ) ) ;
r . in . path = talloc_asprintf ( mem_ctx , " %s " , " ADMIN$ " ) ;
2003-11-20 01:10:20 +03:00
r . in . ctr . ctr0 = & c0 ;
r . in . ctr . ctr0 - > count = 0 ;
r . in . ctr . ctr0 - > array = NULL ;
2004-05-25 20:24:13 +04:00
r . in . max_buffer = ( uint32_t ) - 1 ;
2003-11-19 12:15:46 +03:00
r . in . resume_handle = NULL ;
for ( i = 0 ; i < ARRAY_SIZE ( levels ) ; i + + ) {
2004-07-16 12:56:11 +04:00
ZERO_STRUCT ( r . out ) ;
2003-11-20 01:10:20 +03:00
r . in . level = levels [ i ] ;
2006-09-11 05:22:28 +04:00
d_printf ( " testing NetConnEnum level %u \n " , r . in . level ) ;
2003-11-19 12:15:46 +03:00
status = dcerpc_srvsvc_NetConnEnum ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetConnEnum level %u failed - %s \n " , r . in . level , nt_errstr ( status ) ) ;
2003-11-19 12:15:46 +03:00
ret = False ;
2004-07-16 12:56:11 +04:00
continue ;
}
if ( ! W_ERROR_IS_OK ( r . out . result ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetConnEnum level %u failed - %s \n " , r . in . level , win_errstr ( r . out . result ) ) ;
2004-07-16 12:56:11 +04:00
continue ;
2003-11-19 12:15:46 +03:00
}
}
2004-07-19 10:41:34 +04:00
return ret ;
2003-11-19 12:15:46 +03:00
}
2004-07-19 10:41:34 +04:00
/**************************/
/* srvsvc_NetFile */
/**************************/
2003-11-19 12:15:46 +03:00
static BOOL test_NetFileEnum ( struct dcerpc_pipe * p ,
TALLOC_CTX * mem_ctx )
{
NTSTATUS status ;
struct srvsvc_NetFileEnum r ;
struct srvsvc_NetFileCtr3 c3 ;
2004-05-25 20:24:13 +04:00
uint32_t levels [ ] = { 2 , 3 } ;
2003-11-19 12:15:46 +03:00
int i ;
BOOL ret = True ;
r . in . server_unc = talloc_asprintf ( mem_ctx , " \\ \\ %s " , dcerpc_server_name ( p ) ) ;
r . in . path = NULL ;
r . in . user = NULL ;
2003-11-20 01:10:20 +03:00
r . in . ctr . ctr3 = & c3 ;
r . in . ctr . ctr3 - > count = 0 ;
r . in . ctr . ctr3 - > array = NULL ;
2004-05-25 20:24:13 +04:00
r . in . max_buffer = ( uint32_t ) 4096 ;
2003-11-19 12:15:46 +03:00
r . in . resume_handle = NULL ;
for ( i = 0 ; i < ARRAY_SIZE ( levels ) ; i + + ) {
2004-07-16 12:56:11 +04:00
ZERO_STRUCT ( r . out ) ;
2003-11-20 01:10:20 +03:00
r . in . level = levels [ i ] ;
2006-09-11 05:22:28 +04:00
d_printf ( " testing NetFileEnum level %u \n " , r . in . level ) ;
2003-11-19 12:15:46 +03:00
status = dcerpc_srvsvc_NetFileEnum ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetFileEnum level %u failed - %s \n " , r . in . level , nt_errstr ( status ) ) ;
2003-11-19 12:15:46 +03:00
ret = False ;
2004-07-16 12:56:11 +04:00
continue ;
}
if ( ! W_ERROR_IS_OK ( r . out . result ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetFileEnum level %u failed - %s \n " , r . in . level , win_errstr ( r . out . result ) ) ;
2004-07-16 12:56:11 +04:00
continue ;
2003-11-19 12:15:46 +03:00
}
}
2004-07-19 10:41:34 +04:00
return ret ;
2003-11-19 12:15:46 +03:00
}
2004-07-19 10:41:34 +04:00
/**************************/
/* srvsvc_NetSess */
/**************************/
2003-11-19 12:15:46 +03:00
static BOOL test_NetSessEnum ( struct dcerpc_pipe * p ,
TALLOC_CTX * mem_ctx )
{
NTSTATUS status ;
struct srvsvc_NetSessEnum r ;
struct srvsvc_NetSessCtr0 c0 ;
2004-05-25 20:24:13 +04:00
uint32_t levels [ ] = { 0 , 1 , 2 , 10 , 502 } ;
2003-11-19 12:15:46 +03:00
int i ;
BOOL ret = True ;
r . in . server_unc = talloc_asprintf ( mem_ctx , " \\ \\ %s " , dcerpc_server_name ( p ) ) ;
r . in . client = NULL ;
r . in . user = NULL ;
2003-11-20 01:10:20 +03:00
r . in . ctr . ctr0 = & c0 ;
r . in . ctr . ctr0 - > count = 0 ;
r . in . ctr . ctr0 - > array = NULL ;
2004-05-25 20:24:13 +04:00
r . in . max_buffer = ( uint32_t ) - 1 ;
2003-11-19 12:15:46 +03:00
r . in . resume_handle = NULL ;
for ( i = 0 ; i < ARRAY_SIZE ( levels ) ; i + + ) {
2004-07-16 12:56:11 +04:00
ZERO_STRUCT ( r . out ) ;
2003-11-20 01:10:20 +03:00
r . in . level = levels [ i ] ;
2006-09-11 05:22:28 +04:00
d_printf ( " testing NetSessEnum level %u \n " , r . in . level ) ;
2003-11-19 12:15:46 +03:00
status = dcerpc_srvsvc_NetSessEnum ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetSessEnum level %u failed - %s \n " , r . in . level , nt_errstr ( status ) ) ;
2003-11-19 12:15:46 +03:00
ret = False ;
2004-07-16 12:56:11 +04:00
continue ;
}
if ( ! W_ERROR_IS_OK ( r . out . result ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetSessEnum level %u failed - %s \n " , r . in . level , win_errstr ( r . out . result ) ) ;
2004-07-16 12:56:11 +04:00
continue ;
2003-11-19 12:15:46 +03:00
}
}
2004-07-19 10:41:34 +04:00
return ret ;
2003-11-19 12:15:46 +03:00
}
2004-07-16 12:56:11 +04:00
/**************************/
/* srvsvc_NetShare */
/**************************/
2006-03-31 17:41:49 +04:00
static BOOL test_NetShareCheck ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx ,
const char * device_name )
{
NTSTATUS status ;
struct srvsvc_NetShareCheck r ;
BOOL ret = True ;
r . in . server_unc = talloc_asprintf ( mem_ctx , " \\ \\ %s " , dcerpc_server_name ( p ) ) ;
r . in . device_name = device_name ;
2006-09-11 05:22:28 +04:00
d_printf ( " testing NetShareCheck on device '%s' \n " , r . in . device_name ) ;
2006-03-31 17:41:49 +04:00
status = dcerpc_srvsvc_NetShareCheck ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " dcerpc_srvsvc_NetShareCheck on device '%s' failed - %s \n " ,
2006-03-31 17:41:49 +04:00
r . in . device_name , nt_errstr ( status ) ) ;
ret = False ;
} else if ( ! W_ERROR_IS_OK ( r . out . result ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetShareCheck on device '%s' failed - %s \n " ,
2006-03-31 17:41:49 +04:00
r . in . device_name , win_errstr ( r . out . result ) ) ;
ret = False ;
}
return ret ;
}
2004-05-14 04:24:00 +04:00
static BOOL test_NetShareGetInfo ( struct dcerpc_pipe * p , TALLOC_CTX * mem_ctx ,
2006-03-31 17:41:49 +04:00
const char * sharename , BOOL admin )
2004-05-14 04:24:00 +04:00
{
NTSTATUS status ;
struct srvsvc_NetShareGetInfo r ;
2006-03-31 17:41:49 +04:00
struct {
uint32_t level ;
WERROR anon_status ;
WERROR admin_status ;
} levels [ ] = {
{ 0 , WERR_OK , WERR_OK } ,
{ 1 , WERR_OK , WERR_OK } ,
{ 2 , WERR_ACCESS_DENIED , WERR_OK } ,
{ 501 , WERR_OK , WERR_OK } ,
{ 502 , WERR_ACCESS_DENIED , WERR_OK } ,
{ 1005 , WERR_OK , WERR_OK } ,
} ;
2004-05-14 04:24:00 +04:00
int i ;
BOOL ret = True ;
r . in . server_unc = talloc_asprintf ( mem_ctx , " \\ \\ %s " , dcerpc_server_name ( p ) ) ;
2004-07-16 12:56:11 +04:00
r . in . share_name = sharename ;
2004-05-14 04:24:00 +04:00
for ( i = 0 ; i < ARRAY_SIZE ( levels ) ; i + + ) {
2006-03-31 17:41:49 +04:00
WERROR expected ;
r . in . level = levels [ i ] . level ;
expected = levels [ i ] . anon_status ;
if ( admin ) expected = levels [ i ] . admin_status ;
2004-07-16 12:56:11 +04:00
ZERO_STRUCT ( r . out ) ;
2004-05-14 04:24:00 +04:00
2006-09-11 05:22:28 +04:00
d_printf ( " testing NetShareGetInfo level %u on share '%s' \n " ,
2004-05-14 04:24:00 +04:00
r . in . level , r . in . share_name ) ;
status = dcerpc_srvsvc_NetShareGetInfo ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetShareGetInfo level %u on share '%s' failed - %s \n " ,
2004-07-16 12:56:11 +04:00
r . in . level , r . in . share_name , nt_errstr ( status ) ) ;
2004-05-14 04:24:00 +04:00
ret = False ;
2004-07-16 12:56:11 +04:00
continue ;
}
2006-03-31 17:41:49 +04:00
if ( ! W_ERROR_EQUAL ( r . out . result , expected ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetShareGetInfo level %u on share '%s' failed - %s (expected %s) \n " ,
2006-03-31 17:41:49 +04:00
r . in . level , r . in . share_name , win_errstr ( r . out . result ) ,
win_errstr ( expected ) ) ;
ret = False ;
2004-07-16 12:56:11 +04:00
continue ;
2004-05-14 04:24:00 +04:00
}
2006-03-31 17:41:49 +04:00
if ( r . in . level ! = 2 ) continue ;
if ( ! r . out . info . info2 | | ! r . out . info . info2 - > path ) continue ;
if ( ! test_NetShareCheck ( p , mem_ctx , r . out . info . info2 - > path ) ) {
ret = False ;
}
2004-12-31 09:19:05 +03:00
}
return ret ;
}
2004-07-19 10:41:34 +04:00
/**************************/
/* srvsvc_NetShare */
/**************************/
2003-11-19 12:15:46 +03:00
static BOOL test_NetShareEnumAll ( struct dcerpc_pipe * p ,
2006-03-31 17:41:49 +04:00
TALLOC_CTX * mem_ctx , BOOL admin )
2003-11-19 12:15:46 +03:00
{
NTSTATUS status ;
struct srvsvc_NetShareEnumAll r ;
struct srvsvc_NetShareCtr0 c0 ;
2006-03-31 17:41:49 +04:00
struct {
uint32_t level ;
WERROR anon_status ;
WERROR admin_status ;
} levels [ ] = {
{ 0 , WERR_OK , WERR_OK } ,
{ 1 , WERR_OK , WERR_OK } ,
{ 2 , WERR_ACCESS_DENIED , WERR_OK } ,
{ 501 , WERR_ACCESS_DENIED , WERR_OK } ,
{ 502 , WERR_ACCESS_DENIED , WERR_OK } ,
} ;
2003-11-19 12:15:46 +03:00
int i ;
BOOL ret = True ;
2004-05-25 20:24:13 +04:00
uint32_t resume_handle ;
2003-11-20 08:34:02 +03:00
ZERO_STRUCT ( c0 ) ;
2003-11-19 12:15:46 +03:00
r . in . server_unc = talloc_asprintf ( mem_ctx , " \\ \\ %s " , dcerpc_server_name ( p ) ) ;
2003-11-20 01:10:20 +03:00
r . in . ctr . ctr0 = & c0 ;
2004-05-25 20:24:13 +04:00
r . in . max_buffer = ( uint32_t ) - 1 ;
2003-11-20 08:34:02 +03:00
r . in . resume_handle = & resume_handle ;
r . out . resume_handle = & resume_handle ;
2003-11-19 12:15:46 +03:00
for ( i = 0 ; i < ARRAY_SIZE ( levels ) ; i + + ) {
2004-05-14 04:24:00 +04:00
int j ;
2006-03-31 17:41:49 +04:00
WERROR expected ;
r . in . level = levels [ i ] . level ;
expected = levels [ i ] . anon_status ;
if ( admin ) expected = levels [ i ] . admin_status ;
2004-05-14 04:24:00 +04:00
2004-07-16 12:56:11 +04:00
ZERO_STRUCT ( r . out ) ;
2003-11-20 08:34:02 +03:00
resume_handle = 0 ;
2006-03-31 17:41:49 +04:00
2006-09-11 05:22:28 +04:00
d_printf ( " testing NetShareEnumAll level %u \n " , r . in . level ) ;
2003-11-19 12:15:46 +03:00
status = dcerpc_srvsvc_NetShareEnumAll ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetShareEnumAll level %u failed - %s \n " , r . in . level , nt_errstr ( status ) ) ;
2003-11-19 12:15:46 +03:00
ret = False ;
2004-05-14 04:24:00 +04:00
continue ;
}
2006-03-31 17:41:49 +04:00
if ( ! W_ERROR_EQUAL ( r . out . result , expected ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetShareEnumAll level %u failed - %s (expected %s) \n " ,
2006-03-31 17:41:49 +04:00
r . in . level , win_errstr ( r . out . result ) ,
win_errstr ( expected ) ) ;
2004-07-16 12:56:11 +04:00
continue ;
}
2004-05-14 04:24:00 +04:00
/* call srvsvc_NetShareGetInfo for each returned share */
2006-03-31 17:41:49 +04:00
if ( r . in . level = = 2 & & r . out . ctr . ctr2 ) {
2004-12-31 09:19:05 +03:00
for ( j = 0 ; j < r . out . ctr . ctr2 - > count ; j + + ) {
2004-05-14 04:24:00 +04:00
const char * name ;
2004-12-31 09:19:05 +03:00
name = r . out . ctr . ctr2 - > array [ j ] . name ;
2006-03-31 17:41:49 +04:00
if ( ! test_NetShareGetInfo ( p , mem_ctx , name , admin ) ) {
2004-12-31 09:19:05 +03:00
ret = False ;
}
2004-05-14 04:24:00 +04:00
}
2003-11-19 12:15:46 +03:00
}
}
2004-07-16 12:56:11 +04:00
return ret ;
2003-11-19 12:15:46 +03:00
}
2004-07-16 12:56:11 +04:00
static BOOL test_NetShareEnum ( struct dcerpc_pipe * p ,
2006-03-31 17:41:49 +04:00
TALLOC_CTX * mem_ctx , BOOL admin )
2004-07-16 12:56:11 +04:00
{
NTSTATUS status ;
struct srvsvc_NetShareEnum r ;
struct srvsvc_NetShareCtr0 c0 ;
2006-03-31 17:41:49 +04:00
struct {
uint32_t level ;
WERROR anon_status ;
WERROR admin_status ;
} levels [ ] = {
{ 0 , WERR_OK , WERR_OK } ,
{ 1 , WERR_OK , WERR_OK } ,
{ 2 , WERR_ACCESS_DENIED , WERR_OK } ,
{ 501 , WERR_UNKNOWN_LEVEL , WERR_UNKNOWN_LEVEL } ,
{ 502 , WERR_ACCESS_DENIED , WERR_OK } ,
} ;
2004-07-16 12:56:11 +04:00
int i ;
BOOL ret = True ;
r . in . server_unc = talloc_asprintf ( mem_ctx , " \\ \\ %s " , dcerpc_server_name ( p ) ) ;
r . in . ctr . ctr0 = & c0 ;
r . in . ctr . ctr0 - > count = 0 ;
r . in . ctr . ctr0 - > array = NULL ;
r . in . max_buffer = ( uint32_t ) - 1 ;
r . in . resume_handle = NULL ;
2003-12-15 12:18:53 +03:00
2004-07-16 12:56:11 +04:00
for ( i = 0 ; i < ARRAY_SIZE ( levels ) ; i + + ) {
2006-03-31 17:41:49 +04:00
WERROR expected ;
r . in . level = levels [ i ] . level ;
expected = levels [ i ] . anon_status ;
if ( admin ) expected = levels [ i ] . admin_status ;
2004-07-16 12:56:11 +04:00
ZERO_STRUCT ( r . out ) ;
2006-03-31 17:41:49 +04:00
2006-09-11 05:22:28 +04:00
d_printf ( " testing NetShareEnum level %u \n " , r . in . level ) ;
2004-07-16 12:56:11 +04:00
status = dcerpc_srvsvc_NetShareEnum ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetShareEnum level %u failed - %s \n " , r . in . level , nt_errstr ( status ) ) ;
2004-07-16 12:56:11 +04:00
ret = False ;
continue ;
}
2006-03-31 17:41:49 +04:00
if ( ! W_ERROR_EQUAL ( r . out . result , expected ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetShareEnum level %u failed - %s (expected %s) \n " ,
2006-03-31 17:41:49 +04:00
r . in . level , win_errstr ( r . out . result ) ,
win_errstr ( expected ) ) ;
2004-07-16 12:56:11 +04:00
continue ;
}
}
2004-07-19 10:41:34 +04:00
return ret ;
2004-07-16 12:56:11 +04:00
}
2004-07-16 13:36:17 +04:00
/**************************/
/* srvsvc_NetSrv */
/**************************/
static BOOL test_NetSrvGetInfo ( struct dcerpc_pipe * p ,
TALLOC_CTX * mem_ctx )
{
NTSTATUS status ;
struct srvsvc_NetSrvGetInfo r ;
struct srvsvc_NetSrvInfo503 i503 ;
uint32_t levels [ ] = { 100 , 101 , 102 , 502 , 503 } ;
int i ;
BOOL ret = True ;
uint32_t resume_handle ;
ZERO_STRUCT ( i503 ) ;
r . in . server_unc = talloc_asprintf ( mem_ctx , " \\ \\ %s " , dcerpc_server_name ( p ) ) ;
for ( i = 0 ; i < ARRAY_SIZE ( levels ) ; i + + ) {
ZERO_STRUCT ( r . out ) ;
resume_handle = 0 ;
r . in . level = levels [ i ] ;
2006-09-11 05:22:28 +04:00
d_printf ( " testing NetSrvGetInfo level %u \n " , r . in . level ) ;
2004-07-16 13:36:17 +04:00
status = dcerpc_srvsvc_NetSrvGetInfo ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetSrvGetInfo level %u failed - %s \n " , r . in . level , nt_errstr ( status ) ) ;
2004-07-16 13:36:17 +04:00
ret = False ;
continue ;
}
if ( ! W_ERROR_IS_OK ( r . out . result ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetSrvGetInfo level %u failed - %s \n " , r . in . level , win_errstr ( r . out . result ) ) ;
2004-07-16 13:36:17 +04:00
continue ;
}
}
return ret ;
}
2004-07-16 12:56:11 +04:00
/**************************/
/* srvsvc_NetDisk */
/**************************/
2003-11-19 12:15:46 +03:00
static BOOL test_NetDiskEnum ( struct dcerpc_pipe * p ,
TALLOC_CTX * mem_ctx )
{
NTSTATUS status ;
struct srvsvc_NetDiskEnum r ;
2004-05-25 20:24:13 +04:00
uint32_t levels [ ] = { 0 } ;
2003-11-19 12:15:46 +03:00
int i ;
BOOL ret = True ;
2004-05-25 20:24:13 +04:00
uint32_t resume_handle = 0 ;
2003-11-19 12:15:46 +03:00
2006-03-31 17:41:49 +04:00
ZERO_STRUCT ( r . in ) ;
2003-12-15 13:21:09 +03:00
r . in . server_unc = NULL ;
r . in . resume_handle = & resume_handle ;
2003-11-19 12:15:46 +03:00
for ( i = 0 ; i < ARRAY_SIZE ( levels ) ; i + + ) {
ZERO_STRUCT ( r . out ) ;
2004-07-16 12:56:11 +04:00
r . in . level = levels [ i ] ;
2006-09-11 05:22:28 +04:00
d_printf ( " testing NetDiskEnum level %u \n " , r . in . level ) ;
2003-11-19 12:15:46 +03:00
status = dcerpc_srvsvc_NetDiskEnum ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
NDR_PRINT_OUT_DEBUG ( srvsvc_NetDiskEnum , & r ) ;
2006-09-11 05:22:28 +04:00
d_printf ( " NetDiskEnum level %u failed - %s \n " , r . in . level , nt_errstr ( status ) ) ;
2003-11-19 12:15:46 +03:00
ret = False ;
2004-07-16 12:56:11 +04:00
continue ;
}
if ( ! W_ERROR_IS_OK ( r . out . result ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetDiskEnum level %u failed - %s \n " , r . in . level , win_errstr ( r . out . result ) ) ;
2004-07-16 12:56:11 +04:00
continue ;
2003-11-19 12:15:46 +03:00
}
}
2004-07-16 12:56:11 +04:00
return ret ;
2003-11-19 12:15:46 +03:00
}
2004-07-16 12:56:11 +04:00
/**************************/
/* srvsvc_NetTransport */
/**************************/
2003-11-19 12:15:46 +03:00
static BOOL test_NetTransportEnum ( struct dcerpc_pipe * p ,
TALLOC_CTX * mem_ctx )
{
NTSTATUS status ;
struct srvsvc_NetTransportEnum r ;
struct srvsvc_NetTransportCtr0 c0 ;
2004-05-25 20:24:13 +04:00
uint32_t levels [ ] = { 0 , 1 } ;
2003-11-19 12:15:46 +03:00
int i ;
BOOL ret = True ;
r . in . server_unc = talloc_asprintf ( mem_ctx , " \\ \\ %s " , dcerpc_server_name ( p ) ) ;
2005-12-25 00:32:52 +03:00
r . in . transports . ctr0 = & c0 ;
r . in . transports . ctr0 - > count = 0 ;
r . in . transports . ctr0 - > array = NULL ;
2004-05-25 20:24:13 +04:00
r . in . max_buffer = ( uint32_t ) - 1 ;
2003-11-19 12:15:46 +03:00
r . in . resume_handle = NULL ;
for ( i = 0 ; i < ARRAY_SIZE ( levels ) ; i + + ) {
2004-07-16 12:56:11 +04:00
ZERO_STRUCT ( r . out ) ;
2003-11-20 01:10:20 +03:00
r . in . level = levels [ i ] ;
2006-09-11 05:22:28 +04:00
d_printf ( " testing NetTransportEnum level %u \n " , r . in . level ) ;
2003-11-19 12:15:46 +03:00
status = dcerpc_srvsvc_NetTransportEnum ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetTransportEnum level %u failed - %s \n " , r . in . level , nt_errstr ( status ) ) ;
2003-11-19 12:15:46 +03:00
ret = False ;
2004-07-16 12:56:11 +04:00
continue ;
2003-11-19 12:15:46 +03:00
}
2004-07-16 12:56:11 +04:00
if ( ! W_ERROR_IS_OK ( r . out . result ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetTransportEnum level %u failed - %s \n " , r . in . level , win_errstr ( r . out . result ) ) ;
2004-07-16 12:56:11 +04:00
continue ;
2003-11-19 12:15:46 +03:00
}
}
2004-07-16 12:56:11 +04:00
return ret ;
2003-11-19 12:15:46 +03:00
}
2004-07-19 10:14:00 +04:00
/**************************/
/* srvsvc_NetRemoteTOD */
/**************************/
static BOOL test_NetRemoteTOD ( struct dcerpc_pipe * p ,
TALLOC_CTX * mem_ctx )
{
NTSTATUS status ;
struct srvsvc_NetRemoteTOD r ;
BOOL ret = True ;
r . in . server_unc = talloc_asprintf ( mem_ctx , " \\ \\ %s " , dcerpc_server_name ( p ) ) ;
ZERO_STRUCT ( r . out ) ;
2006-09-11 05:22:28 +04:00
d_printf ( " testing NetRemoteTOD \n " ) ;
2004-07-19 10:14:00 +04:00
status = dcerpc_srvsvc_NetRemoteTOD ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetRemoteTOD failed - %s \n " , nt_errstr ( status ) ) ;
2004-07-19 10:14:00 +04:00
ret = False ;
}
if ( ! W_ERROR_IS_OK ( r . out . result ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetRemoteTOD failed - %s \n " , win_errstr ( r . out . result ) ) ;
2004-07-19 10:14:00 +04:00
}
return ret ;
}
2006-09-11 01:51:46 +04:00
/**************************/
/* srvsvc_NetName */
/**************************/
static BOOL test_NetNameValidate ( struct dcerpc_pipe * p ,
TALLOC_CTX * mem_ctx )
{
NTSTATUS status ;
struct srvsvc_NetNameValidate r ;
char * invalidc ;
int i , n ;
r . in . server_unc = talloc_asprintf ( mem_ctx , " \\ \\ %s " , dcerpc_server_name ( p ) ) ;
r . in . flags = 0x0 ;
2006-09-11 05:22:28 +04:00
d_printf ( " testing NetNameValidate \n " ) ;
2006-09-11 01:51:46 +04:00
/* valid path types only between 1 and 13 */
for ( i = 1 ; i < 14 ; i + + ) {
2006-09-11 02:06:49 +04:00
again :
2006-09-11 01:51:46 +04:00
/* Find maximum length accepted by this type */
2006-09-11 04:24:21 +04:00
ZERO_STRUCT ( r . out ) ;
2006-09-11 01:51:46 +04:00
r . in . name_type = i ;
r . in . name = talloc_strdup ( mem_ctx , " A " ) ;
n = 0 ;
while ( W_ERROR_IS_OK ( r . out . result ) ) {
status = dcerpc_srvsvc_NetNameValidate ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetNameValidate failed while checking maximum size (%s) \n " ,
2006-09-11 01:51:46 +04:00
nt_errstr ( status ) ) ;
break ;
}
r . in . name = talloc_append_string ( mem_ctx , r . in . name , " A " ) ;
if ( ! r . in . name ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetNameValidate: Out of memory! \n " ) ;
2006-09-11 01:51:46 +04:00
return False ;
}
n + + ;
}
talloc_free ( r . in . name ) ;
2006-09-11 05:22:28 +04:00
d_printf ( " Maximum length for type %2d, flags %08x: %d \n " , i , r . in . flags , n ) ;
2006-09-11 01:51:46 +04:00
/* find invalid chars for this type check only ASCII between 0x20 and 0x7e */
2006-09-11 04:24:21 +04:00
invalidc = talloc_strdup ( mem_ctx , " " ) ;
2006-09-11 01:51:46 +04:00
for ( n = 0x20 ; n < 0x7e ; n + + ) {
r . in . name = talloc_asprintf ( mem_ctx , " %c " , ( char ) n ) ;
status = dcerpc_srvsvc_NetNameValidate ( p , mem_ctx , & r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2006-09-11 05:22:28 +04:00
d_printf ( " NetNameValidate failed while checking valid chars (%s) \n " ,
2006-09-11 01:51:46 +04:00
nt_errstr ( status ) ) ;
break ;
}
if ( ! W_ERROR_IS_OK ( r . out . result ) ) {
invalidc = talloc_asprintf_append ( invalidc , " %c " , ( char ) n ) ;
}
talloc_free ( r . in . name ) ;
}
2006-09-11 05:22:28 +04:00
d_printf ( " Invalid chars for type %2d, flags %08x: \" %s \" \n " , i , r . in . flags , invalidc ) ;
2006-09-11 02:06:49 +04:00
/* only two values are accepted for flags: 0x0 and 0x80000000 */
if ( r . in . flags = = 0x0 ) {
r . in . flags = 0x80000000 ;
goto again ;
}
r . in . flags = 0x0 ;
2006-09-11 01:51:46 +04:00
}
return True ;
}
2006-03-25 19:01:28 +03:00
BOOL torture_rpc_srvsvc ( struct torture_context * torture )
2003-11-19 12:15:46 +03:00
{
NTSTATUS status ;
struct dcerpc_pipe * p ;
TALLOC_CTX * mem_ctx ;
BOOL ret = True ;
2006-03-31 17:41:49 +04:00
const char * binding = lp_parm_string ( - 1 , " torture " , " binding " ) ;
struct cli_credentials * anon_credentials ;
2003-11-19 12:15:46 +03:00
mem_ctx = talloc_init ( " torture_rpc_srvsvc " ) ;
2005-12-27 17:28:01 +03:00
status = torture_rpc_connection ( mem_ctx , & p , & dcerpc_table_srvsvc ) ;
2003-11-19 12:15:46 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2005-03-22 11:00:45 +03:00
talloc_free ( mem_ctx ) ;
2003-11-19 12:15:46 +03:00
return False ;
}
2006-03-31 17:41:49 +04:00
ret & = test_NetCharDevEnum ( p , mem_ctx ) ;
ret & = test_NetCharDevQEnum ( p , mem_ctx ) ;
ret & = test_NetConnEnum ( p , mem_ctx ) ;
ret & = test_NetFileEnum ( p , mem_ctx ) ;
ret & = test_NetSessEnum ( p , mem_ctx ) ;
ret & = test_NetShareEnumAll ( p , mem_ctx , True ) ;
ret & = test_NetSrvGetInfo ( p , mem_ctx ) ;
ret & = test_NetDiskEnum ( p , mem_ctx ) ;
ret & = test_NetTransportEnum ( p , mem_ctx ) ;
ret & = test_NetRemoteTOD ( p , mem_ctx ) ;
ret & = test_NetShareEnum ( p , mem_ctx , True ) ;
ret & = test_NetShareGetInfo ( p , mem_ctx , " ADMIN$ " , True ) ;
2006-09-11 01:51:46 +04:00
ret & = test_NetNameValidate ( p , mem_ctx ) ;
2006-03-31 17:41:49 +04:00
status = torture_rpc_connection ( mem_ctx , & p , & dcerpc_table_srvsvc ) ;
2004-07-16 13:36:17 +04:00
2006-03-31 17:41:49 +04:00
if ( ! binding ) {
2006-09-11 05:22:28 +04:00
d_printf ( " You must specify a ncacn binding string \n " ) ;
2006-03-31 17:41:49 +04:00
return False ;
2003-11-19 12:15:46 +03:00
}
2003-12-15 13:21:09 +03:00
2006-03-31 17:41:49 +04:00
anon_credentials = cli_credentials_init ( mem_ctx ) ;
cli_credentials_set_conf ( anon_credentials ) ;
cli_credentials_set_anonymous ( anon_credentials ) ;
2003-12-15 12:18:53 +03:00
2006-03-31 17:41:49 +04:00
status = dcerpc_pipe_connect ( mem_ctx ,
& p , binding , & dcerpc_table_srvsvc ,
anon_credentials , NULL ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
talloc_free ( mem_ctx ) ;
return False ;
2004-07-19 10:14:00 +04:00
}
2006-03-31 17:41:49 +04:00
ret & = test_NetShareEnumAll ( p , mem_ctx , False ) ;
ret & = test_NetShareEnum ( p , mem_ctx , False ) ;
ret & = test_NetShareGetInfo ( p , mem_ctx , " ADMIN$ " , False ) ;
2003-11-19 12:15:46 +03:00
2005-01-27 10:08:20 +03:00
talloc_free ( mem_ctx ) ;
2003-11-22 11:11:32 +03:00
2003-11-19 12:15:46 +03:00
return ret ;
}