Subject: [Patch] [CRYPTO] add alignment for setkey() (rev 2)

setkey() in {cipher,blkcipher,ablkcipher,hash}.c does not respect the
requested alignment by the algorithm. This patch fixes it. The extra
memory is allocated by kmalloc() with GFP_KERNEL flag.

Signed-off-by: Sebastian Siewior <[email protected]>
Index: ps3-linux/crypto/cipher.c
===================================================================
--- ps3-linux.orig/crypto/cipher.c
+++ ps3-linux/crypto/cipher.c
@@ -20,16 +20,43 @@
#include <linux/string.h>
#include "internal.h"

+static int setkey_unaligned(struct crypto_tfm *tfm, const u8 *key, unsigned int keylen,
+ unsigned long alignmask)
+{
+ struct cipher_alg *cia = &tfm->__crt_alg->cra_cipher;
+ int ret;
+ u8 *buffer, *alignbuffer;
+ unsigned long absize;
+
+ absize = keylen + alignmask;
+ buffer = kmalloc(absize, GFP_KERNEL);
+ if (!buffer)
+ return -ENOMEM;
+
+ alignbuffer = (u8 *)ALIGN((unsigned long)buffer, alignmask + 1);
+ memcpy(alignbuffer, key, keylen);
+ ret = cia->cia_setkey(tfm, alignbuffer, keylen);
+ memset(alignbuffer, 0, absize);
+ kfree(buffer);
+ return ret;
+
+}
+
static int setkey(struct crypto_tfm *tfm, const u8 *key, unsigned int keylen)
{
struct cipher_alg *cia = &tfm->__crt_alg->cra_cipher;
-
+ unsigned long alignmask = crypto_tfm_alg_alignmask(tfm);
+
tfm->crt_flags &= ~CRYPTO_TFM_RES_MASK;
if (keylen < cia->cia_min_keysize || keylen > cia->cia_max_keysize) {
tfm->crt_flags |= CRYPTO_TFM_RES_BAD_KEY_LEN;
return -EINVAL;
- } else
- return cia->cia_setkey(tfm, key, keylen);
+ }
+
+ if ((unsigned long) key & alignmask)
+ return setkey_unaligned(tfm, key, keylen, alignmask);
+
+ return cia->cia_setkey(tfm, key, keylen);
}

static void cipher_crypt_unaligned(void (*fn)(struct crypto_tfm *, u8 *,
Index: ps3-linux/crypto/ablkcipher.c
===================================================================
--- ps3-linux.orig/crypto/ablkcipher.c
+++ ps3-linux/crypto/ablkcipher.c
@@ -19,16 +19,41 @@
#include <linux/module.h>
#include <linux/seq_file.h>

+static int setkey_unaligned(struct crypto_ablkcipher *tfm, const u8 *key, unsigned int keylen,
+ unsigned long alignmask)
+{
+ struct ablkcipher_alg *cipher = crypto_ablkcipher_alg(tfm);
+ int ret;
+ u8 *buffer, *alignbuffer;
+ unsigned long absize;
+
+ absize = keylen + alignmask;
+ buffer = kmalloc(absize, GFP_KERNEL);
+ if (!buffer)
+ return -ENOMEM;
+
+ alignbuffer = (u8 *)ALIGN((unsigned long)buffer, alignmask + 1);
+ memcpy(alignbuffer, key, keylen);
+ ret = cipher->setkey(tfm, alignbuffer, keylen);
+ memset(alignbuffer, 0, absize);
+ kfree(buffer);
+ return ret;
+}
+
static int setkey(struct crypto_ablkcipher *tfm, const u8 *key,
unsigned int keylen)
{
struct ablkcipher_alg *cipher = crypto_ablkcipher_alg(tfm);
+ unsigned long alignmask = crypto_ablkcipher_alignmask(tfm);

if (keylen < cipher->min_keysize || keylen > cipher->max_keysize) {
crypto_ablkcipher_set_flags(tfm, CRYPTO_TFM_RES_BAD_KEY_LEN);
return -EINVAL;
}

+ if ((unsigned long) key & alignmask)
+ return setkey_unaligned(tfm, key, keylen, alignmask);
+
return cipher->setkey(tfm, key, keylen);
}

Index: ps3-linux/crypto/blkcipher.c
===================================================================
--- ps3-linux.orig/crypto/blkcipher.c
+++ ps3-linux/crypto/blkcipher.c
@@ -336,16 +336,41 @@ static int blkcipher_walk_first(struct b
return blkcipher_walk_next(desc, walk);
}

+static int setkey_unaligned(struct crypto_tfm *tfm, const u8 *key, unsigned int keylen,
+ unsigned long alignmask)
+{
+ struct blkcipher_alg *cipher = &tfm->__crt_alg->cra_blkcipher;
+ int ret;
+ u8 *buffer, *alignbuffer;
+ unsigned long absize;
+
+ absize = keylen + alignmask;
+ buffer = kmalloc(absize, GFP_KERNEL);
+ if (!buffer)
+ return -ENOMEM;
+
+ alignbuffer = (u8 *)ALIGN((unsigned long)buffer, alignmask + 1);
+ memcpy(alignbuffer, key, keylen);
+ ret = cipher->setkey(tfm, alignbuffer, keylen);
+ memset(alignbuffer, 0, absize);
+ kfree(buffer);
+ return ret;
+}
+
static int setkey(struct crypto_tfm *tfm, const u8 *key,
unsigned int keylen)
{
struct blkcipher_alg *cipher = &tfm->__crt_alg->cra_blkcipher;
+ unsigned long alignmask = crypto_tfm_alg_alignmask(tfm);

if (keylen < cipher->min_keysize || keylen > cipher->max_keysize) {
tfm->crt_flags |= CRYPTO_TFM_RES_BAD_KEY_LEN;
return -EINVAL;
}

+ if ((unsigned long) key & alignmask)
+ return setkey_unaligned(tfm, key, keylen, alignmask);
+
return cipher->setkey(tfm, key, keylen);
}

Index: ps3-linux/crypto/hash.c
===================================================================
--- ps3-linux.orig/crypto/hash.c
+++ ps3-linux/crypto/hash.c
@@ -22,6 +22,41 @@ static unsigned int crypto_hash_ctxsize(
return alg->cra_ctxsize;
}

+static int hash_setkey_unaligned(struct crypto_hash *crt, const u8 *key,
+ unsigned int keylen, unsigned long alignmask)
+{
+ struct crypto_tfm *tfm = crypto_hash_tfm(crt);
+ struct hash_alg *alg = &tfm->__crt_alg->cra_hash;
+ int ret;
+ u8 *buffer, *alignbuffer;
+ unsigned long absize;
+
+ absize = keylen + alignmask;
+ buffer = kmalloc(absize, GFP_KERNEL);
+ if (!buffer)
+ return -ENOMEM;
+
+ alignbuffer = (u8 *)ALIGN((unsigned long)buffer, alignmask + 1);
+ memcpy(alignbuffer, key, keylen);
+ ret = alg->setkey(crt, alignbuffer, keylen);
+ memset(alignbuffer, 0, absize);
+ kfree(buffer);
+ return ret;
+}
+
+static int hash_setkey(struct crypto_hash *crt, const u8 *key,
+ unsigned int keylen)
+{
+ struct crypto_tfm *tfm = crypto_hash_tfm(crt);
+ struct hash_alg *alg = &tfm->__crt_alg->cra_hash;
+ unsigned int alignmask = crypto_hash_alignmask(crt);
+
+ if ((unsigned long) key & alignmask)
+ return hash_setkey_unaligned(crt, key, keylen, alignmask);
+
+ return alg->setkey(crt, key, keylen);
+}
+
static int crypto_init_hash_ops(struct crypto_tfm *tfm, u32 type, u32 mask)
{
struct hash_tfm *crt = &tfm->crt_hash;
@@ -34,7 +69,7 @@ static int crypto_init_hash_ops(struct c
crt->update = alg->update;
crt->final = alg->final;
crt->digest = alg->digest;
- crt->setkey = alg->setkey;
+ crt->setkey = hash_setkey;
crt->digestsize = alg->digestsize;

return 0;


2007-05-19 07:41:40

by Herbert Xu

[permalink] [raw]
Subject: Re: [Patch] [CRYPTO] add alignment for setkey() (rev 2)

Hi Sebastian:

On Fri, May 18, 2007 at 04:01:30PM +0200, Sebastian Siewior wrote:
> setkey() in {cipher,blkcipher,ablkcipher,hash}.c does not respect the
> requested alignment by the algorithm. This patch fixes it. The extra
> memory is allocated by kmalloc() with GFP_KERNEL flag.

I think we should stick with GFP_ATOMIC for now. Once all the atomic
users have been fixed (if they can be), then we can start changing this
interface.

> Index: ps3-linux/crypto/cipher.c
> ===================================================================
> --- ps3-linux.orig/crypto/cipher.c
> +++ ps3-linux/crypto/cipher.c
> @@ -20,16 +20,43 @@
> #include <linux/string.h>
> #include "internal.h"
>
> +static int setkey_unaligned(struct crypto_tfm *tfm, const u8 *key, unsigned int keylen,
> + unsigned long alignmask)

Minor nit. Could you get rid of the alignmask parameter here and just
read it again from tfm? This should make it easier for certain compiler/
arch combos to generate a tail call.

> static int setkey(struct crypto_tfm *tfm, const u8 *key, unsigned int keylen)
> {
> struct cipher_alg *cia = &tfm->__crt_alg->cra_cipher;
> -
> + unsigned long alignmask = crypto_tfm_alg_alignmask(tfm);
> +
> tfm->crt_flags &= ~CRYPTO_TFM_RES_MASK;
> if (keylen < cia->cia_min_keysize || keylen > cia->cia_max_keysize) {
> tfm->crt_flags |= CRYPTO_TFM_RES_BAD_KEY_LEN;
> return -EINVAL;
> - } else
> - return cia->cia_setkey(tfm, key, keylen);
> + }
> +
> + if ((unsigned long) key & alignmask)

Please kill the space before key.

Thanks,
--
Visit Openswan at http://www.openswan.org/
Email: Herbert Xu ~{PmV>HI~} <[email protected]>
Home Page: http://gondor.apana.org.au/~herbert/
PGP Key: http://gondor.apana.org.au/~herbert/pubkey.txt

Subject: Re: [Patch] [CRYPTO] add alignment for setkey() (rev 2)

* Herbert Xu | 2007-05-19 17:41:36 [+1000]:

>I think we should stick with GFP_ATOMIC for now. Once all the atomic
>users have been fixed (if they can be), then we can start changing this
>interface.
okey.

>> +static int setkey_unaligned(struct crypto_tfm *tfm, const u8 *key, unsigned int keylen,
>> + unsigned long alignmask)
>
>Minor nit. Could you get rid of the alignmask parameter here and just
>read it again from tfm? This should make it easier for certain compiler/
>arch combos to generate a tail call.
sure

>> + if ((unsigned long) key & alignmask)
>
>Please kill the space before key.
okey.

>Thanks,
Sebastian