From f757761b310a899dc8ece7ad7845ed0a22ff752b Mon Sep 17 00:00:00 2001 From: Mattias Andrée Date: Thu, 26 Jan 2017 07:26:42 +0100 Subject: Add blind-compress and blind-decompress MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Signed-off-by: Mattias Andrée --- src/blind-compress.c | 69 ++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 69 insertions(+) create mode 100644 src/blind-compress.c (limited to 'src/blind-compress.c') diff --git a/src/blind-compress.c b/src/blind-compress.c new file mode 100644 index 0000000..768ef06 --- /dev/null +++ b/src/blind-compress.c @@ -0,0 +1,69 @@ +/* See LICENSE file for copyright and license details. */ +#include "stream.h" +#include "util.h" + +#include +#include + +USAGE("") + +static size_t +compare(const char *restrict new, const char *restrict old, size_t n, size_t **cmp, size_t *cmpsize) +{ + size_t i, start1, start2, ptr, same, diff; + for (ptr = i = 0; i < n;) { + for (start1 = i; i < n && old[i] == new[i]; i++); + for (start2 = i; i < n && old[i] != new[i]; i++); + same = start2 - start1; + diff = i - start2; + if (ptr && same < 2 * sizeof(size_t) && same + diff <= SIZE_MAX - (*cmp)[ptr - 1]) { + (*cmp)[ptr - 1] += same + diff; + } else { + if (ptr + 2 > *cmpsize) + *cmp = erealloc(*cmp, (*cmpsize += 128) * sizeof(size_t)); + (*cmp)[ptr++] = same; + (*cmp)[ptr++] = diff; + } + } + return ptr; +} + +int +main(int argc, char *argv[]) +{ + struct stream stream; + char *buf[2]; + size_t n, parts, part, off; + int i; + size_t *cmp = NULL; + size_t cmpsize = 0; + + ENOFLAGS(argc); + + stream.file = ""; + stream.fd = STDIN_FILENO; + einit_stream(&stream); + fprint_stream_head(stdout, &stream); + efflush(stdout, ""); + + echeck_frame_size(stream.width, stream.height, stream.pixel_size, 0, ""); + n = stream.width * stream.height * stream.pixel_size; + buf[0] = emalloc(n); + buf[1] = ecalloc(1, n); + + memcpy(buf[0], stream.buf, stream.ptr); + for (i = 0; eread_frame(&stream, buf[i], n); i ^= 1) { + parts = compare(buf[i], buf[i ^ 1], n, &cmp, &cmpsize); + for (off = part = 0; part < parts; part += 2) { + off += cmp[part]; + ewriteall(STDOUT_FILENO, cmp + part, 2 * sizeof(size_t), ""); + ewriteall(STDOUT_FILENO, buf[i] + off, cmp[part + 1], ""); + off += cmp[part + 1]; + } + } + + free(cmp); + free(buf[0]); + free(buf[1]); + return 0; +} -- cgit v1.2.3-70-g09d2