18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
28c2ecf20Sopenharmony_ci/*
38c2ecf20Sopenharmony_ci * uncompress.c
48c2ecf20Sopenharmony_ci *
58c2ecf20Sopenharmony_ci * (C) Copyright 1999 Linus Torvalds
68c2ecf20Sopenharmony_ci *
78c2ecf20Sopenharmony_ci * cramfs interfaces to the uncompression library. There's really just
88c2ecf20Sopenharmony_ci * three entrypoints:
98c2ecf20Sopenharmony_ci *
108c2ecf20Sopenharmony_ci *  - cramfs_uncompress_init() - called to initialize the thing.
118c2ecf20Sopenharmony_ci *  - cramfs_uncompress_exit() - tell me when you're done
128c2ecf20Sopenharmony_ci *  - cramfs_uncompress_block() - uncompress a block.
138c2ecf20Sopenharmony_ci *
148c2ecf20Sopenharmony_ci * NOTE NOTE NOTE! The uncompression is entirely single-threaded. We
158c2ecf20Sopenharmony_ci * only have one stream, and we'll initialize it only once even if it
168c2ecf20Sopenharmony_ci * then is used by multiple filesystems.
178c2ecf20Sopenharmony_ci */
188c2ecf20Sopenharmony_ci
198c2ecf20Sopenharmony_ci#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
208c2ecf20Sopenharmony_ci
218c2ecf20Sopenharmony_ci#include <linux/kernel.h>
228c2ecf20Sopenharmony_ci#include <linux/errno.h>
238c2ecf20Sopenharmony_ci#include <linux/vmalloc.h>
248c2ecf20Sopenharmony_ci#include <linux/zlib.h>
258c2ecf20Sopenharmony_ci#include "internal.h"
268c2ecf20Sopenharmony_ci
278c2ecf20Sopenharmony_cistatic z_stream stream;
288c2ecf20Sopenharmony_cistatic int initialized;
298c2ecf20Sopenharmony_ci
308c2ecf20Sopenharmony_ci/* Returns length of decompressed data. */
318c2ecf20Sopenharmony_ciint cramfs_uncompress_block(void *dst, int dstlen, void *src, int srclen)
328c2ecf20Sopenharmony_ci{
338c2ecf20Sopenharmony_ci	int err;
348c2ecf20Sopenharmony_ci
358c2ecf20Sopenharmony_ci	stream.next_in = src;
368c2ecf20Sopenharmony_ci	stream.avail_in = srclen;
378c2ecf20Sopenharmony_ci
388c2ecf20Sopenharmony_ci	stream.next_out = dst;
398c2ecf20Sopenharmony_ci	stream.avail_out = dstlen;
408c2ecf20Sopenharmony_ci
418c2ecf20Sopenharmony_ci	err = zlib_inflateReset(&stream);
428c2ecf20Sopenharmony_ci	if (err != Z_OK) {
438c2ecf20Sopenharmony_ci		pr_err("zlib_inflateReset error %d\n", err);
448c2ecf20Sopenharmony_ci		zlib_inflateEnd(&stream);
458c2ecf20Sopenharmony_ci		zlib_inflateInit(&stream);
468c2ecf20Sopenharmony_ci	}
478c2ecf20Sopenharmony_ci
488c2ecf20Sopenharmony_ci	err = zlib_inflate(&stream, Z_FINISH);
498c2ecf20Sopenharmony_ci	if (err != Z_STREAM_END)
508c2ecf20Sopenharmony_ci		goto err;
518c2ecf20Sopenharmony_ci	return stream.total_out;
528c2ecf20Sopenharmony_ci
538c2ecf20Sopenharmony_cierr:
548c2ecf20Sopenharmony_ci	pr_err("Error %d while decompressing!\n", err);
558c2ecf20Sopenharmony_ci	pr_err("%p(%d)->%p(%d)\n", src, srclen, dst, dstlen);
568c2ecf20Sopenharmony_ci	return -EIO;
578c2ecf20Sopenharmony_ci}
588c2ecf20Sopenharmony_ci
598c2ecf20Sopenharmony_ciint cramfs_uncompress_init(void)
608c2ecf20Sopenharmony_ci{
618c2ecf20Sopenharmony_ci	if (!initialized++) {
628c2ecf20Sopenharmony_ci		stream.workspace = vmalloc(zlib_inflate_workspacesize());
638c2ecf20Sopenharmony_ci		if (!stream.workspace) {
648c2ecf20Sopenharmony_ci			initialized = 0;
658c2ecf20Sopenharmony_ci			return -ENOMEM;
668c2ecf20Sopenharmony_ci		}
678c2ecf20Sopenharmony_ci		stream.next_in = NULL;
688c2ecf20Sopenharmony_ci		stream.avail_in = 0;
698c2ecf20Sopenharmony_ci		zlib_inflateInit(&stream);
708c2ecf20Sopenharmony_ci	}
718c2ecf20Sopenharmony_ci	return 0;
728c2ecf20Sopenharmony_ci}
738c2ecf20Sopenharmony_ci
748c2ecf20Sopenharmony_civoid cramfs_uncompress_exit(void)
758c2ecf20Sopenharmony_ci{
768c2ecf20Sopenharmony_ci	if (!--initialized) {
778c2ecf20Sopenharmony_ci		zlib_inflateEnd(&stream);
788c2ecf20Sopenharmony_ci		vfree(stream.workspace);
798c2ecf20Sopenharmony_ci	}
808c2ecf20Sopenharmony_ci}
81