Blame view

crypto/cts.c 12.2 KB
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
  /*
   * CTS: Cipher Text Stealing mode
   *
   * COPYRIGHT (c) 2008
   * The Regents of the University of Michigan
   * ALL RIGHTS RESERVED
   *
   * Permission is granted to use, copy, create derivative works
   * and redistribute this software and such derivative works
   * for any purpose, so long as the name of The University of
   * Michigan is not used in any advertising or publicity
   * pertaining to the use of distribution of this software
   * without specific, written prior authorization.  If the
   * above copyright notice or any other identification of the
   * University of Michigan is included in any copy of any
   * portion of this software, then the disclaimer below must
   * also be included.
   *
   * THIS SOFTWARE IS PROVIDED AS IS, WITHOUT REPRESENTATION
   * FROM THE UNIVERSITY OF MICHIGAN AS TO ITS FITNESS FOR ANY
   * PURPOSE, AND WITHOUT WARRANTY BY THE UNIVERSITY OF
   * MICHIGAN OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING
   * WITHOUT LIMITATION THE IMPLIED WARRANTIES OF
   * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE
   * REGENTS OF THE UNIVERSITY OF MICHIGAN SHALL NOT BE LIABLE
   * FOR ANY DAMAGES, INCLUDING SPECIAL, INDIRECT, INCIDENTAL, OR
   * CONSEQUENTIAL DAMAGES, WITH RESPECT TO ANY CLAIM ARISING
   * OUT OF OR IN CONNECTION WITH THE USE OF THE SOFTWARE, EVEN
   * IF IT HAS BEEN OR IS HEREAFTER ADVISED OF THE POSSIBILITY OF
   * SUCH DAMAGES.
   */
  
  /* Derived from various:
   *	Copyright (c) 2006 Herbert Xu <herbert@gondor.apana.org.au>
   */
  
  /*
   * This is the Cipher Text Stealing mode as described by
   * Section 8 of rfc2040 and referenced by rfc3962.
   * rfc3962 includes errata information in its Appendix A.
   */
0605c41cc   Herbert Xu   crypto: cts - Con...
42
  #include <crypto/internal/skcipher.h>
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
43
44
45
46
47
48
49
50
51
52
  #include <linux/err.h>
  #include <linux/init.h>
  #include <linux/kernel.h>
  #include <linux/log2.h>
  #include <linux/module.h>
  #include <linux/scatterlist.h>
  #include <crypto/scatterwalk.h>
  #include <linux/slab.h>
  
  struct crypto_cts_ctx {
0605c41cc   Herbert Xu   crypto: cts - Con...
53
  	struct crypto_skcipher *child;
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
54
  };
0605c41cc   Herbert Xu   crypto: cts - Con...
55
56
57
58
59
60
61
  struct crypto_cts_reqctx {
  	struct scatterlist sg[2];
  	unsigned offset;
  	struct skcipher_request subreq;
  };
  
  static inline u8 *crypto_cts_reqctx_space(struct skcipher_request *req)
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
62
  {
0605c41cc   Herbert Xu   crypto: cts - Con...
63
64
65
66
  	struct crypto_cts_reqctx *rctx = skcipher_request_ctx(req);
  	struct crypto_skcipher *tfm = crypto_skcipher_reqtfm(req);
  	struct crypto_cts_ctx *ctx = crypto_skcipher_ctx(tfm);
  	struct crypto_skcipher *child = ctx->child;
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
67

0605c41cc   Herbert Xu   crypto: cts - Con...
68
69
  	return PTR_ALIGN((u8 *)(rctx + 1) + crypto_skcipher_reqsize(child),
  			 crypto_skcipher_alignmask(tfm) + 1);
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
70
  }
0605c41cc   Herbert Xu   crypto: cts - Con...
71
72
  static int crypto_cts_setkey(struct crypto_skcipher *parent, const u8 *key,
  			     unsigned int keylen)
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
73
  {
0605c41cc   Herbert Xu   crypto: cts - Con...
74
75
  	struct crypto_cts_ctx *ctx = crypto_skcipher_ctx(parent);
  	struct crypto_skcipher *child = ctx->child;
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
76
  	int err;
0605c41cc   Herbert Xu   crypto: cts - Con...
77
78
79
80
81
82
83
84
  	crypto_skcipher_clear_flags(child, CRYPTO_TFM_REQ_MASK);
  	crypto_skcipher_set_flags(child, crypto_skcipher_get_flags(parent) &
  					 CRYPTO_TFM_REQ_MASK);
  	err = crypto_skcipher_setkey(child, key, keylen);
  	crypto_skcipher_set_flags(parent, crypto_skcipher_get_flags(child) &
  					  CRYPTO_TFM_RES_MASK);
  	return err;
  }
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
85

0605c41cc   Herbert Xu   crypto: cts - Con...
86
87
88
  static void cts_cbc_crypt_done(struct crypto_async_request *areq, int err)
  {
  	struct skcipher_request *req = areq->data;
c4913c7b7   Alexey Dobriyan   [CRYPTO] cts: Ini...
89

0605c41cc   Herbert Xu   crypto: cts - Con...
90
91
  	if (err == -EINPROGRESS)
  		return;
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
92

0605c41cc   Herbert Xu   crypto: cts - Con...
93
94
  	skcipher_request_complete(req, err);
  }
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
95

0605c41cc   Herbert Xu   crypto: cts - Con...
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
  static int cts_cbc_encrypt(struct skcipher_request *req)
  {
  	struct crypto_cts_reqctx *rctx = skcipher_request_ctx(req);
  	struct crypto_skcipher *tfm = crypto_skcipher_reqtfm(req);
  	struct skcipher_request *subreq = &rctx->subreq;
  	int bsize = crypto_skcipher_blocksize(tfm);
  	u8 d[bsize * 2] __attribute__ ((aligned(__alignof__(u32))));
  	struct scatterlist *sg;
  	unsigned int offset;
  	int lastn;
  
  	offset = rctx->offset;
  	lastn = req->cryptlen - offset;
  
  	sg = scatterwalk_ffwd(rctx->sg, req->dst, offset - bsize);
  	scatterwalk_map_and_copy(d + bsize, sg, 0, bsize, 0);
  
  	memset(d, 0, bsize);
  	scatterwalk_map_and_copy(d, req->src, offset, lastn, 0);
  
  	scatterwalk_map_and_copy(d, sg, 0, bsize + lastn, 1);
  	memzero_explicit(d, sizeof(d));
  
  	skcipher_request_set_callback(subreq, req->base.flags &
  					      CRYPTO_TFM_REQ_MAY_BACKLOG,
  				      cts_cbc_crypt_done, req);
  	skcipher_request_set_crypt(subreq, sg, sg, bsize, req->iv);
  	return crypto_skcipher_encrypt(subreq);
  }
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
125

0605c41cc   Herbert Xu   crypto: cts - Con...
126
127
128
  static void crypto_cts_encrypt_done(struct crypto_async_request *areq, int err)
  {
  	struct skcipher_request *req = areq->data;
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
129

0605c41cc   Herbert Xu   crypto: cts - Con...
130
131
  	if (err)
  		goto out;
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
132

0605c41cc   Herbert Xu   crypto: cts - Con...
133
134
135
136
  	err = cts_cbc_encrypt(req);
  	if (err == -EINPROGRESS ||
  	    (err == -EBUSY && req->base.flags & CRYPTO_TFM_REQ_MAY_BACKLOG))
  		return;
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
137

0605c41cc   Herbert Xu   crypto: cts - Con...
138
139
140
  out:
  	skcipher_request_complete(req, err);
  }
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
141

0605c41cc   Herbert Xu   crypto: cts - Con...
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
  static int crypto_cts_encrypt(struct skcipher_request *req)
  {
  	struct crypto_skcipher *tfm = crypto_skcipher_reqtfm(req);
  	struct crypto_cts_reqctx *rctx = skcipher_request_ctx(req);
  	struct crypto_cts_ctx *ctx = crypto_skcipher_ctx(tfm);
  	struct skcipher_request *subreq = &rctx->subreq;
  	int bsize = crypto_skcipher_blocksize(tfm);
  	unsigned int nbytes = req->cryptlen;
  	int cbc_blocks = (nbytes + bsize - 1) / bsize - 1;
  	unsigned int offset;
  
  	skcipher_request_set_tfm(subreq, ctx->child);
  
  	if (cbc_blocks <= 0) {
  		skcipher_request_set_callback(subreq, req->base.flags,
  					      req->base.complete,
  					      req->base.data);
  		skcipher_request_set_crypt(subreq, req->src, req->dst, nbytes,
  					   req->iv);
  		return crypto_skcipher_encrypt(subreq);
  	}
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
163

0605c41cc   Herbert Xu   crypto: cts - Con...
164
165
  	offset = cbc_blocks * bsize;
  	rctx->offset = offset;
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
166

0605c41cc   Herbert Xu   crypto: cts - Con...
167
168
169
170
  	skcipher_request_set_callback(subreq, req->base.flags,
  				      crypto_cts_encrypt_done, req);
  	skcipher_request_set_crypt(subreq, req->src, req->dst,
  				   offset, req->iv);
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
171

0605c41cc   Herbert Xu   crypto: cts - Con...
172
173
  	return crypto_skcipher_encrypt(subreq) ?:
  	       cts_cbc_encrypt(req);
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
174
  }
0605c41cc   Herbert Xu   crypto: cts - Con...
175
  static int cts_cbc_decrypt(struct skcipher_request *req)
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
176
  {
0605c41cc   Herbert Xu   crypto: cts - Con...
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
  	struct crypto_cts_reqctx *rctx = skcipher_request_ctx(req);
  	struct crypto_skcipher *tfm = crypto_skcipher_reqtfm(req);
  	struct skcipher_request *subreq = &rctx->subreq;
  	int bsize = crypto_skcipher_blocksize(tfm);
  	u8 d[bsize * 2] __attribute__ ((aligned(__alignof__(u32))));
  	struct scatterlist *sg;
  	unsigned int offset;
  	u8 *space;
  	int lastn;
  
  	offset = rctx->offset;
  	lastn = req->cryptlen - offset;
  
  	sg = scatterwalk_ffwd(rctx->sg, req->dst, offset - bsize);
  
  	/* 1. Decrypt Cn-1 (s) to create Dn */
  	scatterwalk_map_and_copy(d + bsize, sg, 0, bsize, 0);
  	space = crypto_cts_reqctx_space(req);
  	crypto_xor(d + bsize, space, bsize);
  	/* 2. Pad Cn with zeros at the end to create C of length BB */
  	memset(d, 0, bsize);
  	scatterwalk_map_and_copy(d, req->src, offset, lastn, 0);
  	/* 3. Exclusive-or Dn with C to create Xn */
  	/* 4. Select the first Ln bytes of Xn to create Pn */
  	crypto_xor(d + bsize, d, lastn);
  
  	/* 5. Append the tail (BB - Ln) bytes of Xn to Cn to create En */
  	memcpy(d + lastn, d + bsize + lastn, bsize - lastn);
  	/* 6. Decrypt En to create Pn-1 */
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
206

0605c41cc   Herbert Xu   crypto: cts - Con...
207
208
  	scatterwalk_map_and_copy(d, sg, 0, bsize + lastn, 1);
  	memzero_explicit(d, sizeof(d));
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
209

0605c41cc   Herbert Xu   crypto: cts - Con...
210
211
212
213
214
215
  	skcipher_request_set_callback(subreq, req->base.flags &
  					      CRYPTO_TFM_REQ_MAY_BACKLOG,
  				      cts_cbc_crypt_done, req);
  
  	skcipher_request_set_crypt(subreq, sg, sg, bsize, space);
  	return crypto_skcipher_decrypt(subreq);
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
216
  }
0605c41cc   Herbert Xu   crypto: cts - Con...
217
  static void crypto_cts_decrypt_done(struct crypto_async_request *areq, int err)
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
218
  {
0605c41cc   Herbert Xu   crypto: cts - Con...
219
  	struct skcipher_request *req = areq->data;
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
220

0605c41cc   Herbert Xu   crypto: cts - Con...
221
222
  	if (err)
  		goto out;
c4913c7b7   Alexey Dobriyan   [CRYPTO] cts: Ini...
223

0605c41cc   Herbert Xu   crypto: cts - Con...
224
225
226
227
  	err = cts_cbc_decrypt(req);
  	if (err == -EINPROGRESS ||
  	    (err == -EBUSY && req->base.flags & CRYPTO_TFM_REQ_MAY_BACKLOG))
  		return;
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
228

0605c41cc   Herbert Xu   crypto: cts - Con...
229
230
231
  out:
  	skcipher_request_complete(req, err);
  }
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
232

0605c41cc   Herbert Xu   crypto: cts - Con...
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
  static int crypto_cts_decrypt(struct skcipher_request *req)
  {
  	struct crypto_skcipher *tfm = crypto_skcipher_reqtfm(req);
  	struct crypto_cts_reqctx *rctx = skcipher_request_ctx(req);
  	struct crypto_cts_ctx *ctx = crypto_skcipher_ctx(tfm);
  	struct skcipher_request *subreq = &rctx->subreq;
  	int bsize = crypto_skcipher_blocksize(tfm);
  	unsigned int nbytes = req->cryptlen;
  	int cbc_blocks = (nbytes + bsize - 1) / bsize - 1;
  	unsigned int offset;
  	u8 *space;
  
  	skcipher_request_set_tfm(subreq, ctx->child);
  
  	if (cbc_blocks <= 0) {
  		skcipher_request_set_callback(subreq, req->base.flags,
  					      req->base.complete,
  					      req->base.data);
  		skcipher_request_set_crypt(subreq, req->src, req->dst, nbytes,
  					   req->iv);
  		return crypto_skcipher_decrypt(subreq);
  	}
7185ad267   Daniel Borkmann   crypto: memzero_e...
255

0605c41cc   Herbert Xu   crypto: cts - Con...
256
257
  	skcipher_request_set_callback(subreq, req->base.flags,
  				      crypto_cts_decrypt_done, req);
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
258

0605c41cc   Herbert Xu   crypto: cts - Con...
259
  	space = crypto_cts_reqctx_space(req);
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
260

0605c41cc   Herbert Xu   crypto: cts - Con...
261
262
  	offset = cbc_blocks * bsize;
  	rctx->offset = offset;
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
263

0605c41cc   Herbert Xu   crypto: cts - Con...
264
265
266
267
268
  	if (cbc_blocks <= 1)
  		memcpy(space, req->iv, bsize);
  	else
  		scatterwalk_map_and_copy(space, req->src, offset - 2 * bsize,
  					 bsize, 0);
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
269

0605c41cc   Herbert Xu   crypto: cts - Con...
270
271
  	skcipher_request_set_crypt(subreq, req->src, req->dst,
  				   offset, req->iv);
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
272

0605c41cc   Herbert Xu   crypto: cts - Con...
273
274
  	return crypto_skcipher_decrypt(subreq) ?:
  	       cts_cbc_decrypt(req);
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
275
  }
0605c41cc   Herbert Xu   crypto: cts - Con...
276
  static int crypto_cts_init_tfm(struct crypto_skcipher *tfm)
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
277
  {
0605c41cc   Herbert Xu   crypto: cts - Con...
278
279
280
281
282
283
284
  	struct skcipher_instance *inst = skcipher_alg_instance(tfm);
  	struct crypto_skcipher_spawn *spawn = skcipher_instance_ctx(inst);
  	struct crypto_cts_ctx *ctx = crypto_skcipher_ctx(tfm);
  	struct crypto_skcipher *cipher;
  	unsigned reqsize;
  	unsigned bsize;
  	unsigned align;
60425a8ba   Eric Biggers   crypto: skcipher ...
285
  	cipher = crypto_spawn_skcipher(spawn);
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
286
287
288
289
  	if (IS_ERR(cipher))
  		return PTR_ERR(cipher);
  
  	ctx->child = cipher;
0605c41cc   Herbert Xu   crypto: cts - Con...
290
291
292
293
294
295
296
297
298
  
  	align = crypto_skcipher_alignmask(tfm);
  	bsize = crypto_skcipher_blocksize(cipher);
  	reqsize = ALIGN(sizeof(struct crypto_cts_reqctx) +
  			crypto_skcipher_reqsize(cipher),
  			crypto_tfm_ctx_alignment()) +
  		  (align & ~(crypto_tfm_ctx_alignment() - 1)) + bsize;
  
  	crypto_skcipher_set_reqsize(tfm, reqsize);
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
299
300
  	return 0;
  }
0605c41cc   Herbert Xu   crypto: cts - Con...
301
  static void crypto_cts_exit_tfm(struct crypto_skcipher *tfm)
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
302
  {
0605c41cc   Herbert Xu   crypto: cts - Con...
303
304
305
  	struct crypto_cts_ctx *ctx = crypto_skcipher_ctx(tfm);
  
  	crypto_free_skcipher(ctx->child);
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
306
  }
0605c41cc   Herbert Xu   crypto: cts - Con...
307
  static void crypto_cts_free(struct skcipher_instance *inst)
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
308
  {
0605c41cc   Herbert Xu   crypto: cts - Con...
309
310
311
312
313
314
315
316
317
318
319
  	crypto_drop_skcipher(skcipher_instance_ctx(inst));
  	kfree(inst);
  }
  
  static int crypto_cts_create(struct crypto_template *tmpl, struct rtattr **tb)
  {
  	struct crypto_skcipher_spawn *spawn;
  	struct skcipher_instance *inst;
  	struct crypto_attr_type *algt;
  	struct skcipher_alg *alg;
  	const char *cipher_name;
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
320
  	int err;
0605c41cc   Herbert Xu   crypto: cts - Con...
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
  	algt = crypto_get_attr_type(tb);
  	if (IS_ERR(algt))
  		return PTR_ERR(algt);
  
  	if ((algt->type ^ CRYPTO_ALG_TYPE_SKCIPHER) & algt->mask)
  		return -EINVAL;
  
  	cipher_name = crypto_attr_alg_name(tb[1]);
  	if (IS_ERR(cipher_name))
  		return PTR_ERR(cipher_name);
  
  	inst = kzalloc(sizeof(*inst) + sizeof(*spawn), GFP_KERNEL);
  	if (!inst)
  		return -ENOMEM;
  
  	spawn = skcipher_instance_ctx(inst);
  
  	crypto_set_skcipher_spawn(spawn, skcipher_crypto_instance(inst));
a35528eca   Eric Biggers   crypto: skcipher ...
339
340
341
  	err = crypto_grab_skcipher(spawn, cipher_name, 0,
  				   crypto_requires_sync(algt->type,
  							algt->mask));
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
342
  	if (err)
0605c41cc   Herbert Xu   crypto: cts - Con...
343
  		goto err_free_inst;
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
344

0605c41cc   Herbert Xu   crypto: cts - Con...
345
  	alg = crypto_spawn_skcipher_alg(spawn);
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
346

0605c41cc   Herbert Xu   crypto: cts - Con...
347
348
349
  	err = -EINVAL;
  	if (crypto_skcipher_alg_ivsize(alg) != alg->base.cra_blocksize)
  		goto err_drop_spawn;
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
350

0605c41cc   Herbert Xu   crypto: cts - Con...
351
352
  	if (strncmp(alg->base.cra_name, "cbc(", 4))
  		goto err_drop_spawn;
988dc0174   Herbert Xu   crypto: cts - Wee...
353

0605c41cc   Herbert Xu   crypto: cts - Con...
354
355
356
357
  	err = crypto_inst_setname(skcipher_crypto_instance(inst), "cts",
  				  &alg->base);
  	if (err)
  		goto err_drop_spawn;
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
358

0605c41cc   Herbert Xu   crypto: cts - Con...
359
360
361
362
  	inst->alg.base.cra_flags = alg->base.cra_flags & CRYPTO_ALG_ASYNC;
  	inst->alg.base.cra_priority = alg->base.cra_priority;
  	inst->alg.base.cra_blocksize = alg->base.cra_blocksize;
  	inst->alg.base.cra_alignmask = alg->base.cra_alignmask;
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
363
364
  
  	/* We access the data as u32s when xoring. */
0605c41cc   Herbert Xu   crypto: cts - Con...
365
  	inst->alg.base.cra_alignmask |= __alignof__(u32) - 1;
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
366

0605c41cc   Herbert Xu   crypto: cts - Con...
367
368
369
370
  	inst->alg.ivsize = alg->base.cra_blocksize;
  	inst->alg.chunksize = crypto_skcipher_alg_chunksize(alg);
  	inst->alg.min_keysize = crypto_skcipher_alg_min_keysize(alg);
  	inst->alg.max_keysize = crypto_skcipher_alg_max_keysize(alg);
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
371

0605c41cc   Herbert Xu   crypto: cts - Con...
372
  	inst->alg.base.cra_ctxsize = sizeof(struct crypto_cts_ctx);
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
373

0605c41cc   Herbert Xu   crypto: cts - Con...
374
375
  	inst->alg.init = crypto_cts_init_tfm;
  	inst->alg.exit = crypto_cts_exit_tfm;
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
376

0605c41cc   Herbert Xu   crypto: cts - Con...
377
378
379
  	inst->alg.setkey = crypto_cts_setkey;
  	inst->alg.encrypt = crypto_cts_encrypt;
  	inst->alg.decrypt = crypto_cts_decrypt;
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
380

0605c41cc   Herbert Xu   crypto: cts - Con...
381
  	inst->free = crypto_cts_free;
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
382

0605c41cc   Herbert Xu   crypto: cts - Con...
383
384
385
386
387
388
389
390
391
392
  	err = skcipher_register_instance(tmpl, inst);
  	if (err)
  		goto err_drop_spawn;
  
  out:
  	return err;
  
  err_drop_spawn:
  	crypto_drop_skcipher(spawn);
  err_free_inst:
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
393
  	kfree(inst);
0605c41cc   Herbert Xu   crypto: cts - Con...
394
  	goto out;
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
395
396
397
398
  }
  
  static struct crypto_template crypto_cts_tmpl = {
  	.name = "cts",
0605c41cc   Herbert Xu   crypto: cts - Con...
399
  	.create = crypto_cts_create,
76cb95217   Kevin Coffman   [CRYPTO] cts: Add...
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
  	.module = THIS_MODULE,
  };
  
  static int __init crypto_cts_module_init(void)
  {
  	return crypto_register_template(&crypto_cts_tmpl);
  }
  
  static void __exit crypto_cts_module_exit(void)
  {
  	crypto_unregister_template(&crypto_cts_tmpl);
  }
  
  module_init(crypto_cts_module_init);
  module_exit(crypto_cts_module_exit);
  
  MODULE_LICENSE("Dual BSD/GPL");
  MODULE_DESCRIPTION("CTS-CBC CipherText Stealing for CBC");
4943ba16b   Kees Cook   crypto: include c...
418
  MODULE_ALIAS_CRYPTO("cts");