mirror of
https://github.com/samba-team/samba.git
synced 2025-01-27 14:04:05 +03:00
86a604429e
* added a NDR validator. The way it works is that when the DCERPC_DEBUG_VALIDATE_* flags are set the dcerpc system will perform NDR buffer validation. On sending a request the packet is first marshalled, then unmarahslled, then marshalled again, and it is confirmed that the two marshalling results are idential. This ensures that our pull and push routines are absolutely in sync, so that we can be very confident that if a routine works in the client then the corresponding routine must work on the server side. A similar validation is performed on all replies. * a result of this change is that pidl is fussier about the [ref] tag. You can only use it on pointers (which is the only place it makes sense) * fixed a basic alignment bug in the push side of the NDR code * added server side pull/push support. Our dcerpc system is now fully ready to be used on the server side. * fixed the relative offset pointer list. It must be traversed in reverse order on push * added automatic value setting for the size parameter in outgoing SdBuf structures. * expanded the ndr debugging code to always give a message on any failure * fixed the subcontext push code * fixed some memory leaks in smbtorture RPC tests (This used to be commit 8ecf720206a2eef3f8ea7cbdb1f460664a5dba9a)
262 lines
5.0 KiB
C
262 lines
5.0 KiB
C
/*
|
|
Unix SMB/CIFS implementation.
|
|
test suite for echo rpc operations
|
|
|
|
Copyright (C) Andrew Tridgell 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 02139, USA.
|
|
*/
|
|
|
|
#include "includes.h"
|
|
|
|
|
|
/*
|
|
test the AddOne interface
|
|
*/
|
|
static BOOL test_addone(struct dcerpc_pipe *p, TALLOC_CTX *mem_ctx)
|
|
{
|
|
int i;
|
|
NTSTATUS status;
|
|
|
|
printf("\nTesting AddOne\n");
|
|
|
|
for (i=0;i<10;i++) {
|
|
uint32 n = i;
|
|
struct echo_AddOne r;
|
|
r.in.v = &n;
|
|
r.out.v = &n;
|
|
status = dcerpc_echo_AddOne(p, mem_ctx, &r);
|
|
if (!NT_STATUS_IS_OK(status)) {
|
|
printf("AddOne(%d) failed - %s\n", i, nt_errstr(status));
|
|
return False;
|
|
}
|
|
printf("%d + 1 = %u\n", i, n);
|
|
}
|
|
|
|
return True;
|
|
}
|
|
|
|
/*
|
|
test the EchoData interface
|
|
*/
|
|
static BOOL test_echodata(struct dcerpc_pipe *p, TALLOC_CTX *mem_ctx)
|
|
{
|
|
int i;
|
|
NTSTATUS status;
|
|
char *data_in, *data_out;
|
|
int len = 1 + (random() % 5000);
|
|
struct echo_EchoData r;
|
|
|
|
printf("\nTesting EchoData\n");
|
|
|
|
data_in = talloc(mem_ctx, len);
|
|
data_out = talloc(mem_ctx, len);
|
|
for (i=0;i<len;i++) {
|
|
data_in[i] = i;
|
|
}
|
|
|
|
r.in.len = len;
|
|
r.in.in_data = data_in;
|
|
|
|
status = dcerpc_echo_EchoData(p, mem_ctx, &r);
|
|
if (!NT_STATUS_IS_OK(status)) {
|
|
printf("EchoData(%d) failed - %s\n", len, nt_errstr(status));
|
|
return False;
|
|
}
|
|
|
|
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");
|
|
dump_data(0, data_in+i, MIN(len-i, 16));
|
|
printf("out:\n");
|
|
dump_data(0, data_out+i, MIN(len-1, 16));
|
|
return False;
|
|
}
|
|
}
|
|
|
|
|
|
return True;
|
|
}
|
|
|
|
|
|
/*
|
|
test the SourceData interface
|
|
*/
|
|
static BOOL test_sourcedata(struct dcerpc_pipe *p, TALLOC_CTX *mem_ctx)
|
|
{
|
|
int i;
|
|
NTSTATUS status;
|
|
int len = 200000 + (random() % 5000);
|
|
char *data_out;
|
|
struct echo_SourceData r;
|
|
|
|
printf("\nTesting SourceData\n");
|
|
|
|
data_out = talloc(mem_ctx, len);
|
|
|
|
r.in.len = len;
|
|
r.out.data = data_out;
|
|
|
|
status = dcerpc_echo_SourceData(p, mem_ctx, &r);
|
|
if (!NT_STATUS_IS_OK(status)) {
|
|
printf("SourceData(%d) failed - %s\n", len, nt_errstr(status));
|
|
return False;
|
|
}
|
|
|
|
for (i=0;i<len;i++) {
|
|
unsigned char *v = (unsigned char *)data_out;
|
|
if (v[i] != (i & 0xFF)) {
|
|
printf("bad data 0x%x at %d\n", (unsigned char)data_out[i], i);
|
|
return False;
|
|
}
|
|
}
|
|
|
|
return True;
|
|
}
|
|
|
|
/*
|
|
test the SinkData interface
|
|
*/
|
|
static BOOL test_sinkdata(struct dcerpc_pipe *p, TALLOC_CTX *mem_ctx)
|
|
{
|
|
int i;
|
|
NTSTATUS status;
|
|
char *data_in;
|
|
int len = 200000 + (random() % 5000);
|
|
struct echo_SinkData r;
|
|
|
|
printf("\nTesting SinkData\n");
|
|
|
|
data_in = talloc(mem_ctx, len);
|
|
for (i=0;i<len;i++) {
|
|
data_in[i] = i+1;
|
|
}
|
|
|
|
r.in.len = len;
|
|
r.in.data = data_in;
|
|
|
|
status = dcerpc_echo_SinkData(p, mem_ctx, &r);
|
|
if (!NT_STATUS_IS_OK(status)) {
|
|
printf("SinkData(%d) failed - %s\n", len, nt_errstr(status));
|
|
return False;
|
|
}
|
|
|
|
printf("sunk %d bytes\n", len);
|
|
|
|
return True;
|
|
}
|
|
|
|
|
|
/*
|
|
test the testcall interface
|
|
*/
|
|
static BOOL test_testcall(struct dcerpc_pipe *p, TALLOC_CTX *mem_ctx)
|
|
{
|
|
NTSTATUS status;
|
|
struct TestCall r;
|
|
|
|
r.in.s1 = "input string";
|
|
|
|
printf("\nTesting TestCall\n");
|
|
status = dcerpc_TestCall(p, mem_ctx, &r);
|
|
if (!NT_STATUS_IS_OK(status)) {
|
|
printf("TestCall failed - %s\n", nt_errstr(status));
|
|
return False;
|
|
}
|
|
|
|
return True;
|
|
}
|
|
|
|
/*
|
|
test the testcall interface
|
|
*/
|
|
static BOOL test_testcall2(struct dcerpc_pipe *p, TALLOC_CTX *mem_ctx)
|
|
{
|
|
NTSTATUS status;
|
|
struct TestCall2 r;
|
|
int i;
|
|
BOOL ret = True;
|
|
|
|
for (i=1;i<=7;i++) {
|
|
r.in.level = i;
|
|
|
|
printf("\nTesting TestCall2 level %d\n", i);
|
|
status = dcerpc_TestCall2(p, mem_ctx, &r);
|
|
if (!NT_STATUS_IS_OK(status)) {
|
|
printf("TestCall2 failed - %s\n", nt_errstr(status));
|
|
ret = False;
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
BOOL torture_rpc_echo(int dummy)
|
|
{
|
|
NTSTATUS status;
|
|
struct dcerpc_pipe *p;
|
|
TALLOC_CTX *mem_ctx;
|
|
BOOL ret = True;
|
|
|
|
mem_ctx = talloc_init("torture_rpc_echo");
|
|
|
|
status = torture_rpc_connection(&p,
|
|
DCERPC_RPCECHO_NAME,
|
|
DCERPC_RPCECHO_UUID,
|
|
DCERPC_RPCECHO_VERSION);
|
|
if (!NT_STATUS_IS_OK(status)) {
|
|
return False;
|
|
}
|
|
|
|
#if 1
|
|
if (!test_addone(p, mem_ctx)) {
|
|
ret = False;
|
|
}
|
|
|
|
if (!test_echodata(p, mem_ctx)) {
|
|
ret = False;
|
|
}
|
|
|
|
if (!test_sourcedata(p, mem_ctx)) {
|
|
ret = False;
|
|
}
|
|
|
|
if (!test_sinkdata(p, mem_ctx)) {
|
|
ret = False;
|
|
}
|
|
#endif
|
|
|
|
p->flags |= DCERPC_DEBUG_PRINT_BOTH;
|
|
|
|
if (!test_testcall(p, mem_ctx)) {
|
|
ret = False;
|
|
}
|
|
|
|
if (!test_testcall2(p, mem_ctx)) {
|
|
ret = False;
|
|
}
|
|
|
|
printf("\n");
|
|
|
|
talloc_destroy(mem_ctx);
|
|
|
|
torture_rpc_close(p);
|
|
return ret;
|
|
}
|