4 * Deflate algorithm (RFC 1951), implemented here primarily for use
5 * by IPCOMP (RFC 3173 & RFC 2394).
7 * Copyright (c) 2003 James Morris <jmorris@intercode.com.au>
9 * This program is free software; you can redistribute it and/or modify it
10 * under the terms of the GNU General Public License as published by the Free
11 * Software Foundation; either version 2 of the License, or (at your option)
14 * FIXME: deflate transforms will require up to a total of about 436k of kernel
15 * memory on i386 (390k for compression, the rest for decompression), as the
16 * current zlib kernel code uses a worst case pre-allocation system by default.
17 * This needs to be fixed so that the amount of memory required is properly
18 * related to the winbits and memlevel parameters.
20 * The default winbits of 11 should suit most packets, and it may be something
21 * to configure on a per-tfm basis in the future.
23 * Currently, compression history is not maintained between tfm calls, as
24 * it is not needed for IPCOMP and keeps the code simpler. It can be
25 * implemented if someone wants it.
27 #include <linux/init.h>
28 #include <linux/module.h>
29 #include <linux/crypto.h>
30 #include <linux/zlib.h>
31 #include <linux/vmalloc.h>
32 #include <linux/interrupt.h>
34 #include <linux/net.h>
35 #include <crypto/internal/scompress.h>
37 #define DEFLATE_DEF_LEVEL Z_DEFAULT_COMPRESSION
38 #define DEFLATE_DEF_WINBITS 11
39 #define DEFLATE_DEF_MEMLEVEL MAX_MEM_LEVEL
42 struct z_stream_s comp_stream
;
43 struct z_stream_s decomp_stream
;
46 static int deflate_comp_init(struct deflate_ctx
*ctx
)
49 struct z_stream_s
*stream
= &ctx
->comp_stream
;
51 stream
->workspace
= vzalloc(zlib_deflate_workspacesize(
52 -DEFLATE_DEF_WINBITS
, DEFLATE_DEF_MEMLEVEL
));
53 if (!stream
->workspace
) {
57 ret
= zlib_deflateInit2(stream
, DEFLATE_DEF_LEVEL
, Z_DEFLATED
,
58 -DEFLATE_DEF_WINBITS
, DEFLATE_DEF_MEMLEVEL
,
67 vfree(stream
->workspace
);
71 static int deflate_decomp_init(struct deflate_ctx
*ctx
)
74 struct z_stream_s
*stream
= &ctx
->decomp_stream
;
76 stream
->workspace
= vzalloc(zlib_inflate_workspacesize());
77 if (!stream
->workspace
) {
81 ret
= zlib_inflateInit2(stream
, -DEFLATE_DEF_WINBITS
);
89 vfree(stream
->workspace
);
93 static void deflate_comp_exit(struct deflate_ctx
*ctx
)
95 zlib_deflateEnd(&ctx
->comp_stream
);
96 vfree(ctx
->comp_stream
.workspace
);
99 static void deflate_decomp_exit(struct deflate_ctx
*ctx
)
101 zlib_inflateEnd(&ctx
->decomp_stream
);
102 vfree(ctx
->decomp_stream
.workspace
);
105 static int __deflate_init(void *ctx
)
109 ret
= deflate_comp_init(ctx
);
112 ret
= deflate_decomp_init(ctx
);
114 deflate_comp_exit(ctx
);
119 static void *deflate_alloc_ctx(struct crypto_scomp
*tfm
)
121 struct deflate_ctx
*ctx
;
124 ctx
= kzalloc(sizeof(*ctx
), GFP_KERNEL
);
126 return ERR_PTR(-ENOMEM
);
128 ret
= __deflate_init(ctx
);
137 static int deflate_init(struct crypto_tfm
*tfm
)
139 struct deflate_ctx
*ctx
= crypto_tfm_ctx(tfm
);
141 return __deflate_init(ctx
);
144 static void __deflate_exit(void *ctx
)
146 deflate_comp_exit(ctx
);
147 deflate_decomp_exit(ctx
);
150 static void deflate_free_ctx(struct crypto_scomp
*tfm
, void *ctx
)
156 static void deflate_exit(struct crypto_tfm
*tfm
)
158 struct deflate_ctx
*ctx
= crypto_tfm_ctx(tfm
);
163 static int __deflate_compress(const u8
*src
, unsigned int slen
,
164 u8
*dst
, unsigned int *dlen
, void *ctx
)
167 struct deflate_ctx
*dctx
= ctx
;
168 struct z_stream_s
*stream
= &dctx
->comp_stream
;
170 ret
= zlib_deflateReset(stream
);
176 stream
->next_in
= (u8
*)src
;
177 stream
->avail_in
= slen
;
178 stream
->next_out
= (u8
*)dst
;
179 stream
->avail_out
= *dlen
;
181 ret
= zlib_deflate(stream
, Z_FINISH
);
182 if (ret
!= Z_STREAM_END
) {
187 *dlen
= stream
->total_out
;
192 static int deflate_compress(struct crypto_tfm
*tfm
, const u8
*src
,
193 unsigned int slen
, u8
*dst
, unsigned int *dlen
)
195 struct deflate_ctx
*dctx
= crypto_tfm_ctx(tfm
);
197 return __deflate_compress(src
, slen
, dst
, dlen
, dctx
);
200 static int deflate_scompress(struct crypto_scomp
*tfm
, const u8
*src
,
201 unsigned int slen
, u8
*dst
, unsigned int *dlen
,
204 return __deflate_compress(src
, slen
, dst
, dlen
, ctx
);
207 static int __deflate_decompress(const u8
*src
, unsigned int slen
,
208 u8
*dst
, unsigned int *dlen
, void *ctx
)
212 struct deflate_ctx
*dctx
= ctx
;
213 struct z_stream_s
*stream
= &dctx
->decomp_stream
;
215 ret
= zlib_inflateReset(stream
);
221 stream
->next_in
= (u8
*)src
;
222 stream
->avail_in
= slen
;
223 stream
->next_out
= (u8
*)dst
;
224 stream
->avail_out
= *dlen
;
226 ret
= zlib_inflate(stream
, Z_SYNC_FLUSH
);
228 * Work around a bug in zlib, which sometimes wants to taste an extra
229 * byte when being used in the (undocumented) raw deflate mode.
232 if (ret
== Z_OK
&& !stream
->avail_in
&& stream
->avail_out
) {
234 stream
->next_in
= &zerostuff
;
235 stream
->avail_in
= 1;
236 ret
= zlib_inflate(stream
, Z_FINISH
);
238 if (ret
!= Z_STREAM_END
) {
243 *dlen
= stream
->total_out
;
248 static int deflate_decompress(struct crypto_tfm
*tfm
, const u8
*src
,
249 unsigned int slen
, u8
*dst
, unsigned int *dlen
)
251 struct deflate_ctx
*dctx
= crypto_tfm_ctx(tfm
);
253 return __deflate_decompress(src
, slen
, dst
, dlen
, dctx
);
256 static int deflate_sdecompress(struct crypto_scomp
*tfm
, const u8
*src
,
257 unsigned int slen
, u8
*dst
, unsigned int *dlen
,
260 return __deflate_decompress(src
, slen
, dst
, dlen
, ctx
);
263 static struct crypto_alg alg
= {
264 .cra_name
= "deflate",
265 .cra_flags
= CRYPTO_ALG_TYPE_COMPRESS
,
266 .cra_ctxsize
= sizeof(struct deflate_ctx
),
267 .cra_module
= THIS_MODULE
,
268 .cra_init
= deflate_init
,
269 .cra_exit
= deflate_exit
,
270 .cra_u
= { .compress
= {
271 .coa_compress
= deflate_compress
,
272 .coa_decompress
= deflate_decompress
} }
275 static struct scomp_alg scomp
= {
276 .alloc_ctx
= deflate_alloc_ctx
,
277 .free_ctx
= deflate_free_ctx
,
278 .compress
= deflate_scompress
,
279 .decompress
= deflate_sdecompress
,
281 .cra_name
= "deflate",
282 .cra_driver_name
= "deflate-scomp",
283 .cra_module
= THIS_MODULE
,
287 static int __init
deflate_mod_init(void)
291 ret
= crypto_register_alg(&alg
);
295 ret
= crypto_register_scomp(&scomp
);
297 crypto_unregister_alg(&alg
);
304 static void __exit
deflate_mod_fini(void)
306 crypto_unregister_alg(&alg
);
307 crypto_unregister_scomp(&scomp
);
310 module_init(deflate_mod_init
);
311 module_exit(deflate_mod_fini
);
313 MODULE_LICENSE("GPL");
314 MODULE_DESCRIPTION("Deflate Compression Algorithm for IPCOMP");
315 MODULE_AUTHOR("James Morris <jmorris@intercode.com.au>");
316 MODULE_ALIAS_CRYPTO("deflate");