Linux 2.6.19.7
[linux-2.6/suspend2-2.6.19.git] / arch / s390 / crypto / aes_s390.c
blob15c9eec02928ce0c68682624ccb146bb88d2081e
1 /*
2 * Cryptographic API.
4 * s390 implementation of the AES Cipher Algorithm.
6 * s390 Version:
7 * Copyright (C) 2005 IBM Deutschland GmbH, IBM Corporation
8 * Author(s): Jan Glauber (jang@de.ibm.com)
10 * Derived from "crypto/aes.c"
12 * This program is free software; you can redistribute it and/or modify it
13 * under the terms of the GNU General Public License as published by the Free
14 * Software Foundation; either version 2 of the License, or (at your option)
15 * any later version.
19 #include <crypto/algapi.h>
20 #include <linux/module.h>
21 #include <linux/init.h>
22 #include "crypt_s390.h"
24 #define AES_MIN_KEY_SIZE 16
25 #define AES_MAX_KEY_SIZE 32
27 /* data block size for all key lengths */
28 #define AES_BLOCK_SIZE 16
30 int has_aes_128 = 0;
31 int has_aes_192 = 0;
32 int has_aes_256 = 0;
34 struct s390_aes_ctx {
35 u8 iv[AES_BLOCK_SIZE];
36 u8 key[AES_MAX_KEY_SIZE];
37 long enc;
38 long dec;
39 int key_len;
42 static int aes_set_key(struct crypto_tfm *tfm, const u8 *in_key,
43 unsigned int key_len)
45 struct s390_aes_ctx *sctx = crypto_tfm_ctx(tfm);
46 u32 *flags = &tfm->crt_flags;
48 switch (key_len) {
49 case 16:
50 if (!has_aes_128)
51 goto fail;
52 break;
53 case 24:
54 if (!has_aes_192)
55 goto fail;
57 break;
58 case 32:
59 if (!has_aes_256)
60 goto fail;
61 break;
62 default:
63 /* invalid key length */
64 goto fail;
65 break;
68 sctx->key_len = key_len;
69 memcpy(sctx->key, in_key, key_len);
70 return 0;
71 fail:
72 *flags |= CRYPTO_TFM_RES_BAD_KEY_LEN;
73 return -EINVAL;
76 static void aes_encrypt(struct crypto_tfm *tfm, u8 *out, const u8 *in)
78 const struct s390_aes_ctx *sctx = crypto_tfm_ctx(tfm);
80 switch (sctx->key_len) {
81 case 16:
82 crypt_s390_km(KM_AES_128_ENCRYPT, &sctx->key, out, in,
83 AES_BLOCK_SIZE);
84 break;
85 case 24:
86 crypt_s390_km(KM_AES_192_ENCRYPT, &sctx->key, out, in,
87 AES_BLOCK_SIZE);
88 break;
89 case 32:
90 crypt_s390_km(KM_AES_256_ENCRYPT, &sctx->key, out, in,
91 AES_BLOCK_SIZE);
92 break;
96 static void aes_decrypt(struct crypto_tfm *tfm, u8 *out, const u8 *in)
98 const struct s390_aes_ctx *sctx = crypto_tfm_ctx(tfm);
100 switch (sctx->key_len) {
101 case 16:
102 crypt_s390_km(KM_AES_128_DECRYPT, &sctx->key, out, in,
103 AES_BLOCK_SIZE);
104 break;
105 case 24:
106 crypt_s390_km(KM_AES_192_DECRYPT, &sctx->key, out, in,
107 AES_BLOCK_SIZE);
108 break;
109 case 32:
110 crypt_s390_km(KM_AES_256_DECRYPT, &sctx->key, out, in,
111 AES_BLOCK_SIZE);
112 break;
117 static struct crypto_alg aes_alg = {
118 .cra_name = "aes",
119 .cra_driver_name = "aes-s390",
120 .cra_priority = CRYPT_S390_PRIORITY,
121 .cra_flags = CRYPTO_ALG_TYPE_CIPHER,
122 .cra_blocksize = AES_BLOCK_SIZE,
123 .cra_ctxsize = sizeof(struct s390_aes_ctx),
124 .cra_module = THIS_MODULE,
125 .cra_list = LIST_HEAD_INIT(aes_alg.cra_list),
126 .cra_u = {
127 .cipher = {
128 .cia_min_keysize = AES_MIN_KEY_SIZE,
129 .cia_max_keysize = AES_MAX_KEY_SIZE,
130 .cia_setkey = aes_set_key,
131 .cia_encrypt = aes_encrypt,
132 .cia_decrypt = aes_decrypt,
137 static int ecb_aes_set_key(struct crypto_tfm *tfm, const u8 *in_key,
138 unsigned int key_len)
140 struct s390_aes_ctx *sctx = crypto_tfm_ctx(tfm);
142 switch (key_len) {
143 case 16:
144 sctx->enc = KM_AES_128_ENCRYPT;
145 sctx->dec = KM_AES_128_DECRYPT;
146 break;
147 case 24:
148 sctx->enc = KM_AES_192_ENCRYPT;
149 sctx->dec = KM_AES_192_DECRYPT;
150 break;
151 case 32:
152 sctx->enc = KM_AES_256_ENCRYPT;
153 sctx->dec = KM_AES_256_DECRYPT;
154 break;
157 return aes_set_key(tfm, in_key, key_len);
160 static int ecb_aes_crypt(struct blkcipher_desc *desc, long func, void *param,
161 struct blkcipher_walk *walk)
163 int ret = blkcipher_walk_virt(desc, walk);
164 unsigned int nbytes;
166 while ((nbytes = walk->nbytes)) {
167 /* only use complete blocks */
168 unsigned int n = nbytes & ~(AES_BLOCK_SIZE - 1);
169 u8 *out = walk->dst.virt.addr;
170 u8 *in = walk->src.virt.addr;
172 ret = crypt_s390_km(func, param, out, in, n);
173 BUG_ON((ret < 0) || (ret != n));
175 nbytes &= AES_BLOCK_SIZE - 1;
176 ret = blkcipher_walk_done(desc, walk, nbytes);
179 return ret;
182 static int ecb_aes_encrypt(struct blkcipher_desc *desc,
183 struct scatterlist *dst, struct scatterlist *src,
184 unsigned int nbytes)
186 struct s390_aes_ctx *sctx = crypto_blkcipher_ctx(desc->tfm);
187 struct blkcipher_walk walk;
189 blkcipher_walk_init(&walk, dst, src, nbytes);
190 return ecb_aes_crypt(desc, sctx->enc, sctx->key, &walk);
193 static int ecb_aes_decrypt(struct blkcipher_desc *desc,
194 struct scatterlist *dst, struct scatterlist *src,
195 unsigned int nbytes)
197 struct s390_aes_ctx *sctx = crypto_blkcipher_ctx(desc->tfm);
198 struct blkcipher_walk walk;
200 blkcipher_walk_init(&walk, dst, src, nbytes);
201 return ecb_aes_crypt(desc, sctx->dec, sctx->key, &walk);
204 static struct crypto_alg ecb_aes_alg = {
205 .cra_name = "ecb(aes)",
206 .cra_driver_name = "ecb-aes-s390",
207 .cra_priority = CRYPT_S390_COMPOSITE_PRIORITY,
208 .cra_flags = CRYPTO_ALG_TYPE_BLKCIPHER,
209 .cra_blocksize = AES_BLOCK_SIZE,
210 .cra_ctxsize = sizeof(struct s390_aes_ctx),
211 .cra_type = &crypto_blkcipher_type,
212 .cra_module = THIS_MODULE,
213 .cra_list = LIST_HEAD_INIT(ecb_aes_alg.cra_list),
214 .cra_u = {
215 .blkcipher = {
216 .min_keysize = AES_MIN_KEY_SIZE,
217 .max_keysize = AES_MAX_KEY_SIZE,
218 .setkey = ecb_aes_set_key,
219 .encrypt = ecb_aes_encrypt,
220 .decrypt = ecb_aes_decrypt,
225 static int cbc_aes_set_key(struct crypto_tfm *tfm, const u8 *in_key,
226 unsigned int key_len)
228 struct s390_aes_ctx *sctx = crypto_tfm_ctx(tfm);
230 switch (key_len) {
231 case 16:
232 sctx->enc = KMC_AES_128_ENCRYPT;
233 sctx->dec = KMC_AES_128_DECRYPT;
234 break;
235 case 24:
236 sctx->enc = KMC_AES_192_ENCRYPT;
237 sctx->dec = KMC_AES_192_DECRYPT;
238 break;
239 case 32:
240 sctx->enc = KMC_AES_256_ENCRYPT;
241 sctx->dec = KMC_AES_256_DECRYPT;
242 break;
245 return aes_set_key(tfm, in_key, key_len);
248 static int cbc_aes_crypt(struct blkcipher_desc *desc, long func, void *param,
249 struct blkcipher_walk *walk)
251 int ret = blkcipher_walk_virt(desc, walk);
252 unsigned int nbytes = walk->nbytes;
254 if (!nbytes)
255 goto out;
257 memcpy(param, walk->iv, AES_BLOCK_SIZE);
258 do {
259 /* only use complete blocks */
260 unsigned int n = nbytes & ~(AES_BLOCK_SIZE - 1);
261 u8 *out = walk->dst.virt.addr;
262 u8 *in = walk->src.virt.addr;
264 ret = crypt_s390_kmc(func, param, out, in, n);
265 BUG_ON((ret < 0) || (ret != n));
267 nbytes &= AES_BLOCK_SIZE - 1;
268 ret = blkcipher_walk_done(desc, walk, nbytes);
269 } while ((nbytes = walk->nbytes));
270 memcpy(walk->iv, param, AES_BLOCK_SIZE);
272 out:
273 return ret;
276 static int cbc_aes_encrypt(struct blkcipher_desc *desc,
277 struct scatterlist *dst, struct scatterlist *src,
278 unsigned int nbytes)
280 struct s390_aes_ctx *sctx = crypto_blkcipher_ctx(desc->tfm);
281 struct blkcipher_walk walk;
283 blkcipher_walk_init(&walk, dst, src, nbytes);
284 return cbc_aes_crypt(desc, sctx->enc, sctx->iv, &walk);
287 static int cbc_aes_decrypt(struct blkcipher_desc *desc,
288 struct scatterlist *dst, struct scatterlist *src,
289 unsigned int nbytes)
291 struct s390_aes_ctx *sctx = crypto_blkcipher_ctx(desc->tfm);
292 struct blkcipher_walk walk;
294 blkcipher_walk_init(&walk, dst, src, nbytes);
295 return cbc_aes_crypt(desc, sctx->dec, sctx->iv, &walk);
298 static struct crypto_alg cbc_aes_alg = {
299 .cra_name = "cbc(aes)",
300 .cra_driver_name = "cbc-aes-s390",
301 .cra_priority = CRYPT_S390_COMPOSITE_PRIORITY,
302 .cra_flags = CRYPTO_ALG_TYPE_BLKCIPHER,
303 .cra_blocksize = AES_BLOCK_SIZE,
304 .cra_ctxsize = sizeof(struct s390_aes_ctx),
305 .cra_type = &crypto_blkcipher_type,
306 .cra_module = THIS_MODULE,
307 .cra_list = LIST_HEAD_INIT(cbc_aes_alg.cra_list),
308 .cra_u = {
309 .blkcipher = {
310 .min_keysize = AES_MIN_KEY_SIZE,
311 .max_keysize = AES_MAX_KEY_SIZE,
312 .ivsize = AES_BLOCK_SIZE,
313 .setkey = cbc_aes_set_key,
314 .encrypt = cbc_aes_encrypt,
315 .decrypt = cbc_aes_decrypt,
320 static int __init aes_init(void)
322 int ret;
324 if (crypt_s390_func_available(KM_AES_128_ENCRYPT))
325 has_aes_128 = 1;
326 if (crypt_s390_func_available(KM_AES_192_ENCRYPT))
327 has_aes_192 = 1;
328 if (crypt_s390_func_available(KM_AES_256_ENCRYPT))
329 has_aes_256 = 1;
331 if (!has_aes_128 && !has_aes_192 && !has_aes_256)
332 return -ENOSYS;
334 ret = crypto_register_alg(&aes_alg);
335 if (ret != 0) {
336 printk(KERN_INFO "crypt_s390: aes-s390 couldn't be loaded.\n");
337 goto aes_err;
340 ret = crypto_register_alg(&ecb_aes_alg);
341 if (ret != 0) {
342 printk(KERN_INFO
343 "crypt_s390: ecb-aes-s390 couldn't be loaded.\n");
344 goto ecb_aes_err;
347 ret = crypto_register_alg(&cbc_aes_alg);
348 if (ret != 0) {
349 printk(KERN_INFO
350 "crypt_s390: cbc-aes-s390 couldn't be loaded.\n");
351 goto cbc_aes_err;
354 out:
355 return ret;
357 cbc_aes_err:
358 crypto_unregister_alg(&ecb_aes_alg);
359 ecb_aes_err:
360 crypto_unregister_alg(&aes_alg);
361 aes_err:
362 goto out;
365 static void __exit aes_fini(void)
367 crypto_unregister_alg(&cbc_aes_alg);
368 crypto_unregister_alg(&ecb_aes_alg);
369 crypto_unregister_alg(&aes_alg);
372 module_init(aes_init);
373 module_exit(aes_fini);
375 MODULE_ALIAS("aes");
377 MODULE_DESCRIPTION("Rijndael (AES) Cipher Algorithm");
378 MODULE_LICENSE("GPL");