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