summaryrefslogtreecommitdiff
path: root/source4
diff options
context:
space:
mode:
Diffstat (limited to 'source4')
-rw-r--r--source4/libcli/raw/rawdcerpc.c222
1 files changed, 222 insertions, 0 deletions
diff --git a/source4/libcli/raw/rawdcerpc.c b/source4/libcli/raw/rawdcerpc.c
new file mode 100644
index 0000000000..11bd0057ce
--- /dev/null
+++ b/source4/libcli/raw/rawdcerpc.c
@@ -0,0 +1,222 @@
+/*
+ Unix SMB/CIFS implementation.
+ raw dcerpc operations
+
+ Copyright (C) Tim Potter, 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"
+
+static int put_uuid(char *data, int offset, struct dcerpc_uuid *uuid)
+{
+ int i;
+
+ SIVAL(data, offset, uuid->time_low); offset += 4;
+ SSVAL(data, offset, uuid->time_mid); offset += 2;
+ SSVAL(data, offset, uuid->time_hi_and_version); offset += 2;
+ for (i = 0; i < 8; i++)
+ SCVAL(data, offset + i, uuid->remaining[i]);
+ offset += 8;
+
+ return offset;
+}
+
+DATA_BLOB dcerpc_raw_bind_setup(struct dcerpc_bind *parms)
+{
+ int i, offset, size;
+ char *data;
+
+ /* Allocate storage for bind request */
+
+ size = 28;
+ for (i = 0; i < parms->in.num_contexts; i++) {
+ size += 24; /* as header + uuid */
+ size += 20 * parms->in.ctx_list[i].num_ts; /* xfer syntaxes */
+ }
+ size += parms->in.auth_verifier.length;
+
+ data = smb_xmalloc(size);
+ memset(data, 0, size);
+
+ parms->in.hdr.frag_len = size;
+
+ /* Create bind request */
+
+ SCVAL(data, 0, parms->in.hdr.rpc_vers);
+ SCVAL(data, 1, parms->in.hdr.rpc_vers_minor);
+ SCVAL(data, 2, parms->in.hdr.ptype);
+ SCVAL(data, 3, parms->in.hdr.pfc_flags);
+ for (i = 0; i < 4; i++)
+ SCVAL(data, 4 + i, parms->in.hdr.drep[i]);
+ SSVAL(data, 8, parms->in.hdr.frag_len);
+ SSVAL(data, 10, parms->in.auth_verifier.length);
+ SIVAL(data, 12, parms->in.hdr.call_id);
+
+ SSVAL(data, 16, parms->in.max_xmit_frag);
+ SSVAL(data, 18, parms->in.max_recv_frag);
+ SIVAL(data, 20, parms->in.assoc_group_id);
+ SIVAL(data, 24, parms->in.num_contexts);
+
+ offset = 28;
+ for (i = 0; i < parms->in.num_contexts; i++) {
+ struct p_ctx_list *ctx = &parms->in.ctx_list[i];
+ int j;
+
+ SSVAL(data, offset, ctx->cont_id); offset += 2;
+ SSVAL(data, offset, ctx->num_ts); offset += 2;
+ offset = put_uuid(data, offset, &ctx->as->if_uuid);
+ SIVAL(data, offset, ctx->as->if_version); offset += 4;
+ for (j = 0; j < ctx->num_ts; j++) {
+ offset = put_uuid(data, offset, &ctx->ts[i]->if_uuid);
+ SIVAL(data, offset, ctx->ts[i]->if_version);
+ offset += 4;
+ }
+ }
+
+ if (parms->in.auth_verifier.length)
+ memcpy(&data[offset], parms->in.auth_verifier.data,
+ parms->in.auth_verifier.length);
+
+ return data_blob(data, size);
+}
+
+NTSTATUS dcerpc_raw_bind_send(struct cli_dcerpc_pipe *p,
+ struct dcerpc_bind *parms)
+{
+ struct smb_trans2 trans;
+ DATA_BLOB blob;
+ NTSTATUS result;
+ uint16 setup[2];
+
+ blob = dcerpc_raw_bind_setup(parms);
+
+ ZERO_STRUCT(trans);
+
+ trans.in.max_data = blob.length;
+ trans.in.setup_count = 2;
+ trans.in.setup = setup;
+ trans.in.trans_name = "\\PIPE\\";
+
+ setup[0] = TRANSACT_DCERPCCMD;
+ setup[1] = p->fnum;
+
+ trans.in.data = blob;
+
+ result = smb_raw_trans(p->tree, p->mem_ctx, &trans);
+
+ data_blob_free(&blob);
+
+ return result;
+}
+
+NTSTATUS dcerpc_raw_bind_recv(struct cli_dcerpc_pipe *p,
+ struct dcerpc_bind *parms)
+{
+ return NT_STATUS_UNSUCCESSFUL;
+}
+
+NTSTATUS dcerpc_raw_bind(struct cli_dcerpc_pipe *p, struct dcerpc_bind *parms)
+{
+ NTSTATUS result;
+
+ result = dcerpc_raw_bind_send(p, parms);
+ if (NT_STATUS_IS_ERR(result))
+ return result;
+ return dcerpc_raw_bind_recv(p, parms);
+}
+
+DATA_BLOB dcerpc_raw_request_setup(struct dcerpc_request *parms)
+{
+ int size, i;
+ char *data;
+
+ /* Allocate storage for request */
+
+ size = 24 + parms->in.stub_data.length;
+
+ data = smb_xmalloc(size);
+ memset(data, 0, size);
+
+ parms->in.hdr.frag_len = size;
+ parms->in.alloc_hint = parms->in.stub_data.length;
+
+ SCVAL(data, 0, parms->in.hdr.rpc_vers);
+ SCVAL(data, 1, parms->in.hdr.rpc_vers_minor);
+ SCVAL(data, 2, parms->in.hdr.ptype);
+ SCVAL(data, 3, parms->in.hdr.pfc_flags);
+ for (i = 0; i < 4; i++)
+ SCVAL(data, 4 + i, parms->in.hdr.drep[i]);
+ SSVAL(data, 8, parms->in.hdr.frag_len);
+ SSVAL(data, 10, parms->in.auth_verifier.length);
+ SIVAL(data, 12, parms->in.hdr.call_id);
+
+ SIVAL(data, 16, parms->in.alloc_hint);
+ SSVAL(data, 20, parms->in.cont_id);
+ SSVAL(data, 22, parms->in.opnum);
+
+ if (parms->in.stub_data.length)
+ memcpy(&data[24], parms->in.stub_data.data,
+ parms->in.stub_data.length);
+
+ return data_blob(data, size);
+}
+
+NTSTATUS dcerpc_raw_request_send(struct cli_dcerpc_pipe *p,
+ struct dcerpc_request *parms)
+{
+ struct smb_trans2 trans;
+ DATA_BLOB blob;
+ NTSTATUS result;
+ uint16 setup[2];
+
+ blob = dcerpc_raw_request_setup(parms);
+
+ ZERO_STRUCT(trans);
+
+ trans.in.max_data = blob.length;
+ trans.in.setup_count = 2;
+ trans.in.setup = setup;
+ trans.in.trans_name = "\\PIPE\\";
+
+ setup[0] = TRANSACT_DCERPCCMD;
+ setup[1] = p->fnum;
+
+ trans.in.data = blob;
+
+ result = smb_raw_trans(p->tree, p->mem_ctx, &trans);
+
+ data_blob_free(&blob);
+
+ return result;
+}
+
+NTSTATUS dcerpc_raw_request_recv(struct cli_dcerpc_pipe *p,
+ struct dcerpc_request *parms)
+{
+ return NT_STATUS_UNSUCCESSFUL;
+}
+
+NTSTATUS dcerpc_raw_request(struct cli_dcerpc_pipe *p,
+ struct dcerpc_request *parms)
+{
+ NTSTATUS result;
+
+ result = dcerpc_raw_request_send(p, parms);
+ if (NT_STATUS_IS_ERR(result))
+ return result;
+ return dcerpc_raw_request_recv(p, parms);
+}