1 // SPDX-License-Identifier: GPL-2.0-only 1 2 /* 3 * This file is part of UBIFS. 4 * 5 * Copyright (C) 2006-2008 Nokia Corporation. 6 * Copyright (C) 2006, 2007 University of Szeg 7 * 8 * Authors: Adrian Hunter 9 * Artem Bityutskiy (Битюцкий 10 * Zoltan Sogor 11 */ 12 13 /* 14 * This file provides a single place to access 15 * decompression. 16 */ 17 18 #include <linux/crypto.h> 19 #include "ubifs.h" 20 21 /* Fake description object for the "none" comp 22 static struct ubifs_compressor none_compr = { 23 .compr_type = UBIFS_COMPR_NONE, 24 .name = "none", 25 .capi_name = "", 26 }; 27 28 #ifdef CONFIG_UBIFS_FS_LZO 29 static DEFINE_MUTEX(lzo_mutex); 30 31 static struct ubifs_compressor lzo_compr = { 32 .compr_type = UBIFS_COMPR_LZO, 33 .comp_mutex = &lzo_mutex, 34 .name = "lzo", 35 .capi_name = "lzo", 36 }; 37 #else 38 static struct ubifs_compressor lzo_compr = { 39 .compr_type = UBIFS_COMPR_LZO, 40 .name = "lzo", 41 }; 42 #endif 43 44 #ifdef CONFIG_UBIFS_FS_ZLIB 45 static DEFINE_MUTEX(deflate_mutex); 46 static DEFINE_MUTEX(inflate_mutex); 47 48 static struct ubifs_compressor zlib_compr = { 49 .compr_type = UBIFS_COMPR_ZLIB, 50 .comp_mutex = &deflate_mutex, 51 .decomp_mutex = &inflate_mutex, 52 .name = "zlib", 53 .capi_name = "deflate", 54 }; 55 #else 56 static struct ubifs_compressor zlib_compr = { 57 .compr_type = UBIFS_COMPR_ZLIB, 58 .name = "zlib", 59 }; 60 #endif 61 62 #ifdef CONFIG_UBIFS_FS_ZSTD 63 static DEFINE_MUTEX(zstd_enc_mutex); 64 static DEFINE_MUTEX(zstd_dec_mutex); 65 66 static struct ubifs_compressor zstd_compr = { 67 .compr_type = UBIFS_COMPR_ZSTD, 68 .comp_mutex = &zstd_enc_mutex, 69 .decomp_mutex = &zstd_dec_mutex, 70 .name = "zstd", 71 .capi_name = "zstd", 72 }; 73 #else 74 static struct ubifs_compressor zstd_compr = { 75 .compr_type = UBIFS_COMPR_ZSTD, 76 .name = "zstd", 77 }; 78 #endif 79 80 /* All UBIFS compressors */ 81 struct ubifs_compressor *ubifs_compressors[UBI 82 83 /** 84 * ubifs_compress - compress data. 85 * @c: UBIFS file-system description object 86 * @in_buf: data to compress 87 * @in_len: length of the data to compress 88 * @out_buf: output buffer where compressed da 89 * @out_len: output buffer length is returned 90 * @compr_type: type of compression to use on 91 * type on exit 92 * 93 * This function compresses input buffer @in_b 94 * the result in the output buffer @out_buf an 95 * @out_len. If the input buffer does not comp 96 * @out_buf. The same happens if @compr_type i 97 * compression error occurred. 98 * 99 * Note, if the input buffer was not compresse 100 * buffer and %UBIFS_COMPR_NONE is returned in 101 */ 102 void ubifs_compress(const struct ubifs_info *c 103 int in_len, void *out_buf, 104 { 105 int err; 106 struct ubifs_compressor *compr = ubifs 107 108 if (*compr_type == UBIFS_COMPR_NONE) 109 goto no_compr; 110 111 /* If the input data is small, do not 112 if (in_len < UBIFS_MIN_COMPR_LEN) 113 goto no_compr; 114 115 if (compr->comp_mutex) 116 mutex_lock(compr->comp_mutex); 117 err = crypto_comp_compress(compr->cc, 118 (unsigned i 119 if (compr->comp_mutex) 120 mutex_unlock(compr->comp_mutex 121 if (unlikely(err)) { 122 ubifs_warn(c, "cannot compress 123 in_len, compr->name 124 goto no_compr; 125 } 126 127 /* 128 * If the data compressed only slightl 129 * uncompressed to improve read speed. 130 */ 131 if (in_len - *out_len < UBIFS_MIN_COMP 132 goto no_compr; 133 134 return; 135 136 no_compr: 137 memcpy(out_buf, in_buf, in_len); 138 *out_len = in_len; 139 *compr_type = UBIFS_COMPR_NONE; 140 } 141 142 /** 143 * ubifs_decompress - decompress data. 144 * @c: UBIFS file-system description object 145 * @in_buf: data to decompress 146 * @in_len: length of the data to decompress 147 * @out_buf: output buffer where decompressed 148 * @out_len: output length is returned here 149 * @compr_type: type of compression 150 * 151 * This function decompresses data from buffer 152 * The length of the uncompressed data is retu 153 * returns %0 on success or a negative error c 154 */ 155 int ubifs_decompress(const struct ubifs_info * 156 int in_len, void *out_buf 157 { 158 int err; 159 struct ubifs_compressor *compr; 160 161 if (unlikely(compr_type < 0 || compr_t 162 ubifs_err(c, "invalid compress 163 return -EINVAL; 164 } 165 166 compr = ubifs_compressors[compr_type]; 167 168 if (unlikely(!compr->capi_name)) { 169 ubifs_err(c, "%s compression i 170 return -EINVAL; 171 } 172 173 if (compr_type == UBIFS_COMPR_NONE) { 174 memcpy(out_buf, in_buf, in_len 175 *out_len = in_len; 176 return 0; 177 } 178 179 if (compr->decomp_mutex) 180 mutex_lock(compr->decomp_mutex 181 err = crypto_comp_decompress(compr->cc 182 (unsigned 183 if (compr->decomp_mutex) 184 mutex_unlock(compr->decomp_mut 185 if (err) 186 ubifs_err(c, "cannot decompres 187 in_len, compr->name, 188 189 return err; 190 } 191 192 /** 193 * compr_init - initialize a compressor. 194 * @compr: compressor description object 195 * 196 * This function initializes the requested com 197 * of success or a negative error code in case 198 */ 199 static int __init compr_init(struct ubifs_comp 200 { 201 if (compr->capi_name) { 202 compr->cc = crypto_alloc_comp( 203 if (IS_ERR(compr->cc)) { 204 pr_err("UBIFS error (p 205 current->pid, c 206 return PTR_ERR(compr-> 207 } 208 } 209 210 ubifs_compressors[compr->compr_type] = 211 return 0; 212 } 213 214 /** 215 * compr_exit - de-initialize a compressor. 216 * @compr: compressor description object 217 */ 218 static void compr_exit(struct ubifs_compressor 219 { 220 if (compr->capi_name) 221 crypto_free_comp(compr->cc); 222 } 223 224 /** 225 * ubifs_compressors_init - initialize UBIFS c 226 * 227 * This function initializes the compressor wh 228 * zero in case of success and a negative erro 229 */ 230 int __init ubifs_compressors_init(void) 231 { 232 int err; 233 234 err = compr_init(&lzo_compr); 235 if (err) 236 return err; 237 238 err = compr_init(&zstd_compr); 239 if (err) 240 goto out_lzo; 241 242 err = compr_init(&zlib_compr); 243 if (err) 244 goto out_zstd; 245 246 ubifs_compressors[UBIFS_COMPR_NONE] = 247 return 0; 248 249 out_zstd: 250 compr_exit(&zstd_compr); 251 out_lzo: 252 compr_exit(&lzo_compr); 253 return err; 254 } 255 256 /** 257 * ubifs_compressors_exit - de-initialize UBIF 258 */ 259 void ubifs_compressors_exit(void) 260 { 261 compr_exit(&lzo_compr); 262 compr_exit(&zlib_compr); 263 compr_exit(&zstd_compr); 264 } 265
Linux® is a registered trademark of Linus Torvalds in the United States and other countries.
TOMOYO® is a registered trademark of NTT DATA CORPORATION.