Blame SOURCES/dovecot-2.3.16-keeplzma.patch

0ba288
diff -up dovecot-2.3.16/configure.ac.keeplzma dovecot-2.3.16/configure.ac
0ba288
--- dovecot-2.3.16/configure.ac.keeplzma	2021-08-06 11:25:51.000000000 +0200
0ba288
+++ dovecot-2.3.16/configure.ac	2022-02-28 13:58:02.337149927 +0100
0ba288
@@ -173,7 +173,7 @@ AS_HELP_STRING([--with-bzlib], [Build wi
0ba288
   want_bzlib=auto)
0ba288
 
0ba288
 AC_ARG_WITH(lzma,
0ba288
-AS_HELP_STRING([--with-lzma], [Build with LZMA decompression support (auto)]),
0ba288
+AS_HELP_STRING([--with-lzma], [Build with LZMA compression support (auto)]),
0ba288
   TEST_WITH(lzma, $withval),
0ba288
   want_lzma=auto)
0ba288
 
0ba288
diff -up dovecot-2.3.16/run-test-valgrind.supp.keeplzma dovecot-2.3.16/run-test-valgrind.supp
0ba288
--- dovecot-2.3.16/run-test-valgrind.supp.keeplzma	2021-08-06 11:25:51.000000000 +0200
0ba288
+++ dovecot-2.3.16/run-test-valgrind.supp	2022-02-28 13:58:02.337149927 +0100
0ba288
@@ -5,6 +5,17 @@
0ba288
    obj:*/bash
0ba288
 }
0ba288
 {
0ba288
+   <liblzma>
0ba288
+   Memcheck:Cond
0ba288
+   obj:/lib/x86_64-linux-gnu/liblzma.so.5.*
0ba288
+   obj:/lib/x86_64-linux-gnu/liblzma.so.5.*
0ba288
+   obj:/lib/x86_64-linux-gnu/liblzma.so.5.*
0ba288
+   obj:/lib/x86_64-linux-gnu/liblzma.so.5.*
0ba288
+   obj:/lib/x86_64-linux-gnu/liblzma.so.5.*
0ba288
+   fun:lzma_stream_encoder
0ba288
+   fun:lzma_easy_encoder
0ba288
+}
0ba288
+{
0ba288
    <openssl_centos6_i386_v1_0_1_compression_methods>
0ba288
    Memcheck:Leak
0ba288
    fun:malloc
0ba288
diff -up dovecot-2.3.16/src/lib-compression/compression.c.keeplzma dovecot-2.3.16/src/lib-compression/compression.c
0ba288
--- dovecot-2.3.16/src/lib-compression/compression.c.keeplzma	2021-08-06 11:25:51.000000000 +0200
0ba288
+++ dovecot-2.3.16/src/lib-compression/compression.c	2022-02-28 14:22:32.467944396 +0100
0ba288
@@ -25,6 +25,7 @@
0ba288
 #endif
0ba288
 #ifndef HAVE_LZMA
0ba288
 #  define i_stream_create_lzma NULL
0ba288
+#  define o_stream_create_lzma NULL
0ba288
 #endif
0ba288
 #ifndef HAVE_LZ4
0ba288
 #  define i_stream_create_lz4 NULL
0ba288
@@ -216,7 +217,7 @@ const struct compression_handler compres
0ba288
 		.ext = ".xz",
0ba288
 		.is_compressed = is_compressed_xz,
0ba288
 		.create_istream = i_stream_create_lzma,
0ba288
-		.create_ostream = NULL,
0ba288
+		.create_ostream = o_stream_create_lzma,
0ba288
 		.get_min_level = compression_get_min_level_unsupported,
0ba288
 		.get_default_level = compression_get_default_level_unsupported,
0ba288
 		.get_max_level = compression_get_max_level_unsupported,
0ba288
diff -up dovecot-2.3.16/src/lib-compression/Makefile.am.keeplzma dovecot-2.3.16/src/lib-compression/Makefile.am
0ba288
--- dovecot-2.3.16/src/lib-compression/Makefile.am.keeplzma	2021-08-06 11:25:51.000000000 +0200
0ba288
+++ dovecot-2.3.16/src/lib-compression/Makefile.am	2022-02-28 13:58:02.337149927 +0100
0ba288
@@ -13,6 +13,7 @@ libcompression_la_SOURCES = \
0ba288
 	istream-zlib.c \
0ba288
 	istream-bzlib.c \
0ba288
 	istream-zstd.c \
0ba288
+	ostream-lzma.c \
0ba288
 	ostream-lz4.c \
0ba288
 	ostream-zlib.c \
0ba288
 	ostream-bzlib.c \
0ba288
diff -up dovecot-2.3.16/src/lib-compression/ostream-lzma.c.keeplzma dovecot-2.3.16/src/lib-compression/ostream-lzma.c
0ba288
--- dovecot-2.3.16/src/lib-compression/ostream-lzma.c.keeplzma	2022-02-28 13:58:02.338149934 +0100
0ba288
+++ dovecot-2.3.16/src/lib-compression/ostream-lzma.c	2022-02-28 13:58:02.338149934 +0100
0ba288
@@ -0,0 +1,263 @@
0ba288
+/* Copyright (c) 2010-2018 Dovecot authors, see the included COPYING file */
0ba288
+
0ba288
+#include "lib.h"
0ba288
+
0ba288
+#ifdef HAVE_LZMA
0ba288
+
0ba288
+#include "ostream-private.h"
0ba288
+#include "ostream-zlib.h"
0ba288
+#include <lzma.h>
0ba288
+
0ba288
+#define CHUNK_SIZE (1024*64)
0ba288
+
0ba288
+struct lzma_ostream {
0ba288
+	struct ostream_private ostream;
0ba288
+	lzma_stream strm;
0ba288
+
0ba288
+	unsigned char outbuf[CHUNK_SIZE];
0ba288
+	unsigned int outbuf_offset, outbuf_used;
0ba288
+
0ba288
+	bool flushed:1;
0ba288
+};
0ba288
+
0ba288
+static void o_stream_lzma_close(struct iostream_private *stream,
0ba288
+				bool close_parent)
0ba288
+{
0ba288
+	struct lzma_ostream *zstream = (struct lzma_ostream *)stream;
0ba288
+	i_assert(zstream->ostream.finished ||
0ba288
+		 zstream->ostream.ostream.stream_errno != 0 ||
0ba288
+		 zstream->ostream.error_handling_disabled);
0ba288
+	lzma_end(&zstream->strm);
0ba288
+	if (close_parent)
0ba288
+		o_stream_close(zstream->ostream.parent);
0ba288
+}
0ba288
+
0ba288
+static int o_stream_zlib_send_outbuf(struct lzma_ostream *zstream)
0ba288
+{
0ba288
+	ssize_t ret;
0ba288
+	size_t size;
0ba288
+
0ba288
+	if (zstream->outbuf_used == 0)
0ba288
+		return 1;
0ba288
+
0ba288
+	size = zstream->outbuf_used - zstream->outbuf_offset;
0ba288
+	i_assert(size > 0);
0ba288
+	ret = o_stream_send(zstream->ostream.parent,
0ba288
+			    zstream->outbuf + zstream->outbuf_offset, size);
0ba288
+	if (ret < 0) {
0ba288
+		o_stream_copy_error_from_parent(&zstream->ostream);
0ba288
+		return -1;
0ba288
+	}
0ba288
+	if ((size_t)ret != size) {
0ba288
+		zstream->outbuf_offset += ret;
0ba288
+		return 0;
0ba288
+	}
0ba288
+	zstream->outbuf_offset = 0;
0ba288
+	zstream->outbuf_used = 0;
0ba288
+	return 1;
0ba288
+}
0ba288
+
0ba288
+static ssize_t
0ba288
+o_stream_lzma_send_chunk(struct lzma_ostream *zstream,
0ba288
+			  const void *data, size_t size)
0ba288
+{
0ba288
+	lzma_stream *zs = &zstream->strm;
0ba288
+	int ret;
0ba288
+
0ba288
+	i_assert(zstream->outbuf_used == 0);
0ba288
+
0ba288
+	zs->next_in = (void *)data;
0ba288
+	zs->avail_in = size;
0ba288
+	while (zs->avail_in > 0) {
0ba288
+		if (zs->avail_out == 0) {
0ba288
+			/* previous block was compressed. send it and start
0ba288
+			   compression for a new block. */
0ba288
+			zs->next_out = zstream->outbuf;
0ba288
+			zs->avail_out = sizeof(zstream->outbuf);
0ba288
+
0ba288
+			zstream->outbuf_used = sizeof(zstream->outbuf);
0ba288
+			if ((ret = o_stream_zlib_send_outbuf(zstream)) < 0)
0ba288
+				return -1;
0ba288
+			if (ret == 0) {
0ba288
+				/* parent stream's buffer full */
0ba288
+				break;
0ba288
+			}
0ba288
+		}
0ba288
+
0ba288
+		ret = lzma_code(zs, LZMA_RUN);
0ba288
+		switch (ret) {
0ba288
+		case LZMA_OK:
0ba288
+			break;
0ba288
+		case LZMA_MEM_ERROR:
0ba288
+			i_fatal_status(FATAL_OUTOFMEM,
0ba288
+				       "lzma.write(%s): Out of memory",
0ba288
+				       o_stream_get_name(&zstream->ostream.ostream));
0ba288
+		default:
0ba288
+			i_panic("lzma.write(%s) failed with unexpected code %d",
0ba288
+				o_stream_get_name(&zstream->ostream.ostream), ret);
0ba288
+		}
0ba288
+	}
0ba288
+	size -= zs->avail_in;
0ba288
+
0ba288
+	return size;
0ba288
+}
0ba288
+
0ba288
+static int o_stream_lzma_send_flush(struct lzma_ostream *zstream, bool final)
0ba288
+{
0ba288
+	lzma_stream *zs = &zstream->strm;
0ba288
+	size_t len;
0ba288
+	bool done = FALSE;
0ba288
+	int ret;
0ba288
+
0ba288
+	i_assert(zs->avail_in == 0);
0ba288
+
0ba288
+	if (zstream->flushed) {
0ba288
+		i_assert(zstream->outbuf_used == 0);
0ba288
+		return 1;
0ba288
+	}
0ba288
+
0ba288
+	if ((ret = o_stream_flush_parent_if_needed(&zstream->ostream)) <= 0)
0ba288
+		return ret;
0ba288
+	if ((ret = o_stream_zlib_send_outbuf(zstream)) <= 0)
0ba288
+		return ret;
0ba288
+
0ba288
+	if (!final)
0ba288
+		return 1;
0ba288
+
0ba288
+	i_assert(zstream->outbuf_used == 0);
0ba288
+	do {
0ba288
+		len = sizeof(zstream->outbuf) - zs->avail_out;
0ba288
+		if (len != 0) {
0ba288
+			zs->next_out = zstream->outbuf;
0ba288
+			zs->avail_out = sizeof(zstream->outbuf);
0ba288
+
0ba288
+			zstream->outbuf_used = len;
0ba288
+			if ((ret = o_stream_zlib_send_outbuf(zstream)) <= 0)
0ba288
+				return ret;
0ba288
+			if (done)
0ba288
+				break;
0ba288
+		}
0ba288
+		ret = lzma_code(zs, LZMA_FINISH);
0ba288
+		switch (ret) {
0ba288
+		case LZMA_OK:
0ba288
+			/* still unfinished - need to call lzma_code() again */
0ba288
+			break;
0ba288
+		case LZMA_STREAM_END:
0ba288
+			/* output is fully finished */
0ba288
+			done = TRUE;
0ba288
+			break;
0ba288
+		case LZMA_MEM_ERROR:
0ba288
+			i_fatal_status(FATAL_OUTOFMEM,
0ba288
+				       "lzma.write(%s): Out of memory",
0ba288
+				       o_stream_get_name(&zstream->ostream.ostream));
0ba288
+		default:
0ba288
+			i_panic("lzma.write(%s) flush failed with unexpected code %d",
0ba288
+				o_stream_get_name(&zstream->ostream.ostream), ret);
0ba288
+		}
0ba288
+	} while (zs->avail_out != sizeof(zstream->outbuf));
0ba288
+
0ba288
+	if (final)
0ba288
+		zstream->flushed = TRUE;
0ba288
+	i_assert(zstream->outbuf_used == 0);
0ba288
+	return 1;
0ba288
+}
0ba288
+
0ba288
+static int o_stream_lzma_flush(struct ostream_private *stream)
0ba288
+{
0ba288
+	struct lzma_ostream *zstream = (struct lzma_ostream *)stream;
0ba288
+	int ret;
0ba288
+
0ba288
+	if ((ret = o_stream_lzma_send_flush(zstream, stream->finished)) < 0)
0ba288
+		return -1;
0ba288
+	else if (ret > 0)
0ba288
+		return o_stream_flush_parent(stream);
0ba288
+	return ret;
0ba288
+}
0ba288
+
0ba288
+static size_t
0ba288
+o_stream_lzma_get_buffer_used_size(const struct ostream_private *stream)
0ba288
+{
0ba288
+	const struct lzma_ostream *zstream =
0ba288
+		(const struct lzma_ostream *)stream;
0ba288
+
0ba288
+	/* outbuf has already compressed data that we're trying to send to the
0ba288
+	   parent stream. We're not including lzma's internal compression
0ba288
+	   buffer size. */
0ba288
+	return (zstream->outbuf_used - zstream->outbuf_offset) +
0ba288
+		o_stream_get_buffer_used_size(stream->parent);
0ba288
+}
0ba288
+
0ba288
+static size_t
0ba288
+o_stream_lzma_get_buffer_avail_size(const struct ostream_private *stream)
0ba288
+{
0ba288
+	/* FIXME: not correct - this is counting compressed size, which may be
0ba288
+	   too larger than uncompressed size in some situations. Fixing would
0ba288
+	   require some kind of additional buffering. */
0ba288
+	return o_stream_get_buffer_avail_size(stream->parent);
0ba288
+}
0ba288
+
0ba288
+static ssize_t
0ba288
+o_stream_lzma_sendv(struct ostream_private *stream,
0ba288
+		    const struct const_iovec *iov, unsigned int iov_count)
0ba288
+{
0ba288
+	struct lzma_ostream *zstream = (struct lzma_ostream *)stream;
0ba288
+	ssize_t ret, bytes = 0;
0ba288
+	unsigned int i;
0ba288
+
0ba288
+	if ((ret = o_stream_zlib_send_outbuf(zstream)) <= 0) {
0ba288
+		/* error / we still couldn't flush existing data to
0ba288
+		   parent stream. */
0ba288
+		return ret;
0ba288
+	}
0ba288
+
0ba288
+	for (i = 0; i < iov_count; i++) {
0ba288
+		ret = o_stream_lzma_send_chunk(zstream, iov[i].iov_base,
0ba288
+						iov[i].iov_len);
0ba288
+		if (ret < 0)
0ba288
+			return -1;
0ba288
+		bytes += ret;
0ba288
+		if ((size_t)ret != iov[i].iov_len)
0ba288
+			break;
0ba288
+	}
0ba288
+	stream->ostream.offset += bytes;
0ba288
+
0ba288
+	/* avail_in!=0 check is used to detect errors. if it's non-zero here
0ba288
+	   it simply means we didn't send all the data */
0ba288
+	zstream->strm.avail_in = 0;
0ba288
+	return bytes;
0ba288
+}
0ba288
+
0ba288
+struct ostream *o_stream_create_lzma(struct ostream *output, int level)
0ba288
+{
0ba288
+	struct lzma_ostream *zstream;
0ba288
+	lzma_ret ret;
0ba288
+
0ba288
+	i_assert(level >= 1 && level <= 9);
0ba288
+
0ba288
+	zstream = i_new(struct lzma_ostream, 1);
0ba288
+	zstream->ostream.sendv = o_stream_lzma_sendv;
0ba288
+	zstream->ostream.flush = o_stream_lzma_flush;
0ba288
+	zstream->ostream.get_buffer_used_size =
0ba288
+		o_stream_lzma_get_buffer_used_size;
0ba288
+	zstream->ostream.get_buffer_avail_size =
0ba288
+		o_stream_lzma_get_buffer_avail_size;
0ba288
+	zstream->ostream.iostream.close = o_stream_lzma_close;
0ba288
+
0ba288
+	ret = lzma_easy_encoder(&zstream->strm, level, LZMA_CHECK_CRC64);
0ba288
+	switch (ret) {
0ba288
+	case LZMA_OK:
0ba288
+		break;
0ba288
+	case LZMA_MEM_ERROR:
0ba288
+		i_fatal_status(FATAL_OUTOFMEM, "lzma: Out of memory");
0ba288
+	case LZMA_OPTIONS_ERROR:
0ba288
+		i_fatal("lzma: Invalid level");
0ba288
+	default:
0ba288
+		i_fatal("lzma_easy_encoder() failed with %d", ret);
0ba288
+	}
0ba288
+
0ba288
+	zstream->strm.next_out = zstream->outbuf;
0ba288
+	zstream->strm.avail_out = sizeof(zstream->outbuf);
0ba288
+	return o_stream_create(&zstream->ostream, output,
0ba288
+			       o_stream_get_fd(output));
0ba288
+}
0ba288
+#endif
0ba288
diff -up dovecot-2.3.16/src/lib-compression/ostream-zlib.h.keeplzma dovecot-2.3.16/src/lib-compression/ostream-zlib.h
0ba288
--- dovecot-2.3.16/src/lib-compression/ostream-zlib.h.keeplzma	2021-08-06 11:25:51.000000000 +0200
0ba288
+++ dovecot-2.3.16/src/lib-compression/ostream-zlib.h	2022-02-28 13:58:02.338149934 +0100
0ba288
@@ -4,6 +4,7 @@
0ba288
 struct ostream *o_stream_create_gz(struct ostream *output, int level);
0ba288
 struct ostream *o_stream_create_deflate(struct ostream *output, int level);
0ba288
 struct ostream *o_stream_create_bz2(struct ostream *output, int level);
0ba288
+struct ostream *o_stream_create_lzma(struct ostream *output, int level);
0ba288
 struct ostream *o_stream_create_lz4(struct ostream *output, int level);
0ba288
 struct ostream *o_stream_create_zstd(struct ostream *output, int level);
0ba288
 
0ba288
diff -up dovecot-2.3.16/src/lib-compression/test-compression.c.keeplzma dovecot-2.3.16/src/lib-compression/test-compression.c
0ba288
--- dovecot-2.3.16/src/lib-compression/test-compression.c.keeplzma	2021-08-06 11:25:51.000000000 +0200
0ba288
+++ dovecot-2.3.16/src/lib-compression/test-compression.c	2022-02-28 13:58:02.338149934 +0100
0ba288
@@ -730,7 +730,6 @@ static void test_compression_int(bool au
0ba288
 
0ba288
 	for (i = 0; compression_handlers[i].name != NULL; i++) {
0ba288
 		if (compression_handlers[i].create_istream != NULL &&
0ba288
-		    compression_handlers[i].create_ostream != NULL &&
0ba288
 		    (!autodetect ||
0ba288
 		     compression_handlers[i].is_compressed != NULL)) T_BEGIN {
0ba288
 			if (compression_handlers[i].is_compressed != NULL &&