summaryrefslogtreecommitdiff
path: root/source4/librpc/ndr
diff options
context:
space:
mode:
Diffstat (limited to 'source4/librpc/ndr')
-rw-r--r--source4/librpc/ndr/libndr.h5
-rw-r--r--source4/librpc/ndr/ndr_compression.c177
2 files changed, 182 insertions, 0 deletions
diff --git a/source4/librpc/ndr/libndr.h b/source4/librpc/ndr/libndr.h
index 4df3e7f248..3cc25b0ef2 100644
--- a/source4/librpc/ndr/libndr.h
+++ b/source4/librpc/ndr/libndr.h
@@ -152,6 +152,7 @@ enum ndr_err_code {
NDR_ERR_CHARCNV,
NDR_ERR_LENGTH,
NDR_ERR_SUBCONTEXT,
+ NDR_ERR_COMPRESSION,
NDR_ERR_STRING,
NDR_ERR_VALIDATE,
NDR_ERR_BUFSIZE,
@@ -160,6 +161,10 @@ enum ndr_err_code {
NDR_ERR_TOKEN
};
+enum ndr_compression_alg {
+ NDR_COMPRESSION_ZLIB
+};
+
/*
flags passed to control parse flow
*/
diff --git a/source4/librpc/ndr/ndr_compression.c b/source4/librpc/ndr/ndr_compression.c
new file mode 100644
index 0000000000..d4f1a2a927
--- /dev/null
+++ b/source4/librpc/ndr/ndr_compression.c
@@ -0,0 +1,177 @@
+/*
+ Unix SMB/CIFS implementation.
+
+ libndr compression support
+
+ Copyright (C) Stefan Metzmacher 2005
+
+ 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"
+
+#ifdef HAVE_ZLIB
+#include <zlib.h>
+
+static NTSTATUS ndr_pull_compression_zlib(struct ndr_pull *subndr,
+ struct ndr_pull *comndr,
+ ssize_t decompressed_len)
+{
+ DATA_BLOB inbuf;
+ DATA_BLOB outbuf = data_blob_talloc(comndr, NULL, decompressed_len);
+ uint32_t outbuf_len = outbuf.length;
+ struct z_stream_s zs;
+ int ret;
+
+ ZERO_STRUCT(zs);
+
+ if (subndr->data_size < 10) {
+ return ndr_pull_error(subndr, NDR_ERR_COMPRESSION, "Bad ZLIB compressed header (PULL) subcontext size %d",
+ subndr->data_size);
+ }
+
+ inbuf.data = subndr->data+10;
+ inbuf.length = subndr->data_size-10;
+
+ zs.avail_in = inbuf.length;
+ zs.next_in = inbuf.data;
+ zs.next_out = outbuf.data;
+ zs.avail_out = outbuf.length;
+
+ ret = inflateInit2(&zs, 15);
+ if (ret != Z_OK) {
+ return ndr_pull_error(subndr, NDR_ERR_COMPRESSION, "Bad ZLIB (PULL) inflateInit2 error %d",
+ ret);
+ }
+
+ while(1) {
+ ret = inflate(&zs, Z_SYNC_FLUSH);
+ if (ret == Z_STREAM_END) {
+
+ DEBUG(0,("inbuf.length: %d avail_in: %d, avail_out: %d\n", inbuf.length, zs.avail_in, zs.avail_out));
+ break;
+ }
+ if (ret != Z_OK) {
+ return ndr_pull_error(subndr, NDR_ERR_COMPRESSION, "Bad ZLIB (PULL) inflate error %d",
+ ret);
+ }
+ }
+
+ inflateEnd(&zs);
+
+ /* TODO: check if the decompressed_len == outbuf_len */
+ outbuf.length = outbuf_len - zs.avail_out;
+
+ if (outbuf.length < 16) {
+ return ndr_pull_error(subndr, NDR_ERR_COMPRESSION, "Bad ZLIB uncompressed header (PULL) uncompressed size %d",
+ outbuf.length);
+ }
+
+ outbuf.data += 16;
+ outbuf.length -= 16;
+
+ /* TODO: really decompress the data here */
+ *comndr = *subndr;
+ comndr->data = outbuf.data;
+ comndr->data_size = outbuf.length;
+ comndr->offset = 0;
+
+ return NT_STATUS_OK;
+}
+
+static NTSTATUS ndr_push_compression_zlib(struct ndr_push *subndr,
+ struct ndr_push *comndr)
+{
+ DATA_BLOB inbuf;
+ DATA_BLOB outbuf = data_blob_talloc(comndr, NULL, comndr->offset + 10);
+ struct z_stream_s zs;
+ int ret;
+
+ ZERO_STRUCT(zs);
+
+ inbuf = ndr_push_blob(comndr);
+
+ zs.avail_in = inbuf.length;
+ zs.next_in = inbuf.data;
+ zs.next_out = outbuf.data+10;
+ zs.avail_out = outbuf.length-10;
+
+ ret = deflateInit(&zs, Z_DEFAULT_COMPRESSION);
+ if (ret != Z_OK) {
+ return ndr_push_error(subndr, NDR_ERR_COMPRESSION, "Bad ZLIB (PUSH) deflateInit2 error %d",
+ ret);
+ }
+
+ ret = deflate(&zs, Z_SYNC_FLUSH);
+
+ if (ret != Z_OK && ret != Z_STREAM_END) {
+ return ndr_push_error(subndr, NDR_ERR_COMPRESSION, "Bad ZLIB (PULL) deflate error %d",
+ ret);
+ }
+
+ deflateEnd(&zs);
+
+ /* TODO: push the header here */
+
+
+ NDR_CHECK(ndr_push_bytes(subndr, outbuf.data, outbuf.length));
+
+ return NT_STATUS_OK;
+}
+#endif
+
+/*
+ handle compressed subcontext buffers, which in midl land are user-marshalled, but
+ we use magic in pidl to make them easier to cope with
+*/
+NTSTATUS ndr_pull_compression(struct ndr_pull *subndr,
+ struct ndr_pull *comndr,
+ enum ndr_compression_alg compression_alg,
+ ssize_t decompressed_len)
+{
+ comndr->flags = subndr->flags;
+
+ switch (compression_alg) {
+#ifdef HAVE_ZLIB
+ case NDR_COMPRESSION_ZLIB:
+ return ndr_pull_compression_zlib(subndr, comndr, decompressed_len);
+#endif
+ default:
+ return ndr_pull_error(subndr, NDR_ERR_COMPRESSION, "Bad compression algorithm %d (PULL)",
+ compression_alg);
+ }
+ return NT_STATUS_OK;
+}
+
+/*
+ push a compressed subcontext
+*/
+NTSTATUS ndr_push_compression(struct ndr_push *subndr,
+ struct ndr_push *comndr,
+ enum ndr_compression_alg compression_alg)
+{
+ comndr->flags = subndr->flags;
+
+ switch (compression_alg) {
+#ifdef HAVE_ZLIB
+ case NDR_COMPRESSION_ZLIB:
+ return ndr_push_compression_zlib(subndr, comndr);
+#endif
+ default:
+ return ndr_push_error(subndr, NDR_ERR_COMPRESSION, "Bad compression algorithm %d (PUSH)",
+ compression_alg);
+ }
+ return NT_STATUS_OK;
+}