/* SPDX-License-Identifier: GPL-3.0-or-later */ /* * gzip.c * * Copyright (C) 2019 David Oberhollenzer <goliath@infraroot.at> */ #include "../internal.h" #include <zlib.h> typedef struct { ostream_comp_t base; z_stream strm; } ostream_gzip_t; static int flush_inbuf(ostream_comp_t *base, bool finish) { ostream_gzip_t *gzip = (ostream_gzip_t *)base; size_t have; int ret; if (base->inbuf_used > sizeof(base->inbuf)) base->inbuf_used = sizeof(base->inbuf); if (sizeof(size_t) > sizeof(uInt)) { gzip->strm.avail_in = ~((uInt)0); if ((size_t)gzip->strm.avail_in > base->inbuf_used) gzip->strm.avail_in = (uInt)base->inbuf_used; } else { gzip->strm.avail_in = (uInt)base->inbuf_used; } gzip->strm.next_in = base->inbuf; do { gzip->strm.avail_out = BUFSZ; gzip->strm.next_out = base->outbuf; ret = deflate(&gzip->strm, finish ? Z_FINISH : Z_NO_FLUSH); if (ret == Z_STREAM_ERROR) { fprintf(stderr, "%s: internal error in gzip compressor.\n", base->wrapped->get_filename(base->wrapped)); return -1; } have = BUFSZ - gzip->strm.avail_out; if (base->wrapped->append(base->wrapped, base->outbuf, have)) return -1; } while (gzip->strm.avail_out == 0); base->inbuf_used = 0; return 0; } static void cleanup(ostream_comp_t *base) { ostream_gzip_t *gzip = (ostream_gzip_t *)base; deflateEnd(&gzip->strm); } ostream_comp_t *ostream_gzip_create(const char *filename) { ostream_gzip_t *gzip = calloc(1, sizeof(*gzip)); ostream_comp_t *base = (ostream_comp_t *)gzip; int ret; if (gzip == NULL) { fprintf(stderr, "%s: creating gzip wrapper: %s.\n", filename, strerror(errno)); return NULL; } ret = deflateInit2(&gzip->strm, 9, Z_DEFLATED, 16 + 15, 8, Z_DEFAULT_STRATEGY); if (ret != Z_OK) { fprintf(stderr, "%s: internal error creating gzip compressor.\n", filename); free(gzip); return NULL; } base->flush_inbuf = flush_inbuf; base->cleanup = cleanup; return base; }