mirror of
https://github.com/systemd/systemd.git
synced 2025-01-11 09:18:07 +03:00
Merge pull request #1527 from keszybz/lz4
Using lz4 frame api for coredump files
This commit is contained in:
commit
cb181af9b6
46
configure.ac
46
configure.ac
@ -530,25 +530,27 @@ AC_SUBST(CERTIFICATEROOT)
|
|||||||
# ------------------------------------------------------------------------------
|
# ------------------------------------------------------------------------------
|
||||||
have_xz=no
|
have_xz=no
|
||||||
AC_ARG_ENABLE(xz, AS_HELP_STRING([--disable-xz], [Disable optional XZ support]))
|
AC_ARG_ENABLE(xz, AS_HELP_STRING([--disable-xz], [Disable optional XZ support]))
|
||||||
if test "x$enable_xz" != "xno"; then
|
AS_IF([test "x$enable_xz" != "xno"], [
|
||||||
PKG_CHECK_MODULES(XZ, [ liblzma ],
|
PKG_CHECK_MODULES(XZ, [ liblzma ],
|
||||||
[AC_DEFINE(HAVE_XZ, 1, [Define if XZ is available]) have_xz=yes], have_xz=no)
|
[AC_DEFINE(HAVE_XZ, 1, [Define if XZ is available])
|
||||||
if test "x$have_xz" = xno -a "x$enable_xz" = xyes; then
|
have_xz=yes],
|
||||||
AC_MSG_ERROR([*** XZ support requested but libraries not found])
|
have_xz=no)
|
||||||
fi
|
AS_IF([test "x$have_xz" = xno -a "x$enable_xz" = xyes],
|
||||||
fi
|
[AC_MSG_ERROR([*** XZ support requested but libraries not found])])
|
||||||
|
])
|
||||||
AM_CONDITIONAL(HAVE_XZ, [test "$have_xz" = "yes"])
|
AM_CONDITIONAL(HAVE_XZ, [test "$have_xz" = "yes"])
|
||||||
|
|
||||||
# ------------------------------------------------------------------------------
|
# ------------------------------------------------------------------------------
|
||||||
have_zlib=no
|
have_zlib=no
|
||||||
AC_ARG_ENABLE(zlib, AS_HELP_STRING([--disable-zlib], [Disable optional ZLIB support]))
|
AC_ARG_ENABLE(zlib, AS_HELP_STRING([--disable-zlib], [Disable optional ZLIB support]))
|
||||||
if test "x$enable_zlib" != "xno"; then
|
AS_IF([test "x$enable_zlib" != "xno"], [
|
||||||
PKG_CHECK_MODULES(ZLIB, [ zlib ],
|
PKG_CHECK_MODULES(ZLIB, [ zlib ],
|
||||||
[AC_DEFINE(HAVE_ZLIB, 1, [Define if ZLIB is available]) have_zlib=yes], have_zlib=no)
|
[AC_DEFINE(HAVE_ZLIB, 1, [Define if ZLIB is available])
|
||||||
if test "x$have_zlib" = xno -a "x$enable_zlib" = xyes; then
|
have_zlib=yes],
|
||||||
AC_MSG_ERROR([*** ZLIB support requested but libraries not found])
|
have_zlib=no)
|
||||||
fi
|
AS_IF([test "x$have_zlib" = xno -a "x$enable_zlib" = xyes],
|
||||||
fi
|
[AC_MSG_ERROR([*** ZLIB support requested but libraries not found])])
|
||||||
|
])
|
||||||
AM_CONDITIONAL(HAVE_ZLIB, [test "$have_zlib" = "yes"])
|
AM_CONDITIONAL(HAVE_ZLIB, [test "$have_zlib" = "yes"])
|
||||||
|
|
||||||
# ------------------------------------------------------------------------------
|
# ------------------------------------------------------------------------------
|
||||||
@ -556,20 +558,24 @@ have_bzip2=no
|
|||||||
AC_ARG_ENABLE(bzip2, AS_HELP_STRING([--enable-bzip2], [Disable optional BZIP2 support]))
|
AC_ARG_ENABLE(bzip2, AS_HELP_STRING([--enable-bzip2], [Disable optional BZIP2 support]))
|
||||||
AS_IF([test "x$enable_bzip2" != "xno"], [
|
AS_IF([test "x$enable_bzip2" != "xno"], [
|
||||||
AC_CHECK_HEADERS(bzlib.h,
|
AC_CHECK_HEADERS(bzlib.h,
|
||||||
[AC_DEFINE(HAVE_BZIP2, 1, [Define in BZIP2 is available])
|
[AC_DEFINE(HAVE_BZIP2, 1, [Define if BZIP2 is available])
|
||||||
have_bzip2=yes],
|
have_bzip2=yes],
|
||||||
[AS_IF([test "x$have_bzip2" = xyes], [AC_MSG_ERROR([*** BZIP2 support requested but headers not found])])
|
[AS_IF([test "x$enable_bzip2" = xyes],
|
||||||
])
|
[AC_MSG_ERROR([*** BZIP2 support requested but headers not found])])]
|
||||||
|
)
|
||||||
])
|
])
|
||||||
AM_CONDITIONAL(HAVE_BZIP2, [test "$have_bzip2" = "yes"])
|
AM_CONDITIONAL(HAVE_BZIP2, [test "$have_bzip2" = "yes"])
|
||||||
|
|
||||||
# ------------------------------------------------------------------------------
|
# ------------------------------------------------------------------------------
|
||||||
have_lz4=no
|
have_lz4=no
|
||||||
AC_ARG_ENABLE(lz4, AS_HELP_STRING([--enable-lz4], [Enable optional LZ4 support]))
|
AC_ARG_ENABLE(lz4, AS_HELP_STRING([--disable-lz4], [Disable optional LZ4 support]))
|
||||||
AS_IF([test "x$enable_lz4" = "xyes"], [
|
AS_IF([test "x$enable_lz4" != "xno"], [
|
||||||
AC_CHECK_HEADERS(lz4.h,
|
PKG_CHECK_MODULES(LZ4, [ liblz4 >= 125 ],
|
||||||
[AC_DEFINE(HAVE_LZ4, 1, [Define in LZ4 is available]) have_lz4=yes],
|
[AC_DEFINE(HAVE_LZ4, 1, [Define in LZ4 is available])
|
||||||
[AC_MSG_ERROR([*** LZ4 support requested but headers not found])])
|
have_lz4=yes],
|
||||||
|
have_lz4=no)
|
||||||
|
AS_IF([test "x$have_lz4" = xno -a "x$enable_lz4" = xyes],
|
||||||
|
[AC_MSG_ERROR([*** LZ4 support requested but libraries not found])])
|
||||||
])
|
])
|
||||||
AM_CONDITIONAL(HAVE_LZ4, [test "$have_lz4" = "yes"])
|
AM_CONDITIONAL(HAVE_LZ4, [test "$have_lz4" = "yes"])
|
||||||
|
|
||||||
|
@ -22,6 +22,7 @@
|
|||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
#include <sys/mman.h>
|
||||||
|
|
||||||
#ifdef HAVE_XZ
|
#ifdef HAVE_XZ
|
||||||
# include <lzma.h>
|
# include <lzma.h>
|
||||||
@ -29,6 +30,7 @@
|
|||||||
|
|
||||||
#ifdef HAVE_LZ4
|
#ifdef HAVE_LZ4
|
||||||
# include <lz4.h>
|
# include <lz4.h>
|
||||||
|
# include <lz4frame.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include "compress.h"
|
#include "compress.h"
|
||||||
@ -37,6 +39,11 @@
|
|||||||
#include "sparse-endian.h"
|
#include "sparse-endian.h"
|
||||||
#include "journal-def.h"
|
#include "journal-def.h"
|
||||||
|
|
||||||
|
#ifdef HAVE_LZ4
|
||||||
|
DEFINE_TRIVIAL_CLEANUP_FUNC(LZ4F_compressionContext_t, LZ4F_freeCompressionContext);
|
||||||
|
DEFINE_TRIVIAL_CLEANUP_FUNC(LZ4F_decompressionContext_t, LZ4F_freeDecompressionContext);
|
||||||
|
#endif
|
||||||
|
|
||||||
#define ALIGN_8(l) ALIGN_TO(l, sizeof(size_t))
|
#define ALIGN_8(l) ALIGN_TO(l, sizeof(size_t))
|
||||||
|
|
||||||
static const char* const object_compressed_table[_OBJECT_COMPRESSED_MAX] = {
|
static const char* const object_compressed_table[_OBJECT_COMPRESSED_MAX] = {
|
||||||
@ -416,81 +423,96 @@ int compress_stream_xz(int fdf, int fdt, uint64_t max_bytes) {
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
#define LZ4_BUFSIZE (512*1024)
|
#define LZ4_BUFSIZE (512*1024u)
|
||||||
|
|
||||||
int compress_stream_lz4(int fdf, int fdt, uint64_t max_bytes) {
|
int compress_stream_lz4(int fdf, int fdt, uint64_t max_bytes) {
|
||||||
|
|
||||||
#ifdef HAVE_LZ4
|
#ifdef HAVE_LZ4
|
||||||
|
LZ4F_errorCode_t c;
|
||||||
_cleanup_free_ char *buf1 = NULL, *buf2 = NULL, *out = NULL;
|
_cleanup_(LZ4F_freeCompressionContextp) LZ4F_compressionContext_t ctx = NULL;
|
||||||
char *buf;
|
_cleanup_free_ char *buf = NULL;
|
||||||
LZ4_stream_t lz4_data = {};
|
char *src = NULL;
|
||||||
le32_t header;
|
size_t size, n, total_in = 0, total_out = 0, offset = 0, frame_size;
|
||||||
size_t total_in = 0, total_out = sizeof(header);
|
struct stat st;
|
||||||
ssize_t n;
|
|
||||||
|
|
||||||
assert(fdf >= 0);
|
|
||||||
assert(fdt >= 0);
|
|
||||||
|
|
||||||
buf1 = malloc(LZ4_BUFSIZE);
|
|
||||||
buf2 = malloc(LZ4_BUFSIZE);
|
|
||||||
out = malloc(LZ4_COMPRESSBOUND(LZ4_BUFSIZE));
|
|
||||||
if (!buf1 || !buf2 || !out)
|
|
||||||
return log_oom();
|
|
||||||
|
|
||||||
buf = buf1;
|
|
||||||
for (;;) {
|
|
||||||
size_t m;
|
|
||||||
int r;
|
int r;
|
||||||
|
static const LZ4F_compressOptions_t options = {
|
||||||
|
.stableSrc = 1,
|
||||||
|
};
|
||||||
|
static const LZ4F_preferences_t preferences = {
|
||||||
|
.frameInfo.blockSizeID = 5,
|
||||||
|
};
|
||||||
|
|
||||||
m = LZ4_BUFSIZE;
|
c = LZ4F_createCompressionContext(&ctx, LZ4F_VERSION);
|
||||||
if (max_bytes != (uint64_t) -1 && (uint64_t) m > (max_bytes - total_in))
|
if (LZ4F_isError(c))
|
||||||
m = (size_t) (max_bytes - total_in);
|
return -ENOMEM;
|
||||||
|
|
||||||
n = read(fdf, buf, m);
|
if (fstat(fdf, &st) < 0)
|
||||||
if (n < 0)
|
return log_debug_errno(errno, "fstat() failed: %m");
|
||||||
|
|
||||||
|
frame_size = LZ4F_compressBound(LZ4_BUFSIZE, &preferences);
|
||||||
|
size = frame_size + 64*1024; /* add some space for header and trailer */
|
||||||
|
buf = malloc(size);
|
||||||
|
if (!buf)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
n = offset = LZ4F_compressBegin(ctx, buf, size, &preferences);
|
||||||
|
if (LZ4F_isError(n))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
src = mmap(NULL, st.st_size, PROT_READ, MAP_PRIVATE, fdf, 0);
|
||||||
|
if (src == MAP_FAILED)
|
||||||
return -errno;
|
return -errno;
|
||||||
if (n == 0)
|
|
||||||
break;
|
|
||||||
|
|
||||||
total_in += n;
|
log_debug("Buffer size is %zu bytes, header size %zu bytes.", size, n);
|
||||||
|
|
||||||
r = LZ4_compress_continue(&lz4_data, buf, out, n);
|
while (total_in < (size_t) st.st_size) {
|
||||||
if (r == 0) {
|
ssize_t k;
|
||||||
log_error("LZ4 compression failed.");
|
|
||||||
return -EBADMSG;
|
k = MIN(LZ4_BUFSIZE, st.st_size - total_in);
|
||||||
|
n = LZ4F_compressUpdate(ctx, buf + offset, size - offset,
|
||||||
|
src + total_in, k, &options);
|
||||||
|
if (LZ4F_isError(n)) {
|
||||||
|
r = -ENOTRECOVERABLE;
|
||||||
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
header = htole32(r);
|
total_in += k;
|
||||||
errno = 0;
|
offset += n;
|
||||||
|
total_out += n;
|
||||||
|
|
||||||
n = write(fdt, &header, sizeof(header));
|
if (max_bytes != (uint64_t) -1 && total_out > (size_t) max_bytes) {
|
||||||
if (n < 0)
|
log_debug("Compressed stream longer than %zd bytes", max_bytes);
|
||||||
return -errno;
|
return -EFBIG;
|
||||||
if (n != sizeof(header))
|
|
||||||
return errno ? -errno : -EIO;
|
|
||||||
|
|
||||||
n = loop_write(fdt, out, r, false);
|
|
||||||
if (n < 0)
|
|
||||||
return n;
|
|
||||||
|
|
||||||
total_out += sizeof(header) + r;
|
|
||||||
|
|
||||||
buf = buf == buf1 ? buf2 : buf1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
header = htole32(0);
|
if (size - offset < frame_size + 4) {
|
||||||
n = write(fdt, &header, sizeof(header));
|
k = loop_write(fdt, buf, offset, false);
|
||||||
if (n < 0)
|
if (k < 0) {
|
||||||
return -errno;
|
r = k;
|
||||||
if (n != sizeof(header))
|
goto cleanup;
|
||||||
return errno ? -errno : -EIO;
|
}
|
||||||
|
offset = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
n = LZ4F_compressEnd(ctx, buf + offset, size - offset, &options);
|
||||||
|
if (LZ4F_isError(n)) {
|
||||||
|
r = -ENOTRECOVERABLE;
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
|
||||||
|
offset += n;
|
||||||
|
total_out += n;
|
||||||
|
r = loop_write(fdt, buf, offset, false);
|
||||||
|
if (r < 0)
|
||||||
|
goto cleanup;
|
||||||
|
|
||||||
log_debug("LZ4 compression finished (%zu -> %zu bytes, %.1f%%)",
|
log_debug("LZ4 compression finished (%zu -> %zu bytes, %.1f%%)",
|
||||||
total_in, total_out,
|
total_in, total_out,
|
||||||
(double) total_out / total_in * 100);
|
(double) total_out / total_in * 100);
|
||||||
|
cleanup:
|
||||||
return 0;
|
munmap(src, st.st_size);
|
||||||
|
return r;
|
||||||
#else
|
#else
|
||||||
return -EPROTONOSUPPORT;
|
return -EPROTONOSUPPORT;
|
||||||
#endif
|
#endif
|
||||||
@ -510,7 +532,7 @@ int decompress_stream_xz(int fdf, int fdt, uint64_t max_bytes) {
|
|||||||
|
|
||||||
ret = lzma_stream_decoder(&s, UINT64_MAX, 0);
|
ret = lzma_stream_decoder(&s, UINT64_MAX, 0);
|
||||||
if (ret != LZMA_OK) {
|
if (ret != LZMA_OK) {
|
||||||
log_error("Failed to initialize XZ decoder: code %u", ret);
|
log_debug("Failed to initialize XZ decoder: code %u", ret);
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -536,7 +558,7 @@ int decompress_stream_xz(int fdf, int fdt, uint64_t max_bytes) {
|
|||||||
|
|
||||||
ret = lzma_code(&s, action);
|
ret = lzma_code(&s, action);
|
||||||
if (ret != LZMA_OK && ret != LZMA_STREAM_END) {
|
if (ret != LZMA_OK && ret != LZMA_STREAM_END) {
|
||||||
log_error("Decompression failed: code %u", ret);
|
log_debug("Decompression failed: code %u", ret);
|
||||||
return -EBADMSG;
|
return -EBADMSG;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -566,14 +588,14 @@ int decompress_stream_xz(int fdf, int fdt, uint64_t max_bytes) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
log_error("Cannot decompress file. Compiled without XZ support.");
|
log_debug("Cannot decompress file. Compiled without XZ support.");
|
||||||
return -EPROTONOSUPPORT;
|
return -EPROTONOSUPPORT;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
int decompress_stream_lz4(int fdf, int fdt, uint64_t max_bytes) {
|
|
||||||
|
|
||||||
#ifdef HAVE_LZ4
|
#ifdef HAVE_LZ4
|
||||||
|
static int decompress_stream_lz4_v1(int fdf, int fdt, uint64_t max_bytes) {
|
||||||
|
|
||||||
_cleanup_free_ char *buf = NULL, *out = NULL;
|
_cleanup_free_ char *buf = NULL, *out = NULL;
|
||||||
size_t buf_size = 0;
|
size_t buf_size = 0;
|
||||||
LZ4_streamDecode_t lz4_data = {};
|
LZ4_streamDecode_t lz4_data = {};
|
||||||
@ -585,7 +607,7 @@ int decompress_stream_lz4(int fdf, int fdt, uint64_t max_bytes) {
|
|||||||
|
|
||||||
out = malloc(4*LZ4_BUFSIZE);
|
out = malloc(4*LZ4_BUFSIZE);
|
||||||
if (!out)
|
if (!out)
|
||||||
return log_oom();
|
return -ENOMEM;
|
||||||
|
|
||||||
for (;;) {
|
for (;;) {
|
||||||
ssize_t m;
|
ssize_t m;
|
||||||
@ -606,22 +628,24 @@ int decompress_stream_lz4(int fdf, int fdt, uint64_t max_bytes) {
|
|||||||
* not accept buffers compressed by newer binaries then.
|
* not accept buffers compressed by newer binaries then.
|
||||||
*/
|
*/
|
||||||
if (m > LZ4_COMPRESSBOUND(LZ4_BUFSIZE * 4)) {
|
if (m > LZ4_COMPRESSBOUND(LZ4_BUFSIZE * 4)) {
|
||||||
log_error("Compressed stream block too big: %zd bytes", m);
|
log_debug("Compressed stream block too big: %zd bytes", m);
|
||||||
return -EBADMSG;
|
return -ENOBUFS;
|
||||||
}
|
}
|
||||||
|
|
||||||
total_in += sizeof(header) + m;
|
total_in += sizeof(header) + m;
|
||||||
|
|
||||||
if (!GREEDY_REALLOC(buf, buf_size, m))
|
if (!GREEDY_REALLOC(buf, buf_size, m))
|
||||||
return log_oom();
|
return -ENOMEM;
|
||||||
|
|
||||||
r = loop_read_exact(fdf, buf, m, false);
|
r = loop_read_exact(fdf, buf, m, false);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
r = LZ4_decompress_safe_continue(&lz4_data, buf, out, m, 4*LZ4_BUFSIZE);
|
r = LZ4_decompress_safe_continue(&lz4_data, buf, out, m, 4*LZ4_BUFSIZE);
|
||||||
if (r <= 0)
|
if (r <= 0) {
|
||||||
log_error("LZ4 decompression failed.");
|
log_debug("LZ4 decompression failed (legacy format).");
|
||||||
|
return -EBADMSG;
|
||||||
|
}
|
||||||
|
|
||||||
total_out += r;
|
total_out += r;
|
||||||
|
|
||||||
@ -635,13 +659,80 @@ int decompress_stream_lz4(int fdf, int fdt, uint64_t max_bytes) {
|
|||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
log_debug("LZ4 decompression finished (%zu -> %zu bytes, %.1f%%)",
|
log_debug("LZ4 decompression finished (legacy format, %zu -> %zu bytes, %.1f%%)",
|
||||||
total_in, total_out,
|
total_in, total_out,
|
||||||
(double) total_out / total_in * 100);
|
(double) total_out / total_in * 100);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int decompress_stream_lz4_v2(int in, int out, uint64_t max_bytes) {
|
||||||
|
size_t c;
|
||||||
|
_cleanup_(LZ4F_freeDecompressionContextp) LZ4F_decompressionContext_t ctx = NULL;
|
||||||
|
_cleanup_free_ char *buf = NULL;
|
||||||
|
char *src;
|
||||||
|
struct stat st;
|
||||||
|
int r = 0;
|
||||||
|
size_t total_in = 0, total_out = 0;
|
||||||
|
|
||||||
|
c = LZ4F_createDecompressionContext(&ctx, LZ4F_VERSION);
|
||||||
|
if (LZ4F_isError(c))
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
if (fstat(in, &st) < 0)
|
||||||
|
return log_debug_errno(errno, "fstat() failed: %m");
|
||||||
|
|
||||||
|
buf = malloc(LZ4_BUFSIZE);
|
||||||
|
if (!buf)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
src = mmap(NULL, st.st_size, PROT_READ, MAP_PRIVATE, in, 0);
|
||||||
|
if (src == MAP_FAILED)
|
||||||
|
return -errno;
|
||||||
|
|
||||||
|
while (total_in < (size_t) st.st_size) {
|
||||||
|
size_t produced = LZ4_BUFSIZE;
|
||||||
|
size_t used = st.st_size - total_in;
|
||||||
|
|
||||||
|
c = LZ4F_decompress(ctx, buf, &produced, src + total_in, &used, NULL);
|
||||||
|
if (LZ4F_isError(c)) {
|
||||||
|
r = -EBADMSG;
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
|
||||||
|
total_in += used;
|
||||||
|
total_out += produced;
|
||||||
|
|
||||||
|
if (max_bytes != (uint64_t) -1 && total_out > (size_t) max_bytes) {
|
||||||
|
log_debug("Decompressed stream longer than %zd bytes", max_bytes);
|
||||||
|
r = -EFBIG;
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
|
||||||
|
r = loop_write(out, buf, produced, false);
|
||||||
|
if (r < 0)
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
|
||||||
|
log_debug("LZ4 decompression finished (%zu -> %zu bytes, %.1f%%)",
|
||||||
|
total_in, total_out,
|
||||||
|
(double) total_out / total_in * 100);
|
||||||
|
cleanup:
|
||||||
|
munmap(src, st.st_size);
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
int decompress_stream_lz4(int fdf, int fdt, uint64_t max_bytes) {
|
||||||
|
#ifdef HAVE_LZ4
|
||||||
|
int r;
|
||||||
|
|
||||||
|
r = decompress_stream_lz4_v2(fdf, fdt, max_bytes);
|
||||||
|
if (r == -EBADMSG)
|
||||||
|
r = decompress_stream_lz4_v1(fdf, fdt, max_bytes);
|
||||||
|
return r;
|
||||||
#else
|
#else
|
||||||
log_error("Cannot decompress file. Compiled without LZ4 support.");
|
log_debug("Cannot decompress file. Compiled without LZ4 support.");
|
||||||
return -EPROTONOSUPPORT;
|
return -EPROTONOSUPPORT;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
@ -20,6 +20,7 @@
|
|||||||
#include "compress.h"
|
#include "compress.h"
|
||||||
#include "util.h"
|
#include "util.h"
|
||||||
#include "macro.h"
|
#include "macro.h"
|
||||||
|
#include "random-util.h"
|
||||||
|
|
||||||
typedef int (compress_t)(const void *src, uint64_t src_size, void *dst, size_t *dst_size);
|
typedef int (compress_t)(const void *src, uint64_t src_size, void *dst, size_t *dst_size);
|
||||||
typedef int (decompress_t)(const void *src, uint64_t src_size,
|
typedef int (decompress_t)(const void *src, uint64_t src_size,
|
||||||
@ -27,20 +28,31 @@ typedef int (decompress_t)(const void *src, uint64_t src_size,
|
|||||||
|
|
||||||
#define MAX_SIZE (1024*1024LU)
|
#define MAX_SIZE (1024*1024LU)
|
||||||
|
|
||||||
static char* make_buf(size_t count) {
|
static char* make_buf(size_t count, const char *type) {
|
||||||
char *buf;
|
char *buf;
|
||||||
size_t i;
|
size_t i;
|
||||||
|
|
||||||
buf = malloc(count);
|
buf = malloc(count);
|
||||||
assert_se(buf);
|
assert_se(buf);
|
||||||
|
|
||||||
|
if (streq(type, "zeros"))
|
||||||
|
memzero(buf, count);
|
||||||
|
else if (streq(type, "simple"))
|
||||||
for (i = 0; i < count; i++)
|
for (i = 0; i < count; i++)
|
||||||
buf[i] = 'a' + i % ('z' - 'a' + 1);
|
buf[i] = 'a' + i % ('z' - 'a' + 1);
|
||||||
|
else if (streq(type, "random")) {
|
||||||
|
random_bytes(buf, count/10);
|
||||||
|
random_bytes(buf + 2*count/10, count/10);
|
||||||
|
random_bytes(buf + 4*count/10, count/20);
|
||||||
|
random_bytes(buf + 6*count/10, count/20);
|
||||||
|
random_bytes(buf + 8*count/10, count/20);
|
||||||
|
} else
|
||||||
|
assert_not_reached("here");
|
||||||
|
|
||||||
return buf;
|
return buf;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void test_compress_decompress(const char* label,
|
static void test_compress_decompress(const char* label, const char* type,
|
||||||
compress_t compress, decompress_t decompress) {
|
compress_t compress, decompress_t decompress) {
|
||||||
usec_t n, n2 = 0;
|
usec_t n, n2 = 0;
|
||||||
float dt;
|
float dt;
|
||||||
@ -50,7 +62,7 @@ static void test_compress_decompress(const char* label,
|
|||||||
size_t buf2_allocated = 0;
|
size_t buf2_allocated = 0;
|
||||||
size_t skipped = 0, compressed = 0, total = 0;
|
size_t skipped = 0, compressed = 0, total = 0;
|
||||||
|
|
||||||
text = make_buf(MAX_SIZE);
|
text = make_buf(MAX_SIZE, type);
|
||||||
buf = calloc(MAX_SIZE + 1, 1);
|
buf = calloc(MAX_SIZE + 1, 1);
|
||||||
assert_se(text && buf);
|
assert_se(text && buf);
|
||||||
|
|
||||||
@ -62,7 +74,8 @@ static void test_compress_decompress(const char* label,
|
|||||||
|
|
||||||
r = compress(text, i, buf, &j);
|
r = compress(text, i, buf, &j);
|
||||||
/* assume compression must be successful except for small inputs */
|
/* assume compression must be successful except for small inputs */
|
||||||
assert_se(r == 0 || (i < 2048 && r == -ENOBUFS));
|
assert_se(r == 0 || (i < 2048 && r == -ENOBUFS) || streq(type, "random"));
|
||||||
|
|
||||||
/* check for overwrites */
|
/* check for overwrites */
|
||||||
assert_se(buf[i] == 0);
|
assert_se(buf[i] == 0);
|
||||||
if (r != 0) {
|
if (r != 0) {
|
||||||
@ -91,23 +104,26 @@ static void test_compress_decompress(const char* label,
|
|||||||
|
|
||||||
dt = (n2-n) / 1e6;
|
dt = (n2-n) / 1e6;
|
||||||
|
|
||||||
log_info("%s: compressed & decompressed %zu bytes in %.2fs (%.2fMiB/s), "
|
log_info("%s/%s: compressed & decompressed %zu bytes in %.2fs (%.2fMiB/s), "
|
||||||
"mean compresion %.2f%%, skipped %zu bytes",
|
"mean compresion %.2f%%, skipped %zu bytes",
|
||||||
label, total, dt,
|
label, type, total, dt,
|
||||||
total / 1024. / 1024 / dt,
|
total / 1024. / 1024 / dt,
|
||||||
100 - compressed * 100. / total,
|
100 - compressed * 100. / total,
|
||||||
skipped);
|
skipped);
|
||||||
}
|
}
|
||||||
|
|
||||||
int main(int argc, char *argv[]) {
|
int main(int argc, char *argv[]) {
|
||||||
|
const char *i;
|
||||||
|
|
||||||
log_set_max_level(LOG_DEBUG);
|
log_set_max_level(LOG_DEBUG);
|
||||||
|
|
||||||
|
NULSTR_FOREACH(i, "zeros\0simple\0random\0") {
|
||||||
#ifdef HAVE_XZ
|
#ifdef HAVE_XZ
|
||||||
test_compress_decompress("XZ", compress_blob_xz, decompress_blob_xz);
|
test_compress_decompress("XZ", i, compress_blob_xz, decompress_blob_xz);
|
||||||
#endif
|
#endif
|
||||||
#ifdef HAVE_LZ4
|
#ifdef HAVE_LZ4
|
||||||
test_compress_decompress("LZ4", compress_blob_lz4, decompress_blob_lz4);
|
test_compress_decompress("LZ4", i, compress_blob_lz4, decompress_blob_lz4);
|
||||||
#endif
|
#endif
|
||||||
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -144,8 +144,8 @@ static void test_compress_stream(int compression,
|
|||||||
const char *srcfile) {
|
const char *srcfile) {
|
||||||
|
|
||||||
_cleanup_close_ int src = -1, dst = -1, dst2 = -1;
|
_cleanup_close_ int src = -1, dst = -1, dst2 = -1;
|
||||||
char pattern[] = "/tmp/systemd-test.xz.XXXXXX",
|
char pattern[] = "/tmp/systemd-test.compressed.XXXXXX",
|
||||||
pattern2[] = "/tmp/systemd-test.xz.XXXXXX";
|
pattern2[] = "/tmp/systemd-test.compressed.XXXXXX";
|
||||||
int r;
|
int r;
|
||||||
_cleanup_free_ char *cmd = NULL, *cmd2;
|
_cleanup_free_ char *cmd = NULL, *cmd2;
|
||||||
struct stat st = {};
|
struct stat st = {};
|
||||||
@ -185,7 +185,7 @@ static void test_compress_stream(int compression,
|
|||||||
|
|
||||||
assert_se(lseek(dst, 1, SEEK_SET) == 1);
|
assert_se(lseek(dst, 1, SEEK_SET) == 1);
|
||||||
r = decompress(dst, dst2, st.st_size);
|
r = decompress(dst, dst2, st.st_size);
|
||||||
assert_se(r == -EBADMSG);
|
assert_se(r == -EBADMSG || r == 0);
|
||||||
|
|
||||||
assert_se(lseek(dst, 0, SEEK_SET) == 0);
|
assert_se(lseek(dst, 0, SEEK_SET) == 0);
|
||||||
assert_se(lseek(dst2, 0, SEEK_SET) == 0);
|
assert_se(lseek(dst2, 0, SEEK_SET) == 0);
|
||||||
@ -236,8 +236,7 @@ int main(int argc, char *argv[]) {
|
|||||||
compress_blob_lz4, decompress_startswith_lz4,
|
compress_blob_lz4, decompress_startswith_lz4,
|
||||||
data, sizeof(data), true);
|
data, sizeof(data), true);
|
||||||
|
|
||||||
/* Produced stream is not compatible with lz4 binary, skip lz4cat check. */
|
test_compress_stream(OBJECT_COMPRESSED_LZ4, "lz4cat",
|
||||||
test_compress_stream(OBJECT_COMPRESSED_LZ4, NULL,
|
|
||||||
compress_stream_lz4, decompress_stream_lz4, argv[0]);
|
compress_stream_lz4, decompress_stream_lz4, argv[0]);
|
||||||
#else
|
#else
|
||||||
log_info("/* LZ4 test skipped */");
|
log_info("/* LZ4 test skipped */");
|
||||||
|
Loading…
Reference in New Issue
Block a user