The Design and Implementation of the FreeBSD Operating System, Second Edition
Now available: The Design and Implementation of the FreeBSD Operating System (Second Edition)


[ source navigation ] [ diff markup ] [ identifier search ] [ freetext search ] [ file search ] [ list types ] [ track identifier ]

FreeBSD/Linux Kernel Cross Reference
sys/kern/subr_compressor.c

Version: -  FREEBSD  -  FREEBSD-13-STABLE  -  FREEBSD-13-0  -  FREEBSD-12-STABLE  -  FREEBSD-12-0  -  FREEBSD-11-STABLE  -  FREEBSD-11-0  -  FREEBSD-10-STABLE  -  FREEBSD-10-0  -  FREEBSD-9-STABLE  -  FREEBSD-9-0  -  FREEBSD-8-STABLE  -  FREEBSD-8-0  -  FREEBSD-7-STABLE  -  FREEBSD-7-0  -  FREEBSD-6-STABLE  -  FREEBSD-6-0  -  FREEBSD-5-STABLE  -  FREEBSD-5-0  -  FREEBSD-4-STABLE  -  FREEBSD-3-STABLE  -  FREEBSD22  -  l41  -  OPENBSD  -  linux-2.6  -  MK84  -  PLAN9  -  xnu-8792 
SearchContext: -  none  -  3  -  10 

    1 /*-
    2  * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
    3  *
    4  * Copyright (c) 2014, 2017 Mark Johnston <markj@FreeBSD.org>
    5  * Copyright (c) 2017 Conrad Meyer <cem@FreeBSD.org>
    6  *
    7  * Redistribution and use in source and binary forms, with or without
    8  * modification, are permitted provided that the following conditions are
    9  * met:
   10  * 1. Redistributions of source code must retain the above copyright
   11  *    notice, this list of conditions and the following disclaimer.
   12  * 2. Redistributions in binary form must reproduce the above copyright
   13  *    notice, this list of conditions and the following disclaimer in
   14  *    the documentation and/or other materials provided with the
   15  *    distribution.
   16  *
   17  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
   18  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
   19  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
   20  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
   21  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
   22  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
   23  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
   24  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
   25  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
   26  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
   27  * SUCH DAMAGE.
   28  */
   29 
   30 /*
   31  * Subroutines used for writing compressed user process and kernel core dumps.
   32  */
   33 
   34 #include <sys/cdefs.h>
   35 __FBSDID("$FreeBSD$");
   36 
   37 #include "opt_gzio.h"
   38 #include "opt_zstdio.h"
   39 
   40 #include <sys/param.h>
   41 #include <sys/systm.h>
   42 
   43 #include <sys/compressor.h>
   44 #include <sys/kernel.h>
   45 #include <sys/linker_set.h>
   46 #include <sys/malloc.h>
   47 
   48 MALLOC_DEFINE(M_COMPRESS, "compressor", "kernel compression subroutines");
   49 
   50 struct compressor_methods {
   51         int format;
   52         void *(* const init)(size_t, int);
   53         void (* const reset)(void *);
   54         int (* const write)(void *, void *, size_t, compressor_cb_t, void *);
   55         void (* const fini)(void *);
   56 };
   57 
   58 struct compressor {
   59         const struct compressor_methods *methods;
   60         compressor_cb_t cb;
   61         void *priv;
   62         void *arg;
   63 };
   64 
   65 SET_DECLARE(compressors, struct compressor_methods);
   66 
   67 #ifdef GZIO
   68 
   69 #include <contrib/zlib/zutil.h>
   70 
   71 struct gz_stream {
   72         uint8_t         *gz_buffer;     /* output buffer */
   73         size_t          gz_bufsz;       /* output buffer size */
   74         off_t           gz_off;         /* offset into the output stream */
   75         uint32_t        gz_crc;         /* stream CRC32 */
   76         z_stream        gz_stream;      /* zlib state */
   77 };
   78 
   79 static void     *gz_init(size_t maxiosize, int level);
   80 static void     gz_reset(void *stream);
   81 static int      gz_write(void *stream, void *data, size_t len, compressor_cb_t,
   82                     void *);
   83 static void     gz_fini(void *stream);
   84 
   85 static void *
   86 gz_alloc(void *arg __unused, u_int n, u_int sz)
   87 {
   88 
   89         /*
   90          * Memory for zlib state is allocated using M_NODUMP since it may be
   91          * used to compress a kernel dump, and we don't want zlib to attempt to
   92          * compress its own state.
   93          */
   94         return (malloc(n * sz, M_COMPRESS, M_WAITOK | M_ZERO | M_NODUMP));
   95 }
   96 
   97 static void
   98 gz_free(void *arg __unused, void *ptr)
   99 {
  100 
  101         free(ptr, M_COMPRESS);
  102 }
  103 
  104 static void *
  105 gz_init(size_t maxiosize, int level)
  106 {
  107         struct gz_stream *s;
  108         int error;
  109 
  110         s = gz_alloc(NULL, 1, roundup2(sizeof(*s), PAGE_SIZE));
  111         s->gz_buffer = gz_alloc(NULL, 1, maxiosize);
  112         s->gz_bufsz = maxiosize;
  113 
  114         s->gz_stream.zalloc = gz_alloc;
  115         s->gz_stream.zfree = gz_free;
  116         s->gz_stream.opaque = NULL;
  117         s->gz_stream.next_in = Z_NULL;
  118         s->gz_stream.avail_in = 0;
  119 
  120         if (level != Z_DEFAULT_COMPRESSION) {
  121                 if (level < Z_BEST_SPEED)
  122                         level = Z_BEST_SPEED;
  123                 else if (level > Z_BEST_COMPRESSION)
  124                         level = Z_BEST_COMPRESSION;
  125         }
  126 
  127         error = deflateInit2(&s->gz_stream, level, Z_DEFLATED, -MAX_WBITS,
  128             DEF_MEM_LEVEL, Z_DEFAULT_STRATEGY);
  129         if (error != 0)
  130                 goto fail;
  131 
  132         gz_reset(s);
  133 
  134         return (s);
  135 
  136 fail:
  137         gz_free(NULL, s);
  138         return (NULL);
  139 }
  140 
  141 static void
  142 gz_reset(void *stream)
  143 {
  144         struct gz_stream *s;
  145         uint8_t *hdr;
  146         const size_t hdrlen = 10;
  147 
  148         s = stream;
  149         s->gz_off = 0;
  150         s->gz_crc = crc32(0L, Z_NULL, 0);
  151 
  152         (void)deflateReset(&s->gz_stream);
  153         s->gz_stream.avail_out = s->gz_bufsz;
  154         s->gz_stream.next_out = s->gz_buffer;
  155 
  156         /* Write the gzip header to the output buffer. */
  157         hdr = s->gz_buffer;
  158         memset(hdr, 0, hdrlen);
  159         hdr[0] = 0x1f;
  160         hdr[1] = 0x8b;
  161         hdr[2] = Z_DEFLATED;
  162         hdr[9] = OS_CODE;
  163         s->gz_stream.next_out += hdrlen;
  164         s->gz_stream.avail_out -= hdrlen;
  165 }
  166 
  167 static int
  168 gz_write(void *stream, void *data, size_t len, compressor_cb_t cb,
  169     void *arg)
  170 {
  171         struct gz_stream *s;
  172         uint8_t trailer[8];
  173         size_t room;
  174         int error, zerror, zflag;
  175 
  176         s = stream;
  177         zflag = data == NULL ? Z_FINISH : Z_NO_FLUSH;
  178 
  179         if (len > 0) {
  180                 s->gz_stream.avail_in = len;
  181                 s->gz_stream.next_in = data;
  182                 s->gz_crc = crc32(s->gz_crc, data, len);
  183         }
  184 
  185         error = 0;
  186         do {
  187                 zerror = deflate(&s->gz_stream, zflag);
  188                 if (zerror != Z_OK && zerror != Z_STREAM_END) {
  189                         error = EIO;
  190                         break;
  191                 }
  192 
  193                 if (s->gz_stream.avail_out == 0 || zerror == Z_STREAM_END) {
  194                         /*
  195                          * Our output buffer is full or there's nothing left
  196                          * to produce, so we're flushing the buffer.
  197                          */
  198                         len = s->gz_bufsz - s->gz_stream.avail_out;
  199                         if (zerror == Z_STREAM_END) {
  200                                 /*
  201                                  * Try to pack as much of the trailer into the
  202                                  * output buffer as we can.
  203                                  */
  204                                 ((uint32_t *)trailer)[0] = s->gz_crc;
  205                                 ((uint32_t *)trailer)[1] =
  206                                     s->gz_stream.total_in;
  207                                 room = MIN(sizeof(trailer),
  208                                     s->gz_bufsz - len);
  209                                 memcpy(s->gz_buffer + len, trailer, room);
  210                                 len += room;
  211                         }
  212 
  213                         error = cb(s->gz_buffer, len, s->gz_off, arg);
  214                         if (error != 0)
  215                                 break;
  216 
  217                         s->gz_off += len;
  218                         s->gz_stream.next_out = s->gz_buffer;
  219                         s->gz_stream.avail_out = s->gz_bufsz;
  220 
  221                         /*
  222                          * If we couldn't pack the trailer into the output
  223                          * buffer, write it out now.
  224                          */
  225                         if (zerror == Z_STREAM_END && room < sizeof(trailer))
  226                                 error = cb(trailer + room,
  227                                     sizeof(trailer) - room, s->gz_off, arg);
  228                 }
  229         } while (zerror != Z_STREAM_END &&
  230             (zflag == Z_FINISH || s->gz_stream.avail_in > 0));
  231 
  232         return (error);
  233 }
  234 
  235 static void
  236 gz_fini(void *stream)
  237 {
  238         struct gz_stream *s;
  239 
  240         s = stream;
  241         (void)deflateEnd(&s->gz_stream);
  242         gz_free(NULL, s->gz_buffer);
  243         gz_free(NULL, s);
  244 }
  245 
  246 struct compressor_methods gzip_methods = {
  247         .format = COMPRESS_GZIP,
  248         .init = gz_init,
  249         .reset = gz_reset,
  250         .write = gz_write,
  251         .fini = gz_fini,
  252 };
  253 DATA_SET(compressors, gzip_methods);
  254 
  255 #endif /* GZIO */
  256 
  257 #ifdef ZSTDIO
  258 
  259 #define ZSTD_STATIC_LINKING_ONLY
  260 #include <contrib/zstd/lib/zstd.h>
  261 
  262 struct zstdio_stream {
  263         ZSTD_CCtx       *zst_stream;
  264         ZSTD_inBuffer   zst_inbuffer;
  265         ZSTD_outBuffer  zst_outbuffer;
  266         uint8_t *       zst_buffer;     /* output buffer */
  267         size_t          zst_maxiosz;    /* Max output IO size */
  268         off_t           zst_off;        /* offset into the output stream */
  269         void *          zst_static_wkspc;
  270 };
  271 
  272 static void     *zstdio_init(size_t maxiosize, int level);
  273 static void     zstdio_reset(void *stream);
  274 static int      zstdio_write(void *stream, void *data, size_t len,
  275                     compressor_cb_t, void *);
  276 static void     zstdio_fini(void *stream);
  277 
  278 static void *
  279 zstdio_init(size_t maxiosize, int level)
  280 {
  281         ZSTD_CCtx *dump_compressor;
  282         struct zstdio_stream *s;
  283         void *wkspc, *owkspc, *buffer;
  284         size_t wkspc_size, buf_size, rc;
  285 
  286         s = NULL;
  287         wkspc_size = ZSTD_estimateCStreamSize(level);
  288         owkspc = wkspc = malloc(wkspc_size + 8, M_COMPRESS,
  289             M_WAITOK | M_NODUMP);
  290         /* Zstd API requires 8-byte alignment. */
  291         if ((uintptr_t)wkspc % 8 != 0)
  292                 wkspc = (void *)roundup2((uintptr_t)wkspc, 8);
  293 
  294         dump_compressor = ZSTD_initStaticCCtx(wkspc, wkspc_size);
  295         if (dump_compressor == NULL) {
  296                 printf("%s: workspace too small.\n", __func__);
  297                 goto out;
  298         }
  299 
  300         rc = ZSTD_CCtx_setParameter(dump_compressor, ZSTD_c_checksumFlag, 1);
  301         if (ZSTD_isError(rc)) {
  302                 printf("%s: error setting checksumFlag: %s\n", __func__,
  303                     ZSTD_getErrorName(rc));
  304                 goto out;
  305         }
  306         rc = ZSTD_CCtx_setParameter(dump_compressor, ZSTD_c_compressionLevel,
  307             level);
  308         if (ZSTD_isError(rc)) {
  309                 printf("%s: error setting compressLevel: %s\n", __func__,
  310                     ZSTD_getErrorName(rc));
  311                 goto out;
  312         }
  313 
  314         buf_size = ZSTD_CStreamOutSize() * 2;
  315         buffer = malloc(buf_size, M_COMPRESS, M_WAITOK | M_NODUMP);
  316 
  317         s = malloc(sizeof(*s), M_COMPRESS, M_NODUMP | M_WAITOK);
  318         s->zst_buffer = buffer;
  319         s->zst_outbuffer.dst = buffer;
  320         s->zst_outbuffer.size = buf_size;
  321         s->zst_maxiosz = maxiosize;
  322         s->zst_stream = dump_compressor;
  323         s->zst_static_wkspc = owkspc;
  324 
  325         zstdio_reset(s);
  326 
  327 out:
  328         if (s == NULL)
  329                 free(owkspc, M_COMPRESS);
  330         return (s);
  331 }
  332 
  333 static void
  334 zstdio_reset(void *stream)
  335 {
  336         struct zstdio_stream *s;
  337         size_t res;
  338 
  339         s = stream;
  340         res = ZSTD_resetCStream(s->zst_stream, 0);
  341         if (ZSTD_isError(res))
  342                 panic("%s: could not reset stream %p: %s\n", __func__, s,
  343                     ZSTD_getErrorName(res));
  344 
  345         s->zst_off = 0;
  346         s->zst_inbuffer.src = NULL;
  347         s->zst_inbuffer.size = 0;
  348         s->zst_inbuffer.pos = 0;
  349         s->zst_outbuffer.pos = 0;
  350 }
  351 
  352 static int
  353 zst_flush_intermediate(struct zstdio_stream *s, compressor_cb_t cb, void *arg)
  354 {
  355         size_t bytes_to_dump;
  356         int error;
  357 
  358         /* Flush as many full output blocks as possible. */
  359         /* XXX: 4096 is arbitrary safe HDD block size for kernel dumps */
  360         while (s->zst_outbuffer.pos >= 4096) {
  361                 bytes_to_dump = rounddown(s->zst_outbuffer.pos, 4096);
  362 
  363                 if (bytes_to_dump > s->zst_maxiosz)
  364                         bytes_to_dump = s->zst_maxiosz;
  365 
  366                 error = cb(s->zst_buffer, bytes_to_dump, s->zst_off, arg);
  367                 if (error != 0)
  368                         return (error);
  369 
  370                 /*
  371                  * Shift any non-full blocks up to the front of the output
  372                  * buffer.
  373                  */
  374                 s->zst_outbuffer.pos -= bytes_to_dump;
  375                 memmove(s->zst_outbuffer.dst,
  376                     (char *)s->zst_outbuffer.dst + bytes_to_dump,
  377                     s->zst_outbuffer.pos);
  378                 s->zst_off += bytes_to_dump;
  379         }
  380         return (0);
  381 }
  382 
  383 static int
  384 zstdio_flush(struct zstdio_stream *s, compressor_cb_t cb, void *arg)
  385 {
  386         size_t rc, lastpos;
  387         int error;
  388 
  389         /*
  390          * Positive return indicates unflushed data remaining; need to call
  391          * endStream again after clearing out room in output buffer.
  392          */
  393         rc = 1;
  394         lastpos = s->zst_outbuffer.pos;
  395         while (rc > 0) {
  396                 rc = ZSTD_endStream(s->zst_stream, &s->zst_outbuffer);
  397                 if (ZSTD_isError(rc)) {
  398                         printf("%s: ZSTD_endStream failed (%s)\n", __func__,
  399                             ZSTD_getErrorName(rc));
  400                         return (EIO);
  401                 }
  402                 if (lastpos == s->zst_outbuffer.pos) {
  403                         printf("%s: did not make forward progress endStream %zu\n",
  404                             __func__, lastpos);
  405                         return (EIO);
  406                 }
  407 
  408                 error = zst_flush_intermediate(s, cb, arg);
  409                 if (error != 0)
  410                         return (error);
  411 
  412                 lastpos = s->zst_outbuffer.pos;
  413         }
  414 
  415         /*
  416          * We've already done an intermediate flush, so all full blocks have
  417          * been written.  Only a partial block remains.  Padding happens in a
  418          * higher layer.
  419          */
  420         if (s->zst_outbuffer.pos != 0) {
  421                 error = cb(s->zst_buffer, s->zst_outbuffer.pos, s->zst_off,
  422                     arg);
  423                 if (error != 0)
  424                         return (error);
  425         }
  426 
  427         return (0);
  428 }
  429 
  430 static int
  431 zstdio_write(void *stream, void *data, size_t len, compressor_cb_t cb,
  432     void *arg)
  433 {
  434         struct zstdio_stream *s;
  435         size_t lastpos, rc;
  436         int error;
  437 
  438         s = stream;
  439         if (data == NULL)
  440                 return (zstdio_flush(s, cb, arg));
  441 
  442         s->zst_inbuffer.src = data;
  443         s->zst_inbuffer.size = len;
  444         s->zst_inbuffer.pos = 0;
  445         lastpos = 0;
  446 
  447         while (s->zst_inbuffer.pos < s->zst_inbuffer.size) {
  448                 rc = ZSTD_compressStream(s->zst_stream, &s->zst_outbuffer,
  449                     &s->zst_inbuffer);
  450                 if (ZSTD_isError(rc)) {
  451                         printf("%s: Compress failed on %p! (%s)\n",
  452                             __func__, data, ZSTD_getErrorName(rc));
  453                         return (EIO);
  454                 }
  455 
  456                 if (lastpos == s->zst_inbuffer.pos) {
  457                         /*
  458                          * XXX: May need flushStream to make forward progress
  459                          */
  460                         printf("ZSTD: did not make forward progress @pos %zu\n",
  461                             lastpos);
  462                         return (EIO);
  463                 }
  464                 lastpos = s->zst_inbuffer.pos;
  465 
  466                 error = zst_flush_intermediate(s, cb, arg);
  467                 if (error != 0)
  468                         return (error);
  469         }
  470         return (0);
  471 }
  472 
  473 static void
  474 zstdio_fini(void *stream)
  475 {
  476         struct zstdio_stream *s;
  477 
  478         s = stream;
  479         if (s->zst_static_wkspc != NULL)
  480                 free(s->zst_static_wkspc, M_COMPRESS);
  481         else
  482                 ZSTD_freeCCtx(s->zst_stream);
  483         free(s->zst_buffer, M_COMPRESS);
  484         free(s, M_COMPRESS);
  485 }
  486 
  487 static struct compressor_methods zstd_methods = {
  488         .format = COMPRESS_ZSTD,
  489         .init = zstdio_init,
  490         .reset = zstdio_reset,
  491         .write = zstdio_write,
  492         .fini = zstdio_fini,
  493 };
  494 DATA_SET(compressors, zstd_methods);
  495 
  496 #endif /* ZSTDIO */
  497 
  498 bool
  499 compressor_avail(int format)
  500 {
  501         struct compressor_methods **iter;
  502 
  503         SET_FOREACH(iter, compressors) {
  504                 if ((*iter)->format == format)
  505                         return (true);
  506         }
  507         return (false);
  508 }
  509 
  510 struct compressor *
  511 compressor_init(compressor_cb_t cb, int format, size_t maxiosize, int level,
  512     void *arg)
  513 {
  514         struct compressor_methods **iter;
  515         struct compressor *s;
  516         void *priv;
  517 
  518         SET_FOREACH(iter, compressors) {
  519                 if ((*iter)->format == format)
  520                         break;
  521         }
  522         if (iter == SET_LIMIT(compressors))
  523                 return (NULL);
  524 
  525         priv = (*iter)->init(maxiosize, level);
  526         if (priv == NULL)
  527                 return (NULL);
  528 
  529         s = malloc(sizeof(*s), M_COMPRESS, M_WAITOK | M_ZERO);
  530         s->methods = (*iter);
  531         s->priv = priv;
  532         s->cb = cb;
  533         s->arg = arg;
  534         return (s);
  535 }
  536 
  537 void
  538 compressor_reset(struct compressor *stream)
  539 {
  540 
  541         stream->methods->reset(stream->priv);
  542 }
  543 
  544 int
  545 compressor_write(struct compressor *stream, void *data, size_t len)
  546 {
  547 
  548         return (stream->methods->write(stream->priv, data, len, stream->cb,
  549             stream->arg));
  550 }
  551 
  552 int
  553 compressor_flush(struct compressor *stream)
  554 {
  555 
  556         return (stream->methods->write(stream->priv, NULL, 0, stream->cb,
  557             stream->arg));
  558 }
  559 
  560 void
  561 compressor_fini(struct compressor *stream)
  562 {
  563 
  564         stream->methods->fini(stream->priv);
  565 }

Cache object: 3e0f2f0b0842e2c403ec6e419f847ebb


[ source navigation ] [ diff markup ] [ identifier search ] [ freetext search ] [ file search ] [ list types ] [ track identifier ]


This page is part of the FreeBSD/Linux Linux Kernel Cross-Reference, and was automatically generated using a modified version of the LXR engine.