123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081 |
- // SPDX-License-Identifier: GPL-2.0
- /*
- * uncompress.c
- *
- * (C) Copyright 1999 Linus Torvalds
- *
- * cramfs interfaces to the uncompression library. There's really just
- * three entrypoints:
- *
- * - cramfs_uncompress_init() - called to initialize the thing.
- * - cramfs_uncompress_exit() - tell me when you're done
- * - cramfs_uncompress_block() - uncompress a block.
- *
- * NOTE NOTE NOTE! The uncompression is entirely single-threaded. We
- * only have one stream, and we'll initialize it only once even if it
- * then is used by multiple filesystems.
- */
- #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
- #include <linux/kernel.h>
- #include <linux/errno.h>
- #include <linux/vmalloc.h>
- #include <linux/zlib.h>
- #include "internal.h"
- static z_stream stream;
- static int initialized;
- /* Returns length of decompressed data. */
- int cramfs_uncompress_block(void *dst, int dstlen, void *src, int srclen)
- {
- int err;
- stream.next_in = src;
- stream.avail_in = srclen;
- stream.next_out = dst;
- stream.avail_out = dstlen;
- err = zlib_inflateReset(&stream);
- if (err != Z_OK) {
- pr_err("zlib_inflateReset error %d\n", err);
- zlib_inflateEnd(&stream);
- zlib_inflateInit(&stream);
- }
- err = zlib_inflate(&stream, Z_FINISH);
- if (err != Z_STREAM_END)
- goto err;
- return stream.total_out;
- err:
- pr_err("Error %d while decompressing!\n", err);
- pr_err("%p(%d)->%p(%d)\n", src, srclen, dst, dstlen);
- return -EIO;
- }
- int cramfs_uncompress_init(void)
- {
- if (!initialized++) {
- stream.workspace = vmalloc(zlib_inflate_workspacesize());
- if (!stream.workspace) {
- initialized = 0;
- return -ENOMEM;
- }
- stream.next_in = NULL;
- stream.avail_in = 0;
- zlib_inflateInit(&stream);
- }
- return 0;
- }
- void cramfs_uncompress_exit(void)
- {
- if (!--initialized) {
- zlib_inflateEnd(&stream);
- vfree(stream.workspace);
- }
- }
|