2019-06-04 08:11:33 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
2015-03-10 08:47:48 +00:00
|
|
|
/*
|
|
|
|
* Accelerated GHASH implementation with ARMv8 vmull.p64 instructions.
|
|
|
|
*
|
2018-08-23 14:48:51 +00:00
|
|
|
* Copyright (C) 2015 - 2018 Linaro Ltd. <ard.biesheuvel@linaro.org>
|
2015-03-10 08:47:48 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <asm/hwcap.h>
|
|
|
|
#include <asm/neon.h>
|
|
|
|
#include <asm/simd.h>
|
|
|
|
#include <asm/unaligned.h>
|
2019-07-02 19:41:40 +00:00
|
|
|
#include <crypto/b128ops.h>
|
2015-03-10 08:47:48 +00:00
|
|
|
#include <crypto/cryptd.h>
|
|
|
|
#include <crypto/internal/hash.h>
|
2019-03-13 05:12:49 +00:00
|
|
|
#include <crypto/internal/simd.h>
|
2015-03-10 08:47:48 +00:00
|
|
|
#include <crypto/gf128mul.h>
|
2017-05-21 10:23:37 +00:00
|
|
|
#include <linux/cpufeature.h>
|
2015-03-10 08:47:48 +00:00
|
|
|
#include <linux/crypto.h>
|
2020-06-29 07:39:25 +00:00
|
|
|
#include <linux/jump_label.h>
|
2015-03-10 08:47:48 +00:00
|
|
|
#include <linux/module.h>
|
|
|
|
|
2019-07-20 06:09:18 +00:00
|
|
|
MODULE_DESCRIPTION("GHASH hash function using ARMv8 Crypto Extensions");
|
2015-03-10 08:47:48 +00:00
|
|
|
MODULE_AUTHOR("Ard Biesheuvel <ard.biesheuvel@linaro.org>");
|
|
|
|
MODULE_LICENSE("GPL v2");
|
2017-07-24 10:28:17 +00:00
|
|
|
MODULE_ALIAS_CRYPTO("ghash");
|
2015-03-10 08:47:48 +00:00
|
|
|
|
|
|
|
#define GHASH_BLOCK_SIZE 16
|
|
|
|
#define GHASH_DIGEST_SIZE 16
|
|
|
|
|
|
|
|
struct ghash_key {
|
2019-07-02 19:41:40 +00:00
|
|
|
be128 k;
|
2020-06-29 07:39:25 +00:00
|
|
|
u64 h[][2];
|
2015-03-10 08:47:48 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct ghash_desc_ctx {
|
|
|
|
u64 digest[GHASH_DIGEST_SIZE/sizeof(u64)];
|
|
|
|
u8 buf[GHASH_BLOCK_SIZE];
|
|
|
|
u32 count;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ghash_async_ctx {
|
|
|
|
struct cryptd_ahash *cryptd_tfm;
|
|
|
|
};
|
|
|
|
|
2017-07-24 10:28:17 +00:00
|
|
|
asmlinkage void pmull_ghash_update_p64(int blocks, u64 dg[], const char *src,
|
2020-06-29 07:39:25 +00:00
|
|
|
u64 const h[][2], const char *head);
|
2017-07-24 10:28:17 +00:00
|
|
|
|
|
|
|
asmlinkage void pmull_ghash_update_p8(int blocks, u64 dg[], const char *src,
|
2020-06-29 07:39:25 +00:00
|
|
|
u64 const h[][2], const char *head);
|
2017-07-24 10:28:17 +00:00
|
|
|
|
2020-06-29 07:39:25 +00:00
|
|
|
static __ro_after_init DEFINE_STATIC_KEY_FALSE(use_p64);
|
2015-03-10 08:47:48 +00:00
|
|
|
|
|
|
|
static int ghash_init(struct shash_desc *desc)
|
|
|
|
{
|
|
|
|
struct ghash_desc_ctx *ctx = shash_desc_ctx(desc);
|
|
|
|
|
|
|
|
*ctx = (struct ghash_desc_ctx){};
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-07-02 19:41:40 +00:00
|
|
|
static void ghash_do_update(int blocks, u64 dg[], const char *src,
|
|
|
|
struct ghash_key *key, const char *head)
|
|
|
|
{
|
|
|
|
if (likely(crypto_simd_usable())) {
|
|
|
|
kernel_neon_begin();
|
2020-06-29 07:39:25 +00:00
|
|
|
if (static_branch_likely(&use_p64))
|
|
|
|
pmull_ghash_update_p64(blocks, dg, src, key->h, head);
|
|
|
|
else
|
|
|
|
pmull_ghash_update_p8(blocks, dg, src, key->h, head);
|
2019-07-02 19:41:40 +00:00
|
|
|
kernel_neon_end();
|
|
|
|
} else {
|
|
|
|
be128 dst = { cpu_to_be64(dg[1]), cpu_to_be64(dg[0]) };
|
|
|
|
|
|
|
|
do {
|
|
|
|
const u8 *in = src;
|
|
|
|
|
|
|
|
if (head) {
|
|
|
|
in = head;
|
|
|
|
blocks++;
|
|
|
|
head = NULL;
|
|
|
|
} else {
|
|
|
|
src += GHASH_BLOCK_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
crypto_xor((u8 *)&dst, in, GHASH_BLOCK_SIZE);
|
|
|
|
gf128mul_lle(&dst, &key->k);
|
|
|
|
} while (--blocks);
|
|
|
|
|
|
|
|
dg[0] = be64_to_cpu(dst.b);
|
|
|
|
dg[1] = be64_to_cpu(dst.a);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-10 08:47:48 +00:00
|
|
|
static int ghash_update(struct shash_desc *desc, const u8 *src,
|
|
|
|
unsigned int len)
|
|
|
|
{
|
|
|
|
struct ghash_desc_ctx *ctx = shash_desc_ctx(desc);
|
|
|
|
unsigned int partial = ctx->count % GHASH_BLOCK_SIZE;
|
|
|
|
|
|
|
|
ctx->count += len;
|
|
|
|
|
|
|
|
if ((partial + len) >= GHASH_BLOCK_SIZE) {
|
|
|
|
struct ghash_key *key = crypto_shash_ctx(desc->tfm);
|
|
|
|
int blocks;
|
|
|
|
|
|
|
|
if (partial) {
|
|
|
|
int p = GHASH_BLOCK_SIZE - partial;
|
|
|
|
|
|
|
|
memcpy(ctx->buf + partial, src, p);
|
|
|
|
src += p;
|
|
|
|
len -= p;
|
|
|
|
}
|
|
|
|
|
|
|
|
blocks = len / GHASH_BLOCK_SIZE;
|
|
|
|
len %= GHASH_BLOCK_SIZE;
|
|
|
|
|
2019-07-02 19:41:40 +00:00
|
|
|
ghash_do_update(blocks, ctx->digest, src, key,
|
|
|
|
partial ? ctx->buf : NULL);
|
2015-03-10 08:47:48 +00:00
|
|
|
src += blocks * GHASH_BLOCK_SIZE;
|
|
|
|
partial = 0;
|
|
|
|
}
|
|
|
|
if (len)
|
|
|
|
memcpy(ctx->buf + partial, src, len);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int ghash_final(struct shash_desc *desc, u8 *dst)
|
|
|
|
{
|
|
|
|
struct ghash_desc_ctx *ctx = shash_desc_ctx(desc);
|
|
|
|
unsigned int partial = ctx->count % GHASH_BLOCK_SIZE;
|
|
|
|
|
|
|
|
if (partial) {
|
|
|
|
struct ghash_key *key = crypto_shash_ctx(desc->tfm);
|
|
|
|
|
|
|
|
memset(ctx->buf + partial, 0, GHASH_BLOCK_SIZE - partial);
|
2019-07-02 19:41:40 +00:00
|
|
|
ghash_do_update(1, ctx->digest, ctx->buf, key, NULL);
|
2015-03-10 08:47:48 +00:00
|
|
|
}
|
|
|
|
put_unaligned_be64(ctx->digest[1], dst);
|
|
|
|
put_unaligned_be64(ctx->digest[0], dst + 8);
|
|
|
|
|
|
|
|
*ctx = (struct ghash_desc_ctx){};
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-08-23 14:48:51 +00:00
|
|
|
static void ghash_reflect(u64 h[], const be128 *k)
|
|
|
|
{
|
|
|
|
u64 carry = be64_to_cpu(k->a) >> 63;
|
|
|
|
|
|
|
|
h[0] = (be64_to_cpu(k->b) << 1) | carry;
|
|
|
|
h[1] = (be64_to_cpu(k->a) << 1) | (be64_to_cpu(k->b) >> 63);
|
|
|
|
|
|
|
|
if (carry)
|
|
|
|
h[1] ^= 0xc200000000000000UL;
|
|
|
|
}
|
|
|
|
|
2015-03-10 08:47:48 +00:00
|
|
|
static int ghash_setkey(struct crypto_shash *tfm,
|
|
|
|
const u8 *inkey, unsigned int keylen)
|
|
|
|
{
|
|
|
|
struct ghash_key *key = crypto_shash_ctx(tfm);
|
|
|
|
|
crypto: remove CRYPTO_TFM_RES_BAD_KEY_LEN
The CRYPTO_TFM_RES_BAD_KEY_LEN flag was apparently meant as a way to
make the ->setkey() functions provide more information about errors.
However, no one actually checks for this flag, which makes it pointless.
Also, many algorithms fail to set this flag when given a bad length key.
Reviewing just the generic implementations, this is the case for
aes-fixed-time, cbcmac, echainiv, nhpoly1305, pcrypt, rfc3686, rfc4309,
rfc7539, rfc7539esp, salsa20, seqiv, and xcbc. But there are probably
many more in arch/*/crypto/ and drivers/crypto/.
Some algorithms can even set this flag when the key is the correct
length. For example, authenc and authencesn set it when the key payload
is malformed in any way (not just a bad length), the atmel-sha and ccree
drivers can set it if a memory allocation fails, and the chelsio driver
sets it for bad auth tag lengths, not just bad key lengths.
So even if someone actually wanted to start checking this flag (which
seems unlikely, since it's been unused for a long time), there would be
a lot of work needed to get it working correctly. But it would probably
be much better to go back to the drawing board and just define different
return values, like -EINVAL if the key is invalid for the algorithm vs.
-EKEYREJECTED if the key was rejected by a policy like "no weak keys".
That would be much simpler, less error-prone, and easier to test.
So just remove this flag.
Signed-off-by: Eric Biggers <ebiggers@google.com>
Reviewed-by: Horia Geantă <horia.geanta@nxp.com>
Signed-off-by: Herbert Xu <herbert@gondor.apana.org.au>
2019-12-31 03:19:36 +00:00
|
|
|
if (keylen != GHASH_BLOCK_SIZE)
|
2015-03-10 08:47:48 +00:00
|
|
|
return -EINVAL;
|
|
|
|
|
2019-07-02 19:41:40 +00:00
|
|
|
/* needed for the fallback */
|
|
|
|
memcpy(&key->k, inkey, GHASH_BLOCK_SIZE);
|
2020-06-29 07:39:25 +00:00
|
|
|
ghash_reflect(key->h[0], &key->k);
|
2018-08-23 14:48:51 +00:00
|
|
|
|
2020-06-29 07:39:25 +00:00
|
|
|
if (static_branch_likely(&use_p64)) {
|
|
|
|
be128 h = key->k;
|
2015-03-10 08:47:48 +00:00
|
|
|
|
2020-06-29 07:39:25 +00:00
|
|
|
gf128mul_lle(&h, &key->k);
|
|
|
|
ghash_reflect(key->h[1], &h);
|
2015-03-10 08:47:48 +00:00
|
|
|
|
2020-06-29 07:39:25 +00:00
|
|
|
gf128mul_lle(&h, &key->k);
|
|
|
|
ghash_reflect(key->h[2], &h);
|
2015-03-10 08:47:48 +00:00
|
|
|
|
2020-06-29 07:39:25 +00:00
|
|
|
gf128mul_lle(&h, &key->k);
|
|
|
|
ghash_reflect(key->h[3], &h);
|
|
|
|
}
|
2015-03-10 08:47:48 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct shash_alg ghash_alg = {
|
|
|
|
.digestsize = GHASH_DIGEST_SIZE,
|
|
|
|
.init = ghash_init,
|
|
|
|
.update = ghash_update,
|
|
|
|
.final = ghash_final,
|
|
|
|
.setkey = ghash_setkey,
|
|
|
|
.descsize = sizeof(struct ghash_desc_ctx),
|
2019-07-02 19:41:40 +00:00
|
|
|
|
|
|
|
.base.cra_name = "ghash",
|
|
|
|
.base.cra_driver_name = "ghash-ce-sync",
|
|
|
|
.base.cra_priority = 300 - 1,
|
|
|
|
.base.cra_blocksize = GHASH_BLOCK_SIZE,
|
2020-06-29 07:39:25 +00:00
|
|
|
.base.cra_ctxsize = sizeof(struct ghash_key) + sizeof(u64[2]),
|
2019-07-02 19:41:40 +00:00
|
|
|
.base.cra_module = THIS_MODULE,
|
2015-03-10 08:47:48 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static int ghash_async_init(struct ahash_request *req)
|
|
|
|
{
|
|
|
|
struct crypto_ahash *tfm = crypto_ahash_reqtfm(req);
|
|
|
|
struct ghash_async_ctx *ctx = crypto_ahash_ctx(tfm);
|
|
|
|
struct ahash_request *cryptd_req = ahash_request_ctx(req);
|
|
|
|
struct cryptd_ahash *cryptd_tfm = ctx->cryptd_tfm;
|
2016-06-21 08:55:17 +00:00
|
|
|
struct shash_desc *desc = cryptd_shash_desc(cryptd_req);
|
|
|
|
struct crypto_shash *child = cryptd_ahash_child(cryptd_tfm);
|
2015-03-10 08:47:48 +00:00
|
|
|
|
2016-06-21 08:55:17 +00:00
|
|
|
desc->tfm = child;
|
|
|
|
return crypto_shash_init(desc);
|
2015-03-10 08:47:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int ghash_async_update(struct ahash_request *req)
|
|
|
|
{
|
|
|
|
struct ahash_request *cryptd_req = ahash_request_ctx(req);
|
2016-06-21 08:55:17 +00:00
|
|
|
struct crypto_ahash *tfm = crypto_ahash_reqtfm(req);
|
|
|
|
struct ghash_async_ctx *ctx = crypto_ahash_ctx(tfm);
|
|
|
|
struct cryptd_ahash *cryptd_tfm = ctx->cryptd_tfm;
|
2015-03-10 08:47:48 +00:00
|
|
|
|
2019-03-13 05:12:49 +00:00
|
|
|
if (!crypto_simd_usable() ||
|
2016-06-21 08:55:17 +00:00
|
|
|
(in_atomic() && cryptd_ahash_queued(cryptd_tfm))) {
|
2015-03-10 08:47:48 +00:00
|
|
|
memcpy(cryptd_req, req, sizeof(*req));
|
|
|
|
ahash_request_set_tfm(cryptd_req, &cryptd_tfm->base);
|
|
|
|
return crypto_ahash_update(cryptd_req);
|
|
|
|
} else {
|
|
|
|
struct shash_desc *desc = cryptd_shash_desc(cryptd_req);
|
|
|
|
return shash_ahash_update(req, desc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int ghash_async_final(struct ahash_request *req)
|
|
|
|
{
|
|
|
|
struct ahash_request *cryptd_req = ahash_request_ctx(req);
|
2016-06-21 08:55:17 +00:00
|
|
|
struct crypto_ahash *tfm = crypto_ahash_reqtfm(req);
|
|
|
|
struct ghash_async_ctx *ctx = crypto_ahash_ctx(tfm);
|
|
|
|
struct cryptd_ahash *cryptd_tfm = ctx->cryptd_tfm;
|
2015-03-10 08:47:48 +00:00
|
|
|
|
2019-03-13 05:12:49 +00:00
|
|
|
if (!crypto_simd_usable() ||
|
2016-06-21 08:55:17 +00:00
|
|
|
(in_atomic() && cryptd_ahash_queued(cryptd_tfm))) {
|
2015-03-10 08:47:48 +00:00
|
|
|
memcpy(cryptd_req, req, sizeof(*req));
|
|
|
|
ahash_request_set_tfm(cryptd_req, &cryptd_tfm->base);
|
|
|
|
return crypto_ahash_final(cryptd_req);
|
|
|
|
} else {
|
|
|
|
struct shash_desc *desc = cryptd_shash_desc(cryptd_req);
|
|
|
|
return crypto_shash_final(desc, req->result);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int ghash_async_digest(struct ahash_request *req)
|
|
|
|
{
|
|
|
|
struct crypto_ahash *tfm = crypto_ahash_reqtfm(req);
|
|
|
|
struct ghash_async_ctx *ctx = crypto_ahash_ctx(tfm);
|
|
|
|
struct ahash_request *cryptd_req = ahash_request_ctx(req);
|
|
|
|
struct cryptd_ahash *cryptd_tfm = ctx->cryptd_tfm;
|
|
|
|
|
2019-03-13 05:12:49 +00:00
|
|
|
if (!crypto_simd_usable() ||
|
2016-06-21 08:55:17 +00:00
|
|
|
(in_atomic() && cryptd_ahash_queued(cryptd_tfm))) {
|
2015-03-10 08:47:48 +00:00
|
|
|
memcpy(cryptd_req, req, sizeof(*req));
|
|
|
|
ahash_request_set_tfm(cryptd_req, &cryptd_tfm->base);
|
|
|
|
return crypto_ahash_digest(cryptd_req);
|
|
|
|
} else {
|
|
|
|
struct shash_desc *desc = cryptd_shash_desc(cryptd_req);
|
|
|
|
struct crypto_shash *child = cryptd_ahash_child(cryptd_tfm);
|
|
|
|
|
|
|
|
desc->tfm = child;
|
|
|
|
return shash_ahash_digest(req, desc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-01 13:25:42 +00:00
|
|
|
static int ghash_async_import(struct ahash_request *req, const void *in)
|
|
|
|
{
|
|
|
|
struct ahash_request *cryptd_req = ahash_request_ctx(req);
|
|
|
|
struct crypto_ahash *tfm = crypto_ahash_reqtfm(req);
|
|
|
|
struct ghash_async_ctx *ctx = crypto_ahash_ctx(tfm);
|
|
|
|
struct shash_desc *desc = cryptd_shash_desc(cryptd_req);
|
|
|
|
|
|
|
|
desc->tfm = cryptd_ahash_child(ctx->cryptd_tfm);
|
|
|
|
|
|
|
|
return crypto_shash_import(desc, in);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int ghash_async_export(struct ahash_request *req, void *out)
|
|
|
|
{
|
|
|
|
struct ahash_request *cryptd_req = ahash_request_ctx(req);
|
|
|
|
struct shash_desc *desc = cryptd_shash_desc(cryptd_req);
|
|
|
|
|
|
|
|
return crypto_shash_export(desc, out);
|
|
|
|
}
|
|
|
|
|
2015-03-10 08:47:48 +00:00
|
|
|
static int ghash_async_setkey(struct crypto_ahash *tfm, const u8 *key,
|
|
|
|
unsigned int keylen)
|
|
|
|
{
|
|
|
|
struct ghash_async_ctx *ctx = crypto_ahash_ctx(tfm);
|
|
|
|
struct crypto_ahash *child = &ctx->cryptd_tfm->base;
|
|
|
|
|
|
|
|
crypto_ahash_clear_flags(child, CRYPTO_TFM_REQ_MASK);
|
|
|
|
crypto_ahash_set_flags(child, crypto_ahash_get_flags(tfm)
|
|
|
|
& CRYPTO_TFM_REQ_MASK);
|
2019-12-31 03:19:38 +00:00
|
|
|
return crypto_ahash_setkey(child, key, keylen);
|
2015-03-10 08:47:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int ghash_async_init_tfm(struct crypto_tfm *tfm)
|
|
|
|
{
|
|
|
|
struct cryptd_ahash *cryptd_tfm;
|
|
|
|
struct ghash_async_ctx *ctx = crypto_tfm_ctx(tfm);
|
|
|
|
|
2019-07-02 19:41:40 +00:00
|
|
|
cryptd_tfm = cryptd_alloc_ahash("ghash-ce-sync", 0, 0);
|
2015-03-10 08:47:48 +00:00
|
|
|
if (IS_ERR(cryptd_tfm))
|
|
|
|
return PTR_ERR(cryptd_tfm);
|
|
|
|
ctx->cryptd_tfm = cryptd_tfm;
|
|
|
|
crypto_ahash_set_reqsize(__crypto_ahash_cast(tfm),
|
|
|
|
sizeof(struct ahash_request) +
|
|
|
|
crypto_ahash_reqsize(&cryptd_tfm->base));
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ghash_async_exit_tfm(struct crypto_tfm *tfm)
|
|
|
|
{
|
|
|
|
struct ghash_async_ctx *ctx = crypto_tfm_ctx(tfm);
|
|
|
|
|
|
|
|
cryptd_free_ahash(ctx->cryptd_tfm);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct ahash_alg ghash_async_alg = {
|
|
|
|
.init = ghash_async_init,
|
|
|
|
.update = ghash_async_update,
|
|
|
|
.final = ghash_async_final,
|
|
|
|
.setkey = ghash_async_setkey,
|
|
|
|
.digest = ghash_async_digest,
|
2016-09-01 13:25:42 +00:00
|
|
|
.import = ghash_async_import,
|
|
|
|
.export = ghash_async_export,
|
2015-03-10 08:47:48 +00:00
|
|
|
.halg.digestsize = GHASH_DIGEST_SIZE,
|
2016-09-01 13:25:42 +00:00
|
|
|
.halg.statesize = sizeof(struct ghash_desc_ctx),
|
2015-03-10 08:47:48 +00:00
|
|
|
.halg.base = {
|
|
|
|
.cra_name = "ghash",
|
|
|
|
.cra_driver_name = "ghash-ce",
|
|
|
|
.cra_priority = 300,
|
2018-06-30 22:16:12 +00:00
|
|
|
.cra_flags = CRYPTO_ALG_ASYNC,
|
2015-03-10 08:47:48 +00:00
|
|
|
.cra_blocksize = GHASH_BLOCK_SIZE,
|
|
|
|
.cra_ctxsize = sizeof(struct ghash_async_ctx),
|
|
|
|
.cra_module = THIS_MODULE,
|
|
|
|
.cra_init = ghash_async_init_tfm,
|
|
|
|
.cra_exit = ghash_async_exit_tfm,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
static int __init ghash_ce_mod_init(void)
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
|
2017-07-24 10:28:17 +00:00
|
|
|
if (!(elf_hwcap & HWCAP_NEON))
|
|
|
|
return -ENODEV;
|
|
|
|
|
2020-06-29 07:39:25 +00:00
|
|
|
if (elf_hwcap2 & HWCAP2_PMULL) {
|
|
|
|
ghash_alg.base.cra_ctxsize += 3 * sizeof(u64[2]);
|
|
|
|
static_branch_enable(&use_p64);
|
|
|
|
}
|
2017-07-24 10:28:17 +00:00
|
|
|
|
2015-03-10 08:47:48 +00:00
|
|
|
err = crypto_register_shash(&ghash_alg);
|
|
|
|
if (err)
|
|
|
|
return err;
|
|
|
|
err = crypto_register_ahash(&ghash_async_alg);
|
|
|
|
if (err)
|
|
|
|
goto err_shash;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
err_shash:
|
|
|
|
crypto_unregister_shash(&ghash_alg);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __exit ghash_ce_mod_exit(void)
|
|
|
|
{
|
|
|
|
crypto_unregister_ahash(&ghash_async_alg);
|
|
|
|
crypto_unregister_shash(&ghash_alg);
|
|
|
|
}
|
|
|
|
|
2017-07-24 10:28:17 +00:00
|
|
|
module_init(ghash_ce_mod_init);
|
2015-03-10 08:47:48 +00:00
|
|
|
module_exit(ghash_ce_mod_exit);
|