xref: /linux/fs/squashfs/zstd_wrapper.c (revision 2b64b2ed277ff23e785fbdb65098ee7e1252d64f)
1 /*
2  * Squashfs - a compressed read only filesystem for Linux
3  *
4  * Copyright (c) 2016-present, Facebook, Inc.
5  * All rights reserved.
6  *
7  * This program is free software; you can redistribute it and/or
8  * modify it under the terms of the GNU General Public License
9  * as published by the Free Software Foundation; either version 2,
10  * or (at your option) any later version.
11  *
12  * This program is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15  * GNU General Public License for more details.
16  *
17  * zstd_wrapper.c
18  */
19 
20 #include <linux/mutex.h>
21 #include <linux/buffer_head.h>
22 #include <linux/slab.h>
23 #include <linux/zstd.h>
24 #include <linux/vmalloc.h>
25 
26 #include "squashfs_fs.h"
27 #include "squashfs_fs_sb.h"
28 #include "squashfs.h"
29 #include "decompressor.h"
30 #include "page_actor.h"
31 
32 struct workspace {
33 	void *mem;
34 	size_t mem_size;
35 	size_t window_size;
36 };
37 
38 static void *zstd_init(struct squashfs_sb_info *msblk, void *buff)
39 {
40 	struct workspace *wksp = kmalloc(sizeof(*wksp), GFP_KERNEL);
41 
42 	if (wksp == NULL)
43 		goto failed;
44 	wksp->window_size = max_t(size_t,
45 			msblk->block_size, SQUASHFS_METADATA_SIZE);
46 	wksp->mem_size = ZSTD_DStreamWorkspaceBound(wksp->window_size);
47 	wksp->mem = vmalloc(wksp->mem_size);
48 	if (wksp->mem == NULL)
49 		goto failed;
50 
51 	return wksp;
52 
53 failed:
54 	ERROR("Failed to allocate zstd workspace\n");
55 	kfree(wksp);
56 	return ERR_PTR(-ENOMEM);
57 }
58 
59 
60 static void zstd_free(void *strm)
61 {
62 	struct workspace *wksp = strm;
63 
64 	if (wksp)
65 		vfree(wksp->mem);
66 	kfree(wksp);
67 }
68 
69 
70 static int zstd_uncompress(struct squashfs_sb_info *msblk, void *strm,
71 	struct buffer_head **bh, int b, int offset, int length,
72 	struct squashfs_page_actor *output)
73 {
74 	struct workspace *wksp = strm;
75 	ZSTD_DStream *stream;
76 	size_t total_out = 0;
77 	size_t zstd_err;
78 	int k = 0;
79 	ZSTD_inBuffer in_buf = { NULL, 0, 0 };
80 	ZSTD_outBuffer out_buf = { NULL, 0, 0 };
81 
82 	stream = ZSTD_initDStream(wksp->window_size, wksp->mem, wksp->mem_size);
83 
84 	if (!stream) {
85 		ERROR("Failed to initialize zstd decompressor\n");
86 		goto out;
87 	}
88 
89 	out_buf.size = PAGE_SIZE;
90 	out_buf.dst = squashfs_first_page(output);
91 
92 	do {
93 		if (in_buf.pos == in_buf.size && k < b) {
94 			int avail = min(length, msblk->devblksize - offset);
95 
96 			length -= avail;
97 			in_buf.src = bh[k]->b_data + offset;
98 			in_buf.size = avail;
99 			in_buf.pos = 0;
100 			offset = 0;
101 		}
102 
103 		if (out_buf.pos == out_buf.size) {
104 			out_buf.dst = squashfs_next_page(output);
105 			if (out_buf.dst == NULL) {
106 				/* Shouldn't run out of pages
107 				 * before stream is done.
108 				 */
109 				squashfs_finish_page(output);
110 				goto out;
111 			}
112 			out_buf.pos = 0;
113 			out_buf.size = PAGE_SIZE;
114 		}
115 
116 		total_out -= out_buf.pos;
117 		zstd_err = ZSTD_decompressStream(stream, &out_buf, &in_buf);
118 		total_out += out_buf.pos; /* add the additional data produced */
119 
120 		if (in_buf.pos == in_buf.size && k < b)
121 			put_bh(bh[k++]);
122 	} while (zstd_err != 0 && !ZSTD_isError(zstd_err));
123 
124 	squashfs_finish_page(output);
125 
126 	if (ZSTD_isError(zstd_err)) {
127 		ERROR("zstd decompression error: %d\n",
128 				(int)ZSTD_getErrorCode(zstd_err));
129 		goto out;
130 	}
131 
132 	if (k < b)
133 		goto out;
134 
135 	return (int)total_out;
136 
137 out:
138 	for (; k < b; k++)
139 		put_bh(bh[k]);
140 
141 	return -EIO;
142 }
143 
144 const struct squashfs_decompressor squashfs_zstd_comp_ops = {
145 	.init = zstd_init,
146 	.free = zstd_free,
147 	.decompress = zstd_uncompress,
148 	.id = ZSTD_COMPRESSION,
149 	.name = "zstd",
150 	.supported = 1
151 };
152